@bambamboole/pdf-ua-template-builder 0.3.1 → 0.4.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","names":[],"sources":["../src/api/pdfUaApi.ts","../src/builder/blocks/blockChrome.ts","../src/builder/primitives/Chip.tsx","../src/builder/canvas/BlockCardPreview.tsx","../src/builder/lib/sensors.ts","../src/builder/lib/records.ts","../src/builder/state/dragDrop.ts","../src/builder/state/editorModel.ts","../src/builder/hooks/useBuilderDragDrop.ts","../src/render/usePdfUaApi.ts","../src/render/RenderContext.tsx","../src/builder/schema/schemaAdapter.ts","../src/builder/state/editorReducer.ts","../src/builder/context/BuilderContext.tsx","../src/builder/controls/inputs.tsx","../src/builder/controls/fieldConverters.ts","../src/builder/controls/BuilderField.tsx","../src/builder/controls/Field.tsx","../src/builder/controls/AlignSelect.tsx","../src/builder/primitives/Button.tsx","../src/builder/blocks/BlockPalette.tsx","../src/builder/blocks/Palette.tsx","../src/builder/lib/displayScale.ts","../src/builder/lib/pageSizes.ts","../src/builder/canvas/columns.ts","../src/builder/canvas/ColumnResizer.tsx","../src/builder/canvas/PageSheet.tsx","../src/builder/state/configUpdates.ts","../src/builder/canvas/BlockDataPreview.tsx","../src/builder/canvas/SortableBlock.tsx","../src/builder/canvas/BuilderCanvas.tsx","../src/builder/canvas/Canvas.tsx","../src/builder/inspector/editors/ImageBlockEditor.tsx","../src/builder/inspector/InspectorShell.tsx","../src/builder/inspector/editors/SortableList.tsx","../src/builder/inspector/editors/SortableRow.tsx","../src/builder/inspector/editors/KeyValueBlockEditor.tsx","../src/builder/inspector/editors/TableBlockEditor.tsx","../src/builder/inspector/BlockContentControls.tsx","../src/builder/inspector/alignOptions.ts","../src/builder/inspector/BlockLayoutControls.tsx","../src/builder/inspector/SpacingControls.tsx","../src/builder/inspector/TypographyControls.tsx","../src/builder/inspector/BlockInspector.tsx","../src/builder/inspector/Inspector.tsx","../src/builder/inspector/DocumentSettings.tsx","../src/builder/inspector/PageSettings.tsx","../src/builder/Builder.tsx","../src/render/PdfPane.tsx","../src/render/Preview.tsx","../src/builder/TemplateBuilder.tsx","../src/builder/schema/invoiceExample.ts","../src/editor/parseTemplate.ts","../src/editor/TemplateEditorContext.tsx","../src/editor/editorTheme.ts","../schemas/template.schema.json","../src/editor/templateSchema.ts","../src/editor/CodeEditor.tsx","../src/editor/TemplateEditor.tsx"],"sourcesContent":["import type {\n RenderOptions,\n Template,\n TemplateData,\n TemplateSchemaResponse,\n} from \"../types/template\";\n\ninterface RenderTemplateRequest {\n template: Template;\n data?: TemplateData;\n options?: RenderOptions;\n}\n\nexport function resolveDefaultApiUrl(configuredApiUrl?: string): string {\n return configuredApiUrl ?? \"\";\n}\n\nfunction joinUrl(baseUrl: string, path: string): string {\n return `${baseUrl.replace(/\\/+$/, \"\")}/${path.replace(/^\\/+/, \"\")}`;\n}\n\nasync function parseError(response: Response): Promise<string> {\n const contentType = response.headers.get(\"content-type\") ?? \"\";\n const fallback = `${response.status} ${response.statusText || \"Request failed\"}`;\n\n if (contentType.includes(\"application/json\")) {\n const payload = (await response.json()) as { error?: string };\n return payload.error || fallback;\n }\n\n const body = (await response.text()).trim();\n return body || fallback;\n}\n\nexport async function fetchTemplateSchema(baseUrl: string): Promise<TemplateSchemaResponse> {\n const response = await fetch(joinUrl(baseUrl, \"/schema\"), {\n headers: {\n Accept: \"application/json\",\n },\n });\n\n if (!response.ok) {\n throw new Error(await parseError(response));\n }\n\n return (await response.json()) as TemplateSchemaResponse;\n}\n\nexport async function renderTemplatePdf(\n baseUrl: string,\n request: RenderTemplateRequest,\n): Promise<Blob> {\n const response = await fetch(joinUrl(baseUrl, \"/render/template\"), {\n method: \"POST\",\n headers: {\n Accept: \"application/pdf\",\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({\n data: {},\n options: {},\n ...request,\n }),\n });\n\n if (!response.ok) {\n throw new Error(await parseError(response));\n }\n\n return response.blob();\n}\n","import type { Block } from \"../../types/generated/template\";\n\nexport interface BlockChrome {\n chip: string;\n label: string;\n}\n\nconst CHROME: Record<string, BlockChrome> = {\n heading: { chip: \"H\", label: \"Heading\" },\n text: { chip: \"¶\", label: \"Text\" },\n html: { chip: \"</>\", label: \"HTML\" },\n image: { chip: \"◇\", label: \"Image\" },\n table: { chip: \"▦\", label: \"Table\" },\n \"key-value\": { chip: \"≡\", label: \"Key-Value\" },\n spacer: { chip: \"↕\", label: \"Spacer\" },\n divider: { chip: \"─\", label: \"Divider\" },\n};\n\nexport function getBlockChrome(type: string): BlockChrome {\n return CHROME[type] ?? { chip: \"?\", label: prettify(type) };\n}\n\nexport function getBlockSummary(block: Block): string {\n switch (block.type) {\n case \"heading\":\n case \"text\":\n return truncate(block.text);\n case \"html\":\n return truncate(block.html);\n case \"image\":\n return truncate(block.alt ?? block.src);\n case \"key-value\": {\n const fields = block.config?.fields ?? [];\n\n return fields.length > 0 ? `${fields.length} field${fields.length === 1 ? \"\" : \"s\"}` : \"\";\n }\n case \"table\": {\n const columns = block.config?.columns ?? [];\n\n return columns.length > 0\n ? `${columns.length} column${columns.length === 1 ? \"\" : \"s\"}`\n : \"\";\n }\n case \"spacer\": {\n const height = block.config?.height;\n\n return typeof height === \"number\" ? `${height}mm` : \"\";\n }\n case \"divider\": {\n const style = block.config?.style;\n\n return typeof style === \"string\" ? style : \"\";\n }\n default:\n return \"\";\n }\n}\n\nfunction truncate(value: string | null | undefined, max = 56): string {\n if (!value) {\n return \"\";\n }\n\n const single = value.replace(/\\s+/g, \" \").trim();\n\n return single.length > max ? `${single.slice(0, max - 1)}…` : single;\n}\n\nfunction prettify(type: string): string {\n return type\n .split(\"-\")\n .filter(Boolean)\n .map((part) => `${part.charAt(0).toUpperCase()}${part.slice(1)}`)\n .join(\" \");\n}\n","import type { ReactNode } from \"react\";\n\nexport interface ChipProps {\n children: ReactNode;\n /** Grow to fit multi-character labels instead of staying a 22×22 square. */\n wide?: boolean;\n}\n\nexport function Chip({ children, wide = false }: ChipProps) {\n return (\n <span\n className={`inline-grid h-[22px] place-items-center rounded bg-surface-muted font-mono text-2xs font-semibold text-fg-muted ${wide ? \"min-w-[22px] px-1\" : \"w-[22px]\"}`}\n aria-hidden=\"true\"\n >\n {children}\n </span>\n );\n}\n","import { getBlockChrome } from \"../blocks/blockChrome\";\nimport { Chip } from \"../primitives/Chip\";\n\nexport interface BlockCardPreviewProps {\n type: string;\n summary?: string;\n prefix?: string;\n}\n\nexport function BlockCardPreview({ type, summary, prefix }: BlockCardPreviewProps) {\n const chrome = getBlockChrome(type);\n\n return (\n <div className=\"pointer-events-none origin-top-left rotate-[1.5deg] scale-[1.02] cursor-grabbing drop-shadow-drag\">\n <div className=\"inline-flex min-w-[180px] max-w-[360px] items-center gap-2 rounded-lg border border-solid border-border-strong bg-surface px-3 py-2 text-sm font-medium\">\n <Chip>{chrome.chip}</Chip>\n <span className=\"inline-flex items-baseline gap-2\">\n <span className=\"font-medium\">\n {prefix}\n {chrome.label}\n </span>\n {summary ? <span className=\"font-normal text-fg-muted\">{summary}</span> : null}\n </span>\n </div>\n </div>\n );\n}\n","import { KeyboardSensor, PointerSensor, useSensor, useSensors } from \"@dnd-kit/core\";\nimport { sortableKeyboardCoordinates } from \"@dnd-kit/sortable\";\n\nexport function useBuilderSensors() {\n return useSensors(\n useSensor(PointerSensor, { activationConstraint: { distance: 4 } }),\n useSensor(KeyboardSensor, { coordinateGetter: sortableKeyboardCoordinates }),\n );\n}\n","export function isRecord(value: unknown): value is Record<string, unknown> {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nexport function omitKey<TValue>(\n record: Record<string, TValue>,\n key: string,\n): Record<string, TValue> {\n if (!(key in record)) {\n return record;\n }\n\n const next = { ...record };\n delete next[key];\n return next;\n}\n\nexport function renameKey<TValue>(\n record: Record<string, TValue>,\n previousKey: string,\n nextKey: string,\n): Record<string, TValue> {\n if (!(previousKey in record)) {\n return record;\n }\n\n const next: Record<string, TValue> = {};\n for (const [key, value] of Object.entries(record)) {\n next[key === previousKey ? nextKey : key] = value;\n }\n return next;\n}\n\nexport function nextKeyIndex(keys: readonly string[], prefix: string): number {\n let index = keys.length + 1;\n\n while (keys.includes(`${prefix}${index}`)) {\n index += 1;\n }\n\n return index;\n}\n","import type { UniqueIdentifier } from \"@dnd-kit/core\";\nimport { isRecord } from \"../lib/records\";\nimport type { EditorArea, EditorModel } from \"./editorModel\";\n\nexport interface DragData {\n source?: string;\n type?: string;\n rowUid?: string;\n blockUid?: string;\n area?: EditorArea;\n}\n\nexport interface DropTarget {\n rowUid: string | null;\n index: number;\n area: EditorArea;\n}\n\nexport function getDragData(value: unknown): DragData {\n return isRecord(value) ? (value as DragData) : {};\n}\n\nexport function getDropTarget(\n model: EditorModel,\n overId: UniqueIdentifier | undefined,\n overData: DragData,\n): DropTarget {\n if (overId === \"new-row\") {\n return { rowUid: null, index: 0, area: \"body\" };\n }\n if (overId === \"new-footer-row\") {\n return { rowUid: null, index: 0, area: \"footer\" };\n }\n\n if (overData.type === \"block\" && overData.rowUid && overData.blockUid) {\n return {\n rowUid: overData.rowUid,\n index: getBlockIndex(model, overData.blockUid),\n area: overData.area ?? \"body\",\n };\n }\n\n if (overData.type === \"row\" && overData.rowUid) {\n return {\n rowUid: overData.rowUid,\n index: getRowBlockCount(model, overData.rowUid),\n area: overData.area ?? \"body\",\n };\n }\n\n return { rowUid: null, index: 0, area: \"body\" };\n}\n\nexport function getRowIndex(\n model: EditorModel,\n overId: UniqueIdentifier | undefined,\n overData: DragData,\n): number | null {\n const rowUid =\n (overData.type === \"row\" || overData.type === \"block\") && overData.rowUid\n ? overData.rowUid\n : String(overId);\n const bodyIndex = model.rows.findIndex((row) => row.uid === rowUid);\n if (bodyIndex !== -1) {\n return bodyIndex;\n }\n const footerIndex = model.footerRows.findIndex((row) => row.uid === rowUid);\n return footerIndex === -1 ? null : footerIndex;\n}\n\nfunction getBlockIndex(model: EditorModel, blockUid: string): number {\n for (const row of [...model.rows, ...model.footerRows]) {\n const index = row.blocks.findIndex((block) => block.uid === blockUid);\n\n if (index !== -1) {\n return index;\n }\n }\n\n return 0;\n}\n\nfunction getRowBlockCount(model: EditorModel, rowUid: string): number {\n const row =\n model.rows.find((candidate) => candidate.uid === rowUid) ??\n model.footerRows.find((candidate) => candidate.uid === rowUid);\n return row?.blocks.length ?? 0;\n}\n","import type {\n Align,\n Block,\n CustomPageSize,\n Orientation,\n PageConfig,\n PageFooterConfig,\n PageFormat,\n PageNumbersConfig,\n PageSize,\n PresetPageSize,\n Row,\n Template,\n TemplateConfig,\n} from \"../../types/generated/template\";\n\nexport type EditorArea = \"body\" | \"footer\";\n\nexport type PageNumbersValue = \"disabled\" | Align;\n\nexport interface ResolvedPageSize {\n format: PageFormat;\n orientation: Orientation;\n custom: CustomPageSize | null;\n}\n\nexport interface EditorBlock {\n uid: string;\n block: Block;\n}\n\nexport interface EditorRow {\n uid: string;\n blocks: EditorBlock[];\n}\n\nexport interface EditorModel {\n template: Omit<Template, \"rows\">;\n rows: EditorRow[];\n footerRows: EditorRow[];\n}\n\nlet nextUid = 1;\n\nexport function createEditorModel(template: Template): EditorModel {\n const { rows = [], ...templateFields } = template;\n const footerRows = templateFields.config?.page?.footer?.rows ?? [];\n const strippedTemplate = stripFooterRows(cloneValue(templateFields));\n\n return {\n template: strippedTemplate,\n rows: rows.map((row) => createEditorRow(row)),\n footerRows: footerRows.map((row) => createEditorRow(row)),\n };\n}\n\nexport function serializeTemplate(model: EditorModel): Template {\n const template = cloneValue(model.template) as Template;\n const withFooter = applyFooterRows(template, model.footerRows);\n\n if (model.rows.length === 0) {\n return withFooter;\n }\n\n return {\n ...withFooter,\n rows: model.rows.map((row) => ({\n blocks: row.blocks.map((editorBlock) => cloneBlock(editorBlock.block)),\n })),\n };\n}\n\nexport function addBlockToNewRow(\n model: EditorModel,\n block: Block,\n area: EditorArea = \"body\",\n): EditorModel {\n return updateArea(model, area, (rows) => [\n ...rows,\n { uid: createUid(\"row\"), blocks: [createEditorBlock(block)] },\n ]);\n}\n\nexport function addBlockToRow(\n model: EditorModel,\n rowUid: string,\n block: Block,\n index: number,\n): EditorModel {\n const area = findRowArea(model, rowUid);\n\n if (!area) {\n return model;\n }\n\n return updateArea(model, area, (rows) =>\n rows.map((row) =>\n row.uid === rowUid\n ? { ...row, blocks: insertAt(row.blocks, createEditorBlock(block), index) }\n : row,\n ),\n );\n}\n\nexport function removeBlock(model: EditorModel, blockUid: string): EditorModel {\n return mapAllAreas(model, (rows) =>\n rows\n .map((row) => ({\n ...row,\n blocks: row.blocks.filter((block) => block.uid !== blockUid),\n }))\n .filter((row) => row.blocks.length > 0),\n );\n}\n\nexport function moveBlock(\n model: EditorModel,\n blockUid: string,\n rowUid: string | null,\n index: number,\n area: EditorArea = \"body\",\n): EditorModel {\n const block = findEditorBlock(model, blockUid);\n\n if (!block) {\n return model;\n }\n\n if (rowUid === null) {\n const stripped = mapAllAreas(model, (rows) =>\n rows\n .map((row) => ({\n ...row,\n blocks: row.blocks.filter((candidate) => candidate.uid !== blockUid),\n }))\n .filter((row) => row.blocks.length > 0),\n );\n\n return updateArea(stripped, area, (rows) => [\n ...rows,\n { uid: createUid(\"row\"), blocks: [block] },\n ]);\n }\n\n const targetArea = findRowArea(model, rowUid);\n\n if (!targetArea) {\n return model;\n }\n\n const stripped = mapAllAreas(model, (rows) =>\n rows.map((row) => ({\n ...row,\n blocks: row.blocks.filter((candidate) => candidate.uid !== blockUid),\n })),\n );\n\n const withInsert = updateArea(stripped, targetArea, (rows) =>\n rows.map((row) =>\n row.uid === rowUid ? { ...row, blocks: insertAt(row.blocks, block, index) } : row,\n ),\n );\n\n return mapAllAreas(withInsert, (rows) => rows.filter((row) => row.blocks.length > 0));\n}\n\nexport function moveRow(model: EditorModel, rowUid: string, index: number): EditorModel {\n const area = findRowArea(model, rowUid);\n\n if (!area) {\n return model;\n }\n\n return updateArea(model, area, (rows) => {\n const row = rows.find((candidate) => candidate.uid === rowUid);\n\n if (!row) {\n return rows;\n }\n\n const remaining = rows.filter((candidate) => candidate.uid !== rowUid);\n\n return insertAt(remaining, row, index);\n });\n}\n\nexport function updateBlock(model: EditorModel, blockUid: string, block: Block): EditorModel {\n return mapAllAreas(model, (rows) =>\n rows.map((row) => ({\n ...row,\n blocks: row.blocks.map((editorBlock) =>\n editorBlock.uid === blockUid ? { ...editorBlock, block: cloneBlock(block) } : editorBlock,\n ),\n })),\n );\n}\n\nexport function updateTemplateSettings(model: EditorModel, template: Template): EditorModel {\n return {\n ...model,\n template: createEditorModel(template).template,\n };\n}\n\nexport function getPageSize(model: EditorModel): ResolvedPageSize {\n const size = model.template.config?.page?.size;\n\n if (isCustomPageSize(size)) {\n return { format: \"A4\", orientation: \"portrait\", custom: size };\n }\n\n return {\n format: (size?.format ?? \"A4\") as PageFormat,\n orientation: (size?.orientation ?? \"portrait\") as Orientation,\n custom: null,\n };\n}\n\nexport function setPageSize(\n model: EditorModel,\n format: PageFormat,\n orientation: Orientation,\n): EditorModel {\n const nextSize: PresetPageSize = { format, orientation };\n const previousConfig = model.template.config ?? {};\n const previousPage = previousConfig.page ?? {};\n const nextPage: PageConfig = { ...previousPage, size: nextSize };\n const nextConfig: TemplateConfig = { ...previousConfig, page: nextPage };\n\n return {\n ...model,\n template: { ...model.template, config: nextConfig },\n };\n}\n\nfunction isCustomPageSize(size: PageSize | undefined): size is CustomPageSize {\n return (\n typeof size === \"object\" &&\n size !== null &&\n \"width\" in size &&\n \"height\" in size &&\n typeof (size as CustomPageSize).width === \"number\" &&\n typeof (size as CustomPageSize).height === \"number\"\n );\n}\n\nexport function setRowWidths(model: EditorModel, rowUid: string, widths: string[]): EditorModel {\n const area = findRowArea(model, rowUid);\n\n if (!area) {\n return model;\n }\n\n return updateArea(model, area, (rows) =>\n rows.map((row) =>\n row.uid === rowUid\n ? {\n ...row,\n blocks: row.blocks.map((editorBlock, index) => ({\n ...editorBlock,\n block: setBlockWidth(editorBlock.block, widths[index]),\n })),\n }\n : row,\n ),\n );\n}\n\nexport function getFooterRepeat(model: EditorModel): boolean {\n return model.template.config?.page?.footer?.repeat !== false;\n}\n\nexport function setFooterRepeat(model: EditorModel, repeat: boolean): EditorModel {\n return updatePageConfig(model, (page) => {\n const nextFooter: PageFooterConfig = { ...page.footer, repeat };\n\n return { ...page, footer: nextFooter };\n });\n}\n\nexport function getPageNumbers(model: EditorModel): PageNumbersValue {\n const pageNumbers = model.template.config?.page?.pageNumbers;\n\n if (pageNumbers?.enabled !== true) {\n return \"disabled\";\n }\n\n return (pageNumbers.position ?? \"center\") as Align;\n}\n\nexport function setPageNumbers(model: EditorModel, value: PageNumbersValue): EditorModel {\n return updatePageConfig(model, (page) => {\n if (value === \"disabled\") {\n const next: PageNumbersConfig = { ...page.pageNumbers, enabled: false };\n\n return { ...page, pageNumbers: next };\n }\n\n const next: PageNumbersConfig = { enabled: true, position: value };\n\n return { ...page, pageNumbers: next };\n });\n}\n\nfunction updatePageConfig(\n model: EditorModel,\n updater: (page: PageConfig) => PageConfig,\n): EditorModel {\n const previousConfig = model.template.config ?? {};\n const previousPage = previousConfig.page ?? {};\n const nextPage = updater(previousPage);\n const nextConfig: TemplateConfig = { ...previousConfig, page: nextPage };\n\n return { ...model, template: { ...model.template, config: nextConfig } };\n}\n\nfunction findRowArea(model: EditorModel, rowUid: string): EditorArea | null {\n if (model.rows.some((row) => row.uid === rowUid)) {\n return \"body\";\n }\n if (model.footerRows.some((row) => row.uid === rowUid)) {\n return \"footer\";\n }\n return null;\n}\n\nfunction updateArea(\n model: EditorModel,\n area: EditorArea,\n updater: (rows: EditorRow[]) => EditorRow[],\n): EditorModel {\n if (area === \"body\") {\n return { ...model, rows: updater(model.rows) };\n }\n return { ...model, footerRows: updater(model.footerRows) };\n}\n\nfunction mapAllAreas(\n model: EditorModel,\n updater: (rows: EditorRow[]) => EditorRow[],\n): EditorModel {\n return {\n ...model,\n rows: updater(model.rows),\n footerRows: updater(model.footerRows),\n };\n}\n\nfunction stripFooterRows(template: Omit<Template, \"rows\">): Omit<Template, \"rows\"> {\n const footer = template.config?.page?.footer;\n\n if (!footer) {\n return template;\n }\n\n const { rows: _strippedRows, ...rest } = footer;\n const nextFooter = Object.keys(rest).length === 0 ? undefined : rest;\n const nextPage = { ...template.config?.page, footer: nextFooter };\n\n if (nextPage.footer === undefined) {\n delete nextPage.footer;\n }\n\n return {\n ...template,\n config: { ...template.config, page: nextPage },\n };\n}\n\nfunction applyFooterRows(template: Template, footerRows: EditorRow[]): Template {\n const existingPage = template.config?.page ?? {};\n const existingFooter = existingPage.footer ?? {};\n const hasFooterContext =\n footerRows.length > 0 || existingFooter.repeat !== undefined;\n\n if (!hasFooterContext) {\n return template;\n }\n\n const serializedRows: Row[] = footerRows.map((row) => ({\n blocks: row.blocks.map((editorBlock) => cloneBlock(editorBlock.block)),\n }));\n const nextFooter: PageFooterConfig = {\n ...existingFooter,\n ...(serializedRows.length > 0 ? { rows: serializedRows } : {}),\n };\n const nextPage: PageConfig = { ...existingPage, footer: nextFooter };\n\n return { ...template, config: { ...template.config, page: nextPage } };\n}\n\nfunction createEditorRow(row: Row): EditorRow {\n return {\n uid: createUid(\"row\"),\n blocks: row.blocks.map((block) => createEditorBlock(block)),\n };\n}\n\nfunction createEditorBlock(block: Block): EditorBlock {\n return {\n uid: createUid(\"block\"),\n block: cloneBlock(block),\n };\n}\n\nexport function findEditorBlock(model: EditorModel, blockUid: string): EditorBlock | undefined {\n const allRows = [...model.rows, ...model.footerRows];\n\n return allRows.flatMap((row) => row.blocks).find((block) => block.uid === blockUid);\n}\n\nexport function resolveSelectedEditorBlock(\n model: EditorModel,\n blockUid: string | null,\n): EditorBlock | null {\n return blockUid ? (findEditorBlock(model, blockUid) ?? null) : null;\n}\n\nexport function reconcileSelectedBlockUid(\n model: EditorModel,\n blockUid: string | null,\n): string | null {\n return resolveSelectedEditorBlock(model, blockUid) ? blockUid : null;\n}\n\nexport function createNextBlockId(model: EditorModel, blockType: string): string {\n const template = serializeTemplate(model);\n const allRows = [...(template.rows ?? []), ...(template.config?.page?.footer?.rows ?? [])];\n const usedIds = new Set(\n allRows.flatMap((row) =>\n row.blocks\n .map((block) => block.id)\n .filter((id): id is string => typeof id === \"string\" && id.length > 0),\n ),\n );\n let nextId = 1;\n\n while (usedIds.has(`${blockType}-${nextId}`)) {\n nextId += 1;\n }\n\n return `${blockType}-${nextId}`;\n}\n\nfunction setBlockWidth(block: Block, width: string | undefined): Block {\n const cloned = cloneBlock(block);\n const nextConfig: Record<string, unknown> = { ...cloned.config };\n\n if (width === undefined || width === \"\") {\n delete nextConfig.width;\n } else {\n nextConfig.width = width;\n }\n\n if (Object.keys(nextConfig).length === 0) {\n const { config: _config, ...rest } = cloned;\n\n return rest as Block;\n }\n\n return { ...cloned, config: nextConfig } as Block;\n}\n\nfunction insertAt<T>(items: readonly T[], item: T, index: number): T[] {\n const nextItems = [...items];\n const safeIndex = Math.max(0, Math.min(index, nextItems.length));\n\n nextItems.splice(safeIndex, 0, item);\n\n return nextItems;\n}\n\nfunction cloneBlock(block: Block): Block {\n return cloneValue(block);\n}\n\nfunction cloneValue<T>(value: T): T {\n return structuredClone(value);\n}\n\nfunction createUid(prefix: string): string {\n const uid = `${prefix}-${nextUid}`;\n nextUid += 1;\n\n return uid;\n}\n","import { type DragEndEvent, type DragStartEvent } from \"@dnd-kit/core\";\nimport { useCallback, useState, type Dispatch, type RefObject } from \"react\";\nimport type { Block } from \"../../types/generated/template\";\nimport type { JsonSchemaObject } from \"../../types/template\";\nimport { useBuilderSensors } from \"../lib/sensors\";\nimport { getDragData } from \"../state/dragDrop\";\nimport { findEditorBlock, type EditorModel } from \"../state/editorModel\";\nimport type { EditorAction } from \"../state/editorReducer\";\n\nexport type ActiveDrag =\n | { kind: \"palette\"; type: string }\n | { kind: \"block\"; block: Block }\n | { kind: \"row\" }\n | null;\n\ninterface BuilderDragDrop {\n activeDrag: ActiveDrag;\n sensors: ReturnType<typeof useBuilderSensors>;\n onDragStart: (event: DragStartEvent) => void;\n onDragEnd: (event: DragEndEvent) => void;\n onDragCancel: () => void;\n}\n\nexport function useBuilderDragDrop(\n schema: JsonSchemaObject | null,\n dispatch: Dispatch<EditorAction>,\n modelRef: RefObject<EditorModel>,\n): BuilderDragDrop {\n const [activeDrag, setActiveDrag] = useState<ActiveDrag>(null);\n\n const sensors = useBuilderSensors();\n\n const onDragStart = useCallback(\n (event: DragStartEvent) => {\n const dragData = getDragData(event.active.data.current);\n\n if (dragData.source === \"palette\" && dragData.type) {\n setActiveDrag({ kind: \"palette\", type: dragData.type });\n return;\n }\n if (dragData.type === \"row\") {\n setActiveDrag({ kind: \"row\" });\n return;\n }\n if (dragData.type === \"block\" && dragData.blockUid) {\n const editorBlock = findEditorBlock(modelRef.current, dragData.blockUid);\n setActiveDrag(editorBlock ? { kind: \"block\", block: editorBlock.block } : null);\n }\n },\n [modelRef],\n );\n\n const onDragEnd = useCallback(\n (event: DragEndEvent) => {\n setActiveDrag(null);\n\n if (!event.over) {\n return;\n }\n\n const activeData = getDragData(event.active.data.current);\n const overData = getDragData(event.over.data.current);\n\n if (activeData.source === \"palette\" && activeData.type && schema) {\n dispatch({\n type: \"dropFromPalette\",\n schema,\n blockType: activeData.type,\n overId: event.over.id,\n overData,\n });\n return;\n }\n\n if (activeData.type === \"row\" && activeData.rowUid) {\n dispatch({ type: \"moveRowTo\", rowUid: activeData.rowUid, overId: event.over.id, overData });\n return;\n }\n\n if (activeData.type === \"block\" && activeData.blockUid) {\n dispatch({\n type: \"moveBlockTo\",\n blockUid: activeData.blockUid,\n overId: event.over.id,\n overData,\n });\n }\n },\n [schema, dispatch],\n );\n\n const onDragCancel = useCallback(() => {\n setActiveDrag(null);\n }, []);\n\n return { activeDrag, sensors, onDragStart, onDragEnd, onDragCancel };\n}\n","import { useCallback, useEffect, useRef, useState } from \"react\";\nimport { fetchTemplateSchema, renderTemplatePdf } from \"../api/pdfUaApi\";\nimport type { Template } from \"../types/generated/template\";\nimport type { TemplateData, TemplateSchemaResponse } from \"../types/template\";\n\ninterface UsePdfUaApiOptions {\n initialApiUrl: string;\n apiUrl: string;\n onRendered?: (pdf: Blob) => void;\n loadSchemaOnMount?: boolean;\n}\n\ninterface PdfUaApi {\n schema: TemplateSchemaResponse | null;\n schemaLoading: boolean;\n pdfUrl: string | null;\n pdfLoading: boolean;\n error: string | null;\n loadSchema: (url: string) => Promise<void>;\n renderPdf: (template: Template, data: TemplateData) => Promise<void>;\n}\n\nexport function usePdfUaApi({\n initialApiUrl,\n apiUrl,\n onRendered,\n loadSchemaOnMount = true,\n}: UsePdfUaApiOptions): PdfUaApi {\n const [schema, setSchema] = useState<TemplateSchemaResponse | null>(null);\n const [schemaLoading, setSchemaLoading] = useState(false);\n const [pdfUrl, setPdfUrl] = useState<string | null>(null);\n const [pdfLoading, setPdfLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n const initialApiUrlRef = useRef(initialApiUrl);\n const schemaRequestId = useRef(0);\n const renderRequestId = useRef(0);\n const pdfUrlRef = useRef<string | null>(null);\n const onRenderedRef = useRef(onRendered);\n\n useEffect(() => {\n onRenderedRef.current = onRendered;\n }, [onRendered]);\n\n useEffect(() => {\n return () => {\n revokeObjectUrl(pdfUrlRef.current);\n pdfUrlRef.current = null;\n };\n }, []);\n\n const loadSchema = useCallback(async (url: string) => {\n const requestId = schemaRequestId.current + 1;\n schemaRequestId.current = requestId;\n setSchemaLoading(true);\n setError(null);\n\n try {\n const nextSchema = await fetchTemplateSchema(url);\n\n if (requestId === schemaRequestId.current) {\n setSchema(nextSchema);\n }\n } catch (cause) {\n if (requestId === schemaRequestId.current) {\n setError(errorMessage(cause));\n }\n } finally {\n if (requestId === schemaRequestId.current) {\n setSchemaLoading(false);\n }\n }\n }, []);\n\n useEffect(() => {\n if (loadSchemaOnMount) {\n void loadSchema(initialApiUrlRef.current);\n }\n }, [loadSchema, loadSchemaOnMount]);\n\n const renderPdf = useCallback(\n async (template: Template, data: TemplateData) => {\n const requestId = renderRequestId.current + 1;\n renderRequestId.current = requestId;\n setPdfLoading(true);\n setError(null);\n\n try {\n const pdf = await renderTemplatePdf(apiUrl, {\n template,\n data,\n options: { title: \"Template Preview\" },\n });\n const nextPdfUrl = URL.createObjectURL(pdf);\n\n if (requestId !== renderRequestId.current) {\n revokeObjectUrl(nextPdfUrl);\n return;\n }\n\n setPdfUrl((currentPdfUrl) => {\n revokeObjectUrl(currentPdfUrl);\n pdfUrlRef.current = nextPdfUrl;\n return nextPdfUrl;\n });\n onRenderedRef.current?.(pdf);\n } catch (cause) {\n if (requestId === renderRequestId.current) {\n setError(errorMessage(cause));\n }\n } finally {\n if (requestId === renderRequestId.current) {\n setPdfLoading(false);\n }\n }\n },\n [apiUrl],\n );\n\n return { schema, schemaLoading, pdfUrl, pdfLoading, error, loadSchema, renderPdf };\n}\n\nfunction revokeObjectUrl(url: string | null): void {\n if (url) {\n URL.revokeObjectURL(url);\n }\n}\n\nfunction errorMessage(cause: unknown): string {\n return cause instanceof Error ? cause.message : String(cause);\n}\n","import { createContext, useContext, type ReactNode } from \"react\";\nimport type { Template } from \"../types/generated/template\";\nimport type { TemplateData } from \"../types/template\";\n\nexport interface RenderContextValue {\n template: Template | null;\n data: TemplateData;\n pdfUrl: string | null;\n pdfLoading: boolean;\n error: string | null;\n renderPdf: () => void;\n renderDisabled: boolean;\n}\n\nconst RenderContext = createContext<RenderContextValue | null>(null);\n\nexport function RenderProvider({\n value,\n children,\n}: {\n value: RenderContextValue;\n children: ReactNode;\n}) {\n return <RenderContext.Provider value={value}>{children}</RenderContext.Provider>;\n}\n\nexport function useRenderContext(): RenderContextValue {\n const value = useContext(RenderContext);\n\n if (!value) {\n throw new Error(\"useRenderContext must be used within a render provider.\");\n }\n\n return value;\n}\n","import type { Block } from \"../../types/generated/template\";\nimport type {\n JsonSchemaObject,\n JsonSchemaValue,\n TemplateSchemaMetadata,\n} from \"../../types/template\";\nimport { isRecord } from \"../lib/records\";\n\nexport type { JsonSchemaObject, JsonSchemaValue };\n\nexport function getSchemaMetadata(schema: JsonSchemaObject): TemplateSchemaMetadata {\n return assertTemplateSchemaMetadata(schema[\"x-pdfUa\"]);\n}\n\nexport function resolveRef(schema: JsonSchemaObject, ref: string): JsonSchemaObject {\n if (!ref.startsWith(\"#/\")) {\n throw new Error(`Unsupported schema ref: ${ref}`);\n }\n\n const value = ref\n .slice(2)\n .split(\"/\")\n .reduce<unknown>((current, segment) => {\n if (!isSchemaObject(current)) {\n return undefined;\n }\n\n return current[decodeJsonPointerSegment(segment)];\n }, schema);\n\n if (!isSchemaObject(value)) {\n throw new Error(`Schema ref not found: ${ref}`);\n }\n\n return value;\n}\n\nexport function getBlockTypes(schema: JsonSchemaObject): string[] {\n const discovered = getBlockDefinitions(schema)\n .map((definition) => getBlockType(definition))\n .filter((type): type is string => type !== undefined);\n\n const discoveredSet = new Set(discovered);\n const orderedTypes = getSchemaMetadata(schema).blockOrder.filter((type) =>\n discoveredSet.has(type),\n );\n const remainingTypes = discovered.filter((type) => !orderedTypes.includes(type));\n\n return [...orderedTypes, ...remainingTypes];\n}\n\nexport function getBlockDefinition(\n schema: JsonSchemaObject,\n blockType: string,\n): JsonSchemaObject | undefined {\n return getBlockDefinitions(schema).find((definition) => getBlockType(definition) === blockType);\n}\n\nexport function getBlockFieldSchema(schema: JsonSchemaObject, blockType: string): JsonSchemaObject {\n const definition = requireBlockDefinition(schema, blockType);\n const properties = getProperties(definition);\n const fieldProperties = Object.fromEntries(\n Object.entries(properties).filter(([field]) => field !== \"type\" && field !== \"config\"),\n );\n const required = getStringArray(definition.required).filter((field) => field !== \"type\");\n\n return {\n type: \"object\",\n properties: fieldProperties,\n required,\n $defs: schema.$defs,\n };\n}\n\nexport function getBlockConfigSchema(\n schema: JsonSchemaObject,\n blockType: string,\n): JsonSchemaObject | undefined {\n const definition = requireBlockDefinition(schema, blockType);\n const config = getProperties(definition).config;\n\n if (!isSchemaObject(config)) {\n return undefined;\n }\n\n const ref = config.$ref;\n\n const configSchema = typeof ref === \"string\" ? resolveRef(schema, ref) : config;\n\n return {\n ...configSchema,\n $defs: schema.$defs,\n };\n}\n\nexport function createDefaultBlock(schema: JsonSchemaObject, blockType: string, id: string): Block {\n const definition = requireBlockDefinition(schema, blockType);\n const properties = getProperties(definition);\n const block: Record<string, unknown> = {\n type: blockType,\n id,\n };\n\n for (const field of getStringArray(definition.required)) {\n if (field === \"type\" || field === \"config\" || field === \"id\" || block[field] !== undefined) {\n continue;\n }\n\n const fieldSchema = properties[field];\n\n if (isSchemaObject(fieldSchema)) {\n block[field] = createDefaultValue(schema, fieldSchema);\n }\n }\n\n return block as unknown as Block;\n}\n\nfunction getBlockDefinitions(schema: JsonSchemaObject): JsonSchemaObject[] {\n const defs = schema.$defs;\n const block = isSchemaObject(defs) ? defs.block : undefined;\n const oneOf = isSchemaObject(block) && Array.isArray(block.oneOf) ? block.oneOf : [];\n\n return oneOf\n .map((entry) => {\n if (!isSchemaObject(entry)) {\n return undefined;\n }\n\n const ref = entry.$ref;\n\n return typeof ref === \"string\" ? resolveRef(schema, ref) : entry;\n })\n .filter((entry): entry is JsonSchemaObject => entry !== undefined);\n}\n\nfunction requireBlockDefinition(schema: JsonSchemaObject, blockType: string): JsonSchemaObject {\n const definition = getBlockDefinition(schema, blockType);\n\n if (!definition) {\n throw new Error(`Unknown block type: ${blockType}`);\n }\n\n return definition;\n}\n\nfunction getBlockType(definition: JsonSchemaObject): string | undefined {\n const typeProperty = getProperties(definition).type;\n\n if (!isSchemaObject(typeProperty)) {\n return undefined;\n }\n\n return typeof typeProperty.const === \"string\" ? typeProperty.const : undefined;\n}\n\nfunction getProperties(definition: JsonSchemaObject): Record<string, unknown> {\n return isSchemaObject(definition.properties) ? definition.properties : {};\n}\n\nfunction createDefaultValue(schema: JsonSchemaObject, fieldSchema: JsonSchemaObject): unknown {\n if (\"default\" in fieldSchema) {\n return fieldSchema.default;\n }\n\n if (Array.isArray(fieldSchema.enum)) {\n return fieldSchema.enum.find((value) => value !== null);\n }\n\n if (typeof fieldSchema.const === \"string\" || typeof fieldSchema.const === \"number\") {\n return fieldSchema.const;\n }\n\n if (typeof fieldSchema.$ref === \"string\") {\n return createDefaultValue(schema, resolveRef(schema, fieldSchema.$ref));\n }\n\n if (Array.isArray(fieldSchema.oneOf)) {\n const firstNonNullSchema = fieldSchema.oneOf.find(\n (option) => isSchemaObject(option) && option.type !== \"null\",\n );\n\n return isSchemaObject(firstNonNullSchema)\n ? createDefaultValue(schema, firstNonNullSchema)\n : undefined;\n }\n\n const type = getPrimaryType(fieldSchema.type);\n\n switch (type) {\n case \"string\":\n return \"\";\n case \"integer\":\n case \"number\":\n return 0;\n case \"boolean\":\n return false;\n case \"array\":\n return [];\n case \"object\":\n return {};\n default:\n return undefined;\n }\n}\n\nfunction getPrimaryType(type: unknown): string | undefined {\n if (typeof type === \"string\") {\n return type;\n }\n\n if (!Array.isArray(type)) {\n return undefined;\n }\n\n return type.find((entry): entry is string => typeof entry === \"string\" && entry !== \"null\");\n}\n\nfunction getStringArray(value: unknown): string[] {\n return Array.isArray(value)\n ? value.filter((entry): entry is string => typeof entry === \"string\")\n : [];\n}\n\nfunction assertTemplateSchemaMetadata(value: unknown): TemplateSchemaMetadata {\n if (!isSchemaObject(value) || !isStringArray(value.blockOrder)) {\n throw new Error(\"Invalid template schema metadata\");\n }\n\n return value as unknown as TemplateSchemaMetadata;\n}\n\nfunction isStringArray(value: unknown): value is string[] {\n return Array.isArray(value) && value.every((entry) => typeof entry === \"string\");\n}\n\nfunction isSchemaObject(value: unknown): value is JsonSchemaObject {\n return isRecord(value);\n}\n\nfunction decodeJsonPointerSegment(segment: string): string {\n return segment.replace(/~1/g, \"/\").replace(/~0/g, \"~\");\n}\n","import type { UniqueIdentifier } from \"@dnd-kit/core\";\nimport type { Block, Orientation, PageFormat, Template } from \"../../types/generated/template\";\nimport type { JsonSchemaObject, TemplateData } from \"../../types/template\";\nimport { createDefaultBlock } from \"../schema/schemaAdapter\";\nimport { getDropTarget, getRowIndex, type DragData } from \"./dragDrop\";\nimport {\n addBlockToNewRow,\n addBlockToRow,\n createEditorModel,\n createNextBlockId,\n getPageSize,\n moveBlock,\n moveRow,\n reconcileSelectedBlockUid,\n removeBlock,\n setFooterRepeat,\n setPageNumbers,\n setPageSize,\n setRowWidths,\n updateBlock,\n updateTemplateSettings,\n type EditorModel,\n type PageNumbersValue,\n} from \"./editorModel\";\n\nexport interface EditorState {\n model: EditorModel;\n data: TemplateData;\n selectedBlockUid: string | null;\n}\n\nexport type EditorAction =\n | { type: \"selectBlock\"; blockUid: string }\n | { type: \"deselect\" }\n | { type: \"changeBlock\"; blockUid: string; block: Block }\n | { type: \"removeBlock\"; blockUid: string }\n | { type: \"changeTemplateSettings\"; template: Template }\n | { type: \"setRowWidths\"; rowUid: string; widths: string[] }\n | { type: \"addBlock\"; schema: JsonSchemaObject; blockType: string }\n | { type: \"setFormat\"; format: PageFormat }\n | { type: \"setOrientation\"; orientation: Orientation }\n | { type: \"setFooterRepeat\"; repeat: boolean }\n | { type: \"setPageNumbers\"; value: PageNumbersValue }\n | { type: \"setData\"; data: TemplateData }\n | { type: \"loadExample\"; template: Template; data: TemplateData }\n | {\n type: \"dropFromPalette\";\n schema: JsonSchemaObject;\n blockType: string;\n overId?: UniqueIdentifier;\n overData: DragData;\n }\n | { type: \"moveRowTo\"; rowUid: string; overId?: UniqueIdentifier; overData: DragData }\n | { type: \"moveBlockTo\"; blockUid: string; overId?: UniqueIdentifier; overData: DragData };\n\nexport function createEditorState(template: Template, data: TemplateData): EditorState {\n return { model: createEditorModel(template), data, selectedBlockUid: null };\n}\n\nexport function editorReducer(state: EditorState, action: EditorAction): EditorState {\n switch (action.type) {\n case \"selectBlock\":\n return { ...state, selectedBlockUid: action.blockUid };\n case \"deselect\":\n return { ...state, selectedBlockUid: null };\n case \"changeBlock\":\n return withModel(state, updateBlock(state.model, action.blockUid, action.block));\n case \"removeBlock\":\n return withModel(state, removeBlock(state.model, action.blockUid));\n case \"changeTemplateSettings\":\n return withModel(state, updateTemplateSettings(state.model, action.template));\n case \"setRowWidths\":\n return withModel(state, setRowWidths(state.model, action.rowUid, action.widths));\n case \"addBlock\":\n return withModel(state, addBlockToNewRow(state.model, defaultBlock(state.model, action)));\n case \"setFormat\":\n return withModel(\n state,\n setPageSize(state.model, action.format, getPageSize(state.model).orientation),\n );\n case \"setOrientation\":\n return withModel(\n state,\n setPageSize(state.model, getPageSize(state.model).format, action.orientation),\n );\n case \"setFooterRepeat\":\n return withModel(state, setFooterRepeat(state.model, action.repeat));\n case \"setPageNumbers\":\n return withModel(state, setPageNumbers(state.model, action.value));\n case \"setData\":\n return { ...state, data: action.data };\n case \"loadExample\":\n return createEditorState(action.template, action.data);\n case \"dropFromPalette\": {\n const target = getDropTarget(state.model, action.overId, action.overData);\n const block = defaultBlock(state.model, action);\n const model =\n target.rowUid === null\n ? addBlockToNewRow(state.model, block, target.area)\n : addBlockToRow(state.model, target.rowUid, block, target.index);\n\n return withModel(state, model);\n }\n case \"moveRowTo\": {\n const index = getRowIndex(state.model, action.overId, action.overData);\n\n return index === null ? state : withModel(state, moveRow(state.model, action.rowUid, index));\n }\n case \"moveBlockTo\": {\n const target = getDropTarget(state.model, action.overId, action.overData);\n\n return withModel(\n state,\n moveBlock(state.model, action.blockUid, target.rowUid, target.index, target.area),\n );\n }\n }\n}\n\nfunction withModel(state: EditorState, model: EditorModel): EditorState {\n return { ...state, model, selectedBlockUid: reconcileSelectedBlockUid(model, state.selectedBlockUid) };\n}\n\nfunction defaultBlock(\n model: EditorModel,\n action: { schema: JsonSchemaObject; blockType: string },\n): Block {\n return createDefaultBlock(action.schema, action.blockType, createNextBlockId(model, action.blockType));\n}\n","import { DndContext, DragOverlay, closestCenter } from \"@dnd-kit/core\";\nimport {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useReducer,\n useRef,\n type ReactNode,\n} from \"react\";\nimport { resolveDefaultApiUrl } from \"../../api/pdfUaApi\";\nimport type {\n Block,\n Orientation,\n PageFormat,\n Template,\n} from \"../../types/generated/template\";\nimport type { TemplateData, TemplateSchemaResponse } from \"../../types/template\";\nimport { getBlockSummary } from \"../blocks/blockChrome\";\nimport { BlockCardPreview } from \"../canvas/BlockCardPreview\";\nimport { useBuilderDragDrop, type ActiveDrag } from \"../hooks/useBuilderDragDrop\";\nimport { usePdfUaApi } from \"../../render/usePdfUaApi\";\nimport { RenderProvider, type RenderContextValue } from \"../../render/RenderContext\";\nimport { Chip } from \"../primitives/Chip\";\nimport {\n getFooterRepeat,\n getPageNumbers,\n getPageSize,\n resolveSelectedEditorBlock,\n serializeTemplate,\n type EditorBlock,\n type EditorModel,\n type PageNumbersValue,\n type ResolvedPageSize,\n} from \"../state/editorModel\";\nimport { createEditorState, editorReducer } from \"../state/editorReducer\";\nimport { getBlockTypes, type JsonSchemaObject } from \"../schema/schemaAdapter\";\n\nconst emptyTemplate: Template = {\n version: 1,\n};\n\n/** A loadable example: a template plus its optional runtime data. */\nexport interface TemplateExample {\n template: Template;\n data?: TemplateData;\n}\n\nexport interface TemplateBuilderProviderProps {\n /** Base URL of a running pdf-ua-api instance. Defaults to \"\" (relative URLs / proxy). */\n apiUrl?: string;\n /** Template loaded into the editor on first render. */\n initialTemplate?: Template;\n /** Runtime data keyed by block id (table rows, dynamic key-value overrides). */\n initialData?: TemplateData;\n /** Fires whenever the user edits the template or its runtime data. */\n onChange?: (template: Template, data: TemplateData) => void;\n /** Fires after a successful render with the produced PDF blob. */\n onRendered?: (pdf: Blob) => void;\n /** Pane slots to arrange. They share this provider's state and DnD context. */\n children: ReactNode;\n}\n\n/** Editor state and derived values. Changes as the user edits. */\nexport interface BuilderState {\n schema: TemplateSchemaResponse | null;\n schemaLoading: boolean;\n model: EditorModel;\n data: TemplateData;\n serializedTemplate: Template;\n selectedBlockUid: string | null;\n selectedBlock: EditorBlock | null;\n blockTypes: ReturnType<typeof getBlockTypes>;\n pageSize: ResolvedPageSize;\n footerRepeat: boolean;\n pageNumbers: PageNumbersValue;\n pdfUrl: string | null;\n pdfLoading: boolean;\n error: string | null;\n}\n\n/** Stable editor actions. Identity never changes for the provider's lifetime. */\nexport interface BuilderActions {\n renderPdf: () => void;\n loadExample: (example: TemplateExample) => void;\n addBlock: (type: string) => void;\n changeBlock: (blockUid: string, block: Block) => void;\n changeTemplateSettings: (template: Template) => void;\n removeBlock: (blockUid: string) => void;\n selectBlock: (blockUid: string) => void;\n deselect: () => void;\n setRowWidths: (rowUid: string, widths: string[]) => void;\n changeData: (data: TemplateData) => void;\n changeFormat: (format: PageFormat) => void;\n changeOrientation: (orientation: Orientation) => void;\n toggleFooterRepeat: (repeat: boolean) => void;\n changePageNumbers: (value: PageNumbersValue) => void;\n}\n\nexport type BuilderContextValue = BuilderState & BuilderActions;\n\nconst BuilderStateContext = createContext<BuilderState | null>(null);\nconst BuilderActionsContext = createContext<BuilderActions | null>(null);\n\n/** Subscribe to editor state. Re-renders on edits. */\nexport function useBuilderState(): BuilderState {\n const value = useContext(BuilderStateContext);\n\n if (!value) {\n throw new Error(\"useBuilderState must be used within a <TemplateBuilderProvider>.\");\n }\n\n return value;\n}\n\n/** Read the stable editor actions. Does not re-render on edits. */\nexport function useBuilderActions(): BuilderActions {\n const value = useContext(BuilderActionsContext);\n\n if (!value) {\n throw new Error(\"useBuilderActions must be used within a <TemplateBuilderProvider>.\");\n }\n\n return value;\n}\n\n/** Headless escape hatch returning both state and actions. */\nexport function useTemplateBuilder(): BuilderContextValue {\n return { ...useBuilderState(), ...useBuilderActions() };\n}\n\nexport function TemplateBuilderProvider({\n apiUrl: apiUrlProp,\n initialTemplate,\n initialData,\n onChange,\n onRendered,\n children,\n}: TemplateBuilderProviderProps) {\n const apiUrl = resolveDefaultApiUrl(apiUrlProp);\n const [state, dispatch] = useReducer(editorReducer, undefined, () =>\n createEditorState(initialTemplate ?? emptyTemplate, initialData ?? {}),\n );\n const { model, data, selectedBlockUid } = state;\n const modelRef = useRef(model);\n modelRef.current = model;\n\n const {\n schema,\n schemaLoading,\n pdfUrl,\n pdfLoading,\n error,\n renderPdf: renderPdfRequest,\n } = usePdfUaApi({\n initialApiUrl: apiUrl,\n apiUrl,\n onRendered,\n });\n\n const onChangeRef = useRef(onChange);\n const skipNextChangeRef = useRef(true);\n useEffect(() => {\n onChangeRef.current = onChange;\n }, [onChange]);\n\n const serializedTemplate = useMemo(() => serializeTemplate(model), [model]);\n\n useEffect(() => {\n if (skipNextChangeRef.current) {\n skipNextChangeRef.current = false;\n return;\n }\n onChangeRef.current?.(serializedTemplate, data);\n }, [serializedTemplate, data]);\n\n const schemaObject: JsonSchemaObject | null = schema;\n const blockTypes = useMemo(\n () => (schemaObject ? getBlockTypes(schemaObject) : []),\n [schemaObject],\n );\n const selectedBlock = useMemo(\n () => resolveSelectedEditorBlock(model, selectedBlockUid),\n [model, selectedBlockUid],\n );\n\n // Latest values for actions that would otherwise need them as deps, keeping\n // every action's identity stable for the provider's lifetime.\n const serializedTemplateRef = useRef(serializedTemplate);\n serializedTemplateRef.current = serializedTemplate;\n const dataRef = useRef(data);\n dataRef.current = data;\n const schemaRef = useRef(schemaObject);\n schemaRef.current = schemaObject;\n\n const { activeDrag, sensors, onDragStart, onDragEnd, onDragCancel } = useBuilderDragDrop(\n schemaObject,\n dispatch,\n modelRef,\n );\n\n const loadExample = useCallback((example: TemplateExample) => {\n dispatch({ type: \"loadExample\", template: example.template, data: example.data ?? {} });\n }, []);\n\n const changeBlock = useCallback((blockUid: string, block: Block) => {\n dispatch({ type: \"changeBlock\", blockUid, block });\n }, []);\n\n const changeTemplateSettings = useCallback((template: Template) => {\n dispatch({ type: \"changeTemplateSettings\", template });\n }, []);\n\n const removeBlock = useCallback((blockUid: string) => {\n dispatch({ type: \"removeBlock\", blockUid });\n }, []);\n\n const selectBlock = useCallback((blockUid: string) => {\n dispatch({ type: \"selectBlock\", blockUid });\n }, []);\n\n const deselect = useCallback(() => {\n dispatch({ type: \"deselect\" });\n }, []);\n\n const setRowWidths = useCallback((rowUid: string, widths: string[]) => {\n dispatch({ type: \"setRowWidths\", rowUid, widths });\n }, []);\n\n const changeData = useCallback((nextData: TemplateData) => {\n dispatch({ type: \"setData\", data: nextData });\n }, []);\n\n const addBlock = useCallback((type: string) => {\n const currentSchema = schemaRef.current;\n\n if (!currentSchema) {\n return;\n }\n dispatch({ type: \"addBlock\", schema: currentSchema, blockType: type });\n }, []);\n\n const changeFormat = useCallback((format: PageFormat) => {\n dispatch({ type: \"setFormat\", format });\n }, []);\n\n const changeOrientation = useCallback((orientation: Orientation) => {\n dispatch({ type: \"setOrientation\", orientation });\n }, []);\n\n const toggleFooterRepeat = useCallback((repeat: boolean) => {\n dispatch({ type: \"setFooterRepeat\", repeat });\n }, []);\n\n const changePageNumbers = useCallback((value: PageNumbersValue) => {\n dispatch({ type: \"setPageNumbers\", value });\n }, []);\n\n const renderPdf = useCallback(() => {\n void renderPdfRequest(serializedTemplateRef.current, dataRef.current);\n }, [renderPdfRequest]);\n\n const actions = useMemo<BuilderActions>(\n () => ({\n renderPdf,\n loadExample,\n addBlock,\n changeBlock,\n changeTemplateSettings,\n removeBlock,\n selectBlock,\n deselect,\n setRowWidths,\n changeData,\n changeFormat,\n changeOrientation,\n toggleFooterRepeat,\n changePageNumbers,\n }),\n [\n renderPdf,\n loadExample,\n addBlock,\n changeBlock,\n changeTemplateSettings,\n removeBlock,\n selectBlock,\n deselect,\n setRowWidths,\n changeData,\n changeFormat,\n changeOrientation,\n toggleFooterRepeat,\n changePageNumbers,\n ],\n );\n\n const stateValue = useMemo<BuilderState>(\n () => ({\n schema,\n schemaLoading,\n model,\n data,\n serializedTemplate,\n selectedBlockUid,\n selectedBlock,\n blockTypes,\n pageSize: getPageSize(model),\n footerRepeat: getFooterRepeat(model),\n pageNumbers: getPageNumbers(model),\n pdfUrl,\n pdfLoading,\n error,\n }),\n [\n schema,\n schemaLoading,\n model,\n data,\n serializedTemplate,\n selectedBlockUid,\n selectedBlock,\n blockTypes,\n pdfUrl,\n pdfLoading,\n error,\n ],\n );\n\n const renderValue = useMemo<RenderContextValue>(\n () => ({\n template: serializedTemplate,\n data,\n pdfUrl,\n pdfLoading,\n error,\n renderPdf,\n renderDisabled: !schema || pdfLoading,\n }),\n [serializedTemplate, data, pdfUrl, pdfLoading, error, renderPdf, schema],\n );\n\n return (\n <BuilderActionsContext.Provider value={actions}>\n <BuilderStateContext.Provider value={stateValue}>\n <DndContext\n sensors={sensors}\n collisionDetection={closestCenter}\n onDragStart={onDragStart}\n onDragEnd={onDragEnd}\n onDragCancel={onDragCancel}\n >\n <RenderProvider value={renderValue}>{children}</RenderProvider>\n\n <DragOverlay dropAnimation={null}>\n {activeDrag ? <ActiveDragPreview drag={activeDrag} /> : null}\n </DragOverlay>\n </DndContext>\n </BuilderStateContext.Provider>\n </BuilderActionsContext.Provider>\n );\n}\n\nfunction ActiveDragPreview({ drag }: { drag: NonNullable<ActiveDrag> }) {\n if (drag.kind === \"palette\") {\n return <BlockCardPreview type={drag.type} prefix=\"+ \" />;\n }\n if (drag.kind === \"block\") {\n return <BlockCardPreview type={drag.block.type} summary={getBlockSummary(drag.block)} />;\n }\n return (\n <div className=\"pointer-events-none origin-top-left rotate-[1.5deg] scale-[1.02] cursor-grabbing drop-shadow-drag\">\n <div className=\"inline-flex min-w-[180px] max-w-[360px] items-center gap-2 rounded-lg border border-solid border-border-strong bg-surface px-3 py-2 text-sm font-medium\">\n <Chip>⋮⋮</Chip>\n <span className=\"font-medium\">Row</span>\n </div>\n </div>\n );\n}\n","import type {\n InputHTMLAttributes,\n SelectHTMLAttributes,\n TextareaHTMLAttributes,\n} from \"react\";\n\nconst fieldBase =\n \"min-w-0 rounded-md border border-solid border-border bg-surface text-fg outline-none transition-colors hover:border-border-strong focus-visible:border-accent focus-visible:ring-3 focus-visible:ring-accent/20\";\n\nfunction cx(...names: Array<string | undefined>): string {\n return names.filter(Boolean).join(\" \");\n}\n\nexport function Input({ className, ...rest }: InputHTMLAttributes<HTMLInputElement>) {\n return <input className={cx(\"h-8 w-full px-3\", fieldBase, className)} {...rest} />;\n}\n\nexport function Textarea({ className, ...rest }: TextareaHTMLAttributes<HTMLTextAreaElement>) {\n return (\n <textarea className={cx(\"min-h-24 w-full px-3 py-2 font-mono text-xs\", fieldBase, className)} {...rest} />\n );\n}\n\nexport function Select({ className, ...rest }: SelectHTMLAttributes<HTMLSelectElement>) {\n return (\n <select\n className={cx(\n \"h-8 cursor-pointer pl-3 pr-7 appearance-none bg-[linear-gradient(45deg,transparent_50%,var(--pdfua-fg-muted)_50%),linear-gradient(135deg,var(--pdfua-fg-muted)_50%,transparent_50%)] bg-[length:5px_5px,5px_5px] bg-[position:calc(100%-14px)_50%,calc(100%-9px)_50%] bg-no-repeat\",\n fieldBase,\n className,\n )}\n {...rest}\n />\n );\n}\n\nexport function Checkbox({ className, ...rest }: InputHTMLAttributes<HTMLInputElement>) {\n return <input type=\"checkbox\" className={cx(\"h-3.5 w-3.5 accent-accent\", className)} {...rest} />;\n}\n","import type { EmptyTextValue, SelectFieldOption } from \"./BuilderField\";\n\nexport function textValue(value: string, emptyValue: EmptyTextValue): string | undefined {\n if (value === \"\" && emptyValue === \"undefined\") {\n return undefined;\n }\n\n return value;\n}\n\nexport function numberValue(value: string, valueAsNumber: number): number | undefined {\n if (value === \"\" || Number.isNaN(valueAsNumber)) {\n return undefined;\n }\n\n return valueAsNumber;\n}\n\nexport function selectValue<Value extends string>(\n value: string,\n options: readonly SelectFieldOption<Value>[],\n): Value | undefined {\n if (value === \"\") {\n return undefined;\n }\n\n return options.find((option) => option.value === value)?.value;\n}\n","import type { ReactNode } from \"react\";\nimport { Checkbox, Input, Select, Textarea } from \"./inputs\";\nimport { numberValue, selectValue, textValue } from \"./fieldConverters\";\n\nexport type EmptyTextValue = \"empty-string\" | \"undefined\";\n\nexport interface BuilderControlProps {\n name: string;\n label: string;\n help?: ReactNode;\n error?: ReactNode;\n id?: string;\n className?: string;\n disabled?: boolean;\n}\n\nexport interface TextFieldProps extends BuilderControlProps {\n value?: string;\n placeholder?: string;\n emptyValue?: EmptyTextValue;\n autoComplete?: string;\n onChange: (value: string | undefined) => void;\n}\n\nexport interface TextAreaFieldProps extends BuilderControlProps {\n value?: string;\n placeholder?: string;\n emptyValue?: EmptyTextValue;\n rows?: number;\n onChange: (value: string | undefined) => void;\n}\n\nexport interface NumberFieldProps extends BuilderControlProps {\n value?: number;\n min?: number;\n max?: number;\n step?: number | \"any\";\n placeholder?: string;\n onChange: (value: number | undefined) => void;\n}\n\nexport interface SelectFieldOption<Value extends string> {\n value: Value;\n label: string;\n disabled?: boolean;\n}\n\nexport interface SelectFieldProps<Value extends string> extends BuilderControlProps {\n value?: Value;\n options: readonly SelectFieldOption<Value>[];\n optional?: boolean;\n emptyLabel?: string;\n onChange: (value: Value | undefined) => void;\n}\n\nexport interface CheckboxFieldProps extends BuilderControlProps {\n checked: boolean;\n onChange: (checked: boolean) => void;\n}\n\nexport interface ColorFieldProps extends BuilderControlProps {\n value?: string;\n fallbackValue?: string;\n onChange: (value: string | undefined) => void;\n}\n\nexport interface UnitFieldProps extends BuilderControlProps {\n value?: string;\n placeholder?: string;\n emptyValue?: EmptyTextValue;\n readOnly?: boolean;\n onChange?: (value: string | undefined) => void;\n}\n\ninterface FieldLayoutProps extends BuilderControlProps {\n children: ReactNode;\n}\n\ninterface FieldControlState {\n id: string;\n describedBy?: string;\n invalid: boolean;\n}\n\nexport function TextField({\n value,\n placeholder,\n emptyValue = \"empty-string\",\n autoComplete,\n onChange,\n ...fieldProps\n}: TextFieldProps): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps}>\n <Input\n id={fieldState.id}\n name={fieldProps.name}\n type=\"text\"\n value={value ?? \"\"}\n placeholder={placeholder}\n autoComplete={autoComplete}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={(event) => onChange(textValue(event.currentTarget.value, emptyValue))}\n />\n </BuilderField>\n );\n}\n\nexport function TextAreaField({\n value,\n placeholder,\n emptyValue = \"empty-string\",\n rows = 4,\n onChange,\n ...fieldProps\n}: TextAreaFieldProps): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps}>\n <Textarea\n id={fieldState.id}\n name={fieldProps.name}\n value={value ?? \"\"}\n placeholder={placeholder}\n rows={rows}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={(event) => onChange(textValue(event.currentTarget.value, emptyValue))}\n />\n </BuilderField>\n );\n}\n\nexport function NumberField({\n value,\n min,\n max,\n step,\n placeholder,\n onChange,\n ...fieldProps\n}: NumberFieldProps): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps}>\n <Input\n id={fieldState.id}\n name={fieldProps.name}\n type=\"number\"\n value={value ?? \"\"}\n min={min}\n max={max}\n step={step}\n placeholder={placeholder}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={(event) =>\n onChange(numberValue(event.currentTarget.value, event.currentTarget.valueAsNumber))\n }\n />\n </BuilderField>\n );\n}\n\nexport function SelectField<Value extends string>({\n value,\n options,\n optional = false,\n emptyLabel = \"\",\n onChange,\n ...fieldProps\n}: SelectFieldProps<Value>): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps}>\n <Select\n id={fieldState.id}\n className=\"w-full\"\n name={fieldProps.name}\n value={value ?? \"\"}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={(event) => onChange(selectValue(event.currentTarget.value, options))}\n >\n {optional ? <option value=\"\">{emptyLabel}</option> : null}\n {options.map((option) => (\n <option key={option.value} value={option.value} disabled={option.disabled}>\n {option.label}\n </option>\n ))}\n </Select>\n </BuilderField>\n );\n}\n\nexport function CheckboxField({\n checked,\n onChange,\n ...fieldProps\n}: CheckboxFieldProps): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps} className={fieldProps.className}>\n <Checkbox\n id={fieldState.id}\n name={fieldProps.name}\n checked={checked}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={(event) => onChange(event.currentTarget.checked)}\n />\n </BuilderField>\n );\n}\n\nexport function ColorField({\n value,\n fallbackValue = \"#000000\",\n onChange,\n ...fieldProps\n}: ColorFieldProps): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps}>\n <Input\n id={fieldState.id}\n className=\"p-1\"\n name={fieldProps.name}\n type=\"color\"\n value={value ?? fallbackValue}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={(event) => onChange(textValue(event.currentTarget.value, \"undefined\"))}\n />\n </BuilderField>\n );\n}\n\nexport function UnitField({\n value,\n placeholder = \"auto, 50%, 80mm\",\n emptyValue = \"undefined\",\n readOnly = false,\n onChange,\n ...fieldProps\n}: UnitFieldProps): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps}>\n <Input\n id={fieldState.id}\n name={fieldProps.name}\n type=\"text\"\n inputMode=\"text\"\n spellCheck={false}\n className={readOnly ? \"cursor-default bg-surface-muted text-fg-muted\" : undefined}\n value={value ?? \"\"}\n placeholder={placeholder}\n readOnly={readOnly}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={\n readOnly\n ? undefined\n : (event) => onChange?.(textValue(event.currentTarget.value, emptyValue))\n }\n />\n </BuilderField>\n );\n}\n\nexport function BuilderField({\n name,\n label,\n help,\n error,\n id,\n className,\n children,\n}: FieldLayoutProps): ReactNode {\n const fieldState = createFieldState({ name, label, help, error, id });\n const helpId = help ? `${fieldState.id}-help` : undefined;\n const errorId = error ? `${fieldState.id}-error` : undefined;\n\n return (\n <div className={classNames(\"grid min-w-0 gap-1\", className)}>\n <label className=\"text-2xs font-medium text-fg-muted\" htmlFor={fieldState.id}>\n {label}\n </label>\n <div className=\"min-w-0\">{children}</div>\n {help ? (\n <p className=\"m-0 text-2xs text-fg-muted\" id={helpId}>\n {help}\n </p>\n ) : null}\n {error ? (\n <p className=\"m-0 text-2xs text-danger\" id={errorId} role=\"alert\">\n {error}\n </p>\n ) : null}\n </div>\n );\n}\n\nfunction createFieldState({\n name,\n help,\n error,\n id,\n}: BuilderControlProps): FieldControlState {\n const fieldId = id ?? createFieldId(name);\n const helpId = help ? `${fieldId}-help` : undefined;\n const errorId = error ? `${fieldId}-error` : undefined;\n const describedBy = [helpId, errorId].filter(Boolean).join(\" \") || undefined;\n const invalid = error !== undefined && error !== null && error !== \"\";\n\n return {\n id: fieldId,\n describedBy,\n invalid,\n };\n}\n\nexport function createFieldId(name: string): string {\n return `builder-field-${name.replace(/[^A-Za-z0-9_-]+/g, \"-\")}`;\n}\n\nfunction classNames(...names: Array<string | undefined>): string {\n return names.filter((name): name is string => Boolean(name)).join(\" \");\n}\n","import type { ReactNode } from \"react\";\n\nexport interface FieldProps {\n label: ReactNode;\n children: ReactNode;\n}\n\nexport function Field({ label, children }: FieldProps) {\n return (\n <label className=\"grid min-w-0 gap-1 text-2xs font-medium uppercase tracking-wide text-fg-muted\">\n {label}\n {children}\n </label>\n );\n}\n","import { Field } from \"./Field\";\nimport { Select } from \"./inputs\";\n\nexport interface AlignSelectProps {\n name: string;\n value: string;\n label?: string;\n onChange: (value: string) => void;\n}\n\nexport function AlignSelect({ name, value, label = \"Align\", onChange }: AlignSelectProps) {\n return (\n <Field label={label}>\n <Select\n className=\"w-full\"\n name={name}\n value={value}\n onChange={(event) => onChange(event.currentTarget.value)}\n >\n <option value=\"\" />\n <option value=\"left\">left</option>\n <option value=\"center\">center</option>\n <option value=\"right\">right</option>\n </Select>\n </Field>\n );\n}\n","import type { ButtonHTMLAttributes, ReactNode } from \"react\";\n\nexport type ButtonVariant = \"default\" | \"primary\" | \"danger\" | \"ghost\";\n\nconst base =\n \"inline-flex h-8 cursor-pointer items-center rounded-md border border-solid font-medium transition-colors disabled:cursor-not-allowed focus-visible:outline-none focus-visible:ring-3 focus-visible:ring-accent/20\";\n\nconst variantClass: Record<ButtonVariant, string> = {\n default:\n \"border-border bg-surface text-fg hover:border-border-strong hover:bg-surface-muted focus-visible:border-accent disabled:bg-surface disabled:text-fg-subtle\",\n primary:\n \"border-primary bg-primary font-semibold text-on-dark hover:border-primary-strong hover:bg-primary-strong disabled:border-border-strong disabled:bg-surface-muted disabled:text-fg-subtle\",\n danger:\n \"border-danger bg-danger-soft text-danger hover:border-danger hover:bg-danger hover:text-on-dark\",\n ghost:\n \"border-transparent bg-transparent text-fg hover:border-border hover:bg-surface-muted focus-visible:border-accent\",\n};\n\nexport interface ButtonProps extends ButtonHTMLAttributes<HTMLButtonElement> {\n variant?: ButtonVariant;\n /** Render as a square icon button (no label padding). */\n icon?: boolean;\n children: ReactNode;\n}\n\nexport function Button({\n variant = \"default\",\n icon = false,\n type = \"button\",\n className,\n children,\n ...rest\n}: ButtonProps) {\n const layout = icon ? \"w-8 justify-center p-0\" : \"gap-2 whitespace-nowrap px-3\";\n const classes = [base, layout, variantClass[variant], className].filter(Boolean).join(\" \");\n\n return (\n <button type={type} className={classes} {...rest}>\n {children}\n </button>\n );\n}\n\nexport function AddButton({\n type = \"button\",\n className,\n children,\n ...rest\n}: ButtonHTMLAttributes<HTMLButtonElement>) {\n const classes = [\n \"h-7 w-fit cursor-pointer rounded-md border border-dashed border-border-strong bg-transparent px-3 text-2xs text-fg-muted transition-colors hover:border-accent hover:text-accent disabled:cursor-not-allowed disabled:opacity-50\",\n className,\n ]\n .filter(Boolean)\n .join(\" \");\n\n return (\n <button type={type} className={classes} {...rest}>\n {children}\n </button>\n );\n}\n","import { useDraggable } from \"@dnd-kit/core\";\nimport { Chip } from \"../primitives/Chip\";\nimport { getBlockChrome } from \"./blockChrome\";\n\nexport interface BlockPaletteProps {\n blockTypes: string[];\n onAdd?: (type: string) => void;\n}\n\nexport function BlockPalette({ blockTypes, onAdd }: BlockPaletteProps) {\n return (\n <div\n className=\"flex min-w-0 gap-1 overflow-x-auto [scrollbar-width:thin]\"\n aria-label=\"Block palette\"\n >\n {blockTypes.map((type) => (\n <PaletteItem key={type} type={type} onAdd={onAdd} />\n ))}\n </div>\n );\n}\n\ninterface PaletteItemProps {\n type: string;\n onAdd?: (type: string) => void;\n}\n\nfunction PaletteItem({ type, onAdd }: PaletteItemProps) {\n const chrome = getBlockChrome(type);\n const { attributes, listeners, setNodeRef, transform, isDragging } = useDraggable({\n id: `palette:${type}`,\n data: {\n source: \"palette\",\n type,\n },\n });\n\n return (\n <button\n ref={setNodeRef}\n type=\"button\"\n className=\"flex flex-none cursor-grab items-center gap-3 m-0 min-h-8 rounded-md border border-solid border-transparent bg-transparent px-2 py-1 text-left text-xs font-medium text-fg transition-colors hover:border-border hover:bg-surface-muted active:scale-[0.98] active:cursor-grabbing active:bg-surface-muted\"\n style={{\n transform: transform ? `translate3d(${transform.x}px, ${transform.y}px, 0)` : undefined,\n opacity: isDragging ? 0 : undefined,\n }}\n onClick={() => onAdd?.(type)}\n aria-label={`Add ${chrome.label}`}\n {...listeners}\n {...attributes}\n >\n <Chip>{chrome.chip}</Chip>\n <span>{chrome.label}</span>\n </button>\n );\n}\n","import {\n useBuilderActions,\n useBuilderState,\n type TemplateExample,\n} from \"../context/BuilderContext\";\nimport { Select } from \"../controls\";\nimport { Button } from \"../primitives/Button\";\nimport { BlockPalette } from \"./BlockPalette\";\n\nexport interface PaletteProps {\n className?: string;\n /** Loadable examples keyed by display name. The Load control only renders when non-empty. */\n examples?: Record<string, TemplateExample>;\n}\n\nexport function Palette({ className, examples }: PaletteProps = {}) {\n const { blockTypes, schema } = useBuilderState();\n const { addBlock, loadExample } = useBuilderActions();\n const exampleEntries = examples ? Object.entries(examples) : [];\n\n return (\n <aside\n className={`flex h-[56px] min-w-0 items-center overflow-hidden border-0 border-b border-solid border-border bg-surface px-4${className ? ` ${className}` : \"\"}`}\n aria-label=\"Block palette\"\n >\n <div className=\"flex w-full min-w-0 items-center gap-3\">\n <h2 className=\"m-0 flex-none text-2xs font-medium uppercase tracking-[0.06em] text-fg-subtle\">\n Blocks\n </h2>\n <BlockPalette blockTypes={blockTypes} onAdd={addBlock} />\n {exampleEntries.length > 0 ? (\n <div className=\"ml-auto flex-none\">\n <ExampleLoader entries={exampleEntries} disabled={!schema} onLoad={loadExample} />\n </div>\n ) : null}\n </div>\n </aside>\n );\n}\n\ninterface ExampleLoaderProps {\n entries: [string, TemplateExample][];\n disabled: boolean;\n onLoad: (example: TemplateExample) => void;\n}\n\nfunction ExampleLoader({ entries, disabled, onLoad }: ExampleLoaderProps) {\n if (entries.length === 1) {\n const [, example] = entries[0];\n\n return (\n <Button onClick={() => onLoad(example)} disabled={disabled}>\n Load example\n </Button>\n );\n }\n\n return (\n <Select\n className=\"w-auto py-0\"\n value=\"\"\n disabled={disabled}\n aria-label=\"Load example\"\n onChange={(event) => {\n const match = entries.find(([name]) => name === event.currentTarget.value);\n\n if (match) {\n onLoad(match[1]);\n }\n }}\n >\n <option value=\"\" disabled>\n Load example…\n </option>\n {entries.map(([name]) => (\n <option key={name} value={name}>\n {name}\n </option>\n ))}\n </Select>\n );\n}\n","const CSS_PX_PER_INCH = 96;\nconst MM_PER_INCH = 25.4;\n\nexport function mmToPx(mm: number): number {\n return Math.round((mm / MM_PER_INCH) * CSS_PX_PER_INCH);\n}\n","import type { Orientation, PageFormat } from \"../../types/generated/template\";\n\nexport const PAGE_SIZES_MM: Record<PageFormat, [number, number]> = {\n A3: [297, 420],\n A4: [210, 297],\n A5: [148, 210],\n A6: [105, 148],\n Letter: [215.9, 279.4],\n Legal: [215.9, 355.6],\n Tabloid: [279.4, 431.8],\n};\n\nexport function pageSizeForFormat(\n format: PageFormat | string | undefined,\n orientation: Orientation | string | undefined = \"portrait\",\n): [number, number] {\n const portraitSize = PAGE_SIZES_MM[(format ?? \"A4\") as PageFormat] ?? PAGE_SIZES_MM.A4;\n\n return orientation === \"landscape\" ? [portraitSize[1], portraitSize[0]] : portraitSize;\n}\n","const COLUMN_GAP = \"0.375rem\";\nconst MIN_WIDTH = 5;\nconst TOTAL_WIDTH = 100;\n\nexport function parseWidths(\n widths: readonly (number | string)[] | null | undefined,\n count: number,\n): number[] {\n if (count <= 0) {\n return [];\n }\n\n if (!widths || widths.length !== count) {\n return equalWidths(count);\n }\n\n const parsed = widths.map(parseWidth);\n\n return parsed.every((width) => Number.isFinite(width)) ? parsed : equalWidths(count);\n}\n\nexport function formatWidths(widths: readonly number[]): string[] {\n return widths.map((width) => `${width}%`);\n}\n\nexport function gridTemplateForWidths(\n widths: readonly string[] | null | undefined,\n count: number,\n): string | null {\n if (!widths) {\n return null;\n }\n\n const tracks = parseWidths(widths, count).map((width) => `minmax(0, ${width}fr)`);\n\n return tracks.join(` ${COLUMN_GAP} `);\n}\n\nexport function setBoundary(\n widths: readonly number[],\n leftIndex: number,\n leftPercent: number,\n): number[] {\n const rightIndex = leftIndex + 1;\n\n if (leftIndex < 0 || rightIndex >= widths.length) {\n return [...widths];\n }\n\n const pairTotal = widths[leftIndex] + widths[rightIndex];\n const minLeft = Math.min(MIN_WIDTH, pairTotal);\n const maxLeft = Math.max(minLeft, pairTotal - MIN_WIDTH);\n const nextLeft = Math.round(clamp(leftPercent, minLeft, maxLeft));\n const nextWidths = [...widths];\n\n nextWidths[leftIndex] = nextLeft;\n nextWidths[rightIndex] = pairTotal - nextLeft;\n\n return nextWidths;\n}\n\nexport function labelWidthPercent(width: string | null | undefined, fallback = 30): number {\n const parsed = width == null ? Number.NaN : parseWidth(width);\n const value = Number.isFinite(parsed) ? parsed : fallback;\n\n return Math.round(clamp(value, MIN_WIDTH, TOTAL_WIDTH - MIN_WIDTH));\n}\n\nexport const NUMBER_COLUMN_RESERVE = 5;\n\nexport interface TableColumnTracks {\n tracks: number[];\n data: number[];\n}\n\nexport function percentWidth(width: string | number | null | undefined): number | null {\n if (width == null) {\n return null;\n }\n\n const parsed = parseWidth(width);\n\n return Number.isFinite(parsed) ? parsed : null;\n}\n\nexport function distribute(count: number, total: number): number[] {\n if (count <= 0) {\n return [];\n }\n\n const base = Math.floor(total / count);\n const widths = Array.from({ length: count }, () => base);\n widths[count - 1] += total - base * count;\n\n return widths;\n}\n\nexport function tableColumnTracks(\n widths: readonly (string | null | undefined)[],\n numberRows: boolean,\n): TableColumnTracks {\n const available = numberRows ? TOTAL_WIDTH - NUMBER_COLUMN_RESERVE : TOTAL_WIDTH;\n const parsed = widths.map((width) => percentWidth(width));\n const data =\n widths.length > 0 && parsed.every((value): value is number => value !== null)\n ? parsed\n : distribute(widths.length, available);\n const tracks = numberRows ? [NUMBER_COLUMN_RESERVE, ...data] : [...data];\n\n return { tracks, data };\n}\n\nfunction equalWidths(count: number): number[] {\n return distribute(count, TOTAL_WIDTH);\n}\n\nfunction parseWidth(width: number | string): number {\n if (typeof width === \"number\") {\n return width;\n }\n\n const percentMatch = width.trim().match(/^([+-]?\\d+(?:\\.\\d+)?)%$/);\n\n return percentMatch ? Number.parseFloat(percentMatch[1] ?? \"\") : Number.NaN;\n}\n\nfunction clamp(value: number, min: number, max: number): number {\n return Math.min(Math.max(value, min), max);\n}\n","import type { RefObject } from \"react\";\nimport { formatWidths, parseWidths, setBoundary } from \"./columns\";\n\nexport interface ColumnResizerProps {\n widths: readonly string[] | null;\n count: number;\n leftIndex: number;\n containerRef: RefObject<HTMLElement | null>;\n onResize: (widths: string[]) => void;\n label?: string;\n}\n\nexport function ColumnResizer({\n widths,\n count,\n leftIndex,\n containerRef,\n onResize,\n label,\n}: ColumnResizerProps) {\n function handlePointerDown(event: React.PointerEvent<HTMLButtonElement>): void {\n const bounds = containerRef.current?.getBoundingClientRect();\n\n if (!bounds || bounds.width <= 0) {\n return;\n }\n\n const containerBounds = bounds;\n\n event.preventDefault();\n event.currentTarget.setPointerCapture(event.pointerId);\n\n const parsedWidths = parseWidths(widths, count);\n const pairStart = parsedWidths.slice(0, leftIndex).reduce((total, width) => total + width, 0);\n\n function handlePointerMove(pointerEvent: PointerEvent): void {\n const pointerPercent =\n ((pointerEvent.clientX - containerBounds.left) / containerBounds.width) * 100;\n const leftPercent = pointerPercent - pairStart;\n\n onResize(formatWidths(setBoundary(parsedWidths, leftIndex, leftPercent)));\n }\n\n function handlePointerUp(): void {\n window.removeEventListener(\"pointermove\", handlePointerMove);\n window.removeEventListener(\"pointerup\", handlePointerUp);\n }\n\n window.addEventListener(\"pointermove\", handlePointerMove);\n window.addEventListener(\"pointerup\", handlePointerUp);\n }\n\n return (\n <button\n type=\"button\"\n className=\"group/resizer min-h-full w-1.5 cursor-col-resize select-none self-stretch rounded-full border-0 bg-transparent p-0 transition-colors max-[480px]:hidden\"\n aria-label={label ?? `Resize columns ${leftIndex + 1} and ${leftIndex + 2}`}\n onPointerDown={handlePointerDown}\n >\n <span className=\"mx-auto block h-full w-0.5 rounded-full bg-border-strong transition-colors group-hover/resizer:bg-accent\" />\n </button>\n );\n}\n","import type { CSSProperties, ReactNode } from \"react\";\nimport type { Orientation, PageFormat } from \"../../types/generated/template\";\nimport { mmToPx } from \"../lib/displayScale\";\nimport { pageSizeForFormat } from \"../lib/pageSizes\";\n\nexport interface PageSheetProps {\n format: PageFormat;\n orientation: Orientation;\n children: ReactNode;\n /** Show the \"{format} · {orientation} / {width}mm\" caption. Off for the footer sheet. */\n showMeta?: boolean;\n}\n\nexport function PageSheet({ format, orientation, children, showMeta = true }: PageSheetProps) {\n const [widthMm] = pageSizeForFormat(format, orientation);\n const style: CSSProperties = { maxWidth: `${mmToPx(widthMm)}px` };\n\n return (\n <div\n // The page mirrors the white PDF the backend renders, so it stays light\n // even in dark mode (the surrounding canvas/chrome still go dark).\n data-theme=\"light\"\n className=\"mx-auto grid w-full gap-3 rounded-xl border border-solid border-border bg-page p-6 shadow-page transition-[max-width] duration-200\"\n style={style}\n >\n {showMeta ? (\n <div className=\"mb-2 flex items-center justify-between text-2xs uppercase tracking-[0.06em] text-fg-subtle\">\n <span>\n {format} · {orientation}\n </span>\n <span>{Math.round(widthMm)}mm</span>\n </div>\n ) : null}\n {children}\n </div>\n );\n}\n","import type {\n Block,\n BlockConfig,\n DividerConfig,\n HeadingConfig,\n ImageConfig,\n KeyValueConfig,\n SpacingConfig,\n SpacerConfig,\n TableConfig,\n Template,\n TemplateConfig,\n TypographyConfig,\n} from \"../../types/generated/template\";\n\ntype EmptyConfigValue = \"\" | null | undefined;\ntype BlockType = Block[\"type\"];\ntype BlockByType<TType extends BlockType> = Extract<Block, { type: TType }>;\ntype BlockConfigByType = {\n text: BlockConfig;\n html: BlockConfig;\n heading: HeadingConfig;\n image: ImageConfig;\n \"key-value\": KeyValueConfig;\n spacer: SpacerConfig;\n divider: DividerConfig;\n table: TableConfig;\n};\ntype BlockConfigFor<TType extends BlockType> = BlockConfigByType[TType];\n\nexport function setBlockConfigField<\n TType extends BlockType,\n TKey extends keyof BlockConfigFor<TType>,\n>(\n block: BlockByType<TType>,\n field: TKey,\n value: BlockConfigFor<TType>[TKey] | EmptyConfigValue,\n): BlockByType<TType> {\n const nextConfig = setObjectField<BlockConfigFor<TType>, TKey>(\n block.config as BlockConfigFor<TType> | undefined,\n field,\n value,\n );\n\n return withBlockConfig(block, nextConfig);\n}\n\nexport function setBlockTypographyField<\n TType extends BlockType,\n TKey extends keyof TypographyConfig,\n>(\n block: BlockByType<TType>,\n field: TKey,\n value: TypographyConfig[TKey] | EmptyConfigValue,\n): BlockByType<TType> {\n const nextTypography = setObjectField(block.config?.typography, field, value);\n\n return setCommonBlockConfigField(block, \"typography\", nextTypography);\n}\n\nexport function setBlockSpacingField<TType extends BlockType, TKey extends keyof SpacingConfig>(\n block: BlockByType<TType>,\n field: TKey,\n value: SpacingConfig[TKey] | EmptyConfigValue,\n): BlockByType<TType> {\n const nextSpacing = setObjectField(block.config?.spacing, field, value);\n\n return setCommonBlockConfigField(block, \"spacing\", nextSpacing);\n}\n\nexport function setTemplateTypographyField<TKey extends keyof TypographyConfig>(\n template: Template,\n field: TKey,\n value: TypographyConfig[TKey] | EmptyConfigValue,\n): Template {\n const nextTypography = setObjectField(template.config?.typography, field, value);\n const nextConfig = setObjectField(template.config, \"typography\", nextTypography);\n\n return withTemplateConfig(template, nextConfig);\n}\n\nexport function setTemplatePageMargin<TKey extends keyof SpacingConfig>(\n template: Template,\n field: TKey,\n value: SpacingConfig[TKey] | EmptyConfigValue,\n): Template {\n const nextMargins = setObjectField(template.config?.page?.margins, field, value);\n const nextPage = setObjectField(template.config?.page, \"margins\", nextMargins);\n const nextConfig = setObjectField(template.config, \"page\", nextPage);\n\n return withTemplateConfig(template, nextConfig);\n}\n\nfunction setObjectField<TObject extends object, TKey extends keyof TObject>(\n source: TObject | undefined,\n field: TKey,\n value: TObject[TKey] | EmptyConfigValue,\n): TObject | undefined {\n const next = { ...source } as TObject;\n\n if (isClearedConfigValue(value)) {\n delete next[field];\n } else {\n next[field] = value;\n }\n\n return pruneEmptyObject(next);\n}\n\nfunction withBlockConfig<TType extends BlockType>(\n block: BlockByType<TType>,\n config: BlockConfigFor<TType> | undefined,\n): BlockByType<TType> {\n if (config === undefined) {\n const { config: _config, ...rest } = block;\n\n return rest as BlockByType<TType>;\n }\n\n return { ...block, config } as BlockByType<TType>;\n}\n\nfunction setCommonBlockConfigField<TType extends BlockType, TKey extends keyof BlockConfig>(\n block: BlockByType<TType>,\n field: TKey,\n value: BlockConfig[TKey] | EmptyConfigValue,\n): BlockByType<TType> {\n const nextConfig = setObjectField<BlockConfig, TKey>(block.config, field, value);\n\n return withBlockConfig(block, nextConfig as BlockConfigFor<TType> | undefined);\n}\n\nfunction withTemplateConfig(template: Template, config: TemplateConfig | undefined): Template {\n if (config === undefined) {\n const { config: _config, ...rest } = template;\n\n return rest;\n }\n\n return { ...template, config };\n}\n\nfunction pruneEmptyObject<TObject extends object>(value: TObject): TObject | undefined {\n return Object.keys(value).length === 0 ? undefined : value;\n}\n\nfunction isClearedConfigValue(value: unknown): value is EmptyConfigValue {\n return value === \"\" || value === null || value === undefined;\n}\n","import { useRef, type ReactNode } from \"react\";\nimport type {\n Block,\n DividerBlock,\n HeadingBlock,\n HtmlBlock,\n ImageBlock,\n KeyValueBlock,\n KeyValueValues,\n SpacerBlock,\n TableBlock,\n TextBlock,\n} from \"../../types/generated/template\";\nimport { isRecord } from \"../lib/records\";\nimport { setBlockConfigField } from \"../state/configUpdates\";\nimport { ColumnResizer } from \"./ColumnResizer\";\nimport { formatWidths, labelWidthPercent, tableColumnTracks } from \"./columns\";\n\nexport interface BlockDataPreviewProps {\n block: Block;\n rowData?: unknown;\n onChange?: (block: Block) => void;\n}\n\nconst copyClass =\n \"m-0 line-clamp-3 overflow-hidden text-sm leading-[1.45] text-fg-muted [display:-webkit-box] [-webkit-box-orient:vertical]\";\nconst headingCopyClass =\n \"m-0 line-clamp-2 overflow-hidden text-[17px] font-semibold leading-[1.25] text-fg [display:-webkit-box] [-webkit-box-orient:vertical]\";\n\nfunction EmptyPreview({ children }: { children: ReactNode }) {\n return <p className=\"m-0 text-sm text-fg-subtle\">{children}</p>;\n}\n\nexport function BlockDataPreview({ block, rowData, onChange }: BlockDataPreviewProps) {\n switch (block.type) {\n case \"heading\":\n return <TextPreview block={block} variant=\"heading\" />;\n case \"text\":\n return <TextPreview block={block} variant=\"text\" />;\n case \"html\":\n return <HtmlPreview block={block} />;\n case \"image\":\n return <ImagePreview block={block} />;\n case \"key-value\":\n return <KeyValuePreview block={block} rowData={rowData} onChange={onChange} />;\n case \"table\":\n return <TablePreview block={block} rowData={rowData} onChange={onChange} />;\n case \"spacer\":\n return <SpacerPreview block={block} />;\n case \"divider\":\n return <DividerPreview block={block} />;\n default:\n return null;\n }\n}\n\nfunction TextPreview({\n block,\n variant,\n}: {\n block: TextBlock | HeadingBlock;\n variant: \"text\" | \"heading\";\n}) {\n const text = block.text.trim();\n\n if (text.length === 0) {\n return <EmptyPreview>No text yet</EmptyPreview>;\n }\n\n return <p className={variant === \"heading\" ? headingCopyClass : copyClass}>{text}</p>;\n}\n\nfunction HtmlPreview({ block }: { block: HtmlBlock }) {\n const text = stripMarkup(block.html).trim();\n\n if (text.length === 0) {\n return <EmptyPreview>No HTML content yet</EmptyPreview>;\n }\n\n return <p className={copyClass}>{text}</p>;\n}\n\nfunction ImagePreview({ block }: { block: ImageBlock }) {\n const src = block.src.trim();\n\n if (src.length === 0) {\n return <EmptyPreview>No image selected</EmptyPreview>;\n }\n\n return (\n <div className=\"grid min-h-[72px] place-items-center overflow-hidden rounded-md border border-solid border-border bg-surface-muted\">\n <img\n className=\"block max-h-[120px] max-w-full object-contain\"\n src={src}\n alt={block.alt ?? \"\"}\n />\n </div>\n );\n}\n\nfunction KeyValuePreview({\n block,\n rowData,\n onChange,\n}: {\n block: KeyValueBlock;\n rowData?: unknown;\n onChange?: (block: Block) => void;\n}) {\n const listRef = useRef<HTMLDListElement | null>(null);\n const fields = block.config?.fields ?? [];\n const values = mergeRecordValues(block.values, rowData);\n const entries =\n fields.length > 0\n ? fields.map((field) => ({\n key: field.key,\n label: field.label || field.key,\n value: stringifyPreviewValue(values[field.key]),\n }))\n : Object.entries(values).map(([key, value]) => ({\n key,\n label: key,\n value: stringifyPreviewValue(value),\n }));\n\n if (entries.length === 0) {\n return <EmptyPreview>No fields yet</EmptyPreview>;\n }\n\n const labelPercent = labelWidthPercent(block.config?.labelWidth);\n const columnsStyle = { gridTemplateColumns: `${labelPercent}% minmax(0, 1fr)` };\n\n return (\n <dl ref={listRef} className=\"relative m-0 grid gap-0.5\">\n {entries.slice(0, 5).map((entry) => (\n <div key={entry.key} className=\"grid min-w-0 items-baseline\" style={columnsStyle}>\n <dt className=\"min-w-0 overflow-hidden text-ellipsis whitespace-nowrap pr-2 text-2xs text-fg-subtle\">\n {entry.label}\n </dt>\n <dd className=\"m-0 min-w-0 overflow-hidden text-ellipsis whitespace-nowrap pl-2 text-sm text-fg\">\n {entry.value || \"—\"}\n </dd>\n </div>\n ))}\n {onChange ? (\n <div\n className=\"absolute inset-y-0 flex -translate-x-1/2\"\n style={{ left: `${labelPercent}%` }}\n >\n <ColumnResizer\n widths={formatWidths([labelPercent, 100 - labelPercent])}\n count={2}\n leftIndex={0}\n containerRef={listRef}\n label=\"Resize the label column\"\n onResize={(widths) => onChange(setBlockConfigField(block, \"labelWidth\", widths[0]))}\n />\n </div>\n ) : null}\n </dl>\n );\n}\n\nfunction TablePreview({\n block,\n rowData,\n onChange,\n}: {\n block: TableBlock;\n rowData?: unknown;\n onChange?: (block: Block) => void;\n}) {\n const tableRef = useRef<HTMLDivElement | null>(null);\n const columns = block.config?.columns ?? [];\n const numberRows = block.config?.numberRows === true;\n const rows = Array.isArray(rowData) ? rowData.filter(isRecord) : [];\n\n if (columns.length === 0) {\n return <EmptyPreview>No columns yet</EmptyPreview>;\n }\n\n const { tracks, data } = tableColumnTracks(\n columns.map((column) => column.width),\n numberRows,\n );\n const trackStrings = formatWidths(tracks);\n const trackOffset = numberRows ? 1 : 0;\n\n return (\n <div\n ref={tableRef}\n className=\"relative min-w-0 overflow-hidden rounded-md border border-solid border-border\"\n >\n <table className=\"w-full table-fixed border-collapse text-2xs\">\n <colgroup>\n {numberRows ? <col style={{ width: `${tracks[0]}%` }} /> : null}\n {data.map((width, index) => (\n <col key={columns[index].key} style={{ width: `${width}%` }} />\n ))}\n </colgroup>\n <thead>\n <tr>\n {numberRows ? (\n <th className=\"border-0 border-b border-solid border-border bg-surface-muted px-2 py-[5px] text-left font-semibold text-fg-subtle\">\n #\n </th>\n ) : null}\n {columns.map((column) => (\n <th\n key={column.key}\n className=\"min-w-0 overflow-hidden text-ellipsis whitespace-nowrap border-0 border-b border-solid border-border bg-surface-muted px-2 py-[5px] text-left font-semibold text-fg\"\n >\n {column.label || column.key}\n </th>\n ))}\n </tr>\n </thead>\n <tbody>\n {rows.length > 0 ? (\n rows.map((row, rowIndex) => (\n <tr key={tableRowPreviewKey(row, columns)} className=\"last:[&>td]:border-b-0\">\n {numberRows ? (\n <td className=\"border-0 border-b border-solid border-border px-2 py-[5px] text-left text-fg-subtle\">\n {rowIndex + 1}\n </td>\n ) : null}\n {columns.map((column) => (\n <td\n key={column.key}\n className=\"min-w-0 overflow-hidden text-ellipsis whitespace-nowrap border-0 border-b border-solid border-border px-2 py-[5px] text-left text-fg-muted\"\n >\n {stringifyPreviewValue(row[column.key]) || \"—\"}\n </td>\n ))}\n </tr>\n ))\n ) : (\n <tr>\n <td\n colSpan={columns.length + trackOffset}\n className=\"min-w-0 overflow-hidden text-ellipsis whitespace-nowrap border-0 px-2 py-[5px] text-left text-fg-muted\"\n >\n No runtime rows yet\n </td>\n </tr>\n )}\n </tbody>\n </table>\n {onChange\n ? columns.slice(0, -1).map((column, index) => {\n const leftIndex = trackOffset + index;\n const boundaryPercent = tracks\n .slice(0, leftIndex + 1)\n .reduce((total, value) => total + value, 0);\n\n return (\n <div\n key={`${column.key}:resizer`}\n className=\"absolute inset-y-0 flex -translate-x-1/2\"\n style={{ left: `${boundaryPercent}%` }}\n >\n <ColumnResizer\n widths={trackStrings}\n count={tracks.length}\n leftIndex={leftIndex}\n containerRef={tableRef}\n label={`Resize column ${index + 1}`}\n onResize={(next) =>\n onChange(\n setBlockConfigField(\n block,\n \"columns\",\n columns.map((current, columnIndex) => ({\n ...current,\n width: (numberRows ? next.slice(1) : next)[columnIndex],\n })),\n ),\n )\n }\n />\n </div>\n );\n })\n : null}\n </div>\n );\n}\n\nfunction SpacerPreview({ block }: { block: SpacerBlock }) {\n const height = block.config?.height;\n\n return (\n <div className=\"grid h-9 place-items-center rounded-md border border-dashed border-border-strong text-2xs text-fg-subtle\">\n {typeof height === \"number\" ? `${height}mm spacer` : \"Spacer\"}\n </div>\n );\n}\n\nfunction DividerPreview({ block }: { block: DividerBlock }) {\n const style = block.config?.style ?? \"solid\";\n const styleClass =\n style === \"dashed\"\n ? \"border-dashed\"\n : style === \"dotted\"\n ? \"border-dotted\"\n : style === \"double\"\n ? \"border-double border-t-[3px]\"\n : style === \"none\"\n ? \"border-t-transparent\"\n : \"border-solid\";\n\n return <div className={`my-2 border-0 border-t border-border-strong ${styleClass}`} />;\n}\n\nfunction mergeRecordValues(\n values: KeyValueValues | undefined,\n rowData: unknown,\n): Record<string, unknown> {\n return Object.assign({}, values, isRecord(rowData) ? rowData : undefined);\n}\n\nfunction tableRowPreviewKey(\n row: Record<string, unknown>,\n columns: NonNullable<TableBlock[\"config\"]>[\"columns\"],\n): string {\n return columns?.map((column) => stringifyPreviewValue(row[column.key])).join(\"|\") ?? \"\";\n}\n\nfunction stringifyPreviewValue(value: unknown): string {\n if (value === null || value === undefined) {\n return \"\";\n }\n\n if (typeof value === \"string\") {\n return value;\n }\n\n if (typeof value === \"number\" || typeof value === \"boolean\") {\n return String(value);\n }\n\n return \"\";\n}\n\nfunction stripMarkup(value: string): string {\n return value.replace(/<[^>]*>/g, \" \").replace(/\\s+/g, \" \");\n}\n","import { useSortable } from \"@dnd-kit/sortable\";\nimport { CSS } from \"@dnd-kit/utilities\";\nimport type { CSSProperties, KeyboardEvent, MouseEvent } from \"react\";\nimport type { Block } from \"../../types/generated/template\";\nimport type { EditorArea, EditorBlock } from \"../state/editorModel\";\nimport type { TemplateData } from \"../../types/template\";\nimport { BlockDataPreview } from \"./BlockDataPreview\";\n\nexport interface SortableBlockProps {\n rowUid: string;\n area: EditorArea;\n editorBlock: EditorBlock;\n data: TemplateData;\n selected: boolean;\n onRemoveBlock: (blockUid: string) => void;\n onSelect: (blockUid: string) => void;\n onChangeBlock: (blockUid: string, block: Block) => void;\n style?: CSSProperties;\n}\n\nexport function SortableBlock({\n rowUid,\n area,\n editorBlock,\n data,\n selected,\n onRemoveBlock,\n onSelect,\n onChangeBlock,\n style: layoutStyle,\n}: SortableBlockProps) {\n const {\n attributes,\n listeners,\n setActivatorNodeRef,\n setNodeRef,\n transform,\n transition,\n isDragging,\n } = useSortable({\n id: editorBlock.uid,\n data: {\n type: \"block\",\n rowUid,\n blockUid: editorBlock.uid,\n area,\n },\n });\n\n const blockId = typeof editorBlock.block.id === \"string\" ? editorBlock.block.id : null;\n const rowData = blockId ? data[blockId] : undefined;\n const classes = [\n \"group/card relative grid min-w-0 cursor-pointer overflow-hidden rounded-lg border border-solid bg-surface transition-[border-color,box-shadow,background] hover:border-border-strong\",\n isDragging ? \"border-dashed border-border opacity-45\" : \"border-border\",\n selected ? \"!border-accent ring-2 ring-accent-soft\" : \"\",\n ]\n .filter(Boolean)\n .join(\" \");\n const style: CSSProperties = {\n transform: CSS.Transform.toString(transform),\n transition,\n ...layoutStyle,\n };\n\n function selectBlock() {\n onSelect(editorBlock.uid);\n }\n\n function handleCardKeyDown(event: KeyboardEvent<HTMLElement>) {\n if (event.target !== event.currentTarget) {\n return;\n }\n if (event.key === \"Enter\" || event.key === \" \") {\n event.preventDefault();\n selectBlock();\n }\n }\n\n function handleRemove(event: MouseEvent) {\n event.stopPropagation();\n onRemoveBlock(editorBlock.uid);\n }\n\n const chromeRevealedClass = selected || isDragging ? \"opacity-100\" : \"opacity-0\";\n\n return (\n <article\n ref={setNodeRef}\n className={classes}\n style={style}\n aria-current={selected}\n tabIndex={0}\n onClick={selectBlock}\n onKeyDown={handleCardKeyDown}\n >\n <div\n className={`pointer-events-none absolute inset-x-2 top-2 z-[1] flex items-center justify-between gap-1 transition-opacity group-hover/card:opacity-100 group-focus-within/card:opacity-100 ${chromeRevealedClass}`}\n >\n <button\n ref={setActivatorNodeRef}\n type=\"button\"\n className=\"pointer-events-auto inline-grid h-[22px] w-[22px] flex-none cursor-grab place-items-center rounded border border-solid border-border bg-surface/90 p-0 font-mono text-xs text-fg-subtle transition-[background,border-color,color] hover:border-border-strong hover:bg-surface hover:text-fg active:cursor-grabbing\"\n aria-label=\"Drag to move block\"\n onClick={(event) => event.stopPropagation()}\n {...attributes}\n {...listeners}\n >\n ⋮⋮\n </button>\n <div className=\"pointer-events-auto inline-flex flex-none items-center gap-1\">\n <button\n type=\"button\"\n className=\"inline-grid h-[22px] w-[22px] cursor-pointer place-items-center rounded border border-solid border-border bg-surface/90 p-0 text-[13px] text-fg-muted transition-[background,border-color,color] hover:border-border-strong hover:bg-danger-soft hover:text-danger\"\n aria-label=\"Remove block\"\n onClick={handleRemove}\n >\n ✕\n </button>\n </div>\n </div>\n <div className=\"grid min-w-0 bg-surface p-3\">\n <BlockDataPreview\n block={editorBlock.block}\n rowData={rowData}\n onChange={(block) => onChangeBlock(editorBlock.uid, block)}\n />\n </div>\n </article>\n );\n}\n","import { useDroppable } from \"@dnd-kit/core\";\nimport { SortableContext, useSortable, verticalListSortingStrategy } from \"@dnd-kit/sortable\";\nimport { CSS } from \"@dnd-kit/utilities\";\nimport type { CSSProperties, RefObject } from \"react\";\nimport { Fragment, useRef } from \"react\";\nimport type { Block, Orientation, PageFormat } from \"../../types/generated/template\";\nimport type { TemplateData } from \"../../types/template\";\nimport type {\n EditorArea,\n EditorModel,\n EditorRow,\n PageNumbersValue,\n} from \"../state/editorModel\";\nimport { Checkbox, Select } from \"../controls\";\nimport { mmToPx } from \"../lib/displayScale\";\nimport { pageSizeForFormat } from \"../lib/pageSizes\";\nimport { ColumnResizer } from \"./ColumnResizer\";\nimport { gridTemplateForWidths } from \"./columns\";\nimport { PageSheet } from \"./PageSheet\";\nimport { SortableBlock } from \"./SortableBlock\";\n\nexport interface BuilderCanvasProps {\n model: EditorModel;\n data: TemplateData;\n format: PageFormat;\n orientation: Orientation;\n footerRepeat: boolean;\n pageNumbers: PageNumbersValue;\n selectedBlockUid: string | null;\n onRemoveBlock: (blockUid: string) => void;\n onSelectBlock: (blockUid: string) => void;\n onChangeBlock: (blockUid: string, block: Block) => void;\n onDeselect: () => void;\n onSetRowWidths: (rowUid: string, widths: string[]) => void;\n onToggleFooterRepeat: (repeat: boolean) => void;\n onChangePageNumbers: (value: PageNumbersValue) => void;\n className?: string;\n}\n\nexport function BuilderCanvas({\n model,\n data,\n format,\n orientation,\n footerRepeat,\n pageNumbers,\n selectedBlockUid,\n onRemoveBlock,\n onSelectBlock,\n onChangeBlock,\n onDeselect,\n onSetRowWidths,\n onToggleFooterRepeat,\n onChangePageNumbers,\n className,\n}: BuilderCanvasProps) {\n const [widthMm] = pageSizeForFormat(format, orientation);\n const pageWidth: CSSProperties = { maxWidth: `${mmToPx(widthMm)}px` };\n\n return (\n <div\n className={`grid content-start gap-4 min-w-0 min-h-0 overflow-auto bg-canvas px-4 pb-8 pt-6${className ? ` ${className}` : \"\"}`}\n onPointerDown={(event) => {\n if (event.target === event.currentTarget) {\n onDeselect();\n }\n }}\n >\n <PageSheet format={format} orientation={orientation}>\n <CanvasArea\n area=\"body\"\n rows={model.rows}\n data={data}\n selectedBlockUid={selectedBlockUid}\n newRowId=\"new-row\"\n emptyLabel=\"Drop a block here to begin\"\n fillLabel=\"Drop a block here to add a new row\"\n onRemoveBlock={onRemoveBlock}\n onSelectBlock={onSelectBlock}\n onChangeBlock={onChangeBlock}\n onSetRowWidths={onSetRowWidths}\n />\n </PageSheet>\n\n <PageSheet format={format} orientation={orientation} showMeta={false}>\n <div\n className=\"mb-2 flex items-center justify-between gap-3\"\n aria-label=\"Page footer settings\"\n >\n <h2 className=\"m-0 text-2xs font-semibold uppercase tracking-[0.06em] text-fg-subtle\">\n Footer\n </h2>\n <label className=\"inline-flex items-center gap-2 text-xs font-medium text-fg-muted\">\n <Checkbox\n checked={footerRepeat}\n onChange={(event) => onToggleFooterRepeat(event.currentTarget.checked)}\n />\n Repeat on every page\n </label>\n </div>\n\n <CanvasArea\n area=\"footer\"\n rows={model.footerRows}\n data={data}\n selectedBlockUid={selectedBlockUid}\n newRowId=\"new-footer-row\"\n emptyLabel=\"Drop a block here to start the footer\"\n fillLabel=\"Drop a block here to add a footer row\"\n onRemoveBlock={onRemoveBlock}\n onSelectBlock={onSelectBlock}\n onChangeBlock={onChangeBlock}\n onSetRowWidths={onSetRowWidths}\n />\n </PageSheet>\n\n <div className=\"mx-auto flex w-full justify-center\" style={pageWidth}>\n <label className=\"inline-flex items-center gap-3 text-2xs font-medium uppercase tracking-[0.06em] text-fg-muted\">\n Page numbers\n <Select\n className=\"w-auto py-0 text-sm font-normal normal-case tracking-normal\"\n value={pageNumbers}\n onChange={(event) =>\n onChangePageNumbers(event.currentTarget.value as PageNumbersValue)\n }\n >\n <option value=\"disabled\">Disabled</option>\n <option value=\"left\">Left</option>\n <option value=\"center\">Center</option>\n <option value=\"right\">Right</option>\n </Select>\n </label>\n </div>\n </div>\n );\n}\n\ninterface CanvasAreaProps {\n area: EditorArea;\n rows: EditorRow[];\n data: TemplateData;\n selectedBlockUid: string | null;\n newRowId: string;\n emptyLabel: string;\n fillLabel: string;\n onRemoveBlock: (blockUid: string) => void;\n onSelectBlock: (blockUid: string) => void;\n onChangeBlock: (blockUid: string, block: Block) => void;\n onSetRowWidths: (rowUid: string, widths: string[]) => void;\n}\n\nfunction CanvasArea({\n area,\n rows,\n data,\n selectedBlockUid,\n newRowId,\n emptyLabel,\n fillLabel,\n onRemoveBlock,\n onSelectBlock,\n onChangeBlock,\n onSetRowWidths,\n}: CanvasAreaProps) {\n const { setNodeRef: setNewRowRef, isOver: isNewRowOver } = useDroppable({\n id: newRowId,\n data: { type: \"new-row\", area },\n });\n\n return (\n <div className=\"grid gap-3\">\n <SortableContext\n items={rows.map((row) => row.uid)}\n strategy={verticalListSortingStrategy}\n >\n {rows.map((row) => (\n <CanvasRow\n key={row.uid}\n row={row}\n area={area}\n data={data}\n selectedBlockUid={selectedBlockUid}\n onRemoveBlock={onRemoveBlock}\n onSelectBlock={onSelectBlock}\n onChangeBlock={onChangeBlock}\n onSetRowWidths={onSetRowWidths}\n />\n ))}\n </SortableContext>\n\n <div\n ref={setNewRowRef}\n className={\n isNewRowOver\n ? \"grid min-h-10 place-items-center rounded-lg border border-dashed border-accent bg-accent-soft p-3 text-sm text-accent transition-[border-color,background,color]\"\n : \"grid min-h-10 place-items-center rounded-lg border border-dashed border-border-strong bg-transparent p-3 text-sm text-fg-subtle transition-[border-color,background,color]\"\n }\n >\n {rows.length === 0 ? emptyLabel : fillLabel}\n </div>\n </div>\n );\n}\n\ninterface CanvasRowProps {\n row: EditorRow;\n area: EditorArea;\n data: TemplateData;\n selectedBlockUid: string | null;\n onRemoveBlock: (blockUid: string) => void;\n onSelectBlock: (blockUid: string) => void;\n onChangeBlock: (blockUid: string, block: Block) => void;\n onSetRowWidths: (rowUid: string, widths: string[]) => void;\n}\n\nfunction CanvasRow({\n row,\n area,\n data,\n selectedBlockUid,\n onRemoveBlock,\n onSelectBlock,\n onChangeBlock,\n onSetRowWidths,\n}: CanvasRowProps) {\n const rowRef = useRef<HTMLDivElement | null>(null);\n const {\n attributes,\n listeners,\n setActivatorNodeRef,\n setNodeRef,\n transform,\n transition,\n isDragging,\n } = useSortable({\n id: row.uid,\n data: {\n type: \"row\",\n rowUid: row.uid,\n area,\n },\n });\n const widths = getRowWidths(row);\n const canResizeColumns = row.blocks.length > 1;\n const gridTemplateColumns = gridTemplateForWidths(\n canResizeColumns ? (widths ?? []) : widths,\n row.blocks.length,\n );\n const style: CSSProperties = {\n transform: CSS.Transform.toString(transform),\n transition,\n };\n\n return (\n <section\n ref={setNodeRef}\n className={\n isDragging\n ? \"group/row grid min-w-0 gap-2 rounded-lg opacity-50 transition-[background]\"\n : \"group/row grid min-w-0 gap-2 rounded-lg transition-[background]\"\n }\n style={style}\n >\n <div className=\"-mb-0.5 flex h-[22px] items-center gap-2 opacity-0 transition-opacity focus-within:opacity-100 group-hover/row:opacity-100\">\n <button\n ref={setActivatorNodeRef}\n type=\"button\"\n className=\"inline-flex h-[22px] cursor-grab items-center border-0 bg-transparent px-2 font-mono text-2xs text-fg-subtle hover:text-fg\"\n aria-label=\"Drag to move row\"\n {...attributes}\n {...listeners}\n >\n ⋮⋮ row\n </button>\n </div>\n <div\n ref={rowRef}\n className=\"relative grid min-w-0 items-stretch gap-2 max-[480px]:!grid-cols-1\"\n style={gridTemplateColumns ? { gridTemplateColumns } : undefined}\n >\n <SortableContext items={row.blocks.map((block) => block.uid)}>\n {row.blocks.map((editorBlock, index) => (\n <Fragment key={editorBlock.uid}>\n <SortableBlock\n rowUid={row.uid}\n area={area}\n editorBlock={editorBlock}\n data={data}\n selected={editorBlock.uid === selectedBlockUid}\n onRemoveBlock={onRemoveBlock}\n onSelect={onSelectBlock}\n onChangeBlock={onChangeBlock}\n />\n {canResizeColumns && index < row.blocks.length - 1 ? (\n <ColumnResizer\n key={`${editorBlock.uid}:resizer`}\n widths={widths}\n count={row.blocks.length}\n leftIndex={index}\n containerRef={rowRef as RefObject<HTMLElement | null>}\n onResize={(nextWidths) => onSetRowWidths(row.uid, nextWidths)}\n />\n ) : null}\n </Fragment>\n ))}\n </SortableContext>\n </div>\n </section>\n );\n}\n\nfunction getRowWidths(row: EditorRow): string[] | null {\n const widths = row.blocks.map((editorBlock) => editorBlock.block.config?.width);\n\n return widths.every((width): width is string => typeof width === \"string\") ? widths : null;\n}\n","import { useBuilderActions, useBuilderState } from \"../context/BuilderContext\";\nimport { BuilderCanvas } from \"./BuilderCanvas\";\n\nexport interface CanvasProps {\n className?: string;\n}\n\nexport function Canvas({ className }: CanvasProps = {}) {\n const {\n schema,\n schemaLoading,\n model,\n data,\n pageSize,\n footerRepeat,\n pageNumbers,\n selectedBlockUid,\n } = useBuilderState();\n const {\n removeBlock,\n selectBlock,\n changeBlock,\n deselect,\n setRowWidths,\n toggleFooterRepeat,\n changePageNumbers,\n } = useBuilderActions();\n\n if (!schema) {\n return (\n <div\n className={`min-w-0 min-h-0 overflow-auto bg-canvas px-4 pb-8 pt-6${className ? ` ${className}` : \"\"}`}\n >\n <div className=\"grid h-full place-items-center text-sm text-fg-muted\">\n {schemaLoading ? \"Loading schema…\" : \"Load the schema to start building.\"}\n </div>\n </div>\n );\n }\n\n return (\n <BuilderCanvas\n className={className}\n model={model}\n data={data}\n format={pageSize.format}\n orientation={pageSize.orientation}\n footerRepeat={footerRepeat}\n pageNumbers={pageNumbers}\n selectedBlockUid={selectedBlockUid}\n onRemoveBlock={removeBlock}\n onSelectBlock={selectBlock}\n onChangeBlock={changeBlock}\n onDeselect={deselect}\n onSetRowWidths={setRowWidths}\n onToggleFooterRepeat={toggleFooterRepeat}\n onChangePageNumbers={changePageNumbers}\n />\n );\n}\n","import type { ChangeEvent, ReactNode } from \"react\";\nimport type { ImageBlock } from \"../../../types/generated/template\";\nimport type { BlockEditorProps } from \"./blockEditors\";\nimport { Field, Input } from \"../../controls\";\n\nexport function ImageBlockEditor({ block, onChangeBlock }: BlockEditorProps): ReactNode {\n const imageBlock = block as ImageBlock;\n const src = imageBlock.src ?? \"\";\n const alt = imageBlock.alt ?? \"\";\n\n function handleFileChange(event: ChangeEvent<HTMLInputElement>): void {\n const file = event.currentTarget.files?.[0];\n\n if (!file) {\n return;\n }\n\n const reader = new FileReader();\n\n reader.addEventListener(\"load\", () => {\n if (typeof reader.result === \"string\") {\n onChangeBlock({ ...imageBlock, src: reader.result });\n }\n });\n reader.readAsDataURL(file);\n }\n\n return (\n <div className=\"grid gap-3 [container-type:inline-size]\">\n <div\n className=\"grid min-h-24 place-items-center rounded-md border border-solid border-border bg-surface-muted p-3\"\n data-name=\"preview\"\n >\n {src ? (\n <img src={src} alt={alt} className=\"max-h-[120px] max-w-full object-contain\" />\n ) : (\n <div className=\"text-2xs text-fg-muted\">No image selected</div>\n )}\n </div>\n\n <Field label=\"Source\">\n <Input\n name=\"src\"\n type=\"text\"\n value={src}\n onChange={(event) => onChangeBlock({ ...imageBlock, src: event.currentTarget.value })}\n />\n </Field>\n\n <Field label=\"Upload\">\n <Input name=\"src-file\" type=\"file\" accept=\"image/*\" onChange={handleFileChange} />\n </Field>\n\n <Field label=\"Alt text\">\n <Input\n name=\"alt\"\n type=\"text\"\n value={alt}\n onChange={(event) => onChangeBlock({ ...imageBlock, alt: event.currentTarget.value })}\n />\n </Field>\n <p className=\"-mt-2 m-0 text-2xs text-fg-muted\">\n Alt text is required for PDF/UA accessibility. Describe what the image conveys.\n </p>\n </div>\n );\n}\n","import { forwardRef, type KeyboardEvent, type ReactNode } from \"react\";\n\nexport interface InspectorShellProps {\n ariaLabel: string;\n className?: string;\n onKeyDown?: (event: KeyboardEvent<HTMLElement>) => void;\n children: ReactNode;\n}\n\nexport const InspectorShell = forwardRef<HTMLElement, InspectorShellProps>(function InspectorShell(\n { ariaLabel, className, onKeyDown, children },\n ref,\n) {\n return (\n <aside\n ref={ref}\n tabIndex={-1}\n onKeyDown={onKeyDown}\n className={`grid min-h-0 min-w-0 content-start gap-4 overflow-x-hidden overflow-y-auto bg-surface p-4 outline-none${className ? ` ${className}` : \"\"}`}\n aria-label={ariaLabel}\n >\n {children}\n </aside>\n );\n});\n\nexport interface InspectorHeaderProps {\n title: string;\n chip?: ReactNode;\n subtitle?: string;\n action?: ReactNode;\n}\n\nexport function InspectorHeader({ title, chip, subtitle, action }: InspectorHeaderProps) {\n return (\n <header className=\"flex min-w-0 items-start justify-between gap-2\">\n <div className=\"flex min-w-0 flex-auto items-start gap-2\">\n {chip}\n <div>\n <h2 className=\"m-0 text-[15px] font-semibold text-fg\">{title}</h2>\n {subtitle ? (\n <p className=\"mt-0.5 mb-0 break-words text-xs text-fg-muted\">{subtitle}</p>\n ) : null}\n </div>\n </div>\n {action}\n </header>\n );\n}\n\nexport function InspectorSection({ title, children }: { title: string; children: ReactNode }) {\n return (\n <fieldset className=\"grid min-w-0 gap-2 overflow-hidden rounded-md border border-solid border-border bg-surface-muted p-3\">\n <legend className=\"px-1 text-2xs font-semibold uppercase tracking-wide text-fg-muted\">\n {title}\n </legend>\n {children}\n </fieldset>\n );\n}\n","import { DndContext, closestCenter, type DragEndEvent } from \"@dnd-kit/core\";\nimport { SortableContext, verticalListSortingStrategy } from \"@dnd-kit/sortable\";\nimport type { ReactNode } from \"react\";\nimport { useBuilderSensors } from \"../../lib/sensors\";\n\nexport interface SortableListProps {\n count: number;\n onReorder: (sourceIndex: number, targetIndex: number) => void;\n children: ReactNode;\n}\n\nexport function SortableList({ count, onReorder, children }: SortableListProps) {\n const sensors = useBuilderSensors();\n\n function handleDragEnd(event: DragEndEvent): void {\n const { active, over } = event;\n\n if (!over || active.id === over.id) {\n return;\n }\n\n const sourceIndex = Number(active.id);\n const targetIndex = Number(over.id);\n\n if (!Number.isFinite(sourceIndex) || !Number.isFinite(targetIndex)) {\n return;\n }\n\n onReorder(sourceIndex, targetIndex);\n }\n\n const sortableIds = Array.from({ length: count }, (_, index) => String(index));\n\n return (\n <DndContext sensors={sensors} collisionDetection={closestCenter} onDragEnd={handleDragEnd}>\n <SortableContext items={sortableIds} strategy={verticalListSortingStrategy}>\n {children}\n </SortableContext>\n </DndContext>\n );\n}\n","import { useSortable } from \"@dnd-kit/sortable\";\nimport { CSS } from \"@dnd-kit/utilities\";\nimport type { CSSProperties, ReactNode } from \"react\";\n\nexport interface SortableRowProps {\n id: string;\n dragLabel: string;\n removeLabel: string;\n removeName?: string;\n onRemove: () => void;\n children: ReactNode;\n}\n\nexport function SortableRow({\n id,\n dragLabel,\n removeLabel,\n removeName,\n onRemove,\n children,\n}: SortableRowProps) {\n const {\n attributes,\n listeners,\n setActivatorNodeRef,\n setNodeRef,\n transform,\n transition,\n isDragging,\n } = useSortable({ id });\n const style: CSSProperties = {\n transform: CSS.Transform.toString(transform),\n transition,\n opacity: isDragging ? 0.6 : 1,\n };\n\n return (\n <div\n ref={setNodeRef}\n className=\"relative grid gap-2 rounded-md border border-solid border-border bg-surface-muted py-3 pr-8 pl-8\"\n style={style}\n >\n <button\n ref={setActivatorNodeRef}\n type=\"button\"\n className=\"absolute top-2 left-2 inline-grid h-[22px] w-[22px] cursor-grab place-items-center rounded border-0 bg-transparent p-0 font-mono text-xs tracking-tighter text-fg-muted hover:bg-surface hover:text-fg active:cursor-grabbing\"\n aria-label={dragLabel}\n {...attributes}\n {...listeners}\n >\n ⋮⋮\n </button>\n {children}\n <button\n type=\"button\"\n data-name={removeName}\n className=\"absolute top-2 right-2 inline-grid h-[22px] w-[22px] cursor-pointer place-items-center rounded border-0 bg-transparent p-0 text-xs text-fg-muted hover:bg-danger-soft hover:text-danger\"\n aria-label={removeLabel}\n onClick={onRemove}\n >\n ✕\n </button>\n </div>\n );\n}\n","import { arrayMove } from \"@dnd-kit/sortable\";\nimport type { ReactNode } from \"react\";\nimport type {\n Block,\n KeyValueBlock,\n KeyValueField,\n KeyValueValues,\n} from \"../../../types/generated/template\";\nimport { isRecord, nextKeyIndex, omitKey, renameKey } from \"../../lib/records\";\nimport { AddButton } from \"../../primitives/Button\";\nimport type { BlockEditorProps } from \"./blockEditors\";\nimport { InspectorSection } from \"../InspectorShell\";\nimport { SortableList } from \"./SortableList\";\nimport { SortableRow } from \"./SortableRow\";\nimport { Field, Input } from \"../../controls\";\n\nexport function moveField(\n fields: readonly KeyValueField[],\n sourceIndex: number,\n targetIndex: number,\n): KeyValueField[] {\n return arrayMove([...fields], sourceIndex, targetIndex);\n}\n\nexport function addField(block: KeyValueBlock): KeyValueBlock {\n const fields = getFields(block);\n const values = getValues(block);\n const nextIndex = nextKeyIndex(\n fields.map((field) => field.key),\n \"field\",\n );\n const nextFields = [...fields, { key: `field${nextIndex}`, label: `Field ${nextIndex}` }];\n\n return applyFields(block, nextFields, values);\n}\n\nexport function removeField(block: KeyValueBlock, index: number): KeyValueBlock {\n const fields = getFields(block);\n const removed = fields[index];\n\n if (!removed) {\n return block;\n }\n\n return applyFields(\n block,\n fields.filter((_, currentIndex) => currentIndex !== index),\n omitKey(getValues(block), removed.key),\n );\n}\n\nexport function renameFieldKey(\n block: KeyValueBlock,\n index: number,\n nextKey: string,\n): KeyValueBlock {\n const fields = getFields(block);\n const previous = fields[index];\n\n if (!previous || previous.key === nextKey) {\n return block;\n }\n\n if (fields.some((field, currentIndex) => currentIndex !== index && field.key === nextKey)) {\n return block;\n }\n\n return applyFields(\n block,\n fields.map((field, currentIndex) =>\n currentIndex === index ? { ...field, key: nextKey } : field,\n ),\n renameKey(getValues(block), previous.key, nextKey),\n );\n}\n\nexport function setFieldLabel(\n block: KeyValueBlock,\n index: number,\n nextLabel: string,\n): KeyValueBlock {\n const fields = getFields(block);\n\n return applyFields(\n block,\n fields.map((field, currentIndex) =>\n currentIndex === index ? { ...field, label: nextLabel } : field,\n ),\n getValues(block),\n );\n}\n\nexport function setValue(block: KeyValueBlock, key: string, nextValue: string): KeyValueBlock {\n return applyFields(block, getFields(block), { ...getValues(block), [key]: nextValue });\n}\n\nexport function reorderFields(\n block: KeyValueBlock,\n sourceIndex: number,\n targetIndex: number,\n): KeyValueBlock {\n return applyFields(block, moveField(getFields(block), sourceIndex, targetIndex), getValues(block));\n}\n\nexport function KeyValueBlockEditor({ block, onChangeBlock }: BlockEditorProps): ReactNode {\n const kvBlock = block as KeyValueBlock;\n const fields = getFields(kvBlock);\n const values = getValues(kvBlock);\n\n return (\n <>\n <InspectorSection title=\"Fields\">\n <FieldsEditor block={kvBlock} fields={fields} onChangeBlock={onChangeBlock} />\n </InspectorSection>\n\n {fields.length > 0 ? (\n <InspectorSection title=\"Values\">\n {fields.map((field) => (\n <Field key={field.key} label={field.label || field.key}>\n <Input\n name={`values.${field.key}`}\n type=\"text\"\n value={String(values[field.key] ?? \"\")}\n onChange={(event) =>\n onChangeBlock(setValue(kvBlock, field.key, event.currentTarget.value))\n }\n />\n </Field>\n ))}\n </InspectorSection>\n ) : null}\n </>\n );\n}\n\ninterface FieldsEditorProps {\n block: KeyValueBlock;\n fields: KeyValueField[];\n onChangeBlock: (block: Block) => void;\n}\n\nfunction FieldsEditor({ block, fields, onChangeBlock }: FieldsEditorProps) {\n return (\n <>\n <SortableList\n count={fields.length}\n onReorder={(source, target) => onChangeBlock(reorderFields(block, source, target))}\n >\n {fields.map((field, index) => (\n <FieldRow\n key={fieldRowKey(field, index)}\n id={String(index)}\n index={index}\n field={field}\n onChangeKey={(value) => onChangeBlock(renameFieldKey(block, index, value))}\n onChangeLabel={(value) => onChangeBlock(setFieldLabel(block, index, value))}\n onRemove={() => onChangeBlock(removeField(block, index))}\n />\n ))}\n </SortableList>\n <AddButton data-name=\"add-field\" onClick={() => onChangeBlock(addField(block))}>\n Add field\n </AddButton>\n </>\n );\n}\n\ninterface FieldRowProps {\n id: string;\n index: number;\n field: KeyValueField;\n onChangeKey: (value: string) => void;\n onChangeLabel: (value: string) => void;\n onRemove: () => void;\n}\n\nfunction FieldRow({ id, index, field, onChangeKey, onChangeLabel, onRemove }: FieldRowProps) {\n return (\n <SortableRow\n id={id}\n dragLabel={`Drag to reorder field ${index + 1}`}\n removeLabel={`Remove field ${index + 1}`}\n removeName={`remove-field-${index}`}\n onRemove={onRemove}\n >\n <div className=\"grid grid-cols-2 gap-2\">\n <Field label=\"Key\">\n <Input\n name={`field-key-${index}`}\n type=\"text\"\n value={field.key}\n onChange={(event) => onChangeKey(event.currentTarget.value)}\n />\n </Field>\n <Field label=\"Label\">\n <Input\n name={`field-label-${index}`}\n type=\"text\"\n value={field.label}\n onChange={(event) => onChangeLabel(event.currentTarget.value)}\n />\n </Field>\n </div>\n </SortableRow>\n );\n}\n\nfunction fieldRowKey(field: KeyValueField, index: number): string {\n return field.key.length > 0 ? `key:${field.key}` : `index:${index}`;\n}\n\nfunction getFields(block: KeyValueBlock): KeyValueField[] {\n const candidate = block.config?.fields;\n\n return Array.isArray(candidate) ? (candidate as KeyValueField[]) : [];\n}\n\nfunction getValues(block: KeyValueBlock): KeyValueValues {\n const candidate = block.values;\n\n return isRecord(candidate) ? (candidate as KeyValueValues) : {};\n}\n\nfunction applyFields(\n block: KeyValueBlock,\n fields: KeyValueField[],\n values: KeyValueValues,\n): KeyValueBlock {\n const config = { ...block.config };\n\n if (fields.length === 0) {\n delete (config as { fields?: KeyValueField[] }).fields;\n } else {\n (config as { fields?: KeyValueField[] }).fields = fields;\n }\n\n const nextBlock: KeyValueBlock = {\n ...block,\n config: Object.keys(config).length === 0 ? undefined : config,\n };\n\n if (Object.keys(values).length === 0) {\n delete (nextBlock as { values?: KeyValueValues }).values;\n } else {\n nextBlock.values = values;\n }\n\n if (nextBlock.config === undefined) {\n delete (nextBlock as { config?: KeyValueBlock[\"config\"] }).config;\n }\n\n return nextBlock;\n}\n","import { arrayMove } from \"@dnd-kit/sortable\";\nimport type { ReactNode } from \"react\";\nimport type { Align, Block, TableBlock, TableColumn } from \"../../../types/generated/template\";\nimport { NUMBER_COLUMN_RESERVE, percentWidth } from \"../../canvas/columns\";\nimport { isRecord, nextKeyIndex, omitKey, renameKey } from \"../../lib/records\";\nimport { AddButton } from \"../../primitives/Button\";\nimport { InspectorSection } from \"../InspectorShell\";\nimport { setBlockConfigField } from \"../../state/configUpdates\";\nimport type { BlockEditorProps } from \"./blockEditors\";\nimport { SortableList } from \"./SortableList\";\nimport { SortableRow } from \"./SortableRow\";\nimport { AlignSelect, Field, Input } from \"../../controls\";\n\nconst hintClass = \"m-0 text-2xs text-fg-muted\";\n\ntype TableRow = Record<string, string>;\n\nexport function moveColumn(\n columns: readonly TableColumn[],\n sourceIndex: number,\n targetIndex: number,\n): TableColumn[] {\n return arrayMove([...columns], sourceIndex, targetIndex);\n}\n\nexport function moveRow(\n rows: readonly TableRow[],\n sourceIndex: number,\n targetIndex: number,\n): TableRow[] {\n return arrayMove([...rows], sourceIndex, targetIndex);\n}\n\nexport function addColumn(block: TableBlock): TableBlock {\n const columns = getColumns(block);\n const nextIndex = nextKeyIndex(\n columns.map((column) => column.key),\n \"column\",\n );\n const nextColumns: TableColumn[] = [\n ...columns,\n { key: `column${nextIndex}`, label: `Column ${nextIndex}` },\n ];\n\n return applyColumns(block, nextColumns);\n}\n\nexport function removeColumn(block: TableBlock, index: number): TableBlock {\n const columns = getColumns(block);\n\n if (!columns[index]) {\n return block;\n }\n\n return applyColumns(\n block,\n columns.filter((_, currentIndex) => currentIndex !== index),\n );\n}\n\nexport function renameColumnKey(\n block: TableBlock,\n index: number,\n nextKey: string,\n): TableBlock {\n const columns = getColumns(block);\n const previous = columns[index];\n\n if (!previous || previous.key === nextKey) {\n return block;\n }\n\n if (columns.some((column, currentIndex) => currentIndex !== index && column.key === nextKey)) {\n return block;\n }\n\n return applyColumns(\n block,\n columns.map((column, currentIndex) =>\n currentIndex === index ? { ...column, key: nextKey } : column,\n ),\n );\n}\n\nexport function setColumnLabel(\n block: TableBlock,\n index: number,\n nextLabel: string,\n): TableBlock {\n const columns = getColumns(block);\n\n return applyColumns(\n block,\n columns.map((column, currentIndex) =>\n currentIndex === index ? { ...column, label: nextLabel } : column,\n ),\n );\n}\n\nexport function setColumnAlign(\n block: TableBlock,\n index: number,\n nextAlign: string,\n): TableBlock {\n const columns = getColumns(block);\n\n return applyColumns(\n block,\n columns.map((column, currentIndex) => {\n if (currentIndex !== index) {\n return column;\n }\n\n const next: TableColumn = { ...column };\n if (nextAlign === \"\") {\n delete next.align;\n } else {\n next.align = nextAlign as Align;\n }\n return next;\n }),\n );\n}\n\nexport function setTableNumberRows(block: TableBlock, value: boolean | undefined): TableBlock {\n const previousOn = block.config?.numberRows === true;\n const nextOn = value === true;\n const withFlag = setBlockConfigField(block, \"numberRows\", value);\n\n if (previousOn === nextOn) {\n return withFlag;\n }\n\n return adjustFirstColumnWidth(withFlag, nextOn ? -NUMBER_COLUMN_RESERVE : NUMBER_COLUMN_RESERVE);\n}\n\nfunction adjustFirstColumnWidth(block: TableBlock, delta: number): TableBlock {\n const columns = getColumns(block);\n\n if (columns.length === 0 || !columns.every((column) => percentWidth(column.width) !== null)) {\n return block;\n }\n\n return applyColumns(\n block,\n columns.map((column, index) => {\n if (index !== 0) {\n return column;\n }\n\n const current = percentWidth(column.width) ?? 0;\n const next = Math.max(NUMBER_COLUMN_RESERVE, current + delta);\n\n return { ...column, width: `${next}%` };\n }),\n );\n}\n\nexport function reorderColumns(\n block: TableBlock,\n sourceIndex: number,\n targetIndex: number,\n): TableBlock {\n return applyColumns(block, moveColumn(getColumns(block), sourceIndex, targetIndex));\n}\n\nexport function addRow(rows: readonly TableRow[], columns: readonly TableColumn[]): TableRow[] {\n const empty: TableRow = Object.fromEntries(columns.map((column) => [column.key, \"\"]));\n\n return [...rows, empty];\n}\n\nexport function removeRow(rows: readonly TableRow[], index: number): TableRow[] {\n return rows.filter((_, currentIndex) => currentIndex !== index);\n}\n\nexport function setCellValue(\n rows: readonly TableRow[],\n index: number,\n key: string,\n nextValue: string,\n): TableRow[] {\n return rows.map((row, currentIndex) =>\n currentIndex === index ? { ...row, [key]: nextValue } : row,\n );\n}\n\nexport function removeColumnKeyFromRows(\n rows: readonly TableRow[],\n key: string,\n): TableRow[] {\n return rows.map((row) => omitKey(row, key));\n}\n\nexport function renameColumnKeyInRows(\n rows: readonly TableRow[],\n previousKey: string,\n nextKey: string,\n): TableRow[] {\n return rows.map((row) => renameKey(row, previousKey, nextKey));\n}\n\nexport function TableBlockEditor({\n block,\n rowData,\n onChangeBlock,\n onChangeRowData,\n}: BlockEditorProps): ReactNode {\n const tableBlock = block as TableBlock;\n const columns = getColumns(tableBlock);\n const rows = getRows(rowData);\n const blockId = typeof tableBlock.id === \"string\" ? tableBlock.id : \"\";\n const canEditRows = blockId !== \"\" && onChangeRowData !== undefined;\n\n function handleRemoveColumn(index: number): void {\n const removed = columns[index];\n\n if (!removed) {\n return;\n }\n\n onChangeBlock(removeColumn(tableBlock, index));\n\n if (canEditRows) {\n onChangeRowData?.(removeColumnKeyFromRows(rows, removed.key));\n }\n }\n\n function handleRenameColumnKey(index: number, nextKey: string): void {\n const previous = columns[index];\n\n if (!previous || previous.key === nextKey) {\n return;\n }\n\n if (columns.some((column, currentIndex) => currentIndex !== index && column.key === nextKey)) {\n return;\n }\n\n onChangeBlock(renameColumnKey(tableBlock, index, nextKey));\n\n if (canEditRows) {\n onChangeRowData?.(renameColumnKeyInRows(rows, previous.key, nextKey));\n }\n }\n\n return (\n <>\n <ColumnsEditor\n block={tableBlock}\n columns={columns}\n onChangeBlock={onChangeBlock}\n onRemoveColumn={handleRemoveColumn}\n onRenameColumnKey={handleRenameColumnKey}\n />\n\n <RowsEditor\n canEditRows={canEditRows}\n rows={rows}\n columns={columns}\n onChangeRowData={onChangeRowData}\n />\n </>\n );\n}\n\ninterface ColumnsEditorProps {\n block: TableBlock;\n columns: TableColumn[];\n onChangeBlock: (block: Block) => void;\n onRemoveColumn: (index: number) => void;\n onRenameColumnKey: (index: number, nextKey: string) => void;\n}\n\nfunction ColumnsEditor({\n block,\n columns,\n onChangeBlock,\n onRemoveColumn,\n onRenameColumnKey,\n}: ColumnsEditorProps) {\n return (\n <InspectorSection title=\"Columns\">\n <SortableList\n count={columns.length}\n onReorder={(source, target) => onChangeBlock(reorderColumns(block, source, target))}\n >\n {columns.map((column, index) => (\n <ColumnRow\n key={columnRowKey(column, index)}\n id={String(index)}\n index={index}\n column={column}\n onChangeKey={(value) => onRenameColumnKey(index, value)}\n onChangeLabel={(value) => onChangeBlock(setColumnLabel(block, index, value))}\n onChangeAlign={(value) => onChangeBlock(setColumnAlign(block, index, value))}\n onRemove={() => onRemoveColumn(index)}\n />\n ))}\n </SortableList>\n <AddButton data-name=\"add-column\" onClick={() => onChangeBlock(addColumn(block))}>\n Add column\n </AddButton>\n </InspectorSection>\n );\n}\n\ninterface ColumnRowProps {\n id: string;\n index: number;\n column: TableColumn;\n onChangeKey: (value: string) => void;\n onChangeLabel: (value: string) => void;\n onChangeAlign: (value: string) => void;\n onRemove: () => void;\n}\n\nfunction ColumnRow({\n id,\n index,\n column,\n onChangeKey,\n onChangeLabel,\n onChangeAlign,\n onRemove,\n}: ColumnRowProps) {\n return (\n <SortableRow\n id={id}\n dragLabel={`Drag to reorder column ${index + 1}`}\n removeLabel={`Remove column ${index + 1}`}\n removeName={`remove-column-${index}`}\n onRemove={onRemove}\n >\n <div className=\"grid grid-cols-2 gap-2\">\n <Field label=\"Key\">\n <Input\n name={`column-key-${index}`}\n type=\"text\"\n value={column.key}\n onChange={(event) => onChangeKey(event.currentTarget.value)}\n />\n </Field>\n <Field label=\"Label\">\n <Input\n name={`column-label-${index}`}\n type=\"text\"\n value={column.label}\n onChange={(event) => onChangeLabel(event.currentTarget.value)}\n />\n </Field>\n <AlignSelect\n name={`column-align-${index}`}\n value={(column.align ?? \"\") as string}\n onChange={onChangeAlign}\n />\n </div>\n </SortableRow>\n );\n}\n\ninterface RowsEditorProps {\n canEditRows: boolean;\n rows: TableRow[];\n columns: TableColumn[];\n onChangeRowData?: (data: unknown) => void;\n}\n\nfunction RowsEditor({ canEditRows, rows, columns, onChangeRowData }: RowsEditorProps) {\n if (!canEditRows) {\n return (\n <InspectorSection title=\"Rows\">\n <p className={hintClass}>Give this block an id to edit runtime row data here.</p>\n </InspectorSection>\n );\n }\n\n return (\n <InspectorSection title=\"Rows\">\n {rows.length === 0 ? (\n <p className={hintClass}>No rows yet. Add one to seed runtime data for this table.</p>\n ) : (\n <SortableList\n count={rows.length}\n onReorder={(source, target) => onChangeRowData?.(moveRow(rows, source, target))}\n >\n {rows.map((row, rowIndex) => (\n <DataRow\n key={dataRowKey(row, columns, rowIndex)}\n id={String(rowIndex)}\n index={rowIndex}\n row={row}\n columns={columns}\n onChangeCell={(key, value) =>\n onChangeRowData?.(setCellValue(rows, rowIndex, key, value))\n }\n onRemove={() => onChangeRowData?.(removeRow(rows, rowIndex))}\n />\n ))}\n </SortableList>\n )}\n <AddButton\n data-name=\"add-row\"\n disabled={columns.length === 0}\n onClick={() => onChangeRowData?.(addRow(rows, columns))}\n >\n Add row\n </AddButton>\n </InspectorSection>\n );\n}\n\ninterface DataRowProps {\n id: string;\n index: number;\n row: TableRow;\n columns: TableColumn[];\n onChangeCell: (key: string, value: string) => void;\n onRemove: () => void;\n}\n\nfunction DataRow({ id, index, row, columns, onChangeCell, onRemove }: DataRowProps) {\n return (\n <SortableRow\n id={id}\n dragLabel={`Drag to reorder row ${index + 1}`}\n removeLabel={`Remove row ${index + 1}`}\n removeName={`remove-row-${index}`}\n onRemove={onRemove}\n >\n <div className=\"grid grid-cols-2 gap-2\">\n {columns.map((column) => (\n <Field key={column.key} label={column.label || column.key}>\n <Input\n name={`row-${index}.${column.key}`}\n type=\"text\"\n value={row[column.key] ?? \"\"}\n onChange={(event) => onChangeCell(column.key, event.currentTarget.value)}\n />\n </Field>\n ))}\n </div>\n </SortableRow>\n );\n}\n\nfunction columnRowKey(column: TableColumn, index: number): string {\n return column.key.length > 0 ? `key:${column.key}` : `index:${index}`;\n}\n\nfunction dataRowKey(row: TableRow, columns: TableColumn[], index: number): string {\n const firstColumn = columns[0];\n const candidate = firstColumn ? row[firstColumn.key] : undefined;\n\n return typeof candidate === \"string\" && candidate.length > 0\n ? `first:${candidate}:${index}`\n : `index:${index}`;\n}\n\nfunction getColumns(block: TableBlock): TableColumn[] {\n const candidate = block.config?.columns;\n\n return Array.isArray(candidate) ? (candidate as TableColumn[]) : [];\n}\n\nfunction getRows(rowData: unknown): TableRow[] {\n if (!Array.isArray(rowData)) {\n return [];\n }\n\n return rowData.filter((entry): entry is TableRow => isRecord(entry)) as TableRow[];\n}\n\nfunction applyColumns(block: TableBlock, columns: TableColumn[]): TableBlock {\n const config = { ...block.config } as Record<string, unknown>;\n\n if (columns.length === 0) {\n delete config.columns;\n } else {\n config.columns = columns;\n }\n\n const nextBlock: TableBlock = {\n ...block,\n config:\n Object.keys(config).length === 0 ? undefined : (config as TableBlock[\"config\"]),\n };\n\n if (nextBlock.config === undefined) {\n delete (nextBlock as { config?: TableBlock[\"config\"] }).config;\n }\n\n return nextBlock;\n}\n","import type { ReactNode } from \"react\";\nimport type { Block } from \"../../types/generated/template\";\nimport { ImageBlockEditor } from \"./editors/ImageBlockEditor\";\nimport { KeyValueBlockEditor } from \"./editors/KeyValueBlockEditor\";\nimport { TableBlockEditor } from \"./editors/TableBlockEditor\";\nimport { SelectField, TextAreaField, TextField, type SelectFieldOption } from \"../controls\";\nimport { setBlockConfigField } from \"../state/configUpdates\";\nimport { InspectorSection } from \"./InspectorShell\";\n\nexport interface BlockContentControlsProps {\n block: Block;\n rowData?: unknown;\n onChangeBlock: (block: Block) => void;\n onChangeRowData?: (data: unknown) => void;\n}\n\ntype HeadingLevelValue = \"1\" | \"2\" | \"3\" | \"4\" | \"5\" | \"6\";\n\nconst headingLevelOptions = [\n { value: \"1\", label: \"Heading 1\" },\n { value: \"2\", label: \"Heading 2\" },\n { value: \"3\", label: \"Heading 3\" },\n { value: \"4\", label: \"Heading 4\" },\n { value: \"5\", label: \"Heading 5\" },\n { value: \"6\", label: \"Heading 6\" },\n] as const satisfies readonly SelectFieldOption<HeadingLevelValue>[];\n\nexport function BlockContentControls(props: BlockContentControlsProps): ReactNode {\n if (props.block.type === \"key-value\") {\n return <KeyValueBlockEditor block={props.block} onChangeBlock={props.onChangeBlock} />;\n }\n\n if (props.block.type === \"table\") {\n return (\n <TableBlockEditor\n block={props.block}\n rowData={props.rowData}\n onChangeBlock={props.onChangeBlock}\n onChangeRowData={props.onChangeRowData}\n />\n );\n }\n\n return <InspectorSection title=\"Content\">{renderContentFields(props)}</InspectorSection>;\n}\n\nfunction renderContentFields({ block, onChangeBlock }: BlockContentControlsProps): ReactNode {\n switch (block.type) {\n case \"text\":\n return (\n <div className=\"grid gap-2\">\n <TextAreaField\n name=\"text\"\n label=\"Text\"\n value={block.text}\n rows={4}\n onChange={(value) => onChangeBlock({ ...block, text: value ?? \"\" })}\n />\n </div>\n );\n case \"html\":\n return (\n <div className=\"grid gap-2\">\n <TextAreaField\n name=\"html\"\n label=\"HTML\"\n value={block.html}\n rows={6}\n onChange={(value) => onChangeBlock({ ...block, html: value ?? \"\" })}\n />\n </div>\n );\n case \"heading\":\n return (\n <div className=\"grid gap-2\">\n <TextField\n name=\"text\"\n label=\"Text\"\n value={block.text}\n onChange={(value) => onChangeBlock({ ...block, text: value ?? \"\" })}\n />\n <SelectField\n name=\"config.level\"\n label=\"Level\"\n value={headingLevelValue(block.config?.level)}\n options={headingLevelOptions}\n optional\n emptyLabel=\"Default\"\n onChange={(value) =>\n onChangeBlock(setBlockConfigField(block, \"level\", parseHeadingLevel(value)))\n }\n />\n </div>\n );\n case \"image\":\n return <ImageBlockEditor block={block} onChangeBlock={onChangeBlock} />;\n case \"key-value\":\n case \"table\":\n return null;\n case \"spacer\":\n case \"divider\":\n return <p className=\"m-0 text-xs text-fg-muted\">No content fields for this block.</p>;\n }\n}\n\nfunction headingLevelValue(value: number | undefined): HeadingLevelValue | undefined {\n if (value === undefined) {\n return undefined;\n }\n\n const stringValue = String(value);\n\n return isHeadingLevelValue(stringValue) ? stringValue : undefined;\n}\n\nfunction parseHeadingLevel(value: HeadingLevelValue | undefined): number | undefined {\n return value === undefined ? undefined : Number(value);\n}\n\nfunction isHeadingLevelValue(value: string): value is HeadingLevelValue {\n return headingLevelOptions.some((option) => option.value === value);\n}\n","import type { Align } from \"../../types/generated/template\";\nimport type { SelectFieldOption } from \"../controls\";\n\nexport const ALIGN_OPTIONS = [\n { value: \"left\", label: \"Left\" },\n { value: \"center\", label: \"Center\" },\n { value: \"right\", label: \"Right\" },\n] as const satisfies readonly SelectFieldOption<Align>[];\n","import type { ReactNode } from \"react\";\nimport type { Block, BlockConfig, DividerStyle, TableStyle } from \"../../types/generated/template\";\nimport {\n NumberField,\n SelectField,\n TextField,\n UnitField,\n type SelectFieldOption,\n} from \"../controls\";\nimport { setTableNumberRows } from \"./editors/TableBlockEditor\";\nimport { setBlockConfigField } from \"../state/configUpdates\";\nimport { ALIGN_OPTIONS } from \"./alignOptions\";\n\nexport interface BlockLayoutControlsProps {\n block: Block;\n onChangeBlock: (block: Block) => void;\n}\n\nconst dividerStyleOptions = [\n { value: \"solid\", label: \"Solid\" },\n { value: \"dashed\", label: \"Dashed\" },\n { value: \"dotted\", label: \"Dotted\" },\n { value: \"double\", label: \"Double\" },\n { value: \"none\", label: \"None\" },\n] as const satisfies readonly SelectFieldOption<DividerStyle>[];\n\nconst tableStyleOptions = [\n { value: \"striped\", label: \"Striped\" },\n { value: \"bordered\", label: \"Bordered\" },\n { value: \"minimal\", label: \"Minimal\" },\n] as const satisfies readonly SelectFieldOption<TableStyle>[];\n\nconst numberRowsOptions = [\n { value: \"show\", label: \"Show\" },\n { value: \"hide\", label: \"Hide\" },\n] as const satisfies readonly SelectFieldOption<NumberRowsValue>[];\n\ntype NumberRowsValue = \"show\" | \"hide\";\n\nexport function BlockLayoutControls({ block, onChangeBlock }: BlockLayoutControlsProps): ReactNode {\n return (\n <div className=\"grid gap-2\">\n <div className=\"grid grid-cols-2 items-start gap-2\">\n <UnitField\n name=\"config.width\"\n label=\"Width\"\n value={block.config?.width ?? undefined}\n placeholder=\"Full width\"\n readOnly\n help=\"Drag the column divider on the canvas to resize.\"\n />\n <SelectField\n name=\"config.align\"\n label=\"Align\"\n value={block.config?.align ?? undefined}\n options={ALIGN_OPTIONS}\n optional\n emptyLabel=\"Default\"\n onChange={(value) => onChangeBlock(setCommonConfigField(block, \"align\", value))}\n />\n </div>\n {renderTypeSpecificControls(block, onChangeBlock)}\n </div>\n );\n}\n\nfunction renderTypeSpecificControls(\n block: Block,\n onChangeBlock: (block: Block) => void,\n): ReactNode {\n switch (block.type) {\n case \"spacer\":\n return (\n <NumberField\n name=\"config.height\"\n label=\"Height\"\n value={block.config?.height}\n min={0}\n step={0.5}\n onChange={(value) => onChangeBlock(setBlockConfigField(block, \"height\", value))}\n />\n );\n case \"image\":\n return (\n <NumberField\n name=\"config.maxHeight\"\n label=\"Max height\"\n value={block.config?.maxHeight}\n min={0}\n step={0.5}\n onChange={(value) => onChangeBlock(setBlockConfigField(block, \"maxHeight\", value))}\n />\n );\n case \"divider\":\n return (\n <>\n <NumberField\n name=\"config.thickness\"\n label=\"Thickness\"\n value={block.config?.thickness}\n min={0}\n step={0.5}\n onChange={(value) => onChangeBlock(setBlockConfigField(block, \"thickness\", value))}\n />\n <TextField\n name=\"config.lineColor\"\n label=\"Line color\"\n value={block.config?.lineColor}\n placeholder=\"#334455\"\n emptyValue=\"undefined\"\n onChange={(value) => onChangeBlock(setBlockConfigField(block, \"lineColor\", value))}\n />\n <SelectField\n name=\"config.style\"\n label=\"Line style\"\n value={block.config?.style}\n options={dividerStyleOptions}\n optional\n emptyLabel=\"Default\"\n onChange={(value) => onChangeBlock(setBlockConfigField(block, \"style\", value))}\n />\n </>\n );\n case \"table\":\n return (\n <div className=\"grid grid-cols-2 gap-2\">\n <SelectField\n name=\"config.style\"\n label=\"Table style\"\n value={block.config?.style}\n options={tableStyleOptions}\n optional\n emptyLabel=\"Default\"\n onChange={(value) => onChangeBlock(setBlockConfigField(block, \"style\", value))}\n />\n <SelectField\n name=\"config.numberRows\"\n label=\"Row numbers\"\n value={numberRowsValue(block.config?.numberRows)}\n options={numberRowsOptions}\n optional\n emptyLabel=\"Default\"\n onChange={(value) =>\n onChangeBlock(setTableNumberRows(block, booleanFromNumberRows(value)))\n }\n />\n </div>\n );\n case \"key-value\":\n case \"heading\":\n case \"text\":\n case \"html\":\n return null;\n }\n}\n\nfunction setCommonConfigField<TKey extends \"width\" | \"align\">(\n block: Block,\n field: TKey,\n value: BlockConfig[TKey] | undefined,\n): Block {\n switch (block.type) {\n case \"text\":\n case \"html\":\n case \"heading\":\n case \"image\":\n case \"key-value\":\n case \"spacer\":\n case \"divider\":\n case \"table\":\n return setBlockConfigField(block, field, value);\n }\n}\n\nfunction numberRowsValue(value: boolean | undefined): NumberRowsValue | undefined {\n if (value === undefined) {\n return undefined;\n }\n\n return value ? \"show\" : \"hide\";\n}\n\nfunction booleanFromNumberRows(value: NumberRowsValue | undefined): boolean | undefined {\n if (value === undefined) {\n return undefined;\n }\n\n return value === \"show\";\n}\n","import type { ReactNode } from \"react\";\nimport type { Block, SpacingConfig, Template } from \"../../types/generated/template\";\nimport { NumberField } from \"../controls\";\nimport { setBlockSpacingField, setTemplatePageMargin } from \"../state/configUpdates\";\n\nconst spacingSides = [\"top\", \"right\", \"bottom\", \"left\"] as const;\n\ntype SpacingSide = (typeof spacingSides)[number];\n\ninterface BlockSpacingControlsProps {\n scope: \"block\";\n block: Block;\n onChangeBlock: (block: Block) => void;\n}\n\ninterface PageMarginControlsProps {\n scope: \"page\";\n template: Template;\n onChangeTemplate: (template: Template) => void;\n}\n\nexport type SpacingControlsProps = BlockSpacingControlsProps | PageMarginControlsProps;\n\nexport function SpacingControls(props: SpacingControlsProps): ReactNode {\n const spacing =\n props.scope === \"block\" ? props.block.config?.spacing : props.template.config?.page?.margins;\n\n return (\n <div className=\"grid min-w-0 grid-cols-2 gap-2\">\n {spacingSides.map((side) => (\n <NumberField\n key={side}\n name={fieldName(props.scope, side)}\n label={sideLabel(side)}\n value={fieldValue(spacing, side)}\n min={0}\n step={0.5}\n onChange={(value) => {\n if (props.scope === \"block\") {\n props.onChangeBlock(setBlockSpacingField(props.block, side, value));\n return;\n }\n\n props.onChangeTemplate(setTemplatePageMargin(props.template, side, value));\n }}\n />\n ))}\n </div>\n );\n}\n\nfunction fieldName(scope: SpacingControlsProps[\"scope\"], side: SpacingSide): string {\n return scope === \"block\" ? `config.spacing.${side}` : `config.page.margins.${side}`;\n}\n\nfunction sideLabel(side: SpacingSide): string {\n return `${side[0].toUpperCase()}${side.slice(1)} (mm)`;\n}\n\nfunction fieldValue(spacing: SpacingConfig | undefined, side: SpacingSide): number | undefined {\n return spacing?.[side] ?? undefined;\n}\n","import type { ReactNode } from \"react\";\nimport type { Block, Template, TypographyConfig } from \"../../types/generated/template\";\nimport type { TemplateSchemaMetadata } from \"../../types/template\";\nimport { ColorField, NumberField, SelectField } from \"../controls\";\nimport { setBlockTypographyField, setTemplateTypographyField } from \"../state/configUpdates\";\nimport { ALIGN_OPTIONS } from \"./alignOptions\";\n\ninterface TypographyControlsBaseProps {\n metadata?: Pick<TemplateSchemaMetadata, \"bundledFonts\">;\n}\n\nexport interface BlockTypographyControlsProps extends TypographyControlsBaseProps {\n target: \"block\";\n block: Block;\n onChangeBlock: (block: Block) => void;\n}\n\nexport interface TemplateTypographyControlsProps extends TypographyControlsBaseProps {\n target: \"template\";\n template: Template;\n onChangeTemplate: (template: Template) => void;\n}\n\nexport type TypographyControlsProps =\n | BlockTypographyControlsProps\n | TemplateTypographyControlsProps;\n\nexport function TypographyControls(props: TypographyControlsProps): ReactNode {\n const typography =\n props.target === \"block\" ? props.block.config?.typography : props.template.config?.typography;\n const namePrefix = props.target === \"block\" ? \"config.typography\" : \"template.config.typography\";\n const fontOptions = familyOptions(props.metadata, typography?.family ?? undefined);\n\n function handleChange<TKey extends keyof TypographyConfig>(\n field: TKey,\n value: TypographyConfig[TKey] | \"\" | null | undefined,\n ): void {\n if (props.target === \"block\") {\n props.onChangeBlock(setBlockTypographyField(props.block, field, value));\n return;\n }\n\n props.onChangeTemplate(setTemplateTypographyField(props.template, field, value));\n }\n\n return (\n <div className=\"grid min-w-0 gap-2\">\n <div className=\"grid grid-cols-2 gap-2\">\n <SelectField\n name={`${namePrefix}.family`}\n label=\"Family\"\n value={typography?.family ?? undefined}\n options={fontOptions}\n optional\n emptyLabel=\"Default\"\n onChange={(value) => handleChange(\"family\", value)}\n />\n <SelectField\n name={`${namePrefix}.align`}\n label=\"Align\"\n value={typography?.align ?? undefined}\n optional\n options={ALIGN_OPTIONS}\n onChange={(value) => handleChange(\"align\", value)}\n />\n </div>\n <div className=\"grid grid-cols-3 gap-2\">\n <NumberField\n name={`${namePrefix}.size`}\n label=\"Size\"\n value={typography?.size ?? undefined}\n min={1}\n step={1}\n placeholder=\"12\"\n onChange={(value) => handleChange(\"size\", value)}\n />\n <NumberField\n name={`${namePrefix}.weight`}\n label=\"Weight\"\n value={typography?.weight ?? undefined}\n min={1}\n step={1}\n placeholder=\"400\"\n onChange={(value) => handleChange(\"weight\", value)}\n />\n <ColorField\n name={`${namePrefix}.color`}\n label=\"Color\"\n value={typography?.color ?? undefined}\n onChange={(value) => handleChange(\"color\", value)}\n />\n </div>\n </div>\n );\n}\n\nfunction familyOptions(\n metadata: Pick<TemplateSchemaMetadata, \"bundledFonts\"> | undefined,\n current: string | undefined,\n): Array<{ value: string; label: string }> {\n const fonts = bundledFontOptions(metadata);\n\n if (current && !fonts.includes(current)) {\n fonts.push(current);\n }\n\n return fonts.map((font) => ({ value: font, label: font }));\n}\n\nfunction bundledFontOptions(\n metadata: Pick<TemplateSchemaMetadata, \"bundledFonts\"> | undefined,\n): string[] {\n const seen = new Set<string>();\n const options: string[] = [];\n\n for (const font of metadata?.bundledFonts ?? []) {\n const normalized = font.trim();\n\n if (normalized === \"\" || seen.has(normalized)) {\n continue;\n }\n\n seen.add(normalized);\n options.push(normalized);\n }\n\n return options;\n}\n","import { useEffect, useRef, type KeyboardEvent } from \"react\";\nimport type { Block } from \"../../types/generated/template\";\nimport type { TemplateData, TemplateSchemaResponse } from \"../../types/template\";\nimport { Button } from \"../primitives/Button\";\nimport { Chip } from \"../primitives/Chip\";\nimport { TextField } from \"../controls\";\nimport { getBlockChrome, getBlockSummary } from \"../blocks/blockChrome\";\nimport type { EditorBlock } from \"../state/editorModel\";\nimport { BlockContentControls } from \"./BlockContentControls\";\nimport { BlockLayoutControls } from \"./BlockLayoutControls\";\nimport { InspectorHeader, InspectorSection, InspectorShell } from \"./InspectorShell\";\nimport { SpacingControls } from \"./SpacingControls\";\nimport { TypographyControls } from \"./TypographyControls\";\n\nexport interface BlockInspectorProps {\n block: EditorBlock | null;\n schema: TemplateSchemaResponse;\n data: TemplateData;\n onChangeBlock: (blockUid: string, block: Block) => void;\n onChangeData?: (data: TemplateData) => void;\n onRemoveBlock: (blockUid: string) => void;\n onClose: () => void;\n className?: string;\n}\n\nconst detailSections = [\"Layout\", \"Typography\", \"Spacing\"] as const;\n\nexport function BlockInspector({\n block,\n schema,\n data,\n onChangeBlock,\n onChangeData,\n onRemoveBlock,\n onClose,\n className,\n}: BlockInspectorProps) {\n const shellRef = useRef<HTMLElement>(null);\n const returnFocusRef = useRef<HTMLElement | null>(null);\n\n // Move focus into the non-modal flyout when it opens and restore it to the\n // triggering element when it closes, so keyboard users are not stranded.\n useEffect(() => {\n if (!block) {\n return;\n }\n\n returnFocusRef.current = document.activeElement as HTMLElement | null;\n shellRef.current?.focus();\n\n return () => {\n returnFocusRef.current?.focus?.();\n };\n // Selecting a different block keeps the panel mounted; only run on open/close.\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n function handleKeyDown(event: KeyboardEvent<HTMLElement>): void {\n if (event.key === \"Escape\") {\n event.stopPropagation();\n onClose();\n }\n }\n\n if (!block) {\n return (\n <InspectorShell ariaLabel=\"Block inspector\" className={className}>\n <InspectorHeader title=\"Inspector\" />\n <p className=\"m-0 text-xs text-fg-muted\">Select a block to inspect it.</p>\n </InspectorShell>\n );\n }\n\n const chrome = getBlockChrome(block.block.type);\n const summary = getBlockSummary(block.block);\n const blockId = typeof block.block.id === \"string\" ? block.block.id : null;\n const contentControls = (\n <BlockContentControls\n block={block.block}\n rowData={blockId ? data[blockId] : undefined}\n onChangeBlock={(nextBlock) => onChangeBlock(block.uid, nextBlock)}\n onChangeRowData={\n blockId && onChangeData\n ? (nextRowData) => onChangeData({ ...data, [blockId]: nextRowData })\n : undefined\n }\n />\n );\n\n return (\n <InspectorShell\n ref={shellRef}\n ariaLabel=\"Block inspector\"\n className={className}\n onKeyDown={handleKeyDown}\n >\n <InspectorHeader\n chip={<Chip>{chrome.chip}</Chip>}\n title={chrome.label}\n subtitle={summary || undefined}\n action={\n <button\n type=\"button\"\n className=\"inline-grid h-[22px] w-[22px] cursor-pointer place-items-center rounded border border-solid border-border bg-surface/90 p-0 text-fg-muted transition-colors hover:border-border-strong hover:text-fg\"\n aria-label=\"Close inspector\"\n onClick={onClose}\n >\n ✕\n </button>\n }\n />\n\n <TextField\n name=\"block.id\"\n label=\"ID\"\n value={blockId ?? \"\"}\n placeholder=\"Optional identifier\"\n onChange={(value) =>\n onChangeBlock(block.uid, { ...block.block, id: value ? value : undefined })\n }\n />\n\n <div className=\"grid gap-2\" aria-label=\"Inspector sections\">\n {contentControls}\n {detailSections.map((section) => (\n <InspectorSection key={section} title={section}>\n {section === \"Layout\" ? (\n <BlockLayoutControls\n block={block.block}\n onChangeBlock={(nextBlock) => onChangeBlock(block.uid, nextBlock)}\n />\n ) : section === \"Spacing\" ? (\n <SpacingControls\n scope=\"block\"\n block={block.block}\n onChangeBlock={(nextBlock) => onChangeBlock(block.uid, nextBlock)}\n />\n ) : (\n <TypographyControls\n target=\"block\"\n block={block.block}\n metadata={schema[\"x-pdfUa\"]}\n onChangeBlock={(nextBlock) => onChangeBlock(block.uid, nextBlock)}\n />\n )}\n </InspectorSection>\n ))}\n </div>\n\n <footer className=\"flex justify-end\">\n <Button variant=\"danger\" onClick={() => onRemoveBlock(block.uid)}>\n Remove block\n </Button>\n </footer>\n </InspectorShell>\n );\n}\n\n","import { useBuilderActions, useBuilderState } from \"../context/BuilderContext\";\nimport { BlockInspector } from \"./BlockInspector\";\n\n// Flyout panel pinned to the right edge of the canvas, shown while a block is selected.\nconst flyoutClass =\n \"absolute inset-y-0 right-0 z-20 w-[min(360px,100%)] border-0 border-l border-solid border-border shadow-pop animate-flyout\";\n\nexport function Inspector() {\n const { schema, selectedBlock, data } = useBuilderState();\n const { changeBlock, changeData, removeBlock, deselect } = useBuilderActions();\n\n if (!schema || !selectedBlock) {\n return null;\n }\n\n return (\n <BlockInspector\n className={flyoutClass}\n block={selectedBlock}\n schema={schema}\n data={data}\n onChangeBlock={changeBlock}\n onChangeData={changeData}\n onRemoveBlock={removeBlock}\n onClose={deselect}\n />\n );\n}\n","import type { ReactNode } from \"react\";\nimport type { Orientation, PageFormat, Template } from \"../../types/generated/template\";\nimport type { TemplateSchemaMetadata } from \"../../types/template\";\nimport { SelectField } from \"../controls\";\nimport { PAGE_SIZES_MM } from \"../lib/pageSizes\";\nimport { InspectorSection } from \"./InspectorShell\";\nimport { SpacingControls } from \"./SpacingControls\";\nimport { TypographyControls } from \"./TypographyControls\";\n\nconst pageFormatOptions = Object.keys(PAGE_SIZES_MM).map((format) => ({\n value: format as PageFormat,\n label: format,\n}));\n\nconst orientationOptions = [\n { value: \"portrait\", label: \"Portrait\" },\n { value: \"landscape\", label: \"Landscape\" },\n] as const satisfies ReadonlyArray<{ value: Orientation; label: string }>;\n\nexport interface DocumentSettingsProps {\n template: Template;\n metadata?: Pick<TemplateSchemaMetadata, \"bundledFonts\">;\n format: PageFormat;\n orientation: Orientation;\n onChangeTemplate: (template: Template) => void;\n onChangeFormat: (format: PageFormat) => void;\n onChangeOrientation: (orientation: Orientation) => void;\n className?: string;\n}\n\n/**\n * Document-wide settings, laid out as a horizontal bar above the block palette.\n * Each group wraps to its own column on narrow widths.\n */\nexport function DocumentSettings({\n template,\n metadata,\n format,\n orientation,\n onChangeTemplate,\n onChangeFormat,\n onChangeOrientation,\n className,\n}: DocumentSettingsProps): ReactNode {\n return (\n <section\n aria-label=\"Page settings\"\n className={`grid grid-cols-[repeat(auto-fit,minmax(220px,1fr))] items-start gap-3 border-0 border-b border-solid border-border bg-surface px-4 py-3${className ? ` ${className}` : \"\"}`}\n >\n <InspectorSection title=\"Page setup\">\n <div className=\"grid gap-2\">\n <SelectField\n name=\"document.page.size.format\"\n label=\"Page size\"\n value={format}\n options={pageFormatOptions}\n onChange={(value) => {\n if (value) {\n onChangeFormat(value);\n }\n }}\n />\n <SelectField\n name=\"document.page.size.orientation\"\n label=\"Orientation\"\n value={orientation}\n options={orientationOptions}\n onChange={(value) => {\n if (value) {\n onChangeOrientation(value);\n }\n }}\n />\n </div>\n </InspectorSection>\n\n <InspectorSection title=\"Page margins\">\n <SpacingControls scope=\"page\" template={template} onChangeTemplate={onChangeTemplate} />\n </InspectorSection>\n\n <InspectorSection title=\"Template typography\">\n <TypographyControls\n target=\"template\"\n template={template}\n metadata={metadata}\n onChangeTemplate={onChangeTemplate}\n />\n </InspectorSection>\n </section>\n );\n}\n","import { useBuilderActions, useBuilderState } from \"../context/BuilderContext\";\nimport { DocumentSettings } from \"./DocumentSettings\";\n\nexport interface PageSettingsProps {\n className?: string;\n}\n\nexport function PageSettings({ className }: PageSettingsProps = {}) {\n const { schema, serializedTemplate, pageSize } = useBuilderState();\n const { changeTemplateSettings, changeFormat, changeOrientation } = useBuilderActions();\n\n if (!schema) {\n return null;\n }\n\n return (\n <DocumentSettings\n className={className}\n template={serializedTemplate}\n metadata={schema[\"x-pdfUa\"]}\n format={pageSize.format}\n orientation={pageSize.orientation}\n onChangeTemplate={changeTemplateSettings}\n onChangeFormat={changeFormat}\n onChangeOrientation={changeOrientation}\n />\n );\n}\n","import { Palette } from \"./blocks/Palette\";\nimport { Canvas } from \"./canvas/Canvas\";\nimport type { TemplateExample } from \"./context/BuilderContext\";\nimport { Inspector } from \"./inspector/Inspector\";\nimport { PageSettings } from \"./inspector/PageSettings\";\n\nexport interface BuilderProps {\n className?: string;\n /** Loadable examples surfaced in the palette, keyed by display name. */\n examples?: Record<string, TemplateExample>;\n}\n\nexport function Builder({ className, examples }: BuilderProps = {}) {\n return (\n <section\n className={`pdfua-template-builder grid h-full min-w-0 min-h-0 bg-app${className ? ` ${className}` : \"\"}`}\n style={{ gridTemplateRows: \"auto auto 1fr\" }}\n aria-label=\"Template authoring\"\n >\n <PageSettings />\n <Palette examples={examples} />\n\n <div className=\"relative mt-2 min-h-0 min-w-0\">\n <Canvas className=\"h-full\" />\n <Inspector />\n </div>\n </section>\n );\n}\n","import { useState, type ReactNode } from \"react\";\nimport type { Template } from \"../types/generated/template\";\nimport type { TemplateData } from \"../types/template\";\nimport { Button } from \"../builder/primitives/Button\";\n\nexport type OutputTab = \"pdf\" | \"data\";\n\ntype OutputStatus = \"ready\" | \"rendering\" | \"empty\";\n\nexport interface PdfPaneProps {\n pdfUrl: string | null;\n error: string | null;\n loading: boolean;\n template?: Template;\n data?: TemplateData;\n className?: string;\n onRender?: () => void;\n renderDisabled?: boolean;\n}\n\nfunction Tab({\n active,\n onClick,\n children,\n}: {\n active: boolean;\n onClick: () => void;\n children: ReactNode;\n}) {\n return (\n <button\n type=\"button\"\n role=\"tab\"\n className={`h-full m-0 cursor-pointer border-0 border-b-2 border-solid bg-transparent px-3 text-sm font-medium tracking-[0.02em] transition-colors hover:text-fg ${active ? \"border-fg text-fg\" : \"border-transparent text-fg-muted\"}`}\n aria-selected={active}\n onClick={onClick}\n >\n {children}\n </button>\n );\n}\n\nconst statusPillVariantClass: Record<OutputStatus, string> = {\n ready: \"bg-success-soft text-success\",\n rendering: \"bg-accent-soft text-accent\",\n empty: \"bg-surface-muted text-fg-muted\",\n};\n\nfunction StatusPill({ status, children }: { status: OutputStatus; children: ReactNode }) {\n return (\n <span\n className={`inline-flex items-center gap-2 rounded-full px-3 py-1 text-2xs font-medium ${statusPillVariantClass[status]}`}\n >\n <span className=\"h-1.5 w-1.5 rounded-full bg-current opacity-60\" aria-hidden=\"true\" />\n {children}\n </span>\n );\n}\n\nexport function PdfPane({\n pdfUrl,\n error,\n loading,\n template,\n data,\n className,\n onRender,\n renderDisabled,\n}: PdfPaneProps) {\n const [tab, setTab] = useState<OutputTab>(\"pdf\");\n const status: OutputStatus = loading\n ? \"rendering\"\n : pdfUrl\n ? \"ready\"\n : \"empty\";\n const statusLabel = status === \"rendering\" ? \"Rendering…\" : status === \"ready\" ? \"Ready\" : \"Idle\";\n\n return (\n <aside\n className={`pdfua-template-builder grid min-w-0 min-h-0 grid-rows-[56px_auto_minmax(0,1fr)] bg-canvas${className ? ` ${className}` : \"\"}`}\n aria-label=\"Output\"\n >\n <header className=\"row-start-1 flex items-center justify-between gap-3 border-0 border-b border-solid border-border bg-surface px-4\">\n <div className=\"inline-flex h-full items-stretch gap-0.5\" role=\"tablist\" aria-label=\"Output view\">\n <Tab active={tab === \"pdf\"} onClick={() => setTab(\"pdf\")}>\n PDF\n </Tab>\n <Tab active={tab === \"data\"} onClick={() => setTab(\"data\")}>\n Data\n </Tab>\n {tab === \"data\" ? <CopyJsonButton template={template} data={data} /> : null}\n </div>\n <div className=\"flex items-center gap-3\">\n <StatusPill status={status}>{statusLabel}</StatusPill>\n {onRender ? (\n <Button variant=\"primary\" onClick={onRender} disabled={renderDisabled}>\n {loading ? \"Rendering…\" : \"Render PDF\"}\n </Button>\n ) : null}\n </div>\n </header>\n\n {error ? (\n <p\n className=\"row-start-2 mx-4 mt-3 mb-0 rounded border-0 border-l-[3px] border-solid border-danger bg-danger-soft px-3 py-3 text-sm text-danger\"\n role=\"alert\"\n >\n {error}\n </p>\n ) : null}\n\n <div className=\"row-start-3 grid min-h-0 p-4\">\n {tab === \"pdf\" ? (\n <PdfView pdfUrl={pdfUrl} loading={loading} />\n ) : (\n <DataView template={template} data={data} />\n )}\n </div>\n </aside>\n );\n}\n\nfunction PdfView({ pdfUrl, loading }: { pdfUrl: string | null; loading: boolean }) {\n if (pdfUrl) {\n return (\n <object\n data={pdfUrl}\n type=\"application/pdf\"\n // The rendered PDF is white paper, so its frame stays light in dark mode.\n data-theme=\"light\"\n className=\"h-full w-full min-h-0 rounded-lg border border-solid border-border bg-page shadow-page max-[1080px]:h-[34rem]\"\n />\n );\n }\n return (\n <div\n data-theme=\"light\"\n className=\"grid h-full place-items-center rounded-lg border border-dashed border-border-strong bg-page p-6 text-center text-sm text-fg-muted max-[1080px]:h-[34rem]\"\n >\n {loading\n ? \"Rendering the latest template…\"\n : \"Render the template to preview the PDF here.\"}\n </div>\n );\n}\n\nfunction DataView({ template, data }: { template?: Template; data?: TemplateData }) {\n const payload = { template: template ?? null, data: data ?? {} };\n\n return (\n <pre\n className=\"h-full w-full min-h-0 rounded-lg border border-solid border-border bg-surface shadow-page m-0 overflow-auto p-4 font-mono text-sm leading-normal text-fg [tab-size:2] whitespace-pre max-[1080px]:h-[34rem]\"\n >\n <code>{JSON.stringify(payload, null, 2)}</code>\n </pre>\n );\n}\n\nfunction CopyJsonButton({ template, data }: { template?: Template; data?: TemplateData }) {\n const [copied, setCopied] = useState(false);\n\n function handleCopy() {\n const payload = JSON.stringify({ template: template ?? null, data: data ?? {} }, null, 2);\n\n void navigator.clipboard.writeText(payload).then(() => {\n setCopied(true);\n setTimeout(() => setCopied(false), 1500);\n });\n }\n\n return (\n <button\n type=\"button\"\n className=\"m-0 ml-2 h-[26px] cursor-pointer self-center rounded-md border border-solid border-border bg-surface px-3 text-2xs font-medium text-fg-muted transition-colors hover:border-border-strong hover:text-fg\"\n onClick={handleCopy}\n >\n {copied ? \"✓ Copied\" : \"⧉ Copy JSON\"}\n </button>\n );\n}\n","import { PdfPane } from \"./PdfPane\";\nimport { useRenderContext } from \"./RenderContext\";\n\nexport interface PreviewProps {\n className?: string;\n}\n\nexport function Preview({ className }: PreviewProps = {}) {\n const { template, data, pdfUrl, pdfLoading, error, renderPdf, renderDisabled } =\n useRenderContext();\n\n return (\n <PdfPane\n className={className}\n pdfUrl={pdfUrl}\n error={error}\n loading={pdfLoading}\n template={template ?? undefined}\n data={data}\n onRender={renderPdf}\n renderDisabled={renderDisabled}\n />\n );\n}\n","import type { Template } from \"../types/generated/template\";\nimport type { TemplateData } from \"../types/template\";\nimport { Builder } from \"./Builder\";\nimport { TemplateBuilderProvider, type TemplateExample } from \"./context/BuilderContext\";\nimport { Preview } from \"../render/Preview\";\n\nexport interface TemplateBuilderProps {\n /** Base URL of a running pdf-ua-api instance. Defaults to \"\" (relative URLs / proxy). */\n apiUrl?: string;\n /** Template loaded into the editor on first render. */\n initialTemplate?: Template;\n /** Runtime data keyed by block id (table rows, dynamic key-value overrides). */\n initialData?: TemplateData;\n /** Loadable examples surfaced in the palette, keyed by display name. */\n examples?: Record<string, TemplateExample>;\n /** Fires whenever the user edits the template or its runtime data. */\n onChange?: (template: Template, data: TemplateData) => void;\n /** Fires after a successful render with the produced PDF blob. */\n onRendered?: (pdf: Blob) => void;\n /** Optional className appended to the root element. */\n className?: string;\n}\n\nfunction DefaultLayout({\n examples,\n className,\n}: {\n examples?: Record<string, TemplateExample>;\n className?: string;\n}) {\n return (\n <main\n className={`pdfua-template-builder grid h-screen overflow-hidden bg-app text-fg grid-cols-[minmax(40rem,1.55fr)_minmax(28rem,0.95fr)] max-[1080px]:h-auto max-[1080px]:grid-cols-1 max-[1080px]:overflow-visible${className ? ` ${className}` : \"\"}`}\n >\n <Builder examples={examples} className=\"border-0 border-r border-solid border-border\" />\n <Preview />\n </main>\n );\n}\n\n/**\n * All-in-one preset: a provider wrapping a Builder and a Preview side by side.\n *\n * For custom layouts (e.g. preview below the builder), compose the parts\n * directly with `TemplateBuilderProvider`, `Builder`, and `Preview`.\n */\nexport function TemplateBuilder({\n apiUrl,\n initialTemplate,\n initialData,\n examples,\n onChange,\n onRendered,\n className,\n}: TemplateBuilderProps = {}) {\n return (\n <TemplateBuilderProvider\n apiUrl={apiUrl}\n initialTemplate={initialTemplate}\n initialData={initialData}\n onChange={onChange}\n onRendered={onRendered}\n >\n <DefaultLayout examples={examples} className={className} />\n </TemplateBuilderProvider>\n );\n}\n","import type { Template } from \"../../types/generated/template\";\n\nexport type TableRow = Record<string, string>;\nexport type InvoiceData = Record<string, TableRow[]>;\n\nconst LOGO_SRC =\n \"data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHdpZHRoPSIyNjAiIGhlaWdodD0iNzIiIHZpZXdCb3g9IjAgMCAyNjAgNzIiPjx0ZXh0IHg9IjAiIHk9IjQyIiBmaWxsPSIjMTExODI3IiBmb250LWZhbWlseT0iQXJpYWwsIHNhbnMtc2VyaWYiIGZvbnQtc2l6ZT0iMzAiIGZvbnQtd2VpZ2h0PSI3MDAiPlBERiBVQSBLaXQ8L3RleHQ+PC9zdmc+\";\n\nexport interface InvoiceExample {\n template: Template;\n data: InvoiceData;\n}\n\nexport function createInvoiceExample(): InvoiceExample {\n const template: Template = {\n version: 1,\n config: {\n page: {\n size: { format: \"A4\", orientation: \"portrait\" },\n locale: \"de_DE\",\n margins: { top: 20, right: 20, bottom: 20, left: 25 },\n pageNumbers: { enabled: true, position: \"center\" },\n footer: {\n repeat: true,\n rows: [\n {\n blocks: [\n {\n type: \"text\",\n id: \"footer-legal\",\n text: \"PDF UA Kit GmbH · Musterstraße 1 · 10115 Berlin · Germany · Invoice was created electronically and is valid without signature.\",\n config: { width: \"68%\" },\n },\n {\n type: \"key-value\",\n id: \"footer-meta\",\n values: {\n registration: \"HRB 123456 B\",\n taxNumber: \"DE123456789\",\n },\n config: {\n width: \"32%\",\n align: \"right\",\n fields: [\n { key: \"registration\", label: \"Registry\" },\n { key: \"taxNumber\", label: \"Tax no.\" },\n ],\n },\n },\n ],\n },\n ],\n },\n },\n typography: { family: \"Inter\", size: 10 },\n },\n rows: [\n {\n blocks: [\n {\n type: \"image\",\n id: \"logo\",\n src: LOGO_SRC,\n alt: \"PDF UA Kit GmbH logo\",\n config: { width: \"58%\", maxHeight: 28 },\n },\n {\n type: \"key-value\",\n id: \"invoice-meta\",\n values: {\n invoiceNumber: \"RE-2026-001234\",\n issueDate: \"2026-02-17\",\n dueDate: \"2026-03-19\",\n currency: \"EUR\",\n },\n config: {\n width: \"42%\",\n align: \"right\",\n fields: [\n { key: \"invoiceNumber\", label: \"Invoice number\" },\n { key: \"issueDate\", label: \"Issue date\" },\n { key: \"dueDate\", label: \"Due date\" },\n { key: \"currency\", label: \"Currency\" },\n ],\n },\n },\n ],\n },\n {\n blocks: [\n {\n type: \"heading\",\n id: \"title\",\n text: \"Invoice\",\n config: { level: 1 },\n },\n ],\n },\n {\n blocks: [\n {\n type: \"key-value\",\n id: \"seller\",\n values: {\n name: \"PDF UA Kit GmbH\",\n address: \"Musterstraße 1, 10115 Berlin, DE\",\n contact: \"Max Mustermann\",\n email: \"billing@pdfua-kit.example\",\n vatId: \"DE123456789\",\n },\n config: {\n width: \"50%\",\n fields: [\n { key: \"name\", label: \"Seller\" },\n { key: \"address\", label: \"Address\" },\n { key: \"contact\", label: \"Contact\" },\n { key: \"email\", label: \"Email\" },\n { key: \"vatId\", label: \"VAT ID\" },\n ],\n },\n },\n {\n type: \"key-value\",\n id: \"buyer\",\n values: {\n name: \"Musterkunde AG\",\n address: \"Käuferweg 2, 80331 München, DE\",\n email: \"invoice@musterkunde.example\",\n reference: \"04011000-12345-67\",\n },\n config: {\n width: \"50%\",\n fields: [\n { key: \"name\", label: \"Buyer\" },\n { key: \"address\", label: \"Address\" },\n { key: \"email\", label: \"Email\" },\n { key: \"reference\", label: \"Buyer reference\" },\n ],\n },\n },\n ],\n },\n {\n blocks: [{ type: \"divider\", id: \"address-rule\" }],\n },\n {\n blocks: [\n {\n type: \"table\",\n id: \"lineItems\",\n config: {\n style: \"striped\",\n numberRows: true,\n columns: [\n { key: \"description\", label: \"Description\", align: \"left\", width: \"38%\" },\n { key: \"quantity\", label: \"Qty\", align: \"right\", width: \"12%\" },\n { key: \"unitPrice\", label: \"Unit price\", align: \"right\", width: \"16%\" },\n { key: \"vatRate\", label: \"VAT\", align: \"right\", width: \"11%\" },\n { key: \"total\", label: \"Total\", align: \"right\", width: \"16%\" },\n ],\n },\n },\n ],\n },\n {\n blocks: [\n {\n type: \"table\",\n id: \"vat-breakdown\",\n config: {\n style: \"minimal\",\n width: \"54%\",\n numberRows: false,\n columns: [\n { key: \"vatCategory\", label: \"VAT category\", align: \"left\" },\n { key: \"rate\", label: \"Rate\", align: \"right\" },\n { key: \"taxableAmount\", label: \"Taxable amount\", align: \"right\" },\n { key: \"vatAmount\", label: \"VAT amount\", align: \"right\" },\n ],\n },\n },\n {\n type: \"key-value\",\n id: \"totals\",\n values: {\n netAmount: \"6.120,00 €\",\n vatAmount: \"1.162,80 €\",\n grandTotal: \"7.282,80 €\",\n amountDue: \"7.282,80 €\",\n },\n config: {\n width: \"46%\",\n align: \"right\",\n fields: [\n { key: \"netAmount\", label: \"Net amount\" },\n { key: \"vatAmount\", label: \"VAT 19%\" },\n { key: \"grandTotal\", label: \"Grand total\" },\n { key: \"amountDue\", label: \"Amount due\" },\n ],\n },\n },\n ],\n },\n {\n blocks: [\n {\n type: \"text\",\n id: \"notice\",\n text: \"Please transfer the amount due within 30 days. Include the invoice number as the payment reference.\",\n config: { width: \"54%\", spacing: { top: 4 } },\n },\n {\n type: \"key-value\",\n id: \"payment\",\n values: {\n bank: \"Musterbank Berlin\",\n iban: \"DE89370400440532013000\",\n bic: \"COBADEFFXXX\",\n reference: \"RE-2026-001234\",\n },\n config: {\n width: \"46%\",\n align: \"right\",\n fields: [\n { key: \"bank\", label: \"Bank\" },\n { key: \"iban\", label: \"IBAN\" },\n { key: \"bic\", label: \"BIC\" },\n { key: \"reference\", label: \"Payment reference\" },\n ],\n },\n },\n ],\n },\n ],\n };\n\n const data: InvoiceData = {\n lineItems: [\n {\n description: \"Accessible PDF template implementation\",\n quantity: \"40\",\n unitPrice: \"95,00 €\",\n vatRate: \"19%\",\n total: \"3.800,00 €\",\n },\n {\n description: \"Document structure and tagging review\",\n quantity: \"16\",\n unitPrice: \"85,00 €\",\n vatRate: \"19%\",\n total: \"1.360,00 €\",\n },\n {\n description: \"Project management and acceptance testing\",\n quantity: \"8\",\n unitPrice: \"90,00 €\",\n vatRate: \"19%\",\n total: \"720,00 €\",\n },\n {\n description: \"Annual hosting and maintenance package\",\n quantity: \"1\",\n unitPrice: \"240,00 €\",\n vatRate: \"19%\",\n total: \"240,00 €\",\n },\n ],\n \"vat-breakdown\": [\n {\n vatCategory: \"Standard rate\",\n rate: \"19%\",\n taxableAmount: \"6.120,00 €\",\n vatAmount: \"1.162,80 €\",\n },\n ],\n };\n\n return { template, data };\n}\n","import type { Template } from \"../types/generated/template\";\n\nexport interface ParsedTemplate {\n template: Template | null;\n error: string | null;\n}\n\nexport function parseTemplate(text: string): ParsedTemplate {\n if (text.trim() === \"\") {\n return { template: null, error: null };\n }\n\n let parsed: unknown;\n try {\n parsed = JSON.parse(text);\n } catch (cause) {\n return { template: null, error: cause instanceof Error ? cause.message : String(cause) };\n }\n\n if (typeof parsed !== \"object\" || parsed === null || Array.isArray(parsed)) {\n return { template: null, error: \"Template must be a JSON object.\" };\n }\n\n return { template: parsed as Template, error: null };\n}\n","import {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useRef,\n useState,\n type ReactNode,\n} from \"react\";\nimport { resolveDefaultApiUrl } from \"../api/pdfUaApi\";\nimport type { Template } from \"../types/generated/template\";\nimport type { TemplateData } from \"../types/template\";\nimport { RenderProvider, type RenderContextValue } from \"../render/RenderContext\";\nimport { usePdfUaApi } from \"../render/usePdfUaApi\";\nimport { parseTemplate } from \"./parseTemplate\";\n\nexport interface TemplateEditorProviderProps {\n apiUrl?: string;\n initialTemplate?: Template;\n data?: TemplateData;\n onChange?: (template: Template | null, text: string) => void;\n onRendered?: (pdf: Blob) => void;\n children: ReactNode;\n}\n\nexport interface TemplateEditorContextValue {\n text: string;\n setText: (text: string) => void;\n template: Template | null;\n error: string | null;\n data: TemplateData;\n}\n\nconst emptyTemplate: Template = { version: 1 };\nconst emptyData: TemplateData = {};\n\nconst TemplateEditorContext = createContext<TemplateEditorContextValue | null>(null);\n\nexport function useTemplateEditor(): TemplateEditorContextValue {\n const value = useContext(TemplateEditorContext);\n\n if (!value) {\n throw new Error(\"useTemplateEditor must be used within a <TemplateEditorProvider>.\");\n }\n\n return value;\n}\n\nexport function TemplateEditorProvider({\n apiUrl: apiUrlProp,\n initialTemplate,\n data = emptyData,\n onChange,\n onRendered,\n children,\n}: TemplateEditorProviderProps) {\n const apiUrl = resolveDefaultApiUrl(apiUrlProp);\n const [text, setText] = useState(() =>\n JSON.stringify(initialTemplate ?? emptyTemplate, null, 2),\n );\n\n const { template, error } = useMemo(() => parseTemplate(text), [text]);\n\n const onChangeRef = useRef(onChange);\n useEffect(() => {\n onChangeRef.current = onChange;\n }, [onChange]);\n const skipNextChange = useRef(true);\n useEffect(() => {\n if (skipNextChange.current) {\n skipNextChange.current = false;\n return;\n }\n onChangeRef.current?.(template, text);\n }, [text, template]);\n\n const {\n pdfUrl,\n pdfLoading,\n error: renderError,\n renderPdf: renderPdfRequest,\n } = usePdfUaApi({ initialApiUrl: apiUrl, apiUrl, onRendered, loadSchemaOnMount: false });\n\n const templateRef = useRef(template);\n templateRef.current = template;\n const dataRef = useRef(data);\n dataRef.current = data;\n\n const renderPdf = useCallback(() => {\n if (templateRef.current) {\n void renderPdfRequest(templateRef.current, dataRef.current);\n }\n }, [renderPdfRequest]);\n\n const editorValue = useMemo<TemplateEditorContextValue>(\n () => ({ text, setText, template, error, data }),\n [text, template, error, data],\n );\n\n const renderValue = useMemo<RenderContextValue>(\n () => ({\n template,\n data,\n pdfUrl,\n pdfLoading,\n error: renderError,\n renderPdf,\n renderDisabled: template === null || pdfLoading,\n }),\n [template, data, pdfUrl, pdfLoading, renderError, renderPdf],\n );\n\n return (\n <TemplateEditorContext.Provider value={editorValue}>\n <RenderProvider value={renderValue}>{children}</RenderProvider>\n </TemplateEditorContext.Provider>\n );\n}\n","import { HighlightStyle, syntaxHighlighting } from \"@codemirror/language\";\nimport { EditorView } from \"@codemirror/view\";\nimport { tags as t } from \"@lezer/highlight\";\n\nconst baseTheme = EditorView.theme({\n \"&\": {\n color: \"var(--pdfua-fg)\",\n backgroundColor: \"var(--pdfua-surface)\",\n fontSize: \"13px\",\n height: \"100%\",\n },\n \"&.cm-focused\": { outline: \"none\" },\n \".cm-content\": {\n fontFamily: \"ui-monospace, SFMono-Regular, Menlo, Consolas, monospace\",\n caretColor: \"var(--pdfua-fg)\",\n },\n \".cm-gutters\": {\n backgroundColor: \"var(--pdfua-surface)\",\n color: \"var(--pdfua-fg-subtle)\",\n border: \"none\",\n },\n \".cm-activeLine\": {\n backgroundColor: \"color-mix(in oklab, var(--pdfua-accent) 8%, transparent)\",\n },\n \".cm-activeLineGutter\": { backgroundColor: \"transparent\" },\n \".cm-cursor, .cm-dropCursor\": { borderLeftColor: \"var(--pdfua-fg)\" },\n \"&.cm-focused .cm-selectionBackground, .cm-selectionBackground, .cm-content ::selection\": {\n backgroundColor: \"color-mix(in oklab, var(--pdfua-accent) 22%, transparent)\",\n },\n});\n\nconst highlightStyle = HighlightStyle.define([\n { tag: t.propertyName, color: \"var(--pdfua-accent)\" },\n { tag: t.string, color: \"var(--pdfua-fg)\" },\n { tag: [t.number, t.bool, t.null], color: \"var(--pdfua-accent)\" },\n { tag: [t.punctuation, t.separator], color: \"var(--pdfua-fg-muted)\" },\n]);\n\n// Editor styling driven by the --pdfua-* tokens, so it follows light/dark automatically.\nexport const editorTheme = [baseTheme, syntaxHighlighting(highlightStyle)];\n","{\n \"$schema\": \"https://json-schema.org/draft/2020-12/schema\",\n \"$id\": \"https://pdf-ua-api.com/schemas/template-v1.json\",\n \"title\": \"Template\",\n \"type\": \"object\",\n \"properties\": {\n \"version\": {\n \"type\": \"integer\",\n \"const\": 1\n },\n \"config\": {\n \"$ref\": \"#/$defs/templateConfig\"\n },\n \"fonts\": {\n \"type\": \"object\",\n \"description\": \"External fonts keyed by font family name.\",\n \"additionalProperties\": {\n \"$ref\": \"#/$defs/fontFace\"\n }\n },\n \"attachments\": {\n \"type\": \"array\",\n \"items\": {\n \"$ref\": \"#/$defs/fileAttachment\"\n }\n },\n \"rows\": {\n \"type\": \"array\",\n \"items\": {\n \"$ref\": \"#/$defs/row\"\n }\n }\n },\n \"additionalProperties\": false,\n \"required\": [\n \"version\"\n ],\n \"$defs\": {\n \"align\": {\n \"type\": \"string\",\n \"title\": \"Align\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\"\n ]\n },\n \"pageFormat\": {\n \"type\": \"string\",\n \"title\": \"PageFormat\",\n \"enum\": [\n \"A3\",\n \"A4\",\n \"A5\",\n \"A6\",\n \"Letter\",\n \"Legal\",\n \"Tabloid\"\n ]\n },\n \"orientation\": {\n \"type\": \"string\",\n \"title\": \"Orientation\",\n \"enum\": [\n \"portrait\",\n \"landscape\"\n ]\n },\n \"dividerStyle\": {\n \"type\": \"string\",\n \"title\": \"DividerStyle\",\n \"enum\": [\n \"solid\",\n \"dashed\",\n \"dotted\",\n \"double\",\n \"none\"\n ]\n },\n \"tableStyle\": {\n \"type\": \"string\",\n \"title\": \"TableStyle\",\n \"enum\": [\n \"striped\",\n \"bordered\",\n \"minimal\"\n ]\n },\n \"pageBackgroundType\": {\n \"type\": \"string\",\n \"title\": \"PageBackgroundType\",\n \"enum\": [\n \"auto\",\n \"image\",\n \"pdf\"\n ]\n },\n \"typographyConfig\": {\n \"type\": \"object\",\n \"title\": \"TypographyConfig\",\n \"properties\": {\n \"family\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Bundled or external font family key.\"\n },\n \"size\": {\n \"type\": [\n \"integer\",\n \"null\"\n ],\n \"description\": \"Font size in points.\",\n \"minimum\": 1.0\n },\n \"weight\": {\n \"type\": [\n \"integer\",\n \"null\"\n ],\n \"description\": \"Numeric font weight.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Text alignment for this typography scope.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"color\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS color value used for text.\"\n }\n },\n \"additionalProperties\": false\n },\n \"spacingConfig\": {\n \"type\": \"object\",\n \"title\": \"SpacingConfig\",\n \"properties\": {\n \"top\": {\n \"type\": [\n \"integer\",\n \"null\"\n ],\n \"description\": \"Top spacing in millimetres.\",\n \"minimum\": 0.0\n },\n \"right\": {\n \"type\": [\n \"integer\",\n \"null\"\n ],\n \"description\": \"Right spacing in millimetres.\",\n \"minimum\": 0.0\n },\n \"bottom\": {\n \"type\": [\n \"integer\",\n \"null\"\n ],\n \"description\": \"Bottom spacing in millimetres.\",\n \"minimum\": 0.0\n },\n \"left\": {\n \"type\": [\n \"integer\",\n \"null\"\n ],\n \"description\": \"Left spacing in millimetres.\",\n \"minimum\": 0.0\n }\n },\n \"additionalProperties\": false\n },\n \"blockConfig\": {\n \"type\": \"object\",\n \"title\": \"BlockConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"{ typography?: TypographyConfig; spacing?: SpacingConfig; width?: string | null; align?: Align | null }\"\n },\n \"headingConfig\": {\n \"type\": \"object\",\n \"title\": \"HeadingConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"level\": {\n \"type\": \"integer\",\n \"default\": 2,\n \"minimum\": 1.0,\n \"maximum\": 6.0\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"BlockConfig & { level?: number }\"\n },\n \"imageConfig\": {\n \"type\": \"object\",\n \"title\": \"ImageConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"maxHeight\": {\n \"type\": \"integer\",\n \"default\": 60,\n \"minimum\": 1.0\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"BlockConfig & { maxHeight?: number }\"\n },\n \"keyValueField\": {\n \"type\": \"object\",\n \"title\": \"KeyValueField\",\n \"properties\": {\n \"key\": {\n \"type\": \"string\",\n \"pattern\": \"^[A-Za-z][A-Za-z0-9_]*$\"\n },\n \"label\": {\n \"type\": \"string\"\n }\n },\n \"required\": [\n \"key\",\n \"label\"\n ],\n \"additionalProperties\": false,\n \"tsType\": \"{ key: string; label: string }\"\n },\n \"keyValueConfig\": {\n \"type\": \"object\",\n \"title\": \"KeyValueConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"labelWidth\": {\n \"type\": \"string\",\n \"default\": \"30mm\"\n },\n \"fields\": {\n \"type\": \"array\",\n \"items\": {\n \"$ref\": \"#/$defs/keyValueField\"\n }\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"BlockConfig & { labelWidth?: string; fields?: KeyValueField[] }\"\n },\n \"spacerConfig\": {\n \"type\": \"object\",\n \"title\": \"SpacerConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"height\": {\n \"type\": \"integer\",\n \"default\": 5,\n \"minimum\": 0.0\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"BlockConfig & { height?: number }\"\n },\n \"dividerConfig\": {\n \"type\": \"object\",\n \"title\": \"DividerConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"thickness\": {\n \"type\": \"integer\",\n \"default\": 1,\n \"minimum\": 0.0\n },\n \"lineColor\": {\n \"type\": \"string\",\n \"default\": \"#d1d5db\",\n \"pattern\": \"^#[0-9A-Fa-f]{3,8}$\"\n },\n \"style\": {\n \"$ref\": \"#/$defs/dividerStyle\"\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"BlockConfig & { thickness?: number; lineColor?: string; style?: DividerStyle }\"\n },\n \"tableColumn\": {\n \"type\": \"object\",\n \"title\": \"TableColumn\",\n \"properties\": {\n \"key\": {\n \"type\": \"string\",\n \"description\": \"Runtime data key used for this table column.\",\n \"pattern\": \"^[A-Za-z][A-Za-z0-9_]*$\"\n },\n \"label\": {\n \"type\": \"string\",\n \"description\": \"Header label rendered for this table column.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Text alignment for this table column.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Column width as a CSS width value, such as 20mm or 15%.\"\n }\n },\n \"required\": [\n \"key\",\n \"label\"\n ],\n \"additionalProperties\": false\n },\n \"tableConfig\": {\n \"type\": \"object\",\n \"title\": \"TableConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"numberRows\": {\n \"type\": \"boolean\",\n \"default\": false\n },\n \"columns\": {\n \"type\": \"array\",\n \"items\": {\n \"$ref\": \"#/$defs/tableColumn\"\n }\n },\n \"style\": {\n \"$ref\": \"#/$defs/tableStyle\"\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"BlockConfig & { numberRows?: boolean; columns?: TableColumn[]; style?: TableStyle }\"\n },\n \"textBlock\": {\n \"type\": \"object\",\n \"title\": \"TextBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"text\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"text\": {\n \"type\": \"string\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/blockConfig\"\n }\n },\n \"required\": [\n \"type\",\n \"text\"\n ],\n \"additionalProperties\": false\n },\n \"htmlBlock\": {\n \"type\": \"object\",\n \"title\": \"HtmlBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"html\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"html\": {\n \"type\": \"string\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/blockConfig\"\n }\n },\n \"required\": [\n \"type\",\n \"html\"\n ],\n \"additionalProperties\": false\n },\n \"headingBlock\": {\n \"type\": \"object\",\n \"title\": \"HeadingBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"heading\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"text\": {\n \"type\": \"string\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/headingConfig\"\n }\n },\n \"required\": [\n \"type\",\n \"text\"\n ],\n \"additionalProperties\": false\n },\n \"imageBlock\": {\n \"type\": \"object\",\n \"title\": \"ImageBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"image\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"src\": {\n \"type\": \"string\",\n \"description\": \"Public image URL, SVG markup, or uploaded image data URL.\"\n },\n \"alt\": {\n \"type\": \"string\",\n \"description\": \"Alternative text for screen readers and PDF accessibility.\",\n \"default\": \"\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/imageConfig\"\n }\n },\n \"required\": [\n \"type\",\n \"src\"\n ],\n \"additionalProperties\": false\n },\n \"keyValueBlock\": {\n \"type\": \"object\",\n \"title\": \"KeyValueBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"key-value\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"values\": {\n \"type\": \"object\",\n \"title\": \"KeyValueValues\",\n \"propertyNames\": {\n \"type\": \"string\",\n \"pattern\": \"^[A-Za-z][A-Za-z0-9_]*$\"\n },\n \"additionalProperties\": {\n \"type\": [\n \"string\",\n \"null\"\n ]\n }\n },\n \"config\": {\n \"$ref\": \"#/$defs/keyValueConfig\"\n }\n },\n \"required\": [\n \"type\"\n ],\n \"additionalProperties\": false\n },\n \"spacerBlock\": {\n \"type\": \"object\",\n \"title\": \"SpacerBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"spacer\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/spacerConfig\"\n }\n },\n \"required\": [\n \"type\"\n ],\n \"additionalProperties\": false\n },\n \"dividerBlock\": {\n \"type\": \"object\",\n \"title\": \"DividerBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"divider\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/dividerConfig\"\n }\n },\n \"required\": [\n \"type\"\n ],\n \"additionalProperties\": false\n },\n \"tableBlock\": {\n \"type\": \"object\",\n \"title\": \"TableBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"table\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/tableConfig\"\n }\n },\n \"required\": [\n \"type\"\n ],\n \"additionalProperties\": false\n },\n \"block\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/textBlock\"\n },\n {\n \"$ref\": \"#/$defs/htmlBlock\"\n },\n {\n \"$ref\": \"#/$defs/headingBlock\"\n },\n {\n \"$ref\": \"#/$defs/imageBlock\"\n },\n {\n \"$ref\": \"#/$defs/keyValueBlock\"\n },\n {\n \"$ref\": \"#/$defs/spacerBlock\"\n },\n {\n \"$ref\": \"#/$defs/dividerBlock\"\n },\n {\n \"$ref\": \"#/$defs/tableBlock\"\n }\n ]\n },\n \"row\": {\n \"type\": \"object\",\n \"title\": \"Row\",\n \"properties\": {\n \"blocks\": {\n \"type\": \"array\",\n \"items\": {\n \"$ref\": \"#/$defs/block\"\n }\n }\n },\n \"required\": [\n \"blocks\"\n ],\n \"additionalProperties\": false\n },\n \"presetPageSize\": {\n \"type\": \"object\",\n \"title\": \"PresetPageSize\",\n \"properties\": {\n \"format\": {\n \"$ref\": \"#/$defs/pageFormat\"\n },\n \"orientation\": {\n \"$ref\": \"#/$defs/orientation\"\n }\n },\n \"additionalProperties\": false\n },\n \"customPageSize\": {\n \"type\": \"object\",\n \"title\": \"CustomPageSize\",\n \"properties\": {\n \"width\": {\n \"type\": \"integer\",\n \"minimum\": 1.0\n },\n \"height\": {\n \"type\": \"integer\",\n \"minimum\": 1.0\n }\n },\n \"required\": [\n \"width\",\n \"height\"\n ],\n \"additionalProperties\": false\n },\n \"pageSize\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/presetPageSize\"\n },\n {\n \"$ref\": \"#/$defs/customPageSize\"\n }\n ],\n \"title\": \"PageSize\"\n },\n \"pageNumbersConfig\": {\n \"type\": \"object\",\n \"title\": \"PageNumbersConfig\",\n \"properties\": {\n \"enabled\": {\n \"type\": \"boolean\",\n \"default\": false\n },\n \"position\": {\n \"$ref\": \"#/$defs/align\"\n }\n },\n \"additionalProperties\": false\n },\n \"pageBackgroundConfig\": {\n \"type\": \"object\",\n \"title\": \"PageBackgroundConfig\",\n \"properties\": {\n \"src\": {\n \"type\": \"string\",\n \"description\": \"HTTP, HTTPS, or base64 data URI for an image or PDF page background.\",\n \"minLength\": 1\n },\n \"type\": {\n \"$ref\": \"#/$defs/pageBackgroundType\"\n }\n },\n \"required\": [\n \"src\"\n ],\n \"additionalProperties\": false\n },\n \"pageFooterConfig\": {\n \"type\": \"object\",\n \"title\": \"PageFooterConfig\",\n \"properties\": {\n \"repeat\": {\n \"type\": \"boolean\",\n \"default\": true\n },\n \"rows\": {\n \"type\": \"array\",\n \"items\": {\n \"$ref\": \"#/$defs/row\"\n }\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"{ repeat?: boolean; rows?: Row[] }\"\n },\n \"pageConfig\": {\n \"type\": \"object\",\n \"title\": \"PageConfig\",\n \"properties\": {\n \"size\": {\n \"$ref\": \"#/$defs/pageSize\"\n },\n \"locale\": {\n \"type\": \"string\",\n \"default\": \"de_DE\"\n },\n \"margins\": {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n \"pageNumbers\": {\n \"$ref\": \"#/$defs/pageNumbersConfig\"\n },\n \"background\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/pageBackgroundConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"footer\": {\n \"$ref\": \"#/$defs/pageFooterConfig\"\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"{ size?: PageSize; locale?: string; margins?: SpacingConfig; pageNumbers?: PageNumbersConfig; background?: PageBackgroundConfig | null; footer?: PageFooterConfig }\"\n },\n \"templateConfig\": {\n \"type\": \"object\",\n \"title\": \"TemplateConfig\",\n \"properties\": {\n \"page\": {\n \"$ref\": \"#/$defs/pageConfig\"\n },\n \"typography\": {\n \"$ref\": \"#/$defs/typographyConfig\"\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"{ page?: PageConfig; typography?: TypographyConfig }\"\n },\n \"fontFace\": {\n \"type\": \"object\",\n \"title\": \"FontFace\",\n \"properties\": {\n \"src\": {\n \"type\": \"string\"\n },\n \"weight\": {\n \"type\": \"integer\",\n \"default\": 400\n },\n \"style\": {\n \"type\": \"string\",\n \"default\": \"normal\"\n }\n },\n \"required\": [\n \"src\"\n ],\n \"additionalProperties\": false\n },\n \"fileAttachment\": {\n \"type\": \"object\",\n \"title\": \"FileAttachment\",\n \"properties\": {\n \"name\": {\n \"type\": \"string\"\n },\n \"content\": {\n \"type\": \"string\",\n \"description\": \"Base64-encoded file content.\"\n },\n \"mimeType\": {\n \"type\": \"string\",\n \"default\": \"application/octet-stream\"\n },\n \"description\": {\n \"type\": [\n \"string\",\n \"null\"\n ]\n },\n \"relationship\": {\n \"type\": \"string\",\n \"default\": \"Alternative\"\n }\n },\n \"required\": [\n \"name\",\n \"content\"\n ],\n \"additionalProperties\": false\n }\n },\n \"x-pdfUa\": {\n \"kind\": \"template\",\n \"templateVersion\": 1,\n \"renderEndpoint\": \"/render/template\",\n \"templateFields\": [\n \"version\",\n \"config\",\n \"fonts\",\n \"attachments\",\n \"rows\"\n ],\n \"attachmentFields\": [\n \"name\",\n \"content\",\n \"mimeType\",\n \"description\",\n \"relationship\"\n ],\n \"externalFontFields\": [\n \"src\",\n \"weight\",\n \"style\"\n ],\n \"bundledFonts\": [\n \"Fira Code\",\n \"Inter\",\n \"JetBrains Mono\",\n \"Lato\",\n \"Liberation Mono\",\n \"Liberation Sans\",\n \"Liberation Serif\",\n \"Merriweather\",\n \"Montserrat\",\n \"Noto Sans\",\n \"Noto Sans Mono\",\n \"Noto Serif\",\n \"Nunito\",\n \"Open Sans\",\n \"Oswald\",\n \"Playfair Display\",\n \"Poppins\",\n \"Raleway\",\n \"Roboto\",\n \"Rubik\"\n ],\n \"blockOrder\": [\n \"text\",\n \"html\",\n \"heading\",\n \"image\",\n \"key-value\",\n \"spacer\",\n \"divider\",\n \"table\"\n ],\n \"pageFormats\": [\n {\n \"name\": \"A3\",\n \"widthMm\": 297.0,\n \"heightMm\": 420.0\n },\n {\n \"name\": \"A4\",\n \"widthMm\": 210.0,\n \"heightMm\": 297.0\n },\n {\n \"name\": \"A5\",\n \"widthMm\": 148.0,\n \"heightMm\": 210.0\n },\n {\n \"name\": \"A6\",\n \"widthMm\": 105.0,\n \"heightMm\": 148.0\n },\n {\n \"name\": \"Letter\",\n \"widthMm\": 215.9,\n \"heightMm\": 279.4\n },\n {\n \"name\": \"Legal\",\n \"widthMm\": 215.9,\n \"heightMm\": 355.6\n },\n {\n \"name\": \"Tabloid\",\n \"widthMm\": 279.4,\n \"heightMm\": 431.8\n }\n ]\n }\n}","import schema from \"../../schemas/template.schema.json\";\n\n// The validation contract shipped with the package, fed to codemirror-json-schema.\nexport const templateSchema = schema as Record<string, unknown>;\n","import { json } from \"@codemirror/lang-json\";\nimport ReactCodeMirror from \"@uiw/react-codemirror\";\nimport { jsonSchema } from \"codemirror-json-schema\";\nimport { useMemo } from \"react\";\nimport { useTemplateEditor } from \"./TemplateEditorContext\";\nimport { editorTheme } from \"./editorTheme\";\nimport { templateSchema } from \"./templateSchema\";\n\nexport interface CodeEditorProps {\n className?: string;\n}\n\nexport function CodeEditor({ className }: CodeEditorProps = {}) {\n const { text, setText } = useTemplateEditor();\n\n const extensions = useMemo(\n () => [json(), jsonSchema(templateSchema as Parameters<typeof jsonSchema>[0]), ...editorTheme],\n [],\n );\n\n return (\n <div\n className={`pdfua-template-builder min-w-0 min-h-0 overflow-auto bg-surface${className ? ` ${className}` : \"\"}`}\n aria-label=\"Template JSON editor\"\n >\n <ReactCodeMirror\n value={text}\n onChange={setText}\n extensions={extensions}\n theme=\"none\"\n height=\"100%\"\n basicSetup={{ foldGutter: true, highlightActiveLine: true, autocompletion: true }}\n />\n </div>\n );\n}\n","import type { Template } from \"../types/generated/template\";\nimport type { TemplateData } from \"../types/template\";\nimport { Preview } from \"../render/Preview\";\nimport { CodeEditor } from \"./CodeEditor\";\nimport { TemplateEditorProvider } from \"./TemplateEditorContext\";\n\nexport interface TemplateEditorProps {\n /** Base URL of a running pdf-ua-api instance. Defaults to \"\" (relative URLs / proxy). */\n apiUrl?: string;\n /** Template seeded into the editor as pretty-printed JSON on first render. */\n initialTemplate?: Template;\n /** Runtime data keyed by block id, passed through to render. */\n data?: TemplateData;\n /** Fires on every edit; `template` is null while the JSON is invalid. */\n onChange?: (template: Template | null, text: string) => void;\n /** Fires after a successful render with the produced PDF blob. */\n onRendered?: (pdf: Blob) => void;\n /** Optional className appended to the root element. */\n className?: string;\n}\n\nfunction DefaultLayout({ className }: { className?: string }) {\n return (\n <main\n className={`pdfua-template-builder grid h-screen overflow-hidden bg-app text-fg grid-cols-[minmax(40rem,1.55fr)_minmax(28rem,0.95fr)] max-[1080px]:h-auto max-[1080px]:grid-cols-1 max-[1080px]:overflow-visible${className ? ` ${className}` : \"\"}`}\n >\n <CodeEditor className=\"border-0 border-r border-solid border-border\" />\n <Preview />\n </main>\n );\n}\n\n/**\n * All-in-one preset: a JSON template editor and a rendered-PDF preview side by side.\n *\n * For custom layouts, compose directly with `TemplateEditorProvider`, `CodeEditor`,\n * and `Preview`.\n */\nexport function TemplateEditor({\n apiUrl,\n initialTemplate,\n data,\n onChange,\n onRendered,\n className,\n}: TemplateEditorProps = {}) {\n return (\n <TemplateEditorProvider\n apiUrl={apiUrl}\n initialTemplate={initialTemplate}\n data={data}\n onChange={onChange}\n onRendered={onRendered}\n >\n <DefaultLayout className={className} />\n </TemplateEditorProvider>\n );\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"index.js","names":[],"sources":["../src/api/pdfUaApi.ts","../src/builder/blocks/blockChrome.ts","../src/builder/primitives/Chip.tsx","../src/builder/canvas/BlockCardPreview.tsx","../src/builder/lib/sensors.ts","../src/builder/lib/records.ts","../src/builder/state/dragDrop.ts","../src/builder/state/editorModel.ts","../src/builder/hooks/useBuilderDragDrop.ts","../src/render/usePdfUaApi.ts","../src/render/RenderContext.tsx","../src/builder/schema/schemaAdapter.ts","../src/builder/state/editorReducer.ts","../src/builder/context/BuilderContext.tsx","../src/builder/controls/inputs.tsx","../src/builder/controls/fieldConverters.ts","../src/builder/controls/BuilderField.tsx","../src/builder/controls/Field.tsx","../src/builder/controls/AlignSelect.tsx","../src/builder/primitives/Button.tsx","../src/builder/blocks/BlockPalette.tsx","../src/builder/blocks/Palette.tsx","../src/builder/lib/displayScale.ts","../src/builder/lib/pageSizes.ts","../src/builder/canvas/columns.ts","../src/builder/canvas/ColumnResizer.tsx","../src/builder/canvas/PageSheet.tsx","../src/builder/state/configUpdates.ts","../src/builder/canvas/BlockDataPreview.tsx","../src/builder/canvas/SortableBlock.tsx","../src/builder/canvas/BuilderCanvas.tsx","../src/builder/canvas/Canvas.tsx","../src/builder/inspector/editors/ImageBlockEditor.tsx","../src/builder/inspector/InspectorShell.tsx","../src/builder/inspector/editors/SortableList.tsx","../src/builder/inspector/editors/SortableRow.tsx","../src/builder/inspector/editors/KeyValueBlockEditor.tsx","../src/builder/inspector/editors/TableBlockEditor.tsx","../src/builder/inspector/BlockContentControls.tsx","../src/builder/inspector/alignOptions.ts","../src/builder/inspector/BlockLayoutControls.tsx","../src/builder/inspector/SpacingControls.tsx","../src/builder/inspector/TypographyControls.tsx","../src/builder/inspector/BlockInspector.tsx","../src/builder/inspector/Inspector.tsx","../src/builder/inspector/DocumentSettings.tsx","../src/builder/inspector/PageSettings.tsx","../src/builder/Builder.tsx","../src/render/previewChrome.tsx","../src/render/PdfPane.tsx","../src/render/Preview.tsx","../src/builder/TemplateBuilder.tsx","../src/builder/schema/invoiceExample.ts","../src/editor/parseTemplate.ts","../src/editor/TemplateEditorContext.tsx","../src/editor/editorTheme.ts","../schemas/template.schema.json","../src/editor/templateSchema.ts","../src/editor/CodeEditor.tsx","../src/editor/TemplateEditor.tsx","../src/html-editor/defaultHtml.ts","../src/html-editor/useHtmlPreview.ts","../src/html-editor/HtmlEditorContext.tsx","../src/html-editor/HtmlCodeEditor.tsx","../src/html-editor/HtmlPreview.tsx","../src/html-editor/HtmlEditor.tsx"],"sourcesContent":["import type { FileAttachment } from \"../types/generated/template\";\nimport type {\n RenderOptions,\n Template,\n TemplateData,\n TemplateSchemaResponse,\n} from \"../types/template\";\n\ninterface RenderTemplateRequest {\n template: Template;\n data?: TemplateData;\n options?: RenderOptions;\n}\n\nexport interface ConvertHtmlRequest {\n /** Raw HTML document to convert to a PDF/UA document. */\n html: string;\n /** Base URL used to resolve relative asset references in the HTML. */\n baseUrl?: string;\n /** Files to embed in the produced PDF/A-3 document. */\n attachments?: FileAttachment[];\n}\n\nexport function resolveDefaultApiUrl(configuredApiUrl?: string): string {\n return configuredApiUrl ?? \"\";\n}\n\nfunction joinUrl(baseUrl: string, path: string): string {\n return `${baseUrl.replace(/\\/+$/, \"\")}/${path.replace(/^\\/+/, \"\")}`;\n}\n\nasync function parseError(response: Response): Promise<string> {\n const contentType = response.headers.get(\"content-type\") ?? \"\";\n const fallback = `${response.status} ${response.statusText || \"Request failed\"}`;\n\n if (contentType.includes(\"application/json\")) {\n const payload = (await response.json()) as { error?: string };\n return payload.error || fallback;\n }\n\n const body = (await response.text()).trim();\n return body || fallback;\n}\n\nexport async function fetchTemplateSchema(baseUrl: string): Promise<TemplateSchemaResponse> {\n const response = await fetch(joinUrl(baseUrl, \"/schema\"), {\n headers: {\n Accept: \"application/json\",\n },\n });\n\n if (!response.ok) {\n throw new Error(await parseError(response));\n }\n\n return (await response.json()) as TemplateSchemaResponse;\n}\n\nexport async function renderTemplatePdf(\n baseUrl: string,\n request: RenderTemplateRequest,\n): Promise<Blob> {\n const response = await fetch(joinUrl(baseUrl, \"/render/template\"), {\n method: \"POST\",\n headers: {\n Accept: \"application/pdf\",\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify({\n data: {},\n options: {},\n ...request,\n }),\n });\n\n if (!response.ok) {\n throw new Error(await parseError(response));\n }\n\n return response.blob();\n}\n\nexport async function renderHtmlPdf(\n baseUrl: string,\n request: ConvertHtmlRequest,\n): Promise<Blob> {\n const response = await fetch(joinUrl(baseUrl, \"/convert\"), {\n method: \"POST\",\n headers: {\n Accept: \"application/pdf\",\n \"Content-Type\": \"application/json\",\n },\n body: JSON.stringify(request),\n });\n\n if (!response.ok) {\n throw new Error(await parseError(response));\n }\n\n return response.blob();\n}\n","import type { Block } from \"../../types/generated/template\";\n\nexport interface BlockChrome {\n chip: string;\n label: string;\n}\n\nconst CHROME: Record<string, BlockChrome> = {\n heading: { chip: \"H\", label: \"Heading\" },\n text: { chip: \"¶\", label: \"Text\" },\n html: { chip: \"</>\", label: \"HTML\" },\n image: { chip: \"◇\", label: \"Image\" },\n table: { chip: \"▦\", label: \"Table\" },\n \"key-value\": { chip: \"≡\", label: \"Key-Value\" },\n spacer: { chip: \"↕\", label: \"Spacer\" },\n divider: { chip: \"─\", label: \"Divider\" },\n};\n\nexport function getBlockChrome(type: string): BlockChrome {\n return CHROME[type] ?? { chip: \"?\", label: prettify(type) };\n}\n\nexport function getBlockSummary(block: Block): string {\n switch (block.type) {\n case \"heading\":\n case \"text\":\n return truncate(block.text);\n case \"html\":\n return truncate(block.html);\n case \"image\":\n return truncate(block.alt ?? block.src);\n case \"key-value\": {\n const fields = block.config?.fields ?? [];\n\n return fields.length > 0 ? `${fields.length} field${fields.length === 1 ? \"\" : \"s\"}` : \"\";\n }\n case \"table\": {\n const columns = block.config?.columns ?? [];\n\n return columns.length > 0\n ? `${columns.length} column${columns.length === 1 ? \"\" : \"s\"}`\n : \"\";\n }\n case \"spacer\": {\n const height = block.config?.height;\n\n return typeof height === \"number\" ? `${height}mm` : \"\";\n }\n case \"divider\": {\n const style = block.config?.style;\n\n return typeof style === \"string\" ? style : \"\";\n }\n default:\n return \"\";\n }\n}\n\nfunction truncate(value: string | null | undefined, max = 56): string {\n if (!value) {\n return \"\";\n }\n\n const single = value.replace(/\\s+/g, \" \").trim();\n\n return single.length > max ? `${single.slice(0, max - 1)}…` : single;\n}\n\nfunction prettify(type: string): string {\n return type\n .split(\"-\")\n .filter(Boolean)\n .map((part) => `${part.charAt(0).toUpperCase()}${part.slice(1)}`)\n .join(\" \");\n}\n","import type { ReactNode } from \"react\";\n\nexport interface ChipProps {\n children: ReactNode;\n /** Grow to fit multi-character labels instead of staying a 22×22 square. */\n wide?: boolean;\n}\n\nexport function Chip({ children, wide = false }: ChipProps) {\n return (\n <span\n className={`inline-grid h-[22px] place-items-center rounded bg-surface-muted font-mono text-2xs font-semibold text-fg-muted ${wide ? \"min-w-[22px] px-1\" : \"w-[22px]\"}`}\n aria-hidden=\"true\"\n >\n {children}\n </span>\n );\n}\n","import { getBlockChrome } from \"../blocks/blockChrome\";\nimport { Chip } from \"../primitives/Chip\";\n\nexport interface BlockCardPreviewProps {\n type: string;\n summary?: string;\n prefix?: string;\n}\n\nexport function BlockCardPreview({ type, summary, prefix }: BlockCardPreviewProps) {\n const chrome = getBlockChrome(type);\n\n return (\n <div className=\"pointer-events-none origin-top-left rotate-[1.5deg] scale-[1.02] cursor-grabbing drop-shadow-drag\">\n <div className=\"inline-flex min-w-[180px] max-w-[360px] items-center gap-2 rounded-lg border border-solid border-border-strong bg-surface px-3 py-2 text-sm font-medium\">\n <Chip>{chrome.chip}</Chip>\n <span className=\"inline-flex items-baseline gap-2\">\n <span className=\"font-medium\">\n {prefix}\n {chrome.label}\n </span>\n {summary ? <span className=\"font-normal text-fg-muted\">{summary}</span> : null}\n </span>\n </div>\n </div>\n );\n}\n","import { KeyboardSensor, PointerSensor, useSensor, useSensors } from \"@dnd-kit/core\";\nimport { sortableKeyboardCoordinates } from \"@dnd-kit/sortable\";\n\nexport function useBuilderSensors() {\n return useSensors(\n useSensor(PointerSensor, { activationConstraint: { distance: 4 } }),\n useSensor(KeyboardSensor, { coordinateGetter: sortableKeyboardCoordinates }),\n );\n}\n","export function isRecord(value: unknown): value is Record<string, unknown> {\n return typeof value === \"object\" && value !== null && !Array.isArray(value);\n}\n\nexport function omitKey<TValue>(\n record: Record<string, TValue>,\n key: string,\n): Record<string, TValue> {\n if (!(key in record)) {\n return record;\n }\n\n const next = { ...record };\n delete next[key];\n return next;\n}\n\nexport function renameKey<TValue>(\n record: Record<string, TValue>,\n previousKey: string,\n nextKey: string,\n): Record<string, TValue> {\n if (!(previousKey in record)) {\n return record;\n }\n\n const next: Record<string, TValue> = {};\n for (const [key, value] of Object.entries(record)) {\n next[key === previousKey ? nextKey : key] = value;\n }\n return next;\n}\n\nexport function nextKeyIndex(keys: readonly string[], prefix: string): number {\n let index = keys.length + 1;\n\n while (keys.includes(`${prefix}${index}`)) {\n index += 1;\n }\n\n return index;\n}\n","import type { UniqueIdentifier } from \"@dnd-kit/core\";\nimport { isRecord } from \"../lib/records\";\nimport type { EditorArea, EditorModel } from \"./editorModel\";\n\nexport interface DragData {\n source?: string;\n type?: string;\n rowUid?: string;\n blockUid?: string;\n area?: EditorArea;\n}\n\nexport interface DropTarget {\n rowUid: string | null;\n index: number;\n area: EditorArea;\n}\n\nexport function getDragData(value: unknown): DragData {\n return isRecord(value) ? (value as DragData) : {};\n}\n\nexport function getDropTarget(\n model: EditorModel,\n overId: UniqueIdentifier | undefined,\n overData: DragData,\n): DropTarget {\n if (overId === \"new-row\") {\n return { rowUid: null, index: 0, area: \"body\" };\n }\n if (overId === \"new-footer-row\") {\n return { rowUid: null, index: 0, area: \"footer\" };\n }\n\n if (overData.type === \"block\" && overData.rowUid && overData.blockUid) {\n return {\n rowUid: overData.rowUid,\n index: getBlockIndex(model, overData.blockUid),\n area: overData.area ?? \"body\",\n };\n }\n\n if (overData.type === \"row\" && overData.rowUid) {\n return {\n rowUid: overData.rowUid,\n index: getRowBlockCount(model, overData.rowUid),\n area: overData.area ?? \"body\",\n };\n }\n\n return { rowUid: null, index: 0, area: \"body\" };\n}\n\nexport function getRowIndex(\n model: EditorModel,\n overId: UniqueIdentifier | undefined,\n overData: DragData,\n): number | null {\n const rowUid =\n (overData.type === \"row\" || overData.type === \"block\") && overData.rowUid\n ? overData.rowUid\n : String(overId);\n const bodyIndex = model.rows.findIndex((row) => row.uid === rowUid);\n if (bodyIndex !== -1) {\n return bodyIndex;\n }\n const footerIndex = model.footerRows.findIndex((row) => row.uid === rowUid);\n return footerIndex === -1 ? null : footerIndex;\n}\n\nfunction getBlockIndex(model: EditorModel, blockUid: string): number {\n for (const row of [...model.rows, ...model.footerRows]) {\n const index = row.blocks.findIndex((block) => block.uid === blockUid);\n\n if (index !== -1) {\n return index;\n }\n }\n\n return 0;\n}\n\nfunction getRowBlockCount(model: EditorModel, rowUid: string): number {\n const row =\n model.rows.find((candidate) => candidate.uid === rowUid) ??\n model.footerRows.find((candidate) => candidate.uid === rowUid);\n return row?.blocks.length ?? 0;\n}\n","import type {\n Align,\n Block,\n CustomPageSize,\n Orientation,\n PageConfig,\n PageFooterConfig,\n PageFormat,\n PageNumbersConfig,\n PageSize,\n PresetPageSize,\n Row,\n Template,\n TemplateConfig,\n} from \"../../types/generated/template\";\n\nexport type EditorArea = \"body\" | \"footer\";\n\nexport type PageNumbersValue = \"disabled\" | Align;\n\nexport interface ResolvedPageSize {\n format: PageFormat;\n orientation: Orientation;\n custom: CustomPageSize | null;\n}\n\nexport interface EditorBlock {\n uid: string;\n block: Block;\n}\n\nexport interface EditorRow {\n uid: string;\n blocks: EditorBlock[];\n}\n\nexport interface EditorModel {\n template: Omit<Template, \"rows\">;\n rows: EditorRow[];\n footerRows: EditorRow[];\n}\n\nlet nextUid = 1;\n\nexport function createEditorModel(template: Template): EditorModel {\n const { rows = [], ...templateFields } = template;\n const footerRows = templateFields.config?.page?.footer?.rows ?? [];\n const strippedTemplate = stripFooterRows(cloneValue(templateFields));\n\n return {\n template: strippedTemplate,\n rows: rows.map((row) => createEditorRow(row)),\n footerRows: footerRows.map((row) => createEditorRow(row)),\n };\n}\n\nexport function serializeTemplate(model: EditorModel): Template {\n const template = cloneValue(model.template) as Template;\n const withFooter = applyFooterRows(template, model.footerRows);\n\n if (model.rows.length === 0) {\n return withFooter;\n }\n\n return {\n ...withFooter,\n rows: model.rows.map((row) => ({\n blocks: row.blocks.map((editorBlock) => cloneBlock(editorBlock.block)),\n })),\n };\n}\n\nexport function addBlockToNewRow(\n model: EditorModel,\n block: Block,\n area: EditorArea = \"body\",\n): EditorModel {\n return updateArea(model, area, (rows) => [\n ...rows,\n { uid: createUid(\"row\"), blocks: [createEditorBlock(block)] },\n ]);\n}\n\nexport function addBlockToRow(\n model: EditorModel,\n rowUid: string,\n block: Block,\n index: number,\n): EditorModel {\n const area = findRowArea(model, rowUid);\n\n if (!area) {\n return model;\n }\n\n return updateArea(model, area, (rows) =>\n rows.map((row) =>\n row.uid === rowUid\n ? { ...row, blocks: insertAt(row.blocks, createEditorBlock(block), index) }\n : row,\n ),\n );\n}\n\nexport function removeBlock(model: EditorModel, blockUid: string): EditorModel {\n return mapAllAreas(model, (rows) =>\n rows\n .map((row) => ({\n ...row,\n blocks: row.blocks.filter((block) => block.uid !== blockUid),\n }))\n .filter((row) => row.blocks.length > 0),\n );\n}\n\nexport function moveBlock(\n model: EditorModel,\n blockUid: string,\n rowUid: string | null,\n index: number,\n area: EditorArea = \"body\",\n): EditorModel {\n const block = findEditorBlock(model, blockUid);\n\n if (!block) {\n return model;\n }\n\n if (rowUid === null) {\n const stripped = mapAllAreas(model, (rows) =>\n rows\n .map((row) => ({\n ...row,\n blocks: row.blocks.filter((candidate) => candidate.uid !== blockUid),\n }))\n .filter((row) => row.blocks.length > 0),\n );\n\n return updateArea(stripped, area, (rows) => [\n ...rows,\n { uid: createUid(\"row\"), blocks: [block] },\n ]);\n }\n\n const targetArea = findRowArea(model, rowUid);\n\n if (!targetArea) {\n return model;\n }\n\n const stripped = mapAllAreas(model, (rows) =>\n rows.map((row) => ({\n ...row,\n blocks: row.blocks.filter((candidate) => candidate.uid !== blockUid),\n })),\n );\n\n const withInsert = updateArea(stripped, targetArea, (rows) =>\n rows.map((row) =>\n row.uid === rowUid ? { ...row, blocks: insertAt(row.blocks, block, index) } : row,\n ),\n );\n\n return mapAllAreas(withInsert, (rows) => rows.filter((row) => row.blocks.length > 0));\n}\n\nexport function moveRow(model: EditorModel, rowUid: string, index: number): EditorModel {\n const area = findRowArea(model, rowUid);\n\n if (!area) {\n return model;\n }\n\n return updateArea(model, area, (rows) => {\n const row = rows.find((candidate) => candidate.uid === rowUid);\n\n if (!row) {\n return rows;\n }\n\n const remaining = rows.filter((candidate) => candidate.uid !== rowUid);\n\n return insertAt(remaining, row, index);\n });\n}\n\nexport function updateBlock(model: EditorModel, blockUid: string, block: Block): EditorModel {\n return mapAllAreas(model, (rows) =>\n rows.map((row) => ({\n ...row,\n blocks: row.blocks.map((editorBlock) =>\n editorBlock.uid === blockUid ? { ...editorBlock, block: cloneBlock(block) } : editorBlock,\n ),\n })),\n );\n}\n\nexport function updateTemplateSettings(model: EditorModel, template: Template): EditorModel {\n return {\n ...model,\n template: createEditorModel(template).template,\n };\n}\n\nexport function getPageSize(model: EditorModel): ResolvedPageSize {\n const size = model.template.config?.page?.size;\n\n if (isCustomPageSize(size)) {\n return { format: \"A4\", orientation: \"portrait\", custom: size };\n }\n\n return {\n format: (size?.format ?? \"A4\") as PageFormat,\n orientation: (size?.orientation ?? \"portrait\") as Orientation,\n custom: null,\n };\n}\n\nexport function setPageSize(\n model: EditorModel,\n format: PageFormat,\n orientation: Orientation,\n): EditorModel {\n const nextSize: PresetPageSize = { format, orientation };\n const previousConfig = model.template.config ?? {};\n const previousPage = previousConfig.page ?? {};\n const nextPage: PageConfig = { ...previousPage, size: nextSize };\n const nextConfig: TemplateConfig = { ...previousConfig, page: nextPage };\n\n return {\n ...model,\n template: { ...model.template, config: nextConfig },\n };\n}\n\nfunction isCustomPageSize(size: PageSize | undefined): size is CustomPageSize {\n return (\n typeof size === \"object\" &&\n size !== null &&\n \"width\" in size &&\n \"height\" in size &&\n typeof (size as CustomPageSize).width === \"number\" &&\n typeof (size as CustomPageSize).height === \"number\"\n );\n}\n\nexport function setRowWidths(model: EditorModel, rowUid: string, widths: string[]): EditorModel {\n const area = findRowArea(model, rowUid);\n\n if (!area) {\n return model;\n }\n\n return updateArea(model, area, (rows) =>\n rows.map((row) =>\n row.uid === rowUid\n ? {\n ...row,\n blocks: row.blocks.map((editorBlock, index) => ({\n ...editorBlock,\n block: setBlockWidth(editorBlock.block, widths[index]),\n })),\n }\n : row,\n ),\n );\n}\n\nexport function getFooterRepeat(model: EditorModel): boolean {\n return model.template.config?.page?.footer?.repeat !== false;\n}\n\nexport function setFooterRepeat(model: EditorModel, repeat: boolean): EditorModel {\n return updatePageConfig(model, (page) => {\n const nextFooter: PageFooterConfig = { ...page.footer, repeat };\n\n return { ...page, footer: nextFooter };\n });\n}\n\nexport function getPageNumbers(model: EditorModel): PageNumbersValue {\n const pageNumbers = model.template.config?.page?.pageNumbers;\n\n if (pageNumbers?.enabled !== true) {\n return \"disabled\";\n }\n\n return (pageNumbers.position ?? \"center\") as Align;\n}\n\nexport function setPageNumbers(model: EditorModel, value: PageNumbersValue): EditorModel {\n return updatePageConfig(model, (page) => {\n if (value === \"disabled\") {\n const next: PageNumbersConfig = { ...page.pageNumbers, enabled: false };\n\n return { ...page, pageNumbers: next };\n }\n\n const next: PageNumbersConfig = { enabled: true, position: value };\n\n return { ...page, pageNumbers: next };\n });\n}\n\nfunction updatePageConfig(\n model: EditorModel,\n updater: (page: PageConfig) => PageConfig,\n): EditorModel {\n const previousConfig = model.template.config ?? {};\n const previousPage = previousConfig.page ?? {};\n const nextPage = updater(previousPage);\n const nextConfig: TemplateConfig = { ...previousConfig, page: nextPage };\n\n return { ...model, template: { ...model.template, config: nextConfig } };\n}\n\nfunction findRowArea(model: EditorModel, rowUid: string): EditorArea | null {\n if (model.rows.some((row) => row.uid === rowUid)) {\n return \"body\";\n }\n if (model.footerRows.some((row) => row.uid === rowUid)) {\n return \"footer\";\n }\n return null;\n}\n\nfunction updateArea(\n model: EditorModel,\n area: EditorArea,\n updater: (rows: EditorRow[]) => EditorRow[],\n): EditorModel {\n if (area === \"body\") {\n return { ...model, rows: updater(model.rows) };\n }\n return { ...model, footerRows: updater(model.footerRows) };\n}\n\nfunction mapAllAreas(\n model: EditorModel,\n updater: (rows: EditorRow[]) => EditorRow[],\n): EditorModel {\n return {\n ...model,\n rows: updater(model.rows),\n footerRows: updater(model.footerRows),\n };\n}\n\nfunction stripFooterRows(template: Omit<Template, \"rows\">): Omit<Template, \"rows\"> {\n const footer = template.config?.page?.footer;\n\n if (!footer) {\n return template;\n }\n\n const { rows: _strippedRows, ...rest } = footer;\n const nextFooter = Object.keys(rest).length === 0 ? undefined : rest;\n const nextPage = { ...template.config?.page, footer: nextFooter };\n\n if (nextPage.footer === undefined) {\n delete nextPage.footer;\n }\n\n return {\n ...template,\n config: { ...template.config, page: nextPage },\n };\n}\n\nfunction applyFooterRows(template: Template, footerRows: EditorRow[]): Template {\n const existingPage = template.config?.page ?? {};\n const existingFooter = existingPage.footer ?? {};\n const hasFooterContext =\n footerRows.length > 0 || existingFooter.repeat !== undefined;\n\n if (!hasFooterContext) {\n return template;\n }\n\n const serializedRows: Row[] = footerRows.map((row) => ({\n blocks: row.blocks.map((editorBlock) => cloneBlock(editorBlock.block)),\n }));\n const nextFooter: PageFooterConfig = {\n ...existingFooter,\n ...(serializedRows.length > 0 ? { rows: serializedRows } : {}),\n };\n const nextPage: PageConfig = { ...existingPage, footer: nextFooter };\n\n return { ...template, config: { ...template.config, page: nextPage } };\n}\n\nfunction createEditorRow(row: Row): EditorRow {\n return {\n uid: createUid(\"row\"),\n blocks: row.blocks.map((block) => createEditorBlock(block)),\n };\n}\n\nfunction createEditorBlock(block: Block): EditorBlock {\n return {\n uid: createUid(\"block\"),\n block: cloneBlock(block),\n };\n}\n\nexport function findEditorBlock(model: EditorModel, blockUid: string): EditorBlock | undefined {\n const allRows = [...model.rows, ...model.footerRows];\n\n return allRows.flatMap((row) => row.blocks).find((block) => block.uid === blockUid);\n}\n\nexport function resolveSelectedEditorBlock(\n model: EditorModel,\n blockUid: string | null,\n): EditorBlock | null {\n return blockUid ? (findEditorBlock(model, blockUid) ?? null) : null;\n}\n\nexport function reconcileSelectedBlockUid(\n model: EditorModel,\n blockUid: string | null,\n): string | null {\n return resolveSelectedEditorBlock(model, blockUid) ? blockUid : null;\n}\n\nexport function createNextBlockId(model: EditorModel, blockType: string): string {\n const template = serializeTemplate(model);\n const allRows = [...(template.rows ?? []), ...(template.config?.page?.footer?.rows ?? [])];\n const usedIds = new Set(\n allRows.flatMap((row) =>\n row.blocks\n .map((block) => block.id)\n .filter((id): id is string => typeof id === \"string\" && id.length > 0),\n ),\n );\n let nextId = 1;\n\n while (usedIds.has(`${blockType}-${nextId}`)) {\n nextId += 1;\n }\n\n return `${blockType}-${nextId}`;\n}\n\nfunction setBlockWidth(block: Block, width: string | undefined): Block {\n const cloned = cloneBlock(block);\n const nextConfig: Record<string, unknown> = { ...cloned.config };\n\n if (width === undefined || width === \"\") {\n delete nextConfig.width;\n } else {\n nextConfig.width = width;\n }\n\n if (Object.keys(nextConfig).length === 0) {\n const { config: _config, ...rest } = cloned;\n\n return rest as Block;\n }\n\n return { ...cloned, config: nextConfig } as Block;\n}\n\nfunction insertAt<T>(items: readonly T[], item: T, index: number): T[] {\n const nextItems = [...items];\n const safeIndex = Math.max(0, Math.min(index, nextItems.length));\n\n nextItems.splice(safeIndex, 0, item);\n\n return nextItems;\n}\n\nfunction cloneBlock(block: Block): Block {\n return cloneValue(block);\n}\n\nfunction cloneValue<T>(value: T): T {\n return structuredClone(value);\n}\n\nfunction createUid(prefix: string): string {\n const uid = `${prefix}-${nextUid}`;\n nextUid += 1;\n\n return uid;\n}\n","import { type DragEndEvent, type DragStartEvent } from \"@dnd-kit/core\";\nimport { useCallback, useState, type Dispatch, type RefObject } from \"react\";\nimport type { Block } from \"../../types/generated/template\";\nimport type { JsonSchemaObject } from \"../../types/template\";\nimport { useBuilderSensors } from \"../lib/sensors\";\nimport { getDragData } from \"../state/dragDrop\";\nimport { findEditorBlock, type EditorModel } from \"../state/editorModel\";\nimport type { EditorAction } from \"../state/editorReducer\";\n\nexport type ActiveDrag =\n | { kind: \"palette\"; type: string }\n | { kind: \"block\"; block: Block }\n | { kind: \"row\" }\n | null;\n\ninterface BuilderDragDrop {\n activeDrag: ActiveDrag;\n sensors: ReturnType<typeof useBuilderSensors>;\n onDragStart: (event: DragStartEvent) => void;\n onDragEnd: (event: DragEndEvent) => void;\n onDragCancel: () => void;\n}\n\nexport function useBuilderDragDrop(\n schema: JsonSchemaObject | null,\n dispatch: Dispatch<EditorAction>,\n modelRef: RefObject<EditorModel>,\n): BuilderDragDrop {\n const [activeDrag, setActiveDrag] = useState<ActiveDrag>(null);\n\n const sensors = useBuilderSensors();\n\n const onDragStart = useCallback(\n (event: DragStartEvent) => {\n const dragData = getDragData(event.active.data.current);\n\n if (dragData.source === \"palette\" && dragData.type) {\n setActiveDrag({ kind: \"palette\", type: dragData.type });\n return;\n }\n if (dragData.type === \"row\") {\n setActiveDrag({ kind: \"row\" });\n return;\n }\n if (dragData.type === \"block\" && dragData.blockUid) {\n const editorBlock = findEditorBlock(modelRef.current, dragData.blockUid);\n setActiveDrag(editorBlock ? { kind: \"block\", block: editorBlock.block } : null);\n }\n },\n [modelRef],\n );\n\n const onDragEnd = useCallback(\n (event: DragEndEvent) => {\n setActiveDrag(null);\n\n if (!event.over) {\n return;\n }\n\n const activeData = getDragData(event.active.data.current);\n const overData = getDragData(event.over.data.current);\n\n if (activeData.source === \"palette\" && activeData.type && schema) {\n dispatch({\n type: \"dropFromPalette\",\n schema,\n blockType: activeData.type,\n overId: event.over.id,\n overData,\n });\n return;\n }\n\n if (activeData.type === \"row\" && activeData.rowUid) {\n dispatch({ type: \"moveRowTo\", rowUid: activeData.rowUid, overId: event.over.id, overData });\n return;\n }\n\n if (activeData.type === \"block\" && activeData.blockUid) {\n dispatch({\n type: \"moveBlockTo\",\n blockUid: activeData.blockUid,\n overId: event.over.id,\n overData,\n });\n }\n },\n [schema, dispatch],\n );\n\n const onDragCancel = useCallback(() => {\n setActiveDrag(null);\n }, []);\n\n return { activeDrag, sensors, onDragStart, onDragEnd, onDragCancel };\n}\n","import { useCallback, useEffect, useRef, useState } from \"react\";\nimport { fetchTemplateSchema, renderTemplatePdf } from \"../api/pdfUaApi\";\nimport type { Template } from \"../types/generated/template\";\nimport type { TemplateData, TemplateSchemaResponse } from \"../types/template\";\n\ninterface UsePdfUaApiOptions {\n initialApiUrl: string;\n apiUrl: string;\n onRendered?: (pdf: Blob) => void;\n loadSchemaOnMount?: boolean;\n}\n\ninterface PdfUaApi {\n schema: TemplateSchemaResponse | null;\n schemaLoading: boolean;\n pdfUrl: string | null;\n pdfLoading: boolean;\n error: string | null;\n loadSchema: (url: string) => Promise<void>;\n renderPdf: (template: Template, data: TemplateData) => Promise<void>;\n}\n\nexport function usePdfUaApi({\n initialApiUrl,\n apiUrl,\n onRendered,\n loadSchemaOnMount = true,\n}: UsePdfUaApiOptions): PdfUaApi {\n const [schema, setSchema] = useState<TemplateSchemaResponse | null>(null);\n const [schemaLoading, setSchemaLoading] = useState(false);\n const [pdfUrl, setPdfUrl] = useState<string | null>(null);\n const [pdfLoading, setPdfLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n const initialApiUrlRef = useRef(initialApiUrl);\n const schemaRequestId = useRef(0);\n const renderRequestId = useRef(0);\n const pdfUrlRef = useRef<string | null>(null);\n const onRenderedRef = useRef(onRendered);\n\n useEffect(() => {\n onRenderedRef.current = onRendered;\n }, [onRendered]);\n\n useEffect(() => {\n return () => {\n revokeObjectUrl(pdfUrlRef.current);\n pdfUrlRef.current = null;\n };\n }, []);\n\n const loadSchema = useCallback(async (url: string) => {\n const requestId = schemaRequestId.current + 1;\n schemaRequestId.current = requestId;\n setSchemaLoading(true);\n setError(null);\n\n try {\n const nextSchema = await fetchTemplateSchema(url);\n\n if (requestId === schemaRequestId.current) {\n setSchema(nextSchema);\n }\n } catch (cause) {\n if (requestId === schemaRequestId.current) {\n setError(errorMessage(cause));\n }\n } finally {\n if (requestId === schemaRequestId.current) {\n setSchemaLoading(false);\n }\n }\n }, []);\n\n useEffect(() => {\n if (loadSchemaOnMount) {\n void loadSchema(initialApiUrlRef.current);\n }\n }, [loadSchema, loadSchemaOnMount]);\n\n const renderPdf = useCallback(\n async (template: Template, data: TemplateData) => {\n const requestId = renderRequestId.current + 1;\n renderRequestId.current = requestId;\n setPdfLoading(true);\n setError(null);\n\n try {\n const pdf = await renderTemplatePdf(apiUrl, {\n template,\n data,\n options: { title: \"Template Preview\" },\n });\n const nextPdfUrl = URL.createObjectURL(pdf);\n\n if (requestId !== renderRequestId.current) {\n revokeObjectUrl(nextPdfUrl);\n return;\n }\n\n setPdfUrl((currentPdfUrl) => {\n revokeObjectUrl(currentPdfUrl);\n pdfUrlRef.current = nextPdfUrl;\n return nextPdfUrl;\n });\n onRenderedRef.current?.(pdf);\n } catch (cause) {\n if (requestId === renderRequestId.current) {\n setError(errorMessage(cause));\n }\n } finally {\n if (requestId === renderRequestId.current) {\n setPdfLoading(false);\n }\n }\n },\n [apiUrl],\n );\n\n return { schema, schemaLoading, pdfUrl, pdfLoading, error, loadSchema, renderPdf };\n}\n\nfunction revokeObjectUrl(url: string | null): void {\n if (url) {\n URL.revokeObjectURL(url);\n }\n}\n\nfunction errorMessage(cause: unknown): string {\n return cause instanceof Error ? cause.message : String(cause);\n}\n","import { createContext, useContext, type ReactNode } from \"react\";\nimport type { Template } from \"../types/generated/template\";\nimport type { TemplateData } from \"../types/template\";\n\nexport interface RenderContextValue {\n template: Template | null;\n data: TemplateData;\n pdfUrl: string | null;\n pdfLoading: boolean;\n error: string | null;\n renderPdf: () => void;\n renderDisabled: boolean;\n}\n\nconst RenderContext = createContext<RenderContextValue | null>(null);\n\nexport function RenderProvider({\n value,\n children,\n}: {\n value: RenderContextValue;\n children: ReactNode;\n}) {\n return <RenderContext.Provider value={value}>{children}</RenderContext.Provider>;\n}\n\nexport function useRenderContext(): RenderContextValue {\n const value = useContext(RenderContext);\n\n if (!value) {\n throw new Error(\"useRenderContext must be used within a render provider.\");\n }\n\n return value;\n}\n","import type { Block } from \"../../types/generated/template\";\nimport type {\n JsonSchemaObject,\n JsonSchemaValue,\n TemplateSchemaMetadata,\n} from \"../../types/template\";\nimport { isRecord } from \"../lib/records\";\n\nexport type { JsonSchemaObject, JsonSchemaValue };\n\nexport function getSchemaMetadata(schema: JsonSchemaObject): TemplateSchemaMetadata {\n return assertTemplateSchemaMetadata(schema[\"x-pdfUa\"]);\n}\n\nexport function resolveRef(schema: JsonSchemaObject, ref: string): JsonSchemaObject {\n if (!ref.startsWith(\"#/\")) {\n throw new Error(`Unsupported schema ref: ${ref}`);\n }\n\n const value = ref\n .slice(2)\n .split(\"/\")\n .reduce<unknown>((current, segment) => {\n if (!isSchemaObject(current)) {\n return undefined;\n }\n\n return current[decodeJsonPointerSegment(segment)];\n }, schema);\n\n if (!isSchemaObject(value)) {\n throw new Error(`Schema ref not found: ${ref}`);\n }\n\n return value;\n}\n\nexport function getBlockTypes(schema: JsonSchemaObject): string[] {\n const discovered = getBlockDefinitions(schema)\n .map((definition) => getBlockType(definition))\n .filter((type): type is string => type !== undefined);\n\n const discoveredSet = new Set(discovered);\n const orderedTypes = getSchemaMetadata(schema).blockOrder.filter((type) =>\n discoveredSet.has(type),\n );\n const remainingTypes = discovered.filter((type) => !orderedTypes.includes(type));\n\n return [...orderedTypes, ...remainingTypes];\n}\n\nexport function getBlockDefinition(\n schema: JsonSchemaObject,\n blockType: string,\n): JsonSchemaObject | undefined {\n return getBlockDefinitions(schema).find((definition) => getBlockType(definition) === blockType);\n}\n\nexport function getBlockFieldSchema(schema: JsonSchemaObject, blockType: string): JsonSchemaObject {\n const definition = requireBlockDefinition(schema, blockType);\n const properties = getProperties(definition);\n const fieldProperties = Object.fromEntries(\n Object.entries(properties).filter(([field]) => field !== \"type\" && field !== \"config\"),\n );\n const required = getStringArray(definition.required).filter((field) => field !== \"type\");\n\n return {\n type: \"object\",\n properties: fieldProperties,\n required,\n $defs: schema.$defs,\n };\n}\n\nexport function getBlockConfigSchema(\n schema: JsonSchemaObject,\n blockType: string,\n): JsonSchemaObject | undefined {\n const definition = requireBlockDefinition(schema, blockType);\n const config = getProperties(definition).config;\n\n if (!isSchemaObject(config)) {\n return undefined;\n }\n\n const ref = config.$ref;\n\n const configSchema = typeof ref === \"string\" ? resolveRef(schema, ref) : config;\n\n return {\n ...configSchema,\n $defs: schema.$defs,\n };\n}\n\nexport function createDefaultBlock(schema: JsonSchemaObject, blockType: string, id: string): Block {\n const definition = requireBlockDefinition(schema, blockType);\n const properties = getProperties(definition);\n const block: Record<string, unknown> = {\n type: blockType,\n id,\n };\n\n for (const field of getStringArray(definition.required)) {\n if (field === \"type\" || field === \"config\" || field === \"id\" || block[field] !== undefined) {\n continue;\n }\n\n const fieldSchema = properties[field];\n\n if (isSchemaObject(fieldSchema)) {\n block[field] = createDefaultValue(schema, fieldSchema);\n }\n }\n\n return block as unknown as Block;\n}\n\nfunction getBlockDefinitions(schema: JsonSchemaObject): JsonSchemaObject[] {\n const defs = schema.$defs;\n const block = isSchemaObject(defs) ? defs.block : undefined;\n const oneOf = isSchemaObject(block) && Array.isArray(block.oneOf) ? block.oneOf : [];\n\n return oneOf\n .map((entry) => {\n if (!isSchemaObject(entry)) {\n return undefined;\n }\n\n const ref = entry.$ref;\n\n return typeof ref === \"string\" ? resolveRef(schema, ref) : entry;\n })\n .filter((entry): entry is JsonSchemaObject => entry !== undefined);\n}\n\nfunction requireBlockDefinition(schema: JsonSchemaObject, blockType: string): JsonSchemaObject {\n const definition = getBlockDefinition(schema, blockType);\n\n if (!definition) {\n throw new Error(`Unknown block type: ${blockType}`);\n }\n\n return definition;\n}\n\nfunction getBlockType(definition: JsonSchemaObject): string | undefined {\n const typeProperty = getProperties(definition).type;\n\n if (!isSchemaObject(typeProperty)) {\n return undefined;\n }\n\n return typeof typeProperty.const === \"string\" ? typeProperty.const : undefined;\n}\n\nfunction getProperties(definition: JsonSchemaObject): Record<string, unknown> {\n return isSchemaObject(definition.properties) ? definition.properties : {};\n}\n\nfunction createDefaultValue(schema: JsonSchemaObject, fieldSchema: JsonSchemaObject): unknown {\n if (\"default\" in fieldSchema) {\n return fieldSchema.default;\n }\n\n if (Array.isArray(fieldSchema.enum)) {\n return fieldSchema.enum.find((value) => value !== null);\n }\n\n if (typeof fieldSchema.const === \"string\" || typeof fieldSchema.const === \"number\") {\n return fieldSchema.const;\n }\n\n if (typeof fieldSchema.$ref === \"string\") {\n return createDefaultValue(schema, resolveRef(schema, fieldSchema.$ref));\n }\n\n if (Array.isArray(fieldSchema.oneOf)) {\n const firstNonNullSchema = fieldSchema.oneOf.find(\n (option) => isSchemaObject(option) && option.type !== \"null\",\n );\n\n return isSchemaObject(firstNonNullSchema)\n ? createDefaultValue(schema, firstNonNullSchema)\n : undefined;\n }\n\n const type = getPrimaryType(fieldSchema.type);\n\n switch (type) {\n case \"string\":\n return \"\";\n case \"integer\":\n case \"number\":\n return 0;\n case \"boolean\":\n return false;\n case \"array\":\n return [];\n case \"object\":\n return {};\n default:\n return undefined;\n }\n}\n\nfunction getPrimaryType(type: unknown): string | undefined {\n if (typeof type === \"string\") {\n return type;\n }\n\n if (!Array.isArray(type)) {\n return undefined;\n }\n\n return type.find((entry): entry is string => typeof entry === \"string\" && entry !== \"null\");\n}\n\nfunction getStringArray(value: unknown): string[] {\n return Array.isArray(value)\n ? value.filter((entry): entry is string => typeof entry === \"string\")\n : [];\n}\n\nfunction assertTemplateSchemaMetadata(value: unknown): TemplateSchemaMetadata {\n if (!isSchemaObject(value) || !isStringArray(value.blockOrder)) {\n throw new Error(\"Invalid template schema metadata\");\n }\n\n return value as unknown as TemplateSchemaMetadata;\n}\n\nfunction isStringArray(value: unknown): value is string[] {\n return Array.isArray(value) && value.every((entry) => typeof entry === \"string\");\n}\n\nfunction isSchemaObject(value: unknown): value is JsonSchemaObject {\n return isRecord(value);\n}\n\nfunction decodeJsonPointerSegment(segment: string): string {\n return segment.replace(/~1/g, \"/\").replace(/~0/g, \"~\");\n}\n","import type { UniqueIdentifier } from \"@dnd-kit/core\";\nimport type { Block, Orientation, PageFormat, Template } from \"../../types/generated/template\";\nimport type { JsonSchemaObject, TemplateData } from \"../../types/template\";\nimport { createDefaultBlock } from \"../schema/schemaAdapter\";\nimport { getDropTarget, getRowIndex, type DragData } from \"./dragDrop\";\nimport {\n addBlockToNewRow,\n addBlockToRow,\n createEditorModel,\n createNextBlockId,\n getPageSize,\n moveBlock,\n moveRow,\n reconcileSelectedBlockUid,\n removeBlock,\n setFooterRepeat,\n setPageNumbers,\n setPageSize,\n setRowWidths,\n updateBlock,\n updateTemplateSettings,\n type EditorModel,\n type PageNumbersValue,\n} from \"./editorModel\";\n\nexport interface EditorState {\n model: EditorModel;\n data: TemplateData;\n selectedBlockUid: string | null;\n}\n\nexport type EditorAction =\n | { type: \"selectBlock\"; blockUid: string }\n | { type: \"deselect\" }\n | { type: \"changeBlock\"; blockUid: string; block: Block }\n | { type: \"removeBlock\"; blockUid: string }\n | { type: \"changeTemplateSettings\"; template: Template }\n | { type: \"setRowWidths\"; rowUid: string; widths: string[] }\n | { type: \"addBlock\"; schema: JsonSchemaObject; blockType: string }\n | { type: \"setFormat\"; format: PageFormat }\n | { type: \"setOrientation\"; orientation: Orientation }\n | { type: \"setFooterRepeat\"; repeat: boolean }\n | { type: \"setPageNumbers\"; value: PageNumbersValue }\n | { type: \"setData\"; data: TemplateData }\n | { type: \"loadExample\"; template: Template; data: TemplateData }\n | {\n type: \"dropFromPalette\";\n schema: JsonSchemaObject;\n blockType: string;\n overId?: UniqueIdentifier;\n overData: DragData;\n }\n | { type: \"moveRowTo\"; rowUid: string; overId?: UniqueIdentifier; overData: DragData }\n | { type: \"moveBlockTo\"; blockUid: string; overId?: UniqueIdentifier; overData: DragData };\n\nexport function createEditorState(template: Template, data: TemplateData): EditorState {\n return { model: createEditorModel(template), data, selectedBlockUid: null };\n}\n\nexport function editorReducer(state: EditorState, action: EditorAction): EditorState {\n switch (action.type) {\n case \"selectBlock\":\n return { ...state, selectedBlockUid: action.blockUid };\n case \"deselect\":\n return { ...state, selectedBlockUid: null };\n case \"changeBlock\":\n return withModel(state, updateBlock(state.model, action.blockUid, action.block));\n case \"removeBlock\":\n return withModel(state, removeBlock(state.model, action.blockUid));\n case \"changeTemplateSettings\":\n return withModel(state, updateTemplateSettings(state.model, action.template));\n case \"setRowWidths\":\n return withModel(state, setRowWidths(state.model, action.rowUid, action.widths));\n case \"addBlock\":\n return withModel(state, addBlockToNewRow(state.model, defaultBlock(state.model, action)));\n case \"setFormat\":\n return withModel(\n state,\n setPageSize(state.model, action.format, getPageSize(state.model).orientation),\n );\n case \"setOrientation\":\n return withModel(\n state,\n setPageSize(state.model, getPageSize(state.model).format, action.orientation),\n );\n case \"setFooterRepeat\":\n return withModel(state, setFooterRepeat(state.model, action.repeat));\n case \"setPageNumbers\":\n return withModel(state, setPageNumbers(state.model, action.value));\n case \"setData\":\n return { ...state, data: action.data };\n case \"loadExample\":\n return createEditorState(action.template, action.data);\n case \"dropFromPalette\": {\n const target = getDropTarget(state.model, action.overId, action.overData);\n const block = defaultBlock(state.model, action);\n const model =\n target.rowUid === null\n ? addBlockToNewRow(state.model, block, target.area)\n : addBlockToRow(state.model, target.rowUid, block, target.index);\n\n return withModel(state, model);\n }\n case \"moveRowTo\": {\n const index = getRowIndex(state.model, action.overId, action.overData);\n\n return index === null ? state : withModel(state, moveRow(state.model, action.rowUid, index));\n }\n case \"moveBlockTo\": {\n const target = getDropTarget(state.model, action.overId, action.overData);\n\n return withModel(\n state,\n moveBlock(state.model, action.blockUid, target.rowUid, target.index, target.area),\n );\n }\n }\n}\n\nfunction withModel(state: EditorState, model: EditorModel): EditorState {\n return { ...state, model, selectedBlockUid: reconcileSelectedBlockUid(model, state.selectedBlockUid) };\n}\n\nfunction defaultBlock(\n model: EditorModel,\n action: { schema: JsonSchemaObject; blockType: string },\n): Block {\n return createDefaultBlock(action.schema, action.blockType, createNextBlockId(model, action.blockType));\n}\n","import { DndContext, DragOverlay, closestCenter } from \"@dnd-kit/core\";\nimport {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useReducer,\n useRef,\n type ReactNode,\n} from \"react\";\nimport { resolveDefaultApiUrl } from \"../../api/pdfUaApi\";\nimport type {\n Block,\n Orientation,\n PageFormat,\n Template,\n} from \"../../types/generated/template\";\nimport type { TemplateData, TemplateSchemaResponse } from \"../../types/template\";\nimport { getBlockSummary } from \"../blocks/blockChrome\";\nimport { BlockCardPreview } from \"../canvas/BlockCardPreview\";\nimport { useBuilderDragDrop, type ActiveDrag } from \"../hooks/useBuilderDragDrop\";\nimport { usePdfUaApi } from \"../../render/usePdfUaApi\";\nimport { RenderProvider, type RenderContextValue } from \"../../render/RenderContext\";\nimport { Chip } from \"../primitives/Chip\";\nimport {\n getFooterRepeat,\n getPageNumbers,\n getPageSize,\n resolveSelectedEditorBlock,\n serializeTemplate,\n type EditorBlock,\n type EditorModel,\n type PageNumbersValue,\n type ResolvedPageSize,\n} from \"../state/editorModel\";\nimport { createEditorState, editorReducer } from \"../state/editorReducer\";\nimport { getBlockTypes, type JsonSchemaObject } from \"../schema/schemaAdapter\";\n\nconst emptyTemplate: Template = {\n version: 1,\n};\n\n/** A loadable example: a template plus its optional runtime data. */\nexport interface TemplateExample {\n template: Template;\n data?: TemplateData;\n}\n\nexport interface TemplateBuilderProviderProps {\n /** Base URL of a running pdf-ua-api instance. Defaults to \"\" (relative URLs / proxy). */\n apiUrl?: string;\n /** Template loaded into the editor on first render. */\n initialTemplate?: Template;\n /** Runtime data keyed by block id (table rows, dynamic key-value overrides). */\n initialData?: TemplateData;\n /** Fires whenever the user edits the template or its runtime data. */\n onChange?: (template: Template, data: TemplateData) => void;\n /** Fires after a successful render with the produced PDF blob. */\n onRendered?: (pdf: Blob) => void;\n /** Pane slots to arrange. They share this provider's state and DnD context. */\n children: ReactNode;\n}\n\n/** Editor state and derived values. Changes as the user edits. */\nexport interface BuilderState {\n schema: TemplateSchemaResponse | null;\n schemaLoading: boolean;\n model: EditorModel;\n data: TemplateData;\n serializedTemplate: Template;\n selectedBlockUid: string | null;\n selectedBlock: EditorBlock | null;\n blockTypes: ReturnType<typeof getBlockTypes>;\n pageSize: ResolvedPageSize;\n footerRepeat: boolean;\n pageNumbers: PageNumbersValue;\n pdfUrl: string | null;\n pdfLoading: boolean;\n error: string | null;\n}\n\n/** Stable editor actions. Identity never changes for the provider's lifetime. */\nexport interface BuilderActions {\n renderPdf: () => void;\n loadExample: (example: TemplateExample) => void;\n addBlock: (type: string) => void;\n changeBlock: (blockUid: string, block: Block) => void;\n changeTemplateSettings: (template: Template) => void;\n removeBlock: (blockUid: string) => void;\n selectBlock: (blockUid: string) => void;\n deselect: () => void;\n setRowWidths: (rowUid: string, widths: string[]) => void;\n changeData: (data: TemplateData) => void;\n changeFormat: (format: PageFormat) => void;\n changeOrientation: (orientation: Orientation) => void;\n toggleFooterRepeat: (repeat: boolean) => void;\n changePageNumbers: (value: PageNumbersValue) => void;\n}\n\nexport type BuilderContextValue = BuilderState & BuilderActions;\n\nconst BuilderStateContext = createContext<BuilderState | null>(null);\nconst BuilderActionsContext = createContext<BuilderActions | null>(null);\n\n/** Subscribe to editor state. Re-renders on edits. */\nexport function useBuilderState(): BuilderState {\n const value = useContext(BuilderStateContext);\n\n if (!value) {\n throw new Error(\"useBuilderState must be used within a <TemplateBuilderProvider>.\");\n }\n\n return value;\n}\n\n/** Read the stable editor actions. Does not re-render on edits. */\nexport function useBuilderActions(): BuilderActions {\n const value = useContext(BuilderActionsContext);\n\n if (!value) {\n throw new Error(\"useBuilderActions must be used within a <TemplateBuilderProvider>.\");\n }\n\n return value;\n}\n\n/** Headless escape hatch returning both state and actions. */\nexport function useTemplateBuilder(): BuilderContextValue {\n return { ...useBuilderState(), ...useBuilderActions() };\n}\n\nexport function TemplateBuilderProvider({\n apiUrl: apiUrlProp,\n initialTemplate,\n initialData,\n onChange,\n onRendered,\n children,\n}: TemplateBuilderProviderProps) {\n const apiUrl = resolveDefaultApiUrl(apiUrlProp);\n const [state, dispatch] = useReducer(editorReducer, undefined, () =>\n createEditorState(initialTemplate ?? emptyTemplate, initialData ?? {}),\n );\n const { model, data, selectedBlockUid } = state;\n const modelRef = useRef(model);\n modelRef.current = model;\n\n const {\n schema,\n schemaLoading,\n pdfUrl,\n pdfLoading,\n error,\n renderPdf: renderPdfRequest,\n } = usePdfUaApi({\n initialApiUrl: apiUrl,\n apiUrl,\n onRendered,\n });\n\n const onChangeRef = useRef(onChange);\n const skipNextChangeRef = useRef(true);\n useEffect(() => {\n onChangeRef.current = onChange;\n }, [onChange]);\n\n const serializedTemplate = useMemo(() => serializeTemplate(model), [model]);\n\n useEffect(() => {\n if (skipNextChangeRef.current) {\n skipNextChangeRef.current = false;\n return;\n }\n onChangeRef.current?.(serializedTemplate, data);\n }, [serializedTemplate, data]);\n\n const schemaObject: JsonSchemaObject | null = schema;\n const blockTypes = useMemo(\n () => (schemaObject ? getBlockTypes(schemaObject) : []),\n [schemaObject],\n );\n const selectedBlock = useMemo(\n () => resolveSelectedEditorBlock(model, selectedBlockUid),\n [model, selectedBlockUid],\n );\n\n // Latest values for actions that would otherwise need them as deps, keeping\n // every action's identity stable for the provider's lifetime.\n const serializedTemplateRef = useRef(serializedTemplate);\n serializedTemplateRef.current = serializedTemplate;\n const dataRef = useRef(data);\n dataRef.current = data;\n const schemaRef = useRef(schemaObject);\n schemaRef.current = schemaObject;\n\n const { activeDrag, sensors, onDragStart, onDragEnd, onDragCancel } = useBuilderDragDrop(\n schemaObject,\n dispatch,\n modelRef,\n );\n\n const loadExample = useCallback((example: TemplateExample) => {\n dispatch({ type: \"loadExample\", template: example.template, data: example.data ?? {} });\n }, []);\n\n const changeBlock = useCallback((blockUid: string, block: Block) => {\n dispatch({ type: \"changeBlock\", blockUid, block });\n }, []);\n\n const changeTemplateSettings = useCallback((template: Template) => {\n dispatch({ type: \"changeTemplateSettings\", template });\n }, []);\n\n const removeBlock = useCallback((blockUid: string) => {\n dispatch({ type: \"removeBlock\", blockUid });\n }, []);\n\n const selectBlock = useCallback((blockUid: string) => {\n dispatch({ type: \"selectBlock\", blockUid });\n }, []);\n\n const deselect = useCallback(() => {\n dispatch({ type: \"deselect\" });\n }, []);\n\n const setRowWidths = useCallback((rowUid: string, widths: string[]) => {\n dispatch({ type: \"setRowWidths\", rowUid, widths });\n }, []);\n\n const changeData = useCallback((nextData: TemplateData) => {\n dispatch({ type: \"setData\", data: nextData });\n }, []);\n\n const addBlock = useCallback((type: string) => {\n const currentSchema = schemaRef.current;\n\n if (!currentSchema) {\n return;\n }\n dispatch({ type: \"addBlock\", schema: currentSchema, blockType: type });\n }, []);\n\n const changeFormat = useCallback((format: PageFormat) => {\n dispatch({ type: \"setFormat\", format });\n }, []);\n\n const changeOrientation = useCallback((orientation: Orientation) => {\n dispatch({ type: \"setOrientation\", orientation });\n }, []);\n\n const toggleFooterRepeat = useCallback((repeat: boolean) => {\n dispatch({ type: \"setFooterRepeat\", repeat });\n }, []);\n\n const changePageNumbers = useCallback((value: PageNumbersValue) => {\n dispatch({ type: \"setPageNumbers\", value });\n }, []);\n\n const renderPdf = useCallback(() => {\n void renderPdfRequest(serializedTemplateRef.current, dataRef.current);\n }, [renderPdfRequest]);\n\n const actions = useMemo<BuilderActions>(\n () => ({\n renderPdf,\n loadExample,\n addBlock,\n changeBlock,\n changeTemplateSettings,\n removeBlock,\n selectBlock,\n deselect,\n setRowWidths,\n changeData,\n changeFormat,\n changeOrientation,\n toggleFooterRepeat,\n changePageNumbers,\n }),\n [\n renderPdf,\n loadExample,\n addBlock,\n changeBlock,\n changeTemplateSettings,\n removeBlock,\n selectBlock,\n deselect,\n setRowWidths,\n changeData,\n changeFormat,\n changeOrientation,\n toggleFooterRepeat,\n changePageNumbers,\n ],\n );\n\n const stateValue = useMemo<BuilderState>(\n () => ({\n schema,\n schemaLoading,\n model,\n data,\n serializedTemplate,\n selectedBlockUid,\n selectedBlock,\n blockTypes,\n pageSize: getPageSize(model),\n footerRepeat: getFooterRepeat(model),\n pageNumbers: getPageNumbers(model),\n pdfUrl,\n pdfLoading,\n error,\n }),\n [\n schema,\n schemaLoading,\n model,\n data,\n serializedTemplate,\n selectedBlockUid,\n selectedBlock,\n blockTypes,\n pdfUrl,\n pdfLoading,\n error,\n ],\n );\n\n const renderValue = useMemo<RenderContextValue>(\n () => ({\n template: serializedTemplate,\n data,\n pdfUrl,\n pdfLoading,\n error,\n renderPdf,\n renderDisabled: !schema || pdfLoading,\n }),\n [serializedTemplate, data, pdfUrl, pdfLoading, error, renderPdf, schema],\n );\n\n return (\n <BuilderActionsContext.Provider value={actions}>\n <BuilderStateContext.Provider value={stateValue}>\n <DndContext\n sensors={sensors}\n collisionDetection={closestCenter}\n onDragStart={onDragStart}\n onDragEnd={onDragEnd}\n onDragCancel={onDragCancel}\n >\n <RenderProvider value={renderValue}>{children}</RenderProvider>\n\n <DragOverlay dropAnimation={null}>\n {activeDrag ? <ActiveDragPreview drag={activeDrag} /> : null}\n </DragOverlay>\n </DndContext>\n </BuilderStateContext.Provider>\n </BuilderActionsContext.Provider>\n );\n}\n\nfunction ActiveDragPreview({ drag }: { drag: NonNullable<ActiveDrag> }) {\n if (drag.kind === \"palette\") {\n return <BlockCardPreview type={drag.type} prefix=\"+ \" />;\n }\n if (drag.kind === \"block\") {\n return <BlockCardPreview type={drag.block.type} summary={getBlockSummary(drag.block)} />;\n }\n return (\n <div className=\"pointer-events-none origin-top-left rotate-[1.5deg] scale-[1.02] cursor-grabbing drop-shadow-drag\">\n <div className=\"inline-flex min-w-[180px] max-w-[360px] items-center gap-2 rounded-lg border border-solid border-border-strong bg-surface px-3 py-2 text-sm font-medium\">\n <Chip>⋮⋮</Chip>\n <span className=\"font-medium\">Row</span>\n </div>\n </div>\n );\n}\n","import type {\n InputHTMLAttributes,\n SelectHTMLAttributes,\n TextareaHTMLAttributes,\n} from \"react\";\n\nconst fieldBase =\n \"min-w-0 rounded-md border border-solid border-border bg-surface text-fg outline-none transition-colors hover:border-border-strong focus-visible:border-accent focus-visible:ring-3 focus-visible:ring-accent/20\";\n\nfunction cx(...names: Array<string | undefined>): string {\n return names.filter(Boolean).join(\" \");\n}\n\nexport function Input({ className, ...rest }: InputHTMLAttributes<HTMLInputElement>) {\n return <input className={cx(\"h-8 w-full px-3\", fieldBase, className)} {...rest} />;\n}\n\nexport function Textarea({ className, ...rest }: TextareaHTMLAttributes<HTMLTextAreaElement>) {\n return (\n <textarea className={cx(\"min-h-24 w-full px-3 py-2 font-mono text-xs\", fieldBase, className)} {...rest} />\n );\n}\n\nexport function Select({ className, ...rest }: SelectHTMLAttributes<HTMLSelectElement>) {\n return (\n <select\n className={cx(\n \"h-8 cursor-pointer pl-3 pr-7 appearance-none bg-[linear-gradient(45deg,transparent_50%,var(--pdfua-fg-muted)_50%),linear-gradient(135deg,var(--pdfua-fg-muted)_50%,transparent_50%)] bg-[length:5px_5px,5px_5px] bg-[position:calc(100%-14px)_50%,calc(100%-9px)_50%] bg-no-repeat\",\n fieldBase,\n className,\n )}\n {...rest}\n />\n );\n}\n\nexport function Checkbox({ className, ...rest }: InputHTMLAttributes<HTMLInputElement>) {\n return <input type=\"checkbox\" className={cx(\"h-3.5 w-3.5 accent-accent\", className)} {...rest} />;\n}\n","import type { EmptyTextValue, SelectFieldOption } from \"./BuilderField\";\n\nexport function textValue(value: string, emptyValue: EmptyTextValue): string | undefined {\n if (value === \"\" && emptyValue === \"undefined\") {\n return undefined;\n }\n\n return value;\n}\n\nexport function numberValue(value: string, valueAsNumber: number): number | undefined {\n if (value === \"\" || Number.isNaN(valueAsNumber)) {\n return undefined;\n }\n\n return valueAsNumber;\n}\n\nexport function selectValue<Value extends string>(\n value: string,\n options: readonly SelectFieldOption<Value>[],\n): Value | undefined {\n if (value === \"\") {\n return undefined;\n }\n\n return options.find((option) => option.value === value)?.value;\n}\n","import type { ReactNode } from \"react\";\nimport { Checkbox, Input, Select, Textarea } from \"./inputs\";\nimport { numberValue, selectValue, textValue } from \"./fieldConverters\";\n\nexport type EmptyTextValue = \"empty-string\" | \"undefined\";\n\nexport interface BuilderControlProps {\n name: string;\n label: string;\n help?: ReactNode;\n error?: ReactNode;\n id?: string;\n className?: string;\n disabled?: boolean;\n}\n\nexport interface TextFieldProps extends BuilderControlProps {\n value?: string;\n placeholder?: string;\n emptyValue?: EmptyTextValue;\n autoComplete?: string;\n onChange: (value: string | undefined) => void;\n}\n\nexport interface TextAreaFieldProps extends BuilderControlProps {\n value?: string;\n placeholder?: string;\n emptyValue?: EmptyTextValue;\n rows?: number;\n onChange: (value: string | undefined) => void;\n}\n\nexport interface NumberFieldProps extends BuilderControlProps {\n value?: number;\n min?: number;\n max?: number;\n step?: number | \"any\";\n placeholder?: string;\n onChange: (value: number | undefined) => void;\n}\n\nexport interface SelectFieldOption<Value extends string> {\n value: Value;\n label: string;\n disabled?: boolean;\n}\n\nexport interface SelectFieldProps<Value extends string> extends BuilderControlProps {\n value?: Value;\n options: readonly SelectFieldOption<Value>[];\n optional?: boolean;\n emptyLabel?: string;\n onChange: (value: Value | undefined) => void;\n}\n\nexport interface CheckboxFieldProps extends BuilderControlProps {\n checked: boolean;\n onChange: (checked: boolean) => void;\n}\n\nexport interface ColorFieldProps extends BuilderControlProps {\n value?: string;\n fallbackValue?: string;\n onChange: (value: string | undefined) => void;\n}\n\nexport interface UnitFieldProps extends BuilderControlProps {\n value?: string;\n placeholder?: string;\n emptyValue?: EmptyTextValue;\n readOnly?: boolean;\n onChange?: (value: string | undefined) => void;\n}\n\ninterface FieldLayoutProps extends BuilderControlProps {\n children: ReactNode;\n}\n\ninterface FieldControlState {\n id: string;\n describedBy?: string;\n invalid: boolean;\n}\n\nexport function TextField({\n value,\n placeholder,\n emptyValue = \"empty-string\",\n autoComplete,\n onChange,\n ...fieldProps\n}: TextFieldProps): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps}>\n <Input\n id={fieldState.id}\n name={fieldProps.name}\n type=\"text\"\n value={value ?? \"\"}\n placeholder={placeholder}\n autoComplete={autoComplete}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={(event) => onChange(textValue(event.currentTarget.value, emptyValue))}\n />\n </BuilderField>\n );\n}\n\nexport function TextAreaField({\n value,\n placeholder,\n emptyValue = \"empty-string\",\n rows = 4,\n onChange,\n ...fieldProps\n}: TextAreaFieldProps): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps}>\n <Textarea\n id={fieldState.id}\n name={fieldProps.name}\n value={value ?? \"\"}\n placeholder={placeholder}\n rows={rows}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={(event) => onChange(textValue(event.currentTarget.value, emptyValue))}\n />\n </BuilderField>\n );\n}\n\nexport function NumberField({\n value,\n min,\n max,\n step,\n placeholder,\n onChange,\n ...fieldProps\n}: NumberFieldProps): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps}>\n <Input\n id={fieldState.id}\n name={fieldProps.name}\n type=\"number\"\n value={value ?? \"\"}\n min={min}\n max={max}\n step={step}\n placeholder={placeholder}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={(event) =>\n onChange(numberValue(event.currentTarget.value, event.currentTarget.valueAsNumber))\n }\n />\n </BuilderField>\n );\n}\n\nexport function SelectField<Value extends string>({\n value,\n options,\n optional = false,\n emptyLabel = \"\",\n onChange,\n ...fieldProps\n}: SelectFieldProps<Value>): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps}>\n <Select\n id={fieldState.id}\n className=\"w-full\"\n name={fieldProps.name}\n value={value ?? \"\"}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={(event) => onChange(selectValue(event.currentTarget.value, options))}\n >\n {optional ? <option value=\"\">{emptyLabel}</option> : null}\n {options.map((option) => (\n <option key={option.value} value={option.value} disabled={option.disabled}>\n {option.label}\n </option>\n ))}\n </Select>\n </BuilderField>\n );\n}\n\nexport function CheckboxField({\n checked,\n onChange,\n ...fieldProps\n}: CheckboxFieldProps): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps} className={fieldProps.className}>\n <Checkbox\n id={fieldState.id}\n name={fieldProps.name}\n checked={checked}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={(event) => onChange(event.currentTarget.checked)}\n />\n </BuilderField>\n );\n}\n\nexport function ColorField({\n value,\n fallbackValue = \"#000000\",\n onChange,\n ...fieldProps\n}: ColorFieldProps): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps}>\n <Input\n id={fieldState.id}\n className=\"p-1\"\n name={fieldProps.name}\n type=\"color\"\n value={value ?? fallbackValue}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={(event) => onChange(textValue(event.currentTarget.value, \"undefined\"))}\n />\n </BuilderField>\n );\n}\n\nexport function UnitField({\n value,\n placeholder = \"auto, 50%, 80mm\",\n emptyValue = \"undefined\",\n readOnly = false,\n onChange,\n ...fieldProps\n}: UnitFieldProps): ReactNode {\n const fieldState = createFieldState(fieldProps);\n\n return (\n <BuilderField {...fieldProps}>\n <Input\n id={fieldState.id}\n name={fieldProps.name}\n type=\"text\"\n inputMode=\"text\"\n spellCheck={false}\n className={readOnly ? \"cursor-default bg-surface-muted text-fg-muted\" : undefined}\n value={value ?? \"\"}\n placeholder={placeholder}\n readOnly={readOnly}\n disabled={fieldProps.disabled}\n aria-describedby={fieldState.describedBy}\n aria-invalid={fieldState.invalid || undefined}\n onChange={\n readOnly\n ? undefined\n : (event) => onChange?.(textValue(event.currentTarget.value, emptyValue))\n }\n />\n </BuilderField>\n );\n}\n\nexport function BuilderField({\n name,\n label,\n help,\n error,\n id,\n className,\n children,\n}: FieldLayoutProps): ReactNode {\n const fieldState = createFieldState({ name, label, help, error, id });\n const helpId = help ? `${fieldState.id}-help` : undefined;\n const errorId = error ? `${fieldState.id}-error` : undefined;\n\n return (\n <div className={classNames(\"grid min-w-0 gap-1\", className)}>\n <label className=\"text-2xs font-medium text-fg-muted\" htmlFor={fieldState.id}>\n {label}\n </label>\n <div className=\"min-w-0\">{children}</div>\n {help ? (\n <p className=\"m-0 text-2xs text-fg-muted\" id={helpId}>\n {help}\n </p>\n ) : null}\n {error ? (\n <p className=\"m-0 text-2xs text-danger\" id={errorId} role=\"alert\">\n {error}\n </p>\n ) : null}\n </div>\n );\n}\n\nfunction createFieldState({\n name,\n help,\n error,\n id,\n}: BuilderControlProps): FieldControlState {\n const fieldId = id ?? createFieldId(name);\n const helpId = help ? `${fieldId}-help` : undefined;\n const errorId = error ? `${fieldId}-error` : undefined;\n const describedBy = [helpId, errorId].filter(Boolean).join(\" \") || undefined;\n const invalid = error !== undefined && error !== null && error !== \"\";\n\n return {\n id: fieldId,\n describedBy,\n invalid,\n };\n}\n\nexport function createFieldId(name: string): string {\n return `builder-field-${name.replace(/[^A-Za-z0-9_-]+/g, \"-\")}`;\n}\n\nfunction classNames(...names: Array<string | undefined>): string {\n return names.filter((name): name is string => Boolean(name)).join(\" \");\n}\n","import type { ReactNode } from \"react\";\n\nexport interface FieldProps {\n label: ReactNode;\n children: ReactNode;\n}\n\nexport function Field({ label, children }: FieldProps) {\n return (\n <label className=\"grid min-w-0 gap-1 text-2xs font-medium uppercase tracking-wide text-fg-muted\">\n {label}\n {children}\n </label>\n );\n}\n","import { Field } from \"./Field\";\nimport { Select } from \"./inputs\";\n\nexport interface AlignSelectProps {\n name: string;\n value: string;\n label?: string;\n onChange: (value: string) => void;\n}\n\nexport function AlignSelect({ name, value, label = \"Align\", onChange }: AlignSelectProps) {\n return (\n <Field label={label}>\n <Select\n className=\"w-full\"\n name={name}\n value={value}\n onChange={(event) => onChange(event.currentTarget.value)}\n >\n <option value=\"\" />\n <option value=\"left\">left</option>\n <option value=\"center\">center</option>\n <option value=\"right\">right</option>\n </Select>\n </Field>\n );\n}\n","import type { ButtonHTMLAttributes, ReactNode } from \"react\";\n\nexport type ButtonVariant = \"default\" | \"primary\" | \"danger\" | \"ghost\";\n\nconst base =\n \"inline-flex h-8 cursor-pointer items-center rounded-md border border-solid font-medium transition-colors disabled:cursor-not-allowed focus-visible:outline-none focus-visible:ring-3 focus-visible:ring-accent/20\";\n\nconst variantClass: Record<ButtonVariant, string> = {\n default:\n \"border-border bg-surface text-fg hover:border-border-strong hover:bg-surface-muted focus-visible:border-accent disabled:bg-surface disabled:text-fg-subtle\",\n primary:\n \"border-primary bg-primary font-semibold text-on-dark hover:border-primary-strong hover:bg-primary-strong disabled:border-border-strong disabled:bg-surface-muted disabled:text-fg-subtle\",\n danger:\n \"border-danger bg-danger-soft text-danger hover:border-danger hover:bg-danger hover:text-on-dark\",\n ghost:\n \"border-transparent bg-transparent text-fg hover:border-border hover:bg-surface-muted focus-visible:border-accent\",\n};\n\nexport interface ButtonProps extends ButtonHTMLAttributes<HTMLButtonElement> {\n variant?: ButtonVariant;\n /** Render as a square icon button (no label padding). */\n icon?: boolean;\n children: ReactNode;\n}\n\nexport function Button({\n variant = \"default\",\n icon = false,\n type = \"button\",\n className,\n children,\n ...rest\n}: ButtonProps) {\n const layout = icon ? \"w-8 justify-center p-0\" : \"gap-2 whitespace-nowrap px-3\";\n const classes = [base, layout, variantClass[variant], className].filter(Boolean).join(\" \");\n\n return (\n <button type={type} className={classes} {...rest}>\n {children}\n </button>\n );\n}\n\nexport function AddButton({\n type = \"button\",\n className,\n children,\n ...rest\n}: ButtonHTMLAttributes<HTMLButtonElement>) {\n const classes = [\n \"h-7 w-fit cursor-pointer rounded-md border border-dashed border-border-strong bg-transparent px-3 text-2xs text-fg-muted transition-colors hover:border-accent hover:text-accent disabled:cursor-not-allowed disabled:opacity-50\",\n className,\n ]\n .filter(Boolean)\n .join(\" \");\n\n return (\n <button type={type} className={classes} {...rest}>\n {children}\n </button>\n );\n}\n","import { useDraggable } from \"@dnd-kit/core\";\nimport { Chip } from \"../primitives/Chip\";\nimport { getBlockChrome } from \"./blockChrome\";\n\nexport interface BlockPaletteProps {\n blockTypes: string[];\n onAdd?: (type: string) => void;\n}\n\nexport function BlockPalette({ blockTypes, onAdd }: BlockPaletteProps) {\n return (\n <div\n className=\"flex min-w-0 gap-1 overflow-x-auto [scrollbar-width:thin]\"\n aria-label=\"Block palette\"\n >\n {blockTypes.map((type) => (\n <PaletteItem key={type} type={type} onAdd={onAdd} />\n ))}\n </div>\n );\n}\n\ninterface PaletteItemProps {\n type: string;\n onAdd?: (type: string) => void;\n}\n\nfunction PaletteItem({ type, onAdd }: PaletteItemProps) {\n const chrome = getBlockChrome(type);\n const { attributes, listeners, setNodeRef, transform, isDragging } = useDraggable({\n id: `palette:${type}`,\n data: {\n source: \"palette\",\n type,\n },\n });\n\n return (\n <button\n ref={setNodeRef}\n type=\"button\"\n className=\"flex flex-none cursor-grab items-center gap-3 m-0 min-h-8 rounded-md border border-solid border-transparent bg-transparent px-2 py-1 text-left text-xs font-medium text-fg transition-colors hover:border-border hover:bg-surface-muted active:scale-[0.98] active:cursor-grabbing active:bg-surface-muted\"\n style={{\n transform: transform ? `translate3d(${transform.x}px, ${transform.y}px, 0)` : undefined,\n opacity: isDragging ? 0 : undefined,\n }}\n onClick={() => onAdd?.(type)}\n aria-label={`Add ${chrome.label}`}\n {...listeners}\n {...attributes}\n >\n <Chip>{chrome.chip}</Chip>\n <span>{chrome.label}</span>\n </button>\n );\n}\n","import {\n useBuilderActions,\n useBuilderState,\n type TemplateExample,\n} from \"../context/BuilderContext\";\nimport { Select } from \"../controls\";\nimport { Button } from \"../primitives/Button\";\nimport { BlockPalette } from \"./BlockPalette\";\n\nexport interface PaletteProps {\n className?: string;\n /** Loadable examples keyed by display name. The Load control only renders when non-empty. */\n examples?: Record<string, TemplateExample>;\n}\n\nexport function Palette({ className, examples }: PaletteProps = {}) {\n const { blockTypes, schema } = useBuilderState();\n const { addBlock, loadExample } = useBuilderActions();\n const exampleEntries = examples ? Object.entries(examples) : [];\n\n return (\n <aside\n className={`flex h-[56px] min-w-0 items-center overflow-hidden border-0 border-b border-solid border-border bg-surface px-4${className ? ` ${className}` : \"\"}`}\n aria-label=\"Block palette\"\n >\n <div className=\"flex w-full min-w-0 items-center gap-3\">\n <h2 className=\"m-0 flex-none text-2xs font-medium uppercase tracking-[0.06em] text-fg-subtle\">\n Blocks\n </h2>\n <BlockPalette blockTypes={blockTypes} onAdd={addBlock} />\n {exampleEntries.length > 0 ? (\n <div className=\"ml-auto flex-none\">\n <ExampleLoader entries={exampleEntries} disabled={!schema} onLoad={loadExample} />\n </div>\n ) : null}\n </div>\n </aside>\n );\n}\n\ninterface ExampleLoaderProps {\n entries: [string, TemplateExample][];\n disabled: boolean;\n onLoad: (example: TemplateExample) => void;\n}\n\nfunction ExampleLoader({ entries, disabled, onLoad }: ExampleLoaderProps) {\n if (entries.length === 1) {\n const [, example] = entries[0];\n\n return (\n <Button onClick={() => onLoad(example)} disabled={disabled}>\n Load example\n </Button>\n );\n }\n\n return (\n <Select\n className=\"w-auto py-0\"\n value=\"\"\n disabled={disabled}\n aria-label=\"Load example\"\n onChange={(event) => {\n const match = entries.find(([name]) => name === event.currentTarget.value);\n\n if (match) {\n onLoad(match[1]);\n }\n }}\n >\n <option value=\"\" disabled>\n Load example…\n </option>\n {entries.map(([name]) => (\n <option key={name} value={name}>\n {name}\n </option>\n ))}\n </Select>\n );\n}\n","const CSS_PX_PER_INCH = 96;\nconst MM_PER_INCH = 25.4;\n\nexport function mmToPx(mm: number): number {\n return Math.round((mm / MM_PER_INCH) * CSS_PX_PER_INCH);\n}\n","import type { Orientation, PageFormat } from \"../../types/generated/template\";\n\nexport const PAGE_SIZES_MM: Record<PageFormat, [number, number]> = {\n A3: [297, 420],\n A4: [210, 297],\n A5: [148, 210],\n A6: [105, 148],\n Letter: [215.9, 279.4],\n Legal: [215.9, 355.6],\n Tabloid: [279.4, 431.8],\n};\n\nexport function pageSizeForFormat(\n format: PageFormat | string | undefined,\n orientation: Orientation | string | undefined = \"portrait\",\n): [number, number] {\n const portraitSize = PAGE_SIZES_MM[(format ?? \"A4\") as PageFormat] ?? PAGE_SIZES_MM.A4;\n\n return orientation === \"landscape\" ? [portraitSize[1], portraitSize[0]] : portraitSize;\n}\n","const COLUMN_GAP = \"0.375rem\";\nconst MIN_WIDTH = 5;\nconst TOTAL_WIDTH = 100;\n\nexport function parseWidths(\n widths: readonly (number | string)[] | null | undefined,\n count: number,\n): number[] {\n if (count <= 0) {\n return [];\n }\n\n if (!widths || widths.length !== count) {\n return equalWidths(count);\n }\n\n const parsed = widths.map(parseWidth);\n\n return parsed.every((width) => Number.isFinite(width)) ? parsed : equalWidths(count);\n}\n\nexport function formatWidths(widths: readonly number[]): string[] {\n return widths.map((width) => `${width}%`);\n}\n\nexport function gridTemplateForWidths(\n widths: readonly string[] | null | undefined,\n count: number,\n): string | null {\n if (!widths) {\n return null;\n }\n\n const tracks = parseWidths(widths, count).map((width) => `minmax(0, ${width}fr)`);\n\n return tracks.join(` ${COLUMN_GAP} `);\n}\n\nexport function setBoundary(\n widths: readonly number[],\n leftIndex: number,\n leftPercent: number,\n): number[] {\n const rightIndex = leftIndex + 1;\n\n if (leftIndex < 0 || rightIndex >= widths.length) {\n return [...widths];\n }\n\n const pairTotal = widths[leftIndex] + widths[rightIndex];\n const minLeft = Math.min(MIN_WIDTH, pairTotal);\n const maxLeft = Math.max(minLeft, pairTotal - MIN_WIDTH);\n const nextLeft = Math.round(clamp(leftPercent, minLeft, maxLeft));\n const nextWidths = [...widths];\n\n nextWidths[leftIndex] = nextLeft;\n nextWidths[rightIndex] = pairTotal - nextLeft;\n\n return nextWidths;\n}\n\nexport function labelWidthPercent(width: string | null | undefined, fallback = 30): number {\n const parsed = width == null ? Number.NaN : parseWidth(width);\n const value = Number.isFinite(parsed) ? parsed : fallback;\n\n return Math.round(clamp(value, MIN_WIDTH, TOTAL_WIDTH - MIN_WIDTH));\n}\n\nexport const NUMBER_COLUMN_RESERVE = 5;\n\nexport interface TableColumnTracks {\n tracks: number[];\n data: number[];\n}\n\nexport function percentWidth(width: string | number | null | undefined): number | null {\n if (width == null) {\n return null;\n }\n\n const parsed = parseWidth(width);\n\n return Number.isFinite(parsed) ? parsed : null;\n}\n\nexport function distribute(count: number, total: number): number[] {\n if (count <= 0) {\n return [];\n }\n\n const base = Math.floor(total / count);\n const widths = Array.from({ length: count }, () => base);\n widths[count - 1] += total - base * count;\n\n return widths;\n}\n\nexport function tableColumnTracks(\n widths: readonly (string | null | undefined)[],\n numberRows: boolean,\n): TableColumnTracks {\n const available = numberRows ? TOTAL_WIDTH - NUMBER_COLUMN_RESERVE : TOTAL_WIDTH;\n const parsed = widths.map((width) => percentWidth(width));\n const data =\n widths.length > 0 && parsed.every((value): value is number => value !== null)\n ? parsed\n : distribute(widths.length, available);\n const tracks = numberRows ? [NUMBER_COLUMN_RESERVE, ...data] : [...data];\n\n return { tracks, data };\n}\n\nfunction equalWidths(count: number): number[] {\n return distribute(count, TOTAL_WIDTH);\n}\n\nfunction parseWidth(width: number | string): number {\n if (typeof width === \"number\") {\n return width;\n }\n\n const percentMatch = width.trim().match(/^([+-]?\\d+(?:\\.\\d+)?)%$/);\n\n return percentMatch ? Number.parseFloat(percentMatch[1] ?? \"\") : Number.NaN;\n}\n\nfunction clamp(value: number, min: number, max: number): number {\n return Math.min(Math.max(value, min), max);\n}\n","import type { RefObject } from \"react\";\nimport { formatWidths, parseWidths, setBoundary } from \"./columns\";\n\nexport interface ColumnResizerProps {\n widths: readonly string[] | null;\n count: number;\n leftIndex: number;\n containerRef: RefObject<HTMLElement | null>;\n onResize: (widths: string[]) => void;\n label?: string;\n}\n\nexport function ColumnResizer({\n widths,\n count,\n leftIndex,\n containerRef,\n onResize,\n label,\n}: ColumnResizerProps) {\n function handlePointerDown(event: React.PointerEvent<HTMLButtonElement>): void {\n const bounds = containerRef.current?.getBoundingClientRect();\n\n if (!bounds || bounds.width <= 0) {\n return;\n }\n\n const containerBounds = bounds;\n\n event.preventDefault();\n event.currentTarget.setPointerCapture(event.pointerId);\n\n const parsedWidths = parseWidths(widths, count);\n const pairStart = parsedWidths.slice(0, leftIndex).reduce((total, width) => total + width, 0);\n\n function handlePointerMove(pointerEvent: PointerEvent): void {\n const pointerPercent =\n ((pointerEvent.clientX - containerBounds.left) / containerBounds.width) * 100;\n const leftPercent = pointerPercent - pairStart;\n\n onResize(formatWidths(setBoundary(parsedWidths, leftIndex, leftPercent)));\n }\n\n function handlePointerUp(): void {\n window.removeEventListener(\"pointermove\", handlePointerMove);\n window.removeEventListener(\"pointerup\", handlePointerUp);\n }\n\n window.addEventListener(\"pointermove\", handlePointerMove);\n window.addEventListener(\"pointerup\", handlePointerUp);\n }\n\n return (\n <button\n type=\"button\"\n className=\"group/resizer min-h-full w-1.5 cursor-col-resize select-none self-stretch rounded-full border-0 bg-transparent p-0 transition-colors max-[480px]:hidden\"\n aria-label={label ?? `Resize columns ${leftIndex + 1} and ${leftIndex + 2}`}\n onPointerDown={handlePointerDown}\n >\n <span className=\"mx-auto block h-full w-0.5 rounded-full bg-border-strong transition-colors group-hover/resizer:bg-accent\" />\n </button>\n );\n}\n","import type { CSSProperties, ReactNode } from \"react\";\nimport type { Orientation, PageFormat } from \"../../types/generated/template\";\nimport { mmToPx } from \"../lib/displayScale\";\nimport { pageSizeForFormat } from \"../lib/pageSizes\";\n\nexport interface PageSheetProps {\n format: PageFormat;\n orientation: Orientation;\n children: ReactNode;\n /** Show the \"{format} · {orientation} / {width}mm\" caption. Off for the footer sheet. */\n showMeta?: boolean;\n}\n\nexport function PageSheet({ format, orientation, children, showMeta = true }: PageSheetProps) {\n const [widthMm] = pageSizeForFormat(format, orientation);\n const style: CSSProperties = { maxWidth: `${mmToPx(widthMm)}px` };\n\n return (\n <div\n // The page mirrors the white PDF the backend renders, so it stays light\n // even in dark mode (the surrounding canvas/chrome still go dark).\n data-theme=\"light\"\n className=\"mx-auto grid w-full gap-3 rounded-xl border border-solid border-border bg-page p-6 shadow-page transition-[max-width] duration-200\"\n style={style}\n >\n {showMeta ? (\n <div className=\"mb-2 flex items-center justify-between text-2xs uppercase tracking-[0.06em] text-fg-subtle\">\n <span>\n {format} · {orientation}\n </span>\n <span>{Math.round(widthMm)}mm</span>\n </div>\n ) : null}\n {children}\n </div>\n );\n}\n","import type {\n Block,\n BlockConfig,\n DividerConfig,\n HeadingConfig,\n ImageConfig,\n KeyValueConfig,\n SpacingConfig,\n SpacerConfig,\n TableConfig,\n Template,\n TemplateConfig,\n TypographyConfig,\n} from \"../../types/generated/template\";\n\ntype EmptyConfigValue = \"\" | null | undefined;\ntype BlockType = Block[\"type\"];\ntype BlockByType<TType extends BlockType> = Extract<Block, { type: TType }>;\ntype BlockConfigByType = {\n text: BlockConfig;\n html: BlockConfig;\n heading: HeadingConfig;\n image: ImageConfig;\n \"key-value\": KeyValueConfig;\n spacer: SpacerConfig;\n divider: DividerConfig;\n table: TableConfig;\n};\ntype BlockConfigFor<TType extends BlockType> = BlockConfigByType[TType];\n\nexport function setBlockConfigField<\n TType extends BlockType,\n TKey extends keyof BlockConfigFor<TType>,\n>(\n block: BlockByType<TType>,\n field: TKey,\n value: BlockConfigFor<TType>[TKey] | EmptyConfigValue,\n): BlockByType<TType> {\n const nextConfig = setObjectField<BlockConfigFor<TType>, TKey>(\n block.config as BlockConfigFor<TType> | undefined,\n field,\n value,\n );\n\n return withBlockConfig(block, nextConfig);\n}\n\nexport function setBlockTypographyField<\n TType extends BlockType,\n TKey extends keyof TypographyConfig,\n>(\n block: BlockByType<TType>,\n field: TKey,\n value: TypographyConfig[TKey] | EmptyConfigValue,\n): BlockByType<TType> {\n const nextTypography = setObjectField(block.config?.typography, field, value);\n\n return setCommonBlockConfigField(block, \"typography\", nextTypography);\n}\n\nexport function setBlockSpacingField<TType extends BlockType, TKey extends keyof SpacingConfig>(\n block: BlockByType<TType>,\n field: TKey,\n value: SpacingConfig[TKey] | EmptyConfigValue,\n): BlockByType<TType> {\n const nextSpacing = setObjectField(block.config?.spacing, field, value);\n\n return setCommonBlockConfigField(block, \"spacing\", nextSpacing);\n}\n\nexport function setTemplateTypographyField<TKey extends keyof TypographyConfig>(\n template: Template,\n field: TKey,\n value: TypographyConfig[TKey] | EmptyConfigValue,\n): Template {\n const nextTypography = setObjectField(template.config?.typography, field, value);\n const nextConfig = setObjectField(template.config, \"typography\", nextTypography);\n\n return withTemplateConfig(template, nextConfig);\n}\n\nexport function setTemplatePageMargin<TKey extends keyof SpacingConfig>(\n template: Template,\n field: TKey,\n value: SpacingConfig[TKey] | EmptyConfigValue,\n): Template {\n const nextMargins = setObjectField(template.config?.page?.margins, field, value);\n const nextPage = setObjectField(template.config?.page, \"margins\", nextMargins);\n const nextConfig = setObjectField(template.config, \"page\", nextPage);\n\n return withTemplateConfig(template, nextConfig);\n}\n\nfunction setObjectField<TObject extends object, TKey extends keyof TObject>(\n source: TObject | undefined,\n field: TKey,\n value: TObject[TKey] | EmptyConfigValue,\n): TObject | undefined {\n const next = { ...source } as TObject;\n\n if (isClearedConfigValue(value)) {\n delete next[field];\n } else {\n next[field] = value;\n }\n\n return pruneEmptyObject(next);\n}\n\nfunction withBlockConfig<TType extends BlockType>(\n block: BlockByType<TType>,\n config: BlockConfigFor<TType> | undefined,\n): BlockByType<TType> {\n if (config === undefined) {\n const { config: _config, ...rest } = block;\n\n return rest as BlockByType<TType>;\n }\n\n return { ...block, config } as BlockByType<TType>;\n}\n\nfunction setCommonBlockConfigField<TType extends BlockType, TKey extends keyof BlockConfig>(\n block: BlockByType<TType>,\n field: TKey,\n value: BlockConfig[TKey] | EmptyConfigValue,\n): BlockByType<TType> {\n const nextConfig = setObjectField<BlockConfig, TKey>(block.config, field, value);\n\n return withBlockConfig(block, nextConfig as BlockConfigFor<TType> | undefined);\n}\n\nfunction withTemplateConfig(template: Template, config: TemplateConfig | undefined): Template {\n if (config === undefined) {\n const { config: _config, ...rest } = template;\n\n return rest;\n }\n\n return { ...template, config };\n}\n\nfunction pruneEmptyObject<TObject extends object>(value: TObject): TObject | undefined {\n return Object.keys(value).length === 0 ? undefined : value;\n}\n\nfunction isClearedConfigValue(value: unknown): value is EmptyConfigValue {\n return value === \"\" || value === null || value === undefined;\n}\n","import { useRef, type ReactNode } from \"react\";\nimport type {\n Block,\n DividerBlock,\n HeadingBlock,\n HtmlBlock,\n ImageBlock,\n KeyValueBlock,\n KeyValueValues,\n SpacerBlock,\n TableBlock,\n TextBlock,\n} from \"../../types/generated/template\";\nimport { isRecord } from \"../lib/records\";\nimport { setBlockConfigField } from \"../state/configUpdates\";\nimport { ColumnResizer } from \"./ColumnResizer\";\nimport { formatWidths, labelWidthPercent, tableColumnTracks } from \"./columns\";\n\nexport interface BlockDataPreviewProps {\n block: Block;\n rowData?: unknown;\n onChange?: (block: Block) => void;\n}\n\nconst copyClass =\n \"m-0 line-clamp-3 overflow-hidden text-sm leading-[1.45] text-fg-muted [display:-webkit-box] [-webkit-box-orient:vertical]\";\nconst headingCopyClass =\n \"m-0 line-clamp-2 overflow-hidden text-[17px] font-semibold leading-[1.25] text-fg [display:-webkit-box] [-webkit-box-orient:vertical]\";\n\nfunction EmptyPreview({ children }: { children: ReactNode }) {\n return <p className=\"m-0 text-sm text-fg-subtle\">{children}</p>;\n}\n\nexport function BlockDataPreview({ block, rowData, onChange }: BlockDataPreviewProps) {\n switch (block.type) {\n case \"heading\":\n return <TextPreview block={block} variant=\"heading\" />;\n case \"text\":\n return <TextPreview block={block} variant=\"text\" />;\n case \"html\":\n return <HtmlPreview block={block} />;\n case \"image\":\n return <ImagePreview block={block} />;\n case \"key-value\":\n return <KeyValuePreview block={block} rowData={rowData} onChange={onChange} />;\n case \"table\":\n return <TablePreview block={block} rowData={rowData} onChange={onChange} />;\n case \"spacer\":\n return <SpacerPreview block={block} />;\n case \"divider\":\n return <DividerPreview block={block} />;\n default:\n return null;\n }\n}\n\nfunction TextPreview({\n block,\n variant,\n}: {\n block: TextBlock | HeadingBlock;\n variant: \"text\" | \"heading\";\n}) {\n const text = block.text.trim();\n\n if (text.length === 0) {\n return <EmptyPreview>No text yet</EmptyPreview>;\n }\n\n return <p className={variant === \"heading\" ? headingCopyClass : copyClass}>{text}</p>;\n}\n\nfunction HtmlPreview({ block }: { block: HtmlBlock }) {\n const text = stripMarkup(block.html).trim();\n\n if (text.length === 0) {\n return <EmptyPreview>No HTML content yet</EmptyPreview>;\n }\n\n return <p className={copyClass}>{text}</p>;\n}\n\nfunction ImagePreview({ block }: { block: ImageBlock }) {\n const src = block.src.trim();\n\n if (src.length === 0) {\n return <EmptyPreview>No image selected</EmptyPreview>;\n }\n\n return (\n <div className=\"grid min-h-[72px] place-items-center overflow-hidden rounded-md border border-solid border-border bg-surface-muted\">\n <img\n className=\"block max-h-[120px] max-w-full object-contain\"\n src={src}\n alt={block.alt ?? \"\"}\n />\n </div>\n );\n}\n\nfunction KeyValuePreview({\n block,\n rowData,\n onChange,\n}: {\n block: KeyValueBlock;\n rowData?: unknown;\n onChange?: (block: Block) => void;\n}) {\n const listRef = useRef<HTMLDListElement | null>(null);\n const fields = block.config?.fields ?? [];\n const values = mergeRecordValues(block.values, rowData);\n const entries =\n fields.length > 0\n ? fields.map((field) => ({\n key: field.key,\n label: field.label || field.key,\n value: stringifyPreviewValue(values[field.key]),\n }))\n : Object.entries(values).map(([key, value]) => ({\n key,\n label: key,\n value: stringifyPreviewValue(value),\n }));\n\n if (entries.length === 0) {\n return <EmptyPreview>No fields yet</EmptyPreview>;\n }\n\n const labelPercent = labelWidthPercent(block.config?.labelWidth);\n const columnsStyle = { gridTemplateColumns: `${labelPercent}% minmax(0, 1fr)` };\n\n return (\n <dl ref={listRef} className=\"relative m-0 grid gap-0.5\">\n {entries.slice(0, 5).map((entry) => (\n <div key={entry.key} className=\"grid min-w-0 items-baseline\" style={columnsStyle}>\n <dt className=\"min-w-0 overflow-hidden text-ellipsis whitespace-nowrap pr-2 text-2xs text-fg-subtle\">\n {entry.label}\n </dt>\n <dd className=\"m-0 min-w-0 overflow-hidden text-ellipsis whitespace-nowrap pl-2 text-sm text-fg\">\n {entry.value || \"—\"}\n </dd>\n </div>\n ))}\n {onChange ? (\n <div\n className=\"absolute inset-y-0 flex -translate-x-1/2\"\n style={{ left: `${labelPercent}%` }}\n >\n <ColumnResizer\n widths={formatWidths([labelPercent, 100 - labelPercent])}\n count={2}\n leftIndex={0}\n containerRef={listRef}\n label=\"Resize the label column\"\n onResize={(widths) => onChange(setBlockConfigField(block, \"labelWidth\", widths[0]))}\n />\n </div>\n ) : null}\n </dl>\n );\n}\n\nfunction TablePreview({\n block,\n rowData,\n onChange,\n}: {\n block: TableBlock;\n rowData?: unknown;\n onChange?: (block: Block) => void;\n}) {\n const tableRef = useRef<HTMLDivElement | null>(null);\n const columns = block.config?.columns ?? [];\n const numberRows = block.config?.numberRows === true;\n const rows = Array.isArray(rowData) ? rowData.filter(isRecord) : [];\n\n if (columns.length === 0) {\n return <EmptyPreview>No columns yet</EmptyPreview>;\n }\n\n const { tracks, data } = tableColumnTracks(\n columns.map((column) => column.width),\n numberRows,\n );\n const trackStrings = formatWidths(tracks);\n const trackOffset = numberRows ? 1 : 0;\n\n return (\n <div\n ref={tableRef}\n className=\"relative min-w-0 overflow-hidden rounded-md border border-solid border-border\"\n >\n <table className=\"w-full table-fixed border-collapse text-2xs\">\n <colgroup>\n {numberRows ? <col style={{ width: `${tracks[0]}%` }} /> : null}\n {data.map((width, index) => (\n <col key={columns[index].key} style={{ width: `${width}%` }} />\n ))}\n </colgroup>\n <thead>\n <tr>\n {numberRows ? (\n <th className=\"border-0 border-b border-solid border-border bg-surface-muted px-2 py-[5px] text-left font-semibold text-fg-subtle\">\n #\n </th>\n ) : null}\n {columns.map((column) => (\n <th\n key={column.key}\n className=\"min-w-0 overflow-hidden text-ellipsis whitespace-nowrap border-0 border-b border-solid border-border bg-surface-muted px-2 py-[5px] text-left font-semibold text-fg\"\n >\n {column.label || column.key}\n </th>\n ))}\n </tr>\n </thead>\n <tbody>\n {rows.length > 0 ? (\n rows.map((row, rowIndex) => (\n <tr key={tableRowPreviewKey(row, columns)} className=\"last:[&>td]:border-b-0\">\n {numberRows ? (\n <td className=\"border-0 border-b border-solid border-border px-2 py-[5px] text-left text-fg-subtle\">\n {rowIndex + 1}\n </td>\n ) : null}\n {columns.map((column) => (\n <td\n key={column.key}\n className=\"min-w-0 overflow-hidden text-ellipsis whitespace-nowrap border-0 border-b border-solid border-border px-2 py-[5px] text-left text-fg-muted\"\n >\n {stringifyPreviewValue(row[column.key]) || \"—\"}\n </td>\n ))}\n </tr>\n ))\n ) : (\n <tr>\n <td\n colSpan={columns.length + trackOffset}\n className=\"min-w-0 overflow-hidden text-ellipsis whitespace-nowrap border-0 px-2 py-[5px] text-left text-fg-muted\"\n >\n No runtime rows yet\n </td>\n </tr>\n )}\n </tbody>\n </table>\n {onChange\n ? columns.slice(0, -1).map((column, index) => {\n const leftIndex = trackOffset + index;\n const boundaryPercent = tracks\n .slice(0, leftIndex + 1)\n .reduce((total, value) => total + value, 0);\n\n return (\n <div\n key={`${column.key}:resizer`}\n className=\"absolute inset-y-0 flex -translate-x-1/2\"\n style={{ left: `${boundaryPercent}%` }}\n >\n <ColumnResizer\n widths={trackStrings}\n count={tracks.length}\n leftIndex={leftIndex}\n containerRef={tableRef}\n label={`Resize column ${index + 1}`}\n onResize={(next) =>\n onChange(\n setBlockConfigField(\n block,\n \"columns\",\n columns.map((current, columnIndex) => ({\n ...current,\n width: (numberRows ? next.slice(1) : next)[columnIndex],\n })),\n ),\n )\n }\n />\n </div>\n );\n })\n : null}\n </div>\n );\n}\n\nfunction SpacerPreview({ block }: { block: SpacerBlock }) {\n const height = block.config?.height;\n\n return (\n <div className=\"grid h-9 place-items-center rounded-md border border-dashed border-border-strong text-2xs text-fg-subtle\">\n {typeof height === \"number\" ? `${height}mm spacer` : \"Spacer\"}\n </div>\n );\n}\n\nfunction DividerPreview({ block }: { block: DividerBlock }) {\n const style = block.config?.style ?? \"solid\";\n const styleClass =\n style === \"dashed\"\n ? \"border-dashed\"\n : style === \"dotted\"\n ? \"border-dotted\"\n : style === \"double\"\n ? \"border-double border-t-[3px]\"\n : style === \"none\"\n ? \"border-t-transparent\"\n : \"border-solid\";\n\n return <div className={`my-2 border-0 border-t border-border-strong ${styleClass}`} />;\n}\n\nfunction mergeRecordValues(\n values: KeyValueValues | undefined,\n rowData: unknown,\n): Record<string, unknown> {\n return Object.assign({}, values, isRecord(rowData) ? rowData : undefined);\n}\n\nfunction tableRowPreviewKey(\n row: Record<string, unknown>,\n columns: NonNullable<TableBlock[\"config\"]>[\"columns\"],\n): string {\n return columns?.map((column) => stringifyPreviewValue(row[column.key])).join(\"|\") ?? \"\";\n}\n\nfunction stringifyPreviewValue(value: unknown): string {\n if (value === null || value === undefined) {\n return \"\";\n }\n\n if (typeof value === \"string\") {\n return value;\n }\n\n if (typeof value === \"number\" || typeof value === \"boolean\") {\n return String(value);\n }\n\n return \"\";\n}\n\nfunction stripMarkup(value: string): string {\n return value.replace(/<[^>]*>/g, \" \").replace(/\\s+/g, \" \");\n}\n","import { useSortable } from \"@dnd-kit/sortable\";\nimport { CSS } from \"@dnd-kit/utilities\";\nimport type { CSSProperties, KeyboardEvent, MouseEvent } from \"react\";\nimport type { Block } from \"../../types/generated/template\";\nimport type { EditorArea, EditorBlock } from \"../state/editorModel\";\nimport type { TemplateData } from \"../../types/template\";\nimport { BlockDataPreview } from \"./BlockDataPreview\";\n\nexport interface SortableBlockProps {\n rowUid: string;\n area: EditorArea;\n editorBlock: EditorBlock;\n data: TemplateData;\n selected: boolean;\n onRemoveBlock: (blockUid: string) => void;\n onSelect: (blockUid: string) => void;\n onChangeBlock: (blockUid: string, block: Block) => void;\n style?: CSSProperties;\n}\n\nexport function SortableBlock({\n rowUid,\n area,\n editorBlock,\n data,\n selected,\n onRemoveBlock,\n onSelect,\n onChangeBlock,\n style: layoutStyle,\n}: SortableBlockProps) {\n const {\n attributes,\n listeners,\n setActivatorNodeRef,\n setNodeRef,\n transform,\n transition,\n isDragging,\n } = useSortable({\n id: editorBlock.uid,\n data: {\n type: \"block\",\n rowUid,\n blockUid: editorBlock.uid,\n area,\n },\n });\n\n const blockId = typeof editorBlock.block.id === \"string\" ? editorBlock.block.id : null;\n const rowData = blockId ? data[blockId] : undefined;\n const classes = [\n \"group/card relative grid min-w-0 cursor-pointer overflow-hidden rounded-lg border border-solid bg-surface transition-[border-color,box-shadow,background] hover:border-border-strong\",\n isDragging ? \"border-dashed border-border opacity-45\" : \"border-border\",\n selected ? \"!border-accent ring-2 ring-accent-soft\" : \"\",\n ]\n .filter(Boolean)\n .join(\" \");\n const style: CSSProperties = {\n transform: CSS.Transform.toString(transform),\n transition,\n ...layoutStyle,\n };\n\n function selectBlock() {\n onSelect(editorBlock.uid);\n }\n\n function handleCardKeyDown(event: KeyboardEvent<HTMLElement>) {\n if (event.target !== event.currentTarget) {\n return;\n }\n if (event.key === \"Enter\" || event.key === \" \") {\n event.preventDefault();\n selectBlock();\n }\n }\n\n function handleRemove(event: MouseEvent) {\n event.stopPropagation();\n onRemoveBlock(editorBlock.uid);\n }\n\n const chromeRevealedClass = selected || isDragging ? \"opacity-100\" : \"opacity-0\";\n\n return (\n <article\n ref={setNodeRef}\n className={classes}\n style={style}\n aria-current={selected}\n tabIndex={0}\n onClick={selectBlock}\n onKeyDown={handleCardKeyDown}\n >\n <div\n className={`pointer-events-none absolute inset-x-2 top-2 z-[1] flex items-center justify-between gap-1 transition-opacity group-hover/card:opacity-100 group-focus-within/card:opacity-100 ${chromeRevealedClass}`}\n >\n <button\n ref={setActivatorNodeRef}\n type=\"button\"\n className=\"pointer-events-auto inline-grid h-[22px] w-[22px] flex-none cursor-grab place-items-center rounded border border-solid border-border bg-surface/90 p-0 font-mono text-xs text-fg-subtle transition-[background,border-color,color] hover:border-border-strong hover:bg-surface hover:text-fg active:cursor-grabbing\"\n aria-label=\"Drag to move block\"\n onClick={(event) => event.stopPropagation()}\n {...attributes}\n {...listeners}\n >\n ⋮⋮\n </button>\n <div className=\"pointer-events-auto inline-flex flex-none items-center gap-1\">\n <button\n type=\"button\"\n className=\"inline-grid h-[22px] w-[22px] cursor-pointer place-items-center rounded border border-solid border-border bg-surface/90 p-0 text-[13px] text-fg-muted transition-[background,border-color,color] hover:border-border-strong hover:bg-danger-soft hover:text-danger\"\n aria-label=\"Remove block\"\n onClick={handleRemove}\n >\n ✕\n </button>\n </div>\n </div>\n <div className=\"grid min-w-0 bg-surface p-3\">\n <BlockDataPreview\n block={editorBlock.block}\n rowData={rowData}\n onChange={(block) => onChangeBlock(editorBlock.uid, block)}\n />\n </div>\n </article>\n );\n}\n","import { useDroppable } from \"@dnd-kit/core\";\nimport { SortableContext, useSortable, verticalListSortingStrategy } from \"@dnd-kit/sortable\";\nimport { CSS } from \"@dnd-kit/utilities\";\nimport type { CSSProperties, RefObject } from \"react\";\nimport { Fragment, useRef } from \"react\";\nimport type { Block, Orientation, PageFormat } from \"../../types/generated/template\";\nimport type { TemplateData } from \"../../types/template\";\nimport type {\n EditorArea,\n EditorModel,\n EditorRow,\n PageNumbersValue,\n} from \"../state/editorModel\";\nimport { Checkbox, Select } from \"../controls\";\nimport { mmToPx } from \"../lib/displayScale\";\nimport { pageSizeForFormat } from \"../lib/pageSizes\";\nimport { ColumnResizer } from \"./ColumnResizer\";\nimport { gridTemplateForWidths } from \"./columns\";\nimport { PageSheet } from \"./PageSheet\";\nimport { SortableBlock } from \"./SortableBlock\";\n\nexport interface BuilderCanvasProps {\n model: EditorModel;\n data: TemplateData;\n format: PageFormat;\n orientation: Orientation;\n footerRepeat: boolean;\n pageNumbers: PageNumbersValue;\n selectedBlockUid: string | null;\n onRemoveBlock: (blockUid: string) => void;\n onSelectBlock: (blockUid: string) => void;\n onChangeBlock: (blockUid: string, block: Block) => void;\n onDeselect: () => void;\n onSetRowWidths: (rowUid: string, widths: string[]) => void;\n onToggleFooterRepeat: (repeat: boolean) => void;\n onChangePageNumbers: (value: PageNumbersValue) => void;\n className?: string;\n}\n\nexport function BuilderCanvas({\n model,\n data,\n format,\n orientation,\n footerRepeat,\n pageNumbers,\n selectedBlockUid,\n onRemoveBlock,\n onSelectBlock,\n onChangeBlock,\n onDeselect,\n onSetRowWidths,\n onToggleFooterRepeat,\n onChangePageNumbers,\n className,\n}: BuilderCanvasProps) {\n const [widthMm] = pageSizeForFormat(format, orientation);\n const pageWidth: CSSProperties = { maxWidth: `${mmToPx(widthMm)}px` };\n\n return (\n <div\n className={`grid content-start gap-4 min-w-0 min-h-0 overflow-auto bg-canvas px-4 pb-8 pt-6${className ? ` ${className}` : \"\"}`}\n onPointerDown={(event) => {\n if (event.target === event.currentTarget) {\n onDeselect();\n }\n }}\n >\n <PageSheet format={format} orientation={orientation}>\n <CanvasArea\n area=\"body\"\n rows={model.rows}\n data={data}\n selectedBlockUid={selectedBlockUid}\n newRowId=\"new-row\"\n emptyLabel=\"Drop a block here to begin\"\n fillLabel=\"Drop a block here to add a new row\"\n onRemoveBlock={onRemoveBlock}\n onSelectBlock={onSelectBlock}\n onChangeBlock={onChangeBlock}\n onSetRowWidths={onSetRowWidths}\n />\n </PageSheet>\n\n <PageSheet format={format} orientation={orientation} showMeta={false}>\n <div\n className=\"mb-2 flex items-center justify-between gap-3\"\n aria-label=\"Page footer settings\"\n >\n <h2 className=\"m-0 text-2xs font-semibold uppercase tracking-[0.06em] text-fg-subtle\">\n Footer\n </h2>\n <label className=\"inline-flex items-center gap-2 text-xs font-medium text-fg-muted\">\n <Checkbox\n checked={footerRepeat}\n onChange={(event) => onToggleFooterRepeat(event.currentTarget.checked)}\n />\n Repeat on every page\n </label>\n </div>\n\n <CanvasArea\n area=\"footer\"\n rows={model.footerRows}\n data={data}\n selectedBlockUid={selectedBlockUid}\n newRowId=\"new-footer-row\"\n emptyLabel=\"Drop a block here to start the footer\"\n fillLabel=\"Drop a block here to add a footer row\"\n onRemoveBlock={onRemoveBlock}\n onSelectBlock={onSelectBlock}\n onChangeBlock={onChangeBlock}\n onSetRowWidths={onSetRowWidths}\n />\n </PageSheet>\n\n <div className=\"mx-auto flex w-full justify-center\" style={pageWidth}>\n <label className=\"inline-flex items-center gap-3 text-2xs font-medium uppercase tracking-[0.06em] text-fg-muted\">\n Page numbers\n <Select\n className=\"w-auto py-0 text-sm font-normal normal-case tracking-normal\"\n value={pageNumbers}\n onChange={(event) =>\n onChangePageNumbers(event.currentTarget.value as PageNumbersValue)\n }\n >\n <option value=\"disabled\">Disabled</option>\n <option value=\"left\">Left</option>\n <option value=\"center\">Center</option>\n <option value=\"right\">Right</option>\n </Select>\n </label>\n </div>\n </div>\n );\n}\n\ninterface CanvasAreaProps {\n area: EditorArea;\n rows: EditorRow[];\n data: TemplateData;\n selectedBlockUid: string | null;\n newRowId: string;\n emptyLabel: string;\n fillLabel: string;\n onRemoveBlock: (blockUid: string) => void;\n onSelectBlock: (blockUid: string) => void;\n onChangeBlock: (blockUid: string, block: Block) => void;\n onSetRowWidths: (rowUid: string, widths: string[]) => void;\n}\n\nfunction CanvasArea({\n area,\n rows,\n data,\n selectedBlockUid,\n newRowId,\n emptyLabel,\n fillLabel,\n onRemoveBlock,\n onSelectBlock,\n onChangeBlock,\n onSetRowWidths,\n}: CanvasAreaProps) {\n const { setNodeRef: setNewRowRef, isOver: isNewRowOver } = useDroppable({\n id: newRowId,\n data: { type: \"new-row\", area },\n });\n\n return (\n <div className=\"grid gap-3\">\n <SortableContext\n items={rows.map((row) => row.uid)}\n strategy={verticalListSortingStrategy}\n >\n {rows.map((row) => (\n <CanvasRow\n key={row.uid}\n row={row}\n area={area}\n data={data}\n selectedBlockUid={selectedBlockUid}\n onRemoveBlock={onRemoveBlock}\n onSelectBlock={onSelectBlock}\n onChangeBlock={onChangeBlock}\n onSetRowWidths={onSetRowWidths}\n />\n ))}\n </SortableContext>\n\n <div\n ref={setNewRowRef}\n className={\n isNewRowOver\n ? \"grid min-h-10 place-items-center rounded-lg border border-dashed border-accent bg-accent-soft p-3 text-sm text-accent transition-[border-color,background,color]\"\n : \"grid min-h-10 place-items-center rounded-lg border border-dashed border-border-strong bg-transparent p-3 text-sm text-fg-subtle transition-[border-color,background,color]\"\n }\n >\n {rows.length === 0 ? emptyLabel : fillLabel}\n </div>\n </div>\n );\n}\n\ninterface CanvasRowProps {\n row: EditorRow;\n area: EditorArea;\n data: TemplateData;\n selectedBlockUid: string | null;\n onRemoveBlock: (blockUid: string) => void;\n onSelectBlock: (blockUid: string) => void;\n onChangeBlock: (blockUid: string, block: Block) => void;\n onSetRowWidths: (rowUid: string, widths: string[]) => void;\n}\n\nfunction CanvasRow({\n row,\n area,\n data,\n selectedBlockUid,\n onRemoveBlock,\n onSelectBlock,\n onChangeBlock,\n onSetRowWidths,\n}: CanvasRowProps) {\n const rowRef = useRef<HTMLDivElement | null>(null);\n const {\n attributes,\n listeners,\n setActivatorNodeRef,\n setNodeRef,\n transform,\n transition,\n isDragging,\n } = useSortable({\n id: row.uid,\n data: {\n type: \"row\",\n rowUid: row.uid,\n area,\n },\n });\n const widths = getRowWidths(row);\n const canResizeColumns = row.blocks.length > 1;\n const gridTemplateColumns = gridTemplateForWidths(\n canResizeColumns ? (widths ?? []) : widths,\n row.blocks.length,\n );\n const style: CSSProperties = {\n transform: CSS.Transform.toString(transform),\n transition,\n };\n\n return (\n <section\n ref={setNodeRef}\n className={\n isDragging\n ? \"group/row grid min-w-0 gap-2 rounded-lg opacity-50 transition-[background]\"\n : \"group/row grid min-w-0 gap-2 rounded-lg transition-[background]\"\n }\n style={style}\n >\n <div className=\"-mb-0.5 flex h-[22px] items-center gap-2 opacity-0 transition-opacity focus-within:opacity-100 group-hover/row:opacity-100\">\n <button\n ref={setActivatorNodeRef}\n type=\"button\"\n className=\"inline-flex h-[22px] cursor-grab items-center border-0 bg-transparent px-2 font-mono text-2xs text-fg-subtle hover:text-fg\"\n aria-label=\"Drag to move row\"\n {...attributes}\n {...listeners}\n >\n ⋮⋮ row\n </button>\n </div>\n <div\n ref={rowRef}\n className=\"relative grid min-w-0 items-stretch gap-2 max-[480px]:!grid-cols-1\"\n style={gridTemplateColumns ? { gridTemplateColumns } : undefined}\n >\n <SortableContext items={row.blocks.map((block) => block.uid)}>\n {row.blocks.map((editorBlock, index) => (\n <Fragment key={editorBlock.uid}>\n <SortableBlock\n rowUid={row.uid}\n area={area}\n editorBlock={editorBlock}\n data={data}\n selected={editorBlock.uid === selectedBlockUid}\n onRemoveBlock={onRemoveBlock}\n onSelect={onSelectBlock}\n onChangeBlock={onChangeBlock}\n />\n {canResizeColumns && index < row.blocks.length - 1 ? (\n <ColumnResizer\n key={`${editorBlock.uid}:resizer`}\n widths={widths}\n count={row.blocks.length}\n leftIndex={index}\n containerRef={rowRef as RefObject<HTMLElement | null>}\n onResize={(nextWidths) => onSetRowWidths(row.uid, nextWidths)}\n />\n ) : null}\n </Fragment>\n ))}\n </SortableContext>\n </div>\n </section>\n );\n}\n\nfunction getRowWidths(row: EditorRow): string[] | null {\n const widths = row.blocks.map((editorBlock) => editorBlock.block.config?.width);\n\n return widths.every((width): width is string => typeof width === \"string\") ? widths : null;\n}\n","import { useBuilderActions, useBuilderState } from \"../context/BuilderContext\";\nimport { BuilderCanvas } from \"./BuilderCanvas\";\n\nexport interface CanvasProps {\n className?: string;\n}\n\nexport function Canvas({ className }: CanvasProps = {}) {\n const {\n schema,\n schemaLoading,\n model,\n data,\n pageSize,\n footerRepeat,\n pageNumbers,\n selectedBlockUid,\n } = useBuilderState();\n const {\n removeBlock,\n selectBlock,\n changeBlock,\n deselect,\n setRowWidths,\n toggleFooterRepeat,\n changePageNumbers,\n } = useBuilderActions();\n\n if (!schema) {\n return (\n <div\n className={`min-w-0 min-h-0 overflow-auto bg-canvas px-4 pb-8 pt-6${className ? ` ${className}` : \"\"}`}\n >\n <div className=\"grid h-full place-items-center text-sm text-fg-muted\">\n {schemaLoading ? \"Loading schema…\" : \"Load the schema to start building.\"}\n </div>\n </div>\n );\n }\n\n return (\n <BuilderCanvas\n className={className}\n model={model}\n data={data}\n format={pageSize.format}\n orientation={pageSize.orientation}\n footerRepeat={footerRepeat}\n pageNumbers={pageNumbers}\n selectedBlockUid={selectedBlockUid}\n onRemoveBlock={removeBlock}\n onSelectBlock={selectBlock}\n onChangeBlock={changeBlock}\n onDeselect={deselect}\n onSetRowWidths={setRowWidths}\n onToggleFooterRepeat={toggleFooterRepeat}\n onChangePageNumbers={changePageNumbers}\n />\n );\n}\n","import type { ChangeEvent, ReactNode } from \"react\";\nimport type { ImageBlock } from \"../../../types/generated/template\";\nimport type { BlockEditorProps } from \"./blockEditors\";\nimport { Field, Input } from \"../../controls\";\n\nexport function ImageBlockEditor({ block, onChangeBlock }: BlockEditorProps): ReactNode {\n const imageBlock = block as ImageBlock;\n const src = imageBlock.src ?? \"\";\n const alt = imageBlock.alt ?? \"\";\n\n function handleFileChange(event: ChangeEvent<HTMLInputElement>): void {\n const file = event.currentTarget.files?.[0];\n\n if (!file) {\n return;\n }\n\n const reader = new FileReader();\n\n reader.addEventListener(\"load\", () => {\n if (typeof reader.result === \"string\") {\n onChangeBlock({ ...imageBlock, src: reader.result });\n }\n });\n reader.readAsDataURL(file);\n }\n\n return (\n <div className=\"grid gap-3 [container-type:inline-size]\">\n <div\n className=\"grid min-h-24 place-items-center rounded-md border border-solid border-border bg-surface-muted p-3\"\n data-name=\"preview\"\n >\n {src ? (\n <img src={src} alt={alt} className=\"max-h-[120px] max-w-full object-contain\" />\n ) : (\n <div className=\"text-2xs text-fg-muted\">No image selected</div>\n )}\n </div>\n\n <Field label=\"Source\">\n <Input\n name=\"src\"\n type=\"text\"\n value={src}\n onChange={(event) => onChangeBlock({ ...imageBlock, src: event.currentTarget.value })}\n />\n </Field>\n\n <Field label=\"Upload\">\n <Input name=\"src-file\" type=\"file\" accept=\"image/*\" onChange={handleFileChange} />\n </Field>\n\n <Field label=\"Alt text\">\n <Input\n name=\"alt\"\n type=\"text\"\n value={alt}\n onChange={(event) => onChangeBlock({ ...imageBlock, alt: event.currentTarget.value })}\n />\n </Field>\n <p className=\"-mt-2 m-0 text-2xs text-fg-muted\">\n Alt text is required for PDF/UA accessibility. Describe what the image conveys.\n </p>\n </div>\n );\n}\n","import { forwardRef, type KeyboardEvent, type ReactNode } from \"react\";\n\nexport interface InspectorShellProps {\n ariaLabel: string;\n className?: string;\n onKeyDown?: (event: KeyboardEvent<HTMLElement>) => void;\n children: ReactNode;\n}\n\nexport const InspectorShell = forwardRef<HTMLElement, InspectorShellProps>(function InspectorShell(\n { ariaLabel, className, onKeyDown, children },\n ref,\n) {\n return (\n <aside\n ref={ref}\n tabIndex={-1}\n onKeyDown={onKeyDown}\n className={`grid min-h-0 min-w-0 content-start gap-4 overflow-x-hidden overflow-y-auto bg-surface p-4 outline-none${className ? ` ${className}` : \"\"}`}\n aria-label={ariaLabel}\n >\n {children}\n </aside>\n );\n});\n\nexport interface InspectorHeaderProps {\n title: string;\n chip?: ReactNode;\n subtitle?: string;\n action?: ReactNode;\n}\n\nexport function InspectorHeader({ title, chip, subtitle, action }: InspectorHeaderProps) {\n return (\n <header className=\"flex min-w-0 items-start justify-between gap-2\">\n <div className=\"flex min-w-0 flex-auto items-start gap-2\">\n {chip}\n <div>\n <h2 className=\"m-0 text-[15px] font-semibold text-fg\">{title}</h2>\n {subtitle ? (\n <p className=\"mt-0.5 mb-0 break-words text-xs text-fg-muted\">{subtitle}</p>\n ) : null}\n </div>\n </div>\n {action}\n </header>\n );\n}\n\nexport function InspectorSection({ title, children }: { title: string; children: ReactNode }) {\n return (\n <fieldset className=\"grid min-w-0 gap-2 overflow-hidden rounded-md border border-solid border-border bg-surface-muted p-3\">\n <legend className=\"px-1 text-2xs font-semibold uppercase tracking-wide text-fg-muted\">\n {title}\n </legend>\n {children}\n </fieldset>\n );\n}\n","import { DndContext, closestCenter, type DragEndEvent } from \"@dnd-kit/core\";\nimport { SortableContext, verticalListSortingStrategy } from \"@dnd-kit/sortable\";\nimport type { ReactNode } from \"react\";\nimport { useBuilderSensors } from \"../../lib/sensors\";\n\nexport interface SortableListProps {\n count: number;\n onReorder: (sourceIndex: number, targetIndex: number) => void;\n children: ReactNode;\n}\n\nexport function SortableList({ count, onReorder, children }: SortableListProps) {\n const sensors = useBuilderSensors();\n\n function handleDragEnd(event: DragEndEvent): void {\n const { active, over } = event;\n\n if (!over || active.id === over.id) {\n return;\n }\n\n const sourceIndex = Number(active.id);\n const targetIndex = Number(over.id);\n\n if (!Number.isFinite(sourceIndex) || !Number.isFinite(targetIndex)) {\n return;\n }\n\n onReorder(sourceIndex, targetIndex);\n }\n\n const sortableIds = Array.from({ length: count }, (_, index) => String(index));\n\n return (\n <DndContext sensors={sensors} collisionDetection={closestCenter} onDragEnd={handleDragEnd}>\n <SortableContext items={sortableIds} strategy={verticalListSortingStrategy}>\n {children}\n </SortableContext>\n </DndContext>\n );\n}\n","import { useSortable } from \"@dnd-kit/sortable\";\nimport { CSS } from \"@dnd-kit/utilities\";\nimport type { CSSProperties, ReactNode } from \"react\";\n\nexport interface SortableRowProps {\n id: string;\n dragLabel: string;\n removeLabel: string;\n removeName?: string;\n onRemove: () => void;\n children: ReactNode;\n}\n\nexport function SortableRow({\n id,\n dragLabel,\n removeLabel,\n removeName,\n onRemove,\n children,\n}: SortableRowProps) {\n const {\n attributes,\n listeners,\n setActivatorNodeRef,\n setNodeRef,\n transform,\n transition,\n isDragging,\n } = useSortable({ id });\n const style: CSSProperties = {\n transform: CSS.Transform.toString(transform),\n transition,\n opacity: isDragging ? 0.6 : 1,\n };\n\n return (\n <div\n ref={setNodeRef}\n className=\"relative grid gap-2 rounded-md border border-solid border-border bg-surface-muted py-3 pr-8 pl-8\"\n style={style}\n >\n <button\n ref={setActivatorNodeRef}\n type=\"button\"\n className=\"absolute top-2 left-2 inline-grid h-[22px] w-[22px] cursor-grab place-items-center rounded border-0 bg-transparent p-0 font-mono text-xs tracking-tighter text-fg-muted hover:bg-surface hover:text-fg active:cursor-grabbing\"\n aria-label={dragLabel}\n {...attributes}\n {...listeners}\n >\n ⋮⋮\n </button>\n {children}\n <button\n type=\"button\"\n data-name={removeName}\n className=\"absolute top-2 right-2 inline-grid h-[22px] w-[22px] cursor-pointer place-items-center rounded border-0 bg-transparent p-0 text-xs text-fg-muted hover:bg-danger-soft hover:text-danger\"\n aria-label={removeLabel}\n onClick={onRemove}\n >\n ✕\n </button>\n </div>\n );\n}\n","import { arrayMove } from \"@dnd-kit/sortable\";\nimport type { ReactNode } from \"react\";\nimport type {\n Block,\n KeyValueBlock,\n KeyValueField,\n KeyValueValues,\n} from \"../../../types/generated/template\";\nimport { isRecord, nextKeyIndex, omitKey, renameKey } from \"../../lib/records\";\nimport { AddButton } from \"../../primitives/Button\";\nimport type { BlockEditorProps } from \"./blockEditors\";\nimport { InspectorSection } from \"../InspectorShell\";\nimport { SortableList } from \"./SortableList\";\nimport { SortableRow } from \"./SortableRow\";\nimport { Field, Input } from \"../../controls\";\n\nexport function moveField(\n fields: readonly KeyValueField[],\n sourceIndex: number,\n targetIndex: number,\n): KeyValueField[] {\n return arrayMove([...fields], sourceIndex, targetIndex);\n}\n\nexport function addField(block: KeyValueBlock): KeyValueBlock {\n const fields = getFields(block);\n const values = getValues(block);\n const nextIndex = nextKeyIndex(\n fields.map((field) => field.key),\n \"field\",\n );\n const nextFields = [...fields, { key: `field${nextIndex}`, label: `Field ${nextIndex}` }];\n\n return applyFields(block, nextFields, values);\n}\n\nexport function removeField(block: KeyValueBlock, index: number): KeyValueBlock {\n const fields = getFields(block);\n const removed = fields[index];\n\n if (!removed) {\n return block;\n }\n\n return applyFields(\n block,\n fields.filter((_, currentIndex) => currentIndex !== index),\n omitKey(getValues(block), removed.key),\n );\n}\n\nexport function renameFieldKey(\n block: KeyValueBlock,\n index: number,\n nextKey: string,\n): KeyValueBlock {\n const fields = getFields(block);\n const previous = fields[index];\n\n if (!previous || previous.key === nextKey) {\n return block;\n }\n\n if (fields.some((field, currentIndex) => currentIndex !== index && field.key === nextKey)) {\n return block;\n }\n\n return applyFields(\n block,\n fields.map((field, currentIndex) =>\n currentIndex === index ? { ...field, key: nextKey } : field,\n ),\n renameKey(getValues(block), previous.key, nextKey),\n );\n}\n\nexport function setFieldLabel(\n block: KeyValueBlock,\n index: number,\n nextLabel: string,\n): KeyValueBlock {\n const fields = getFields(block);\n\n return applyFields(\n block,\n fields.map((field, currentIndex) =>\n currentIndex === index ? { ...field, label: nextLabel } : field,\n ),\n getValues(block),\n );\n}\n\nexport function setValue(block: KeyValueBlock, key: string, nextValue: string): KeyValueBlock {\n return applyFields(block, getFields(block), { ...getValues(block), [key]: nextValue });\n}\n\nexport function reorderFields(\n block: KeyValueBlock,\n sourceIndex: number,\n targetIndex: number,\n): KeyValueBlock {\n return applyFields(block, moveField(getFields(block), sourceIndex, targetIndex), getValues(block));\n}\n\nexport function KeyValueBlockEditor({ block, onChangeBlock }: BlockEditorProps): ReactNode {\n const kvBlock = block as KeyValueBlock;\n const fields = getFields(kvBlock);\n const values = getValues(kvBlock);\n\n return (\n <>\n <InspectorSection title=\"Fields\">\n <FieldsEditor block={kvBlock} fields={fields} onChangeBlock={onChangeBlock} />\n </InspectorSection>\n\n {fields.length > 0 ? (\n <InspectorSection title=\"Values\">\n {fields.map((field) => (\n <Field key={field.key} label={field.label || field.key}>\n <Input\n name={`values.${field.key}`}\n type=\"text\"\n value={String(values[field.key] ?? \"\")}\n onChange={(event) =>\n onChangeBlock(setValue(kvBlock, field.key, event.currentTarget.value))\n }\n />\n </Field>\n ))}\n </InspectorSection>\n ) : null}\n </>\n );\n}\n\ninterface FieldsEditorProps {\n block: KeyValueBlock;\n fields: KeyValueField[];\n onChangeBlock: (block: Block) => void;\n}\n\nfunction FieldsEditor({ block, fields, onChangeBlock }: FieldsEditorProps) {\n return (\n <>\n <SortableList\n count={fields.length}\n onReorder={(source, target) => onChangeBlock(reorderFields(block, source, target))}\n >\n {fields.map((field, index) => (\n <FieldRow\n key={fieldRowKey(field, index)}\n id={String(index)}\n index={index}\n field={field}\n onChangeKey={(value) => onChangeBlock(renameFieldKey(block, index, value))}\n onChangeLabel={(value) => onChangeBlock(setFieldLabel(block, index, value))}\n onRemove={() => onChangeBlock(removeField(block, index))}\n />\n ))}\n </SortableList>\n <AddButton data-name=\"add-field\" onClick={() => onChangeBlock(addField(block))}>\n Add field\n </AddButton>\n </>\n );\n}\n\ninterface FieldRowProps {\n id: string;\n index: number;\n field: KeyValueField;\n onChangeKey: (value: string) => void;\n onChangeLabel: (value: string) => void;\n onRemove: () => void;\n}\n\nfunction FieldRow({ id, index, field, onChangeKey, onChangeLabel, onRemove }: FieldRowProps) {\n return (\n <SortableRow\n id={id}\n dragLabel={`Drag to reorder field ${index + 1}`}\n removeLabel={`Remove field ${index + 1}`}\n removeName={`remove-field-${index}`}\n onRemove={onRemove}\n >\n <div className=\"grid grid-cols-2 gap-2\">\n <Field label=\"Key\">\n <Input\n name={`field-key-${index}`}\n type=\"text\"\n value={field.key}\n onChange={(event) => onChangeKey(event.currentTarget.value)}\n />\n </Field>\n <Field label=\"Label\">\n <Input\n name={`field-label-${index}`}\n type=\"text\"\n value={field.label}\n onChange={(event) => onChangeLabel(event.currentTarget.value)}\n />\n </Field>\n </div>\n </SortableRow>\n );\n}\n\nfunction fieldRowKey(field: KeyValueField, index: number): string {\n return field.key.length > 0 ? `key:${field.key}` : `index:${index}`;\n}\n\nfunction getFields(block: KeyValueBlock): KeyValueField[] {\n const candidate = block.config?.fields;\n\n return Array.isArray(candidate) ? (candidate as KeyValueField[]) : [];\n}\n\nfunction getValues(block: KeyValueBlock): KeyValueValues {\n const candidate = block.values;\n\n return isRecord(candidate) ? (candidate as KeyValueValues) : {};\n}\n\nfunction applyFields(\n block: KeyValueBlock,\n fields: KeyValueField[],\n values: KeyValueValues,\n): KeyValueBlock {\n const config = { ...block.config };\n\n if (fields.length === 0) {\n delete (config as { fields?: KeyValueField[] }).fields;\n } else {\n (config as { fields?: KeyValueField[] }).fields = fields;\n }\n\n const nextBlock: KeyValueBlock = {\n ...block,\n config: Object.keys(config).length === 0 ? undefined : config,\n };\n\n if (Object.keys(values).length === 0) {\n delete (nextBlock as { values?: KeyValueValues }).values;\n } else {\n nextBlock.values = values;\n }\n\n if (nextBlock.config === undefined) {\n delete (nextBlock as { config?: KeyValueBlock[\"config\"] }).config;\n }\n\n return nextBlock;\n}\n","import { arrayMove } from \"@dnd-kit/sortable\";\nimport type { ReactNode } from \"react\";\nimport type { Align, Block, TableBlock, TableColumn } from \"../../../types/generated/template\";\nimport { NUMBER_COLUMN_RESERVE, percentWidth } from \"../../canvas/columns\";\nimport { isRecord, nextKeyIndex, omitKey, renameKey } from \"../../lib/records\";\nimport { AddButton } from \"../../primitives/Button\";\nimport { InspectorSection } from \"../InspectorShell\";\nimport { setBlockConfigField } from \"../../state/configUpdates\";\nimport type { BlockEditorProps } from \"./blockEditors\";\nimport { SortableList } from \"./SortableList\";\nimport { SortableRow } from \"./SortableRow\";\nimport { AlignSelect, Field, Input } from \"../../controls\";\n\nconst hintClass = \"m-0 text-2xs text-fg-muted\";\n\ntype TableRow = Record<string, string>;\n\nexport function moveColumn(\n columns: readonly TableColumn[],\n sourceIndex: number,\n targetIndex: number,\n): TableColumn[] {\n return arrayMove([...columns], sourceIndex, targetIndex);\n}\n\nexport function moveRow(\n rows: readonly TableRow[],\n sourceIndex: number,\n targetIndex: number,\n): TableRow[] {\n return arrayMove([...rows], sourceIndex, targetIndex);\n}\n\nexport function addColumn(block: TableBlock): TableBlock {\n const columns = getColumns(block);\n const nextIndex = nextKeyIndex(\n columns.map((column) => column.key),\n \"column\",\n );\n const nextColumns: TableColumn[] = [\n ...columns,\n { key: `column${nextIndex}`, label: `Column ${nextIndex}` },\n ];\n\n return applyColumns(block, nextColumns);\n}\n\nexport function removeColumn(block: TableBlock, index: number): TableBlock {\n const columns = getColumns(block);\n\n if (!columns[index]) {\n return block;\n }\n\n return applyColumns(\n block,\n columns.filter((_, currentIndex) => currentIndex !== index),\n );\n}\n\nexport function renameColumnKey(\n block: TableBlock,\n index: number,\n nextKey: string,\n): TableBlock {\n const columns = getColumns(block);\n const previous = columns[index];\n\n if (!previous || previous.key === nextKey) {\n return block;\n }\n\n if (columns.some((column, currentIndex) => currentIndex !== index && column.key === nextKey)) {\n return block;\n }\n\n return applyColumns(\n block,\n columns.map((column, currentIndex) =>\n currentIndex === index ? { ...column, key: nextKey } : column,\n ),\n );\n}\n\nexport function setColumnLabel(\n block: TableBlock,\n index: number,\n nextLabel: string,\n): TableBlock {\n const columns = getColumns(block);\n\n return applyColumns(\n block,\n columns.map((column, currentIndex) =>\n currentIndex === index ? { ...column, label: nextLabel } : column,\n ),\n );\n}\n\nexport function setColumnAlign(\n block: TableBlock,\n index: number,\n nextAlign: string,\n): TableBlock {\n const columns = getColumns(block);\n\n return applyColumns(\n block,\n columns.map((column, currentIndex) => {\n if (currentIndex !== index) {\n return column;\n }\n\n const next: TableColumn = { ...column };\n if (nextAlign === \"\") {\n delete next.align;\n } else {\n next.align = nextAlign as Align;\n }\n return next;\n }),\n );\n}\n\nexport function setTableNumberRows(block: TableBlock, value: boolean | undefined): TableBlock {\n const previousOn = block.config?.numberRows === true;\n const nextOn = value === true;\n const withFlag = setBlockConfigField(block, \"numberRows\", value);\n\n if (previousOn === nextOn) {\n return withFlag;\n }\n\n return adjustFirstColumnWidth(withFlag, nextOn ? -NUMBER_COLUMN_RESERVE : NUMBER_COLUMN_RESERVE);\n}\n\nfunction adjustFirstColumnWidth(block: TableBlock, delta: number): TableBlock {\n const columns = getColumns(block);\n\n if (columns.length === 0 || !columns.every((column) => percentWidth(column.width) !== null)) {\n return block;\n }\n\n return applyColumns(\n block,\n columns.map((column, index) => {\n if (index !== 0) {\n return column;\n }\n\n const current = percentWidth(column.width) ?? 0;\n const next = Math.max(NUMBER_COLUMN_RESERVE, current + delta);\n\n return { ...column, width: `${next}%` };\n }),\n );\n}\n\nexport function reorderColumns(\n block: TableBlock,\n sourceIndex: number,\n targetIndex: number,\n): TableBlock {\n return applyColumns(block, moveColumn(getColumns(block), sourceIndex, targetIndex));\n}\n\nexport function addRow(rows: readonly TableRow[], columns: readonly TableColumn[]): TableRow[] {\n const empty: TableRow = Object.fromEntries(columns.map((column) => [column.key, \"\"]));\n\n return [...rows, empty];\n}\n\nexport function removeRow(rows: readonly TableRow[], index: number): TableRow[] {\n return rows.filter((_, currentIndex) => currentIndex !== index);\n}\n\nexport function setCellValue(\n rows: readonly TableRow[],\n index: number,\n key: string,\n nextValue: string,\n): TableRow[] {\n return rows.map((row, currentIndex) =>\n currentIndex === index ? { ...row, [key]: nextValue } : row,\n );\n}\n\nexport function removeColumnKeyFromRows(\n rows: readonly TableRow[],\n key: string,\n): TableRow[] {\n return rows.map((row) => omitKey(row, key));\n}\n\nexport function renameColumnKeyInRows(\n rows: readonly TableRow[],\n previousKey: string,\n nextKey: string,\n): TableRow[] {\n return rows.map((row) => renameKey(row, previousKey, nextKey));\n}\n\nexport function TableBlockEditor({\n block,\n rowData,\n onChangeBlock,\n onChangeRowData,\n}: BlockEditorProps): ReactNode {\n const tableBlock = block as TableBlock;\n const columns = getColumns(tableBlock);\n const rows = getRows(rowData);\n const blockId = typeof tableBlock.id === \"string\" ? tableBlock.id : \"\";\n const canEditRows = blockId !== \"\" && onChangeRowData !== undefined;\n\n function handleRemoveColumn(index: number): void {\n const removed = columns[index];\n\n if (!removed) {\n return;\n }\n\n onChangeBlock(removeColumn(tableBlock, index));\n\n if (canEditRows) {\n onChangeRowData?.(removeColumnKeyFromRows(rows, removed.key));\n }\n }\n\n function handleRenameColumnKey(index: number, nextKey: string): void {\n const previous = columns[index];\n\n if (!previous || previous.key === nextKey) {\n return;\n }\n\n if (columns.some((column, currentIndex) => currentIndex !== index && column.key === nextKey)) {\n return;\n }\n\n onChangeBlock(renameColumnKey(tableBlock, index, nextKey));\n\n if (canEditRows) {\n onChangeRowData?.(renameColumnKeyInRows(rows, previous.key, nextKey));\n }\n }\n\n return (\n <>\n <ColumnsEditor\n block={tableBlock}\n columns={columns}\n onChangeBlock={onChangeBlock}\n onRemoveColumn={handleRemoveColumn}\n onRenameColumnKey={handleRenameColumnKey}\n />\n\n <RowsEditor\n canEditRows={canEditRows}\n rows={rows}\n columns={columns}\n onChangeRowData={onChangeRowData}\n />\n </>\n );\n}\n\ninterface ColumnsEditorProps {\n block: TableBlock;\n columns: TableColumn[];\n onChangeBlock: (block: Block) => void;\n onRemoveColumn: (index: number) => void;\n onRenameColumnKey: (index: number, nextKey: string) => void;\n}\n\nfunction ColumnsEditor({\n block,\n columns,\n onChangeBlock,\n onRemoveColumn,\n onRenameColumnKey,\n}: ColumnsEditorProps) {\n return (\n <InspectorSection title=\"Columns\">\n <SortableList\n count={columns.length}\n onReorder={(source, target) => onChangeBlock(reorderColumns(block, source, target))}\n >\n {columns.map((column, index) => (\n <ColumnRow\n key={columnRowKey(column, index)}\n id={String(index)}\n index={index}\n column={column}\n onChangeKey={(value) => onRenameColumnKey(index, value)}\n onChangeLabel={(value) => onChangeBlock(setColumnLabel(block, index, value))}\n onChangeAlign={(value) => onChangeBlock(setColumnAlign(block, index, value))}\n onRemove={() => onRemoveColumn(index)}\n />\n ))}\n </SortableList>\n <AddButton data-name=\"add-column\" onClick={() => onChangeBlock(addColumn(block))}>\n Add column\n </AddButton>\n </InspectorSection>\n );\n}\n\ninterface ColumnRowProps {\n id: string;\n index: number;\n column: TableColumn;\n onChangeKey: (value: string) => void;\n onChangeLabel: (value: string) => void;\n onChangeAlign: (value: string) => void;\n onRemove: () => void;\n}\n\nfunction ColumnRow({\n id,\n index,\n column,\n onChangeKey,\n onChangeLabel,\n onChangeAlign,\n onRemove,\n}: ColumnRowProps) {\n return (\n <SortableRow\n id={id}\n dragLabel={`Drag to reorder column ${index + 1}`}\n removeLabel={`Remove column ${index + 1}`}\n removeName={`remove-column-${index}`}\n onRemove={onRemove}\n >\n <div className=\"grid grid-cols-2 gap-2\">\n <Field label=\"Key\">\n <Input\n name={`column-key-${index}`}\n type=\"text\"\n value={column.key}\n onChange={(event) => onChangeKey(event.currentTarget.value)}\n />\n </Field>\n <Field label=\"Label\">\n <Input\n name={`column-label-${index}`}\n type=\"text\"\n value={column.label}\n onChange={(event) => onChangeLabel(event.currentTarget.value)}\n />\n </Field>\n <AlignSelect\n name={`column-align-${index}`}\n value={(column.align ?? \"\") as string}\n onChange={onChangeAlign}\n />\n </div>\n </SortableRow>\n );\n}\n\ninterface RowsEditorProps {\n canEditRows: boolean;\n rows: TableRow[];\n columns: TableColumn[];\n onChangeRowData?: (data: unknown) => void;\n}\n\nfunction RowsEditor({ canEditRows, rows, columns, onChangeRowData }: RowsEditorProps) {\n if (!canEditRows) {\n return (\n <InspectorSection title=\"Rows\">\n <p className={hintClass}>Give this block an id to edit runtime row data here.</p>\n </InspectorSection>\n );\n }\n\n return (\n <InspectorSection title=\"Rows\">\n {rows.length === 0 ? (\n <p className={hintClass}>No rows yet. Add one to seed runtime data for this table.</p>\n ) : (\n <SortableList\n count={rows.length}\n onReorder={(source, target) => onChangeRowData?.(moveRow(rows, source, target))}\n >\n {rows.map((row, rowIndex) => (\n <DataRow\n key={dataRowKey(row, columns, rowIndex)}\n id={String(rowIndex)}\n index={rowIndex}\n row={row}\n columns={columns}\n onChangeCell={(key, value) =>\n onChangeRowData?.(setCellValue(rows, rowIndex, key, value))\n }\n onRemove={() => onChangeRowData?.(removeRow(rows, rowIndex))}\n />\n ))}\n </SortableList>\n )}\n <AddButton\n data-name=\"add-row\"\n disabled={columns.length === 0}\n onClick={() => onChangeRowData?.(addRow(rows, columns))}\n >\n Add row\n </AddButton>\n </InspectorSection>\n );\n}\n\ninterface DataRowProps {\n id: string;\n index: number;\n row: TableRow;\n columns: TableColumn[];\n onChangeCell: (key: string, value: string) => void;\n onRemove: () => void;\n}\n\nfunction DataRow({ id, index, row, columns, onChangeCell, onRemove }: DataRowProps) {\n return (\n <SortableRow\n id={id}\n dragLabel={`Drag to reorder row ${index + 1}`}\n removeLabel={`Remove row ${index + 1}`}\n removeName={`remove-row-${index}`}\n onRemove={onRemove}\n >\n <div className=\"grid grid-cols-2 gap-2\">\n {columns.map((column) => (\n <Field key={column.key} label={column.label || column.key}>\n <Input\n name={`row-${index}.${column.key}`}\n type=\"text\"\n value={row[column.key] ?? \"\"}\n onChange={(event) => onChangeCell(column.key, event.currentTarget.value)}\n />\n </Field>\n ))}\n </div>\n </SortableRow>\n );\n}\n\nfunction columnRowKey(column: TableColumn, index: number): string {\n return column.key.length > 0 ? `key:${column.key}` : `index:${index}`;\n}\n\nfunction dataRowKey(row: TableRow, columns: TableColumn[], index: number): string {\n const firstColumn = columns[0];\n const candidate = firstColumn ? row[firstColumn.key] : undefined;\n\n return typeof candidate === \"string\" && candidate.length > 0\n ? `first:${candidate}:${index}`\n : `index:${index}`;\n}\n\nfunction getColumns(block: TableBlock): TableColumn[] {\n const candidate = block.config?.columns;\n\n return Array.isArray(candidate) ? (candidate as TableColumn[]) : [];\n}\n\nfunction getRows(rowData: unknown): TableRow[] {\n if (!Array.isArray(rowData)) {\n return [];\n }\n\n return rowData.filter((entry): entry is TableRow => isRecord(entry)) as TableRow[];\n}\n\nfunction applyColumns(block: TableBlock, columns: TableColumn[]): TableBlock {\n const config = { ...block.config } as Record<string, unknown>;\n\n if (columns.length === 0) {\n delete config.columns;\n } else {\n config.columns = columns;\n }\n\n const nextBlock: TableBlock = {\n ...block,\n config:\n Object.keys(config).length === 0 ? undefined : (config as TableBlock[\"config\"]),\n };\n\n if (nextBlock.config === undefined) {\n delete (nextBlock as { config?: TableBlock[\"config\"] }).config;\n }\n\n return nextBlock;\n}\n","import type { ReactNode } from \"react\";\nimport type { Block } from \"../../types/generated/template\";\nimport { ImageBlockEditor } from \"./editors/ImageBlockEditor\";\nimport { KeyValueBlockEditor } from \"./editors/KeyValueBlockEditor\";\nimport { TableBlockEditor } from \"./editors/TableBlockEditor\";\nimport { SelectField, TextAreaField, TextField, type SelectFieldOption } from \"../controls\";\nimport { setBlockConfigField } from \"../state/configUpdates\";\nimport { InspectorSection } from \"./InspectorShell\";\n\nexport interface BlockContentControlsProps {\n block: Block;\n rowData?: unknown;\n onChangeBlock: (block: Block) => void;\n onChangeRowData?: (data: unknown) => void;\n}\n\ntype HeadingLevelValue = \"1\" | \"2\" | \"3\" | \"4\" | \"5\" | \"6\";\n\nconst headingLevelOptions = [\n { value: \"1\", label: \"Heading 1\" },\n { value: \"2\", label: \"Heading 2\" },\n { value: \"3\", label: \"Heading 3\" },\n { value: \"4\", label: \"Heading 4\" },\n { value: \"5\", label: \"Heading 5\" },\n { value: \"6\", label: \"Heading 6\" },\n] as const satisfies readonly SelectFieldOption<HeadingLevelValue>[];\n\nexport function BlockContentControls(props: BlockContentControlsProps): ReactNode {\n if (props.block.type === \"key-value\") {\n return <KeyValueBlockEditor block={props.block} onChangeBlock={props.onChangeBlock} />;\n }\n\n if (props.block.type === \"table\") {\n return (\n <TableBlockEditor\n block={props.block}\n rowData={props.rowData}\n onChangeBlock={props.onChangeBlock}\n onChangeRowData={props.onChangeRowData}\n />\n );\n }\n\n return <InspectorSection title=\"Content\">{renderContentFields(props)}</InspectorSection>;\n}\n\nfunction renderContentFields({ block, onChangeBlock }: BlockContentControlsProps): ReactNode {\n switch (block.type) {\n case \"text\":\n return (\n <div className=\"grid gap-2\">\n <TextAreaField\n name=\"text\"\n label=\"Text\"\n value={block.text}\n rows={4}\n onChange={(value) => onChangeBlock({ ...block, text: value ?? \"\" })}\n />\n </div>\n );\n case \"html\":\n return (\n <div className=\"grid gap-2\">\n <TextAreaField\n name=\"html\"\n label=\"HTML\"\n value={block.html}\n rows={6}\n onChange={(value) => onChangeBlock({ ...block, html: value ?? \"\" })}\n />\n </div>\n );\n case \"heading\":\n return (\n <div className=\"grid gap-2\">\n <TextField\n name=\"text\"\n label=\"Text\"\n value={block.text}\n onChange={(value) => onChangeBlock({ ...block, text: value ?? \"\" })}\n />\n <SelectField\n name=\"config.level\"\n label=\"Level\"\n value={headingLevelValue(block.config?.level)}\n options={headingLevelOptions}\n optional\n emptyLabel=\"Default\"\n onChange={(value) =>\n onChangeBlock(setBlockConfigField(block, \"level\", parseHeadingLevel(value)))\n }\n />\n </div>\n );\n case \"image\":\n return <ImageBlockEditor block={block} onChangeBlock={onChangeBlock} />;\n case \"key-value\":\n case \"table\":\n return null;\n case \"spacer\":\n case \"divider\":\n return <p className=\"m-0 text-xs text-fg-muted\">No content fields for this block.</p>;\n }\n}\n\nfunction headingLevelValue(value: number | undefined): HeadingLevelValue | undefined {\n if (value === undefined) {\n return undefined;\n }\n\n const stringValue = String(value);\n\n return isHeadingLevelValue(stringValue) ? stringValue : undefined;\n}\n\nfunction parseHeadingLevel(value: HeadingLevelValue | undefined): number | undefined {\n return value === undefined ? undefined : Number(value);\n}\n\nfunction isHeadingLevelValue(value: string): value is HeadingLevelValue {\n return headingLevelOptions.some((option) => option.value === value);\n}\n","import type { Align } from \"../../types/generated/template\";\nimport type { SelectFieldOption } from \"../controls\";\n\nexport const ALIGN_OPTIONS = [\n { value: \"left\", label: \"Left\" },\n { value: \"center\", label: \"Center\" },\n { value: \"right\", label: \"Right\" },\n] as const satisfies readonly SelectFieldOption<Align>[];\n","import type { ReactNode } from \"react\";\nimport type { Block, BlockConfig, DividerStyle, TableStyle } from \"../../types/generated/template\";\nimport {\n NumberField,\n SelectField,\n TextField,\n UnitField,\n type SelectFieldOption,\n} from \"../controls\";\nimport { setTableNumberRows } from \"./editors/TableBlockEditor\";\nimport { setBlockConfigField } from \"../state/configUpdates\";\nimport { ALIGN_OPTIONS } from \"./alignOptions\";\n\nexport interface BlockLayoutControlsProps {\n block: Block;\n onChangeBlock: (block: Block) => void;\n}\n\nconst dividerStyleOptions = [\n { value: \"solid\", label: \"Solid\" },\n { value: \"dashed\", label: \"Dashed\" },\n { value: \"dotted\", label: \"Dotted\" },\n { value: \"double\", label: \"Double\" },\n { value: \"none\", label: \"None\" },\n] as const satisfies readonly SelectFieldOption<DividerStyle>[];\n\nconst tableStyleOptions = [\n { value: \"striped\", label: \"Striped\" },\n { value: \"bordered\", label: \"Bordered\" },\n { value: \"minimal\", label: \"Minimal\" },\n] as const satisfies readonly SelectFieldOption<TableStyle>[];\n\nconst numberRowsOptions = [\n { value: \"show\", label: \"Show\" },\n { value: \"hide\", label: \"Hide\" },\n] as const satisfies readonly SelectFieldOption<NumberRowsValue>[];\n\ntype NumberRowsValue = \"show\" | \"hide\";\n\nexport function BlockLayoutControls({ block, onChangeBlock }: BlockLayoutControlsProps): ReactNode {\n return (\n <div className=\"grid gap-2\">\n <div className=\"grid grid-cols-2 items-start gap-2\">\n <UnitField\n name=\"config.width\"\n label=\"Width\"\n value={block.config?.width ?? undefined}\n placeholder=\"Full width\"\n readOnly\n help=\"Drag the column divider on the canvas to resize.\"\n />\n <SelectField\n name=\"config.align\"\n label=\"Align\"\n value={block.config?.align ?? undefined}\n options={ALIGN_OPTIONS}\n optional\n emptyLabel=\"Default\"\n onChange={(value) => onChangeBlock(setCommonConfigField(block, \"align\", value))}\n />\n </div>\n {renderTypeSpecificControls(block, onChangeBlock)}\n </div>\n );\n}\n\nfunction renderTypeSpecificControls(\n block: Block,\n onChangeBlock: (block: Block) => void,\n): ReactNode {\n switch (block.type) {\n case \"spacer\":\n return (\n <NumberField\n name=\"config.height\"\n label=\"Height\"\n value={block.config?.height}\n min={0}\n step={0.5}\n onChange={(value) => onChangeBlock(setBlockConfigField(block, \"height\", value))}\n />\n );\n case \"image\":\n return (\n <NumberField\n name=\"config.maxHeight\"\n label=\"Max height\"\n value={block.config?.maxHeight}\n min={0}\n step={0.5}\n onChange={(value) => onChangeBlock(setBlockConfigField(block, \"maxHeight\", value))}\n />\n );\n case \"divider\":\n return (\n <>\n <NumberField\n name=\"config.thickness\"\n label=\"Thickness\"\n value={block.config?.thickness}\n min={0}\n step={0.5}\n onChange={(value) => onChangeBlock(setBlockConfigField(block, \"thickness\", value))}\n />\n <TextField\n name=\"config.lineColor\"\n label=\"Line color\"\n value={block.config?.lineColor}\n placeholder=\"#334455\"\n emptyValue=\"undefined\"\n onChange={(value) => onChangeBlock(setBlockConfigField(block, \"lineColor\", value))}\n />\n <SelectField\n name=\"config.style\"\n label=\"Line style\"\n value={block.config?.style}\n options={dividerStyleOptions}\n optional\n emptyLabel=\"Default\"\n onChange={(value) => onChangeBlock(setBlockConfigField(block, \"style\", value))}\n />\n </>\n );\n case \"table\":\n return (\n <div className=\"grid grid-cols-2 gap-2\">\n <SelectField\n name=\"config.style\"\n label=\"Table style\"\n value={block.config?.style}\n options={tableStyleOptions}\n optional\n emptyLabel=\"Default\"\n onChange={(value) => onChangeBlock(setBlockConfigField(block, \"style\", value))}\n />\n <SelectField\n name=\"config.numberRows\"\n label=\"Row numbers\"\n value={numberRowsValue(block.config?.numberRows)}\n options={numberRowsOptions}\n optional\n emptyLabel=\"Default\"\n onChange={(value) =>\n onChangeBlock(setTableNumberRows(block, booleanFromNumberRows(value)))\n }\n />\n </div>\n );\n case \"key-value\":\n case \"heading\":\n case \"text\":\n case \"html\":\n return null;\n }\n}\n\nfunction setCommonConfigField<TKey extends \"width\" | \"align\">(\n block: Block,\n field: TKey,\n value: BlockConfig[TKey] | undefined,\n): Block {\n switch (block.type) {\n case \"text\":\n case \"html\":\n case \"heading\":\n case \"image\":\n case \"key-value\":\n case \"spacer\":\n case \"divider\":\n case \"table\":\n return setBlockConfigField(block, field, value);\n }\n}\n\nfunction numberRowsValue(value: boolean | undefined): NumberRowsValue | undefined {\n if (value === undefined) {\n return undefined;\n }\n\n return value ? \"show\" : \"hide\";\n}\n\nfunction booleanFromNumberRows(value: NumberRowsValue | undefined): boolean | undefined {\n if (value === undefined) {\n return undefined;\n }\n\n return value === \"show\";\n}\n","import type { ReactNode } from \"react\";\nimport type { Block, SpacingConfig, Template } from \"../../types/generated/template\";\nimport { NumberField } from \"../controls\";\nimport { setBlockSpacingField, setTemplatePageMargin } from \"../state/configUpdates\";\n\nconst spacingSides = [\"top\", \"right\", \"bottom\", \"left\"] as const;\n\ntype SpacingSide = (typeof spacingSides)[number];\n\ninterface BlockSpacingControlsProps {\n scope: \"block\";\n block: Block;\n onChangeBlock: (block: Block) => void;\n}\n\ninterface PageMarginControlsProps {\n scope: \"page\";\n template: Template;\n onChangeTemplate: (template: Template) => void;\n}\n\nexport type SpacingControlsProps = BlockSpacingControlsProps | PageMarginControlsProps;\n\nexport function SpacingControls(props: SpacingControlsProps): ReactNode {\n const spacing =\n props.scope === \"block\" ? props.block.config?.spacing : props.template.config?.page?.margins;\n\n return (\n <div className=\"grid min-w-0 grid-cols-2 gap-2\">\n {spacingSides.map((side) => (\n <NumberField\n key={side}\n name={fieldName(props.scope, side)}\n label={sideLabel(side)}\n value={fieldValue(spacing, side)}\n min={0}\n step={0.5}\n onChange={(value) => {\n if (props.scope === \"block\") {\n props.onChangeBlock(setBlockSpacingField(props.block, side, value));\n return;\n }\n\n props.onChangeTemplate(setTemplatePageMargin(props.template, side, value));\n }}\n />\n ))}\n </div>\n );\n}\n\nfunction fieldName(scope: SpacingControlsProps[\"scope\"], side: SpacingSide): string {\n return scope === \"block\" ? `config.spacing.${side}` : `config.page.margins.${side}`;\n}\n\nfunction sideLabel(side: SpacingSide): string {\n return `${side[0].toUpperCase()}${side.slice(1)} (mm)`;\n}\n\nfunction fieldValue(spacing: SpacingConfig | undefined, side: SpacingSide): number | undefined {\n return spacing?.[side] ?? undefined;\n}\n","import type { ReactNode } from \"react\";\nimport type { Block, Template, TypographyConfig } from \"../../types/generated/template\";\nimport type { TemplateSchemaMetadata } from \"../../types/template\";\nimport { ColorField, NumberField, SelectField } from \"../controls\";\nimport { setBlockTypographyField, setTemplateTypographyField } from \"../state/configUpdates\";\nimport { ALIGN_OPTIONS } from \"./alignOptions\";\n\ninterface TypographyControlsBaseProps {\n metadata?: Pick<TemplateSchemaMetadata, \"bundledFonts\">;\n}\n\nexport interface BlockTypographyControlsProps extends TypographyControlsBaseProps {\n target: \"block\";\n block: Block;\n onChangeBlock: (block: Block) => void;\n}\n\nexport interface TemplateTypographyControlsProps extends TypographyControlsBaseProps {\n target: \"template\";\n template: Template;\n onChangeTemplate: (template: Template) => void;\n}\n\nexport type TypographyControlsProps =\n | BlockTypographyControlsProps\n | TemplateTypographyControlsProps;\n\nexport function TypographyControls(props: TypographyControlsProps): ReactNode {\n const typography =\n props.target === \"block\" ? props.block.config?.typography : props.template.config?.typography;\n const namePrefix = props.target === \"block\" ? \"config.typography\" : \"template.config.typography\";\n const fontOptions = familyOptions(props.metadata, typography?.family ?? undefined);\n\n function handleChange<TKey extends keyof TypographyConfig>(\n field: TKey,\n value: TypographyConfig[TKey] | \"\" | null | undefined,\n ): void {\n if (props.target === \"block\") {\n props.onChangeBlock(setBlockTypographyField(props.block, field, value));\n return;\n }\n\n props.onChangeTemplate(setTemplateTypographyField(props.template, field, value));\n }\n\n return (\n <div className=\"grid min-w-0 gap-2\">\n <div className=\"grid grid-cols-2 gap-2\">\n <SelectField\n name={`${namePrefix}.family`}\n label=\"Family\"\n value={typography?.family ?? undefined}\n options={fontOptions}\n optional\n emptyLabel=\"Default\"\n onChange={(value) => handleChange(\"family\", value)}\n />\n <SelectField\n name={`${namePrefix}.align`}\n label=\"Align\"\n value={typography?.align ?? undefined}\n optional\n options={ALIGN_OPTIONS}\n onChange={(value) => handleChange(\"align\", value)}\n />\n </div>\n <div className=\"grid grid-cols-3 gap-2\">\n <NumberField\n name={`${namePrefix}.size`}\n label=\"Size\"\n value={typography?.size ?? undefined}\n min={1}\n step={1}\n placeholder=\"12\"\n onChange={(value) => handleChange(\"size\", value)}\n />\n <NumberField\n name={`${namePrefix}.weight`}\n label=\"Weight\"\n value={typography?.weight ?? undefined}\n min={1}\n step={1}\n placeholder=\"400\"\n onChange={(value) => handleChange(\"weight\", value)}\n />\n <ColorField\n name={`${namePrefix}.color`}\n label=\"Color\"\n value={typography?.color ?? undefined}\n onChange={(value) => handleChange(\"color\", value)}\n />\n </div>\n </div>\n );\n}\n\nfunction familyOptions(\n metadata: Pick<TemplateSchemaMetadata, \"bundledFonts\"> | undefined,\n current: string | undefined,\n): Array<{ value: string; label: string }> {\n const fonts = bundledFontOptions(metadata);\n\n if (current && !fonts.includes(current)) {\n fonts.push(current);\n }\n\n return fonts.map((font) => ({ value: font, label: font }));\n}\n\nfunction bundledFontOptions(\n metadata: Pick<TemplateSchemaMetadata, \"bundledFonts\"> | undefined,\n): string[] {\n const seen = new Set<string>();\n const options: string[] = [];\n\n for (const font of metadata?.bundledFonts ?? []) {\n const normalized = font.trim();\n\n if (normalized === \"\" || seen.has(normalized)) {\n continue;\n }\n\n seen.add(normalized);\n options.push(normalized);\n }\n\n return options;\n}\n","import { useEffect, useRef, type KeyboardEvent } from \"react\";\nimport type { Block } from \"../../types/generated/template\";\nimport type { TemplateData, TemplateSchemaResponse } from \"../../types/template\";\nimport { Button } from \"../primitives/Button\";\nimport { Chip } from \"../primitives/Chip\";\nimport { TextField } from \"../controls\";\nimport { getBlockChrome, getBlockSummary } from \"../blocks/blockChrome\";\nimport type { EditorBlock } from \"../state/editorModel\";\nimport { BlockContentControls } from \"./BlockContentControls\";\nimport { BlockLayoutControls } from \"./BlockLayoutControls\";\nimport { InspectorHeader, InspectorSection, InspectorShell } from \"./InspectorShell\";\nimport { SpacingControls } from \"./SpacingControls\";\nimport { TypographyControls } from \"./TypographyControls\";\n\nexport interface BlockInspectorProps {\n block: EditorBlock | null;\n schema: TemplateSchemaResponse;\n data: TemplateData;\n onChangeBlock: (blockUid: string, block: Block) => void;\n onChangeData?: (data: TemplateData) => void;\n onRemoveBlock: (blockUid: string) => void;\n onClose: () => void;\n className?: string;\n}\n\nconst detailSections = [\"Layout\", \"Typography\", \"Spacing\"] as const;\n\nexport function BlockInspector({\n block,\n schema,\n data,\n onChangeBlock,\n onChangeData,\n onRemoveBlock,\n onClose,\n className,\n}: BlockInspectorProps) {\n const shellRef = useRef<HTMLElement>(null);\n const returnFocusRef = useRef<HTMLElement | null>(null);\n\n // Move focus into the non-modal flyout when it opens and restore it to the\n // triggering element when it closes, so keyboard users are not stranded.\n useEffect(() => {\n if (!block) {\n return;\n }\n\n returnFocusRef.current = document.activeElement as HTMLElement | null;\n shellRef.current?.focus();\n\n return () => {\n returnFocusRef.current?.focus?.();\n };\n // Selecting a different block keeps the panel mounted; only run on open/close.\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n function handleKeyDown(event: KeyboardEvent<HTMLElement>): void {\n if (event.key === \"Escape\") {\n event.stopPropagation();\n onClose();\n }\n }\n\n if (!block) {\n return (\n <InspectorShell ariaLabel=\"Block inspector\" className={className}>\n <InspectorHeader title=\"Inspector\" />\n <p className=\"m-0 text-xs text-fg-muted\">Select a block to inspect it.</p>\n </InspectorShell>\n );\n }\n\n const chrome = getBlockChrome(block.block.type);\n const summary = getBlockSummary(block.block);\n const blockId = typeof block.block.id === \"string\" ? block.block.id : null;\n const contentControls = (\n <BlockContentControls\n block={block.block}\n rowData={blockId ? data[blockId] : undefined}\n onChangeBlock={(nextBlock) => onChangeBlock(block.uid, nextBlock)}\n onChangeRowData={\n blockId && onChangeData\n ? (nextRowData) => onChangeData({ ...data, [blockId]: nextRowData })\n : undefined\n }\n />\n );\n\n return (\n <InspectorShell\n ref={shellRef}\n ariaLabel=\"Block inspector\"\n className={className}\n onKeyDown={handleKeyDown}\n >\n <InspectorHeader\n chip={<Chip>{chrome.chip}</Chip>}\n title={chrome.label}\n subtitle={summary || undefined}\n action={\n <button\n type=\"button\"\n className=\"inline-grid h-[22px] w-[22px] cursor-pointer place-items-center rounded border border-solid border-border bg-surface/90 p-0 text-fg-muted transition-colors hover:border-border-strong hover:text-fg\"\n aria-label=\"Close inspector\"\n onClick={onClose}\n >\n ✕\n </button>\n }\n />\n\n <TextField\n name=\"block.id\"\n label=\"ID\"\n value={blockId ?? \"\"}\n placeholder=\"Optional identifier\"\n onChange={(value) =>\n onChangeBlock(block.uid, { ...block.block, id: value ? value : undefined })\n }\n />\n\n <div className=\"grid gap-2\" aria-label=\"Inspector sections\">\n {contentControls}\n {detailSections.map((section) => (\n <InspectorSection key={section} title={section}>\n {section === \"Layout\" ? (\n <BlockLayoutControls\n block={block.block}\n onChangeBlock={(nextBlock) => onChangeBlock(block.uid, nextBlock)}\n />\n ) : section === \"Spacing\" ? (\n <SpacingControls\n scope=\"block\"\n block={block.block}\n onChangeBlock={(nextBlock) => onChangeBlock(block.uid, nextBlock)}\n />\n ) : (\n <TypographyControls\n target=\"block\"\n block={block.block}\n metadata={schema[\"x-pdfUa\"]}\n onChangeBlock={(nextBlock) => onChangeBlock(block.uid, nextBlock)}\n />\n )}\n </InspectorSection>\n ))}\n </div>\n\n <footer className=\"flex justify-end\">\n <Button variant=\"danger\" onClick={() => onRemoveBlock(block.uid)}>\n Remove block\n </Button>\n </footer>\n </InspectorShell>\n );\n}\n\n","import { useBuilderActions, useBuilderState } from \"../context/BuilderContext\";\nimport { BlockInspector } from \"./BlockInspector\";\n\n// Flyout panel pinned to the right edge of the canvas, shown while a block is selected.\nconst flyoutClass =\n \"absolute inset-y-0 right-0 z-20 w-[min(360px,100%)] border-0 border-l border-solid border-border shadow-pop animate-flyout\";\n\nexport function Inspector() {\n const { schema, selectedBlock, data } = useBuilderState();\n const { changeBlock, changeData, removeBlock, deselect } = useBuilderActions();\n\n if (!schema || !selectedBlock) {\n return null;\n }\n\n return (\n <BlockInspector\n className={flyoutClass}\n block={selectedBlock}\n schema={schema}\n data={data}\n onChangeBlock={changeBlock}\n onChangeData={changeData}\n onRemoveBlock={removeBlock}\n onClose={deselect}\n />\n );\n}\n","import type { ReactNode } from \"react\";\nimport type { Orientation, PageFormat, Template } from \"../../types/generated/template\";\nimport type { TemplateSchemaMetadata } from \"../../types/template\";\nimport { SelectField } from \"../controls\";\nimport { PAGE_SIZES_MM } from \"../lib/pageSizes\";\nimport { InspectorSection } from \"./InspectorShell\";\nimport { SpacingControls } from \"./SpacingControls\";\nimport { TypographyControls } from \"./TypographyControls\";\n\nconst pageFormatOptions = Object.keys(PAGE_SIZES_MM).map((format) => ({\n value: format as PageFormat,\n label: format,\n}));\n\nconst orientationOptions = [\n { value: \"portrait\", label: \"Portrait\" },\n { value: \"landscape\", label: \"Landscape\" },\n] as const satisfies ReadonlyArray<{ value: Orientation; label: string }>;\n\nexport interface DocumentSettingsProps {\n template: Template;\n metadata?: Pick<TemplateSchemaMetadata, \"bundledFonts\">;\n format: PageFormat;\n orientation: Orientation;\n onChangeTemplate: (template: Template) => void;\n onChangeFormat: (format: PageFormat) => void;\n onChangeOrientation: (orientation: Orientation) => void;\n className?: string;\n}\n\n/**\n * Document-wide settings, laid out as a horizontal bar above the block palette.\n * Each group wraps to its own column on narrow widths.\n */\nexport function DocumentSettings({\n template,\n metadata,\n format,\n orientation,\n onChangeTemplate,\n onChangeFormat,\n onChangeOrientation,\n className,\n}: DocumentSettingsProps): ReactNode {\n return (\n <section\n aria-label=\"Page settings\"\n className={`grid grid-cols-[repeat(auto-fit,minmax(220px,1fr))] items-start gap-3 border-0 border-b border-solid border-border bg-surface px-4 py-3${className ? ` ${className}` : \"\"}`}\n >\n <InspectorSection title=\"Page setup\">\n <div className=\"grid gap-2\">\n <SelectField\n name=\"document.page.size.format\"\n label=\"Page size\"\n value={format}\n options={pageFormatOptions}\n onChange={(value) => {\n if (value) {\n onChangeFormat(value);\n }\n }}\n />\n <SelectField\n name=\"document.page.size.orientation\"\n label=\"Orientation\"\n value={orientation}\n options={orientationOptions}\n onChange={(value) => {\n if (value) {\n onChangeOrientation(value);\n }\n }}\n />\n </div>\n </InspectorSection>\n\n <InspectorSection title=\"Page margins\">\n <SpacingControls scope=\"page\" template={template} onChangeTemplate={onChangeTemplate} />\n </InspectorSection>\n\n <InspectorSection title=\"Template typography\">\n <TypographyControls\n target=\"template\"\n template={template}\n metadata={metadata}\n onChangeTemplate={onChangeTemplate}\n />\n </InspectorSection>\n </section>\n );\n}\n","import { useBuilderActions, useBuilderState } from \"../context/BuilderContext\";\nimport { DocumentSettings } from \"./DocumentSettings\";\n\nexport interface PageSettingsProps {\n className?: string;\n}\n\nexport function PageSettings({ className }: PageSettingsProps = {}) {\n const { schema, serializedTemplate, pageSize } = useBuilderState();\n const { changeTemplateSettings, changeFormat, changeOrientation } = useBuilderActions();\n\n if (!schema) {\n return null;\n }\n\n return (\n <DocumentSettings\n className={className}\n template={serializedTemplate}\n metadata={schema[\"x-pdfUa\"]}\n format={pageSize.format}\n orientation={pageSize.orientation}\n onChangeTemplate={changeTemplateSettings}\n onChangeFormat={changeFormat}\n onChangeOrientation={changeOrientation}\n />\n );\n}\n","import { Palette } from \"./blocks/Palette\";\nimport { Canvas } from \"./canvas/Canvas\";\nimport type { TemplateExample } from \"./context/BuilderContext\";\nimport { Inspector } from \"./inspector/Inspector\";\nimport { PageSettings } from \"./inspector/PageSettings\";\n\nexport interface BuilderProps {\n className?: string;\n /** Loadable examples surfaced in the palette, keyed by display name. */\n examples?: Record<string, TemplateExample>;\n}\n\nexport function Builder({ className, examples }: BuilderProps = {}) {\n return (\n <section\n className={`pdfua-template-builder grid h-full min-w-0 min-h-0 bg-app${className ? ` ${className}` : \"\"}`}\n style={{ gridTemplateRows: \"auto auto 1fr\" }}\n aria-label=\"Template authoring\"\n >\n <PageSettings />\n <Palette examples={examples} />\n\n <div className=\"relative mt-2 min-h-0 min-w-0\">\n <Canvas className=\"h-full\" />\n <Inspector />\n </div>\n </section>\n );\n}\n","import type { ReactNode } from \"react\";\n\nexport type OutputStatus = \"ready\" | \"rendering\" | \"empty\";\n\nexport function deriveStatus(loading: boolean, pdfUrl: string | null): OutputStatus {\n if (loading) return \"rendering\";\n return pdfUrl ? \"ready\" : \"empty\";\n}\n\nexport function statusLabel(status: OutputStatus): string {\n if (status === \"rendering\") return \"Rendering…\";\n return status === \"ready\" ? \"Ready\" : \"Idle\";\n}\n\nconst statusPillVariantClass: Record<OutputStatus, string> = {\n ready: \"bg-success-soft text-success\",\n rendering: \"bg-accent-soft text-accent\",\n empty: \"bg-surface-muted text-fg-muted\",\n};\n\nexport function StatusPill({ status, children }: { status: OutputStatus; children: ReactNode }) {\n return (\n <span\n className={`inline-flex items-center gap-2 rounded-full px-3 py-1 text-2xs font-medium ${statusPillVariantClass[status]}`}\n >\n <span className=\"h-1.5 w-1.5 rounded-full bg-current opacity-60\" aria-hidden=\"true\" />\n {children}\n </span>\n );\n}\n\nexport function Tab({\n active,\n onClick,\n children,\n}: {\n active: boolean;\n onClick: () => void;\n children: ReactNode;\n}) {\n return (\n <button\n type=\"button\"\n role=\"tab\"\n className={`h-full m-0 cursor-pointer border-0 border-b-2 border-solid bg-transparent px-3 text-sm font-medium tracking-[0.02em] transition-colors hover:text-fg ${active ? \"border-fg text-fg\" : \"border-transparent text-fg-muted\"}`}\n aria-selected={active}\n onClick={onClick}\n >\n {children}\n </button>\n );\n}\n\nexport function PdfView({\n pdfUrl,\n loading,\n emptyLabel = \"Render the template to preview the PDF here.\",\n loadingLabel = \"Rendering the latest changes…\",\n}: {\n pdfUrl: string | null;\n loading: boolean;\n emptyLabel?: string;\n loadingLabel?: string;\n}) {\n if (pdfUrl) {\n return (\n <object\n data={pdfUrl}\n type=\"application/pdf\"\n // The rendered PDF is white paper, so its frame stays light in dark mode.\n data-theme=\"light\"\n className=\"h-full w-full min-h-0 rounded-lg border border-solid border-border bg-page shadow-page max-[1080px]:h-[34rem]\"\n />\n );\n }\n return (\n <div\n data-theme=\"light\"\n className=\"grid h-full place-items-center rounded-lg border border-dashed border-border-strong bg-page p-6 text-center text-sm text-fg-muted max-[1080px]:h-[34rem]\"\n >\n {loading ? loadingLabel : emptyLabel}\n </div>\n );\n}\n","import { useState } from \"react\";\nimport type { Template } from \"../types/generated/template\";\nimport type { TemplateData } from \"../types/template\";\nimport { Button } from \"../builder/primitives/Button\";\nimport { PdfView, StatusPill, Tab, deriveStatus, statusLabel } from \"./previewChrome\";\n\nexport type OutputTab = \"pdf\" | \"data\";\n\nexport interface PdfPaneProps {\n pdfUrl: string | null;\n error: string | null;\n loading: boolean;\n template?: Template;\n data?: TemplateData;\n className?: string;\n onRender?: () => void;\n renderDisabled?: boolean;\n}\n\nexport function PdfPane({\n pdfUrl,\n error,\n loading,\n template,\n data,\n className,\n onRender,\n renderDisabled,\n}: PdfPaneProps) {\n const [tab, setTab] = useState<OutputTab>(\"pdf\");\n const status = deriveStatus(loading, pdfUrl);\n\n return (\n <aside\n className={`pdfua-template-builder grid min-w-0 min-h-0 grid-rows-[56px_auto_minmax(0,1fr)] bg-canvas${className ? ` ${className}` : \"\"}`}\n aria-label=\"Output\"\n >\n <header className=\"row-start-1 flex items-center justify-between gap-3 border-0 border-b border-solid border-border bg-surface px-4\">\n <div className=\"inline-flex h-full items-stretch gap-0.5\" role=\"tablist\" aria-label=\"Output view\">\n <Tab active={tab === \"pdf\"} onClick={() => setTab(\"pdf\")}>\n PDF\n </Tab>\n <Tab active={tab === \"data\"} onClick={() => setTab(\"data\")}>\n Data\n </Tab>\n {tab === \"data\" ? <CopyJsonButton template={template} data={data} /> : null}\n </div>\n <div className=\"flex items-center gap-3\">\n <StatusPill status={status}>{statusLabel(status)}</StatusPill>\n {onRender ? (\n <Button variant=\"primary\" onClick={onRender} disabled={renderDisabled}>\n {loading ? \"Rendering…\" : \"Render PDF\"}\n </Button>\n ) : null}\n </div>\n </header>\n\n {error ? (\n <p\n className=\"row-start-2 mx-4 mt-3 mb-0 rounded border-0 border-l-[3px] border-solid border-danger bg-danger-soft px-3 py-3 text-sm text-danger\"\n role=\"alert\"\n >\n {error}\n </p>\n ) : null}\n\n <div className=\"row-start-3 grid min-h-0 p-4\">\n {tab === \"pdf\" ? (\n <PdfView\n pdfUrl={pdfUrl}\n loading={loading}\n loadingLabel=\"Rendering the latest template…\"\n emptyLabel=\"Render the template to preview the PDF here.\"\n />\n ) : (\n <DataView template={template} data={data} />\n )}\n </div>\n </aside>\n );\n}\n\nfunction DataView({ template, data }: { template?: Template; data?: TemplateData }) {\n const payload = { template: template ?? null, data: data ?? {} };\n\n return (\n <pre\n className=\"h-full w-full min-h-0 rounded-lg border border-solid border-border bg-surface shadow-page m-0 overflow-auto p-4 font-mono text-sm leading-normal text-fg [tab-size:2] whitespace-pre max-[1080px]:h-[34rem]\"\n >\n <code>{JSON.stringify(payload, null, 2)}</code>\n </pre>\n );\n}\n\nfunction CopyJsonButton({ template, data }: { template?: Template; data?: TemplateData }) {\n const [copied, setCopied] = useState(false);\n\n function handleCopy() {\n const payload = JSON.stringify({ template: template ?? null, data: data ?? {} }, null, 2);\n\n void navigator.clipboard.writeText(payload).then(() => {\n setCopied(true);\n setTimeout(() => setCopied(false), 1500);\n });\n }\n\n return (\n <button\n type=\"button\"\n className=\"m-0 ml-2 h-[26px] cursor-pointer self-center rounded-md border border-solid border-border bg-surface px-3 text-2xs font-medium text-fg-muted transition-colors hover:border-border-strong hover:text-fg\"\n onClick={handleCopy}\n >\n {copied ? \"✓ Copied\" : \"⧉ Copy JSON\"}\n </button>\n );\n}\n","import { PdfPane } from \"./PdfPane\";\nimport { useRenderContext } from \"./RenderContext\";\n\nexport interface PreviewProps {\n className?: string;\n}\n\nexport function Preview({ className }: PreviewProps = {}) {\n const { template, data, pdfUrl, pdfLoading, error, renderPdf, renderDisabled } =\n useRenderContext();\n\n return (\n <PdfPane\n className={className}\n pdfUrl={pdfUrl}\n error={error}\n loading={pdfLoading}\n template={template ?? undefined}\n data={data}\n onRender={renderPdf}\n renderDisabled={renderDisabled}\n />\n );\n}\n","import type { Template } from \"../types/generated/template\";\nimport type { TemplateData } from \"../types/template\";\nimport { Builder } from \"./Builder\";\nimport { TemplateBuilderProvider, type TemplateExample } from \"./context/BuilderContext\";\nimport { Preview } from \"../render/Preview\";\n\nexport interface TemplateBuilderProps {\n /** Base URL of a running pdf-ua-api instance. Defaults to \"\" (relative URLs / proxy). */\n apiUrl?: string;\n /** Template loaded into the editor on first render. */\n initialTemplate?: Template;\n /** Runtime data keyed by block id (table rows, dynamic key-value overrides). */\n initialData?: TemplateData;\n /** Loadable examples surfaced in the palette, keyed by display name. */\n examples?: Record<string, TemplateExample>;\n /** Fires whenever the user edits the template or its runtime data. */\n onChange?: (template: Template, data: TemplateData) => void;\n /** Fires after a successful render with the produced PDF blob. */\n onRendered?: (pdf: Blob) => void;\n /** Optional className appended to the root element. */\n className?: string;\n}\n\nfunction DefaultLayout({\n examples,\n className,\n}: {\n examples?: Record<string, TemplateExample>;\n className?: string;\n}) {\n return (\n <main\n className={`pdfua-template-builder grid h-screen overflow-hidden bg-app text-fg grid-cols-[minmax(40rem,1.55fr)_minmax(28rem,0.95fr)] max-[1080px]:h-auto max-[1080px]:grid-cols-1 max-[1080px]:overflow-visible${className ? ` ${className}` : \"\"}`}\n >\n <Builder examples={examples} className=\"border-0 border-r border-solid border-border\" />\n <Preview />\n </main>\n );\n}\n\n/**\n * All-in-one preset: a provider wrapping a Builder and a Preview side by side.\n *\n * For custom layouts (e.g. preview below the builder), compose the parts\n * directly with `TemplateBuilderProvider`, `Builder`, and `Preview`.\n */\nexport function TemplateBuilder({\n apiUrl,\n initialTemplate,\n initialData,\n examples,\n onChange,\n onRendered,\n className,\n}: TemplateBuilderProps = {}) {\n return (\n <TemplateBuilderProvider\n apiUrl={apiUrl}\n initialTemplate={initialTemplate}\n initialData={initialData}\n onChange={onChange}\n onRendered={onRendered}\n >\n <DefaultLayout examples={examples} className={className} />\n </TemplateBuilderProvider>\n );\n}\n","import type { Template } from \"../../types/generated/template\";\n\nexport type TableRow = Record<string, string>;\nexport type InvoiceData = Record<string, TableRow[]>;\n\nconst LOGO_SRC =\n \"data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHdpZHRoPSIyNjAiIGhlaWdodD0iNzIiIHZpZXdCb3g9IjAgMCAyNjAgNzIiPjx0ZXh0IHg9IjAiIHk9IjQyIiBmaWxsPSIjMTExODI3IiBmb250LWZhbWlseT0iQXJpYWwsIHNhbnMtc2VyaWYiIGZvbnQtc2l6ZT0iMzAiIGZvbnQtd2VpZ2h0PSI3MDAiPlBERiBVQSBLaXQ8L3RleHQ+PC9zdmc+\";\n\nexport interface InvoiceExample {\n template: Template;\n data: InvoiceData;\n}\n\nexport function createInvoiceExample(): InvoiceExample {\n const template: Template = {\n version: 1,\n config: {\n page: {\n size: { format: \"A4\", orientation: \"portrait\" },\n locale: \"de_DE\",\n margins: { top: 20, right: 20, bottom: 20, left: 25 },\n pageNumbers: { enabled: true, position: \"center\" },\n footer: {\n repeat: true,\n rows: [\n {\n blocks: [\n {\n type: \"text\",\n id: \"footer-legal\",\n text: \"PDF UA Kit GmbH · Musterstraße 1 · 10115 Berlin · Germany · Invoice was created electronically and is valid without signature.\",\n config: { width: \"68%\" },\n },\n {\n type: \"key-value\",\n id: \"footer-meta\",\n values: {\n registration: \"HRB 123456 B\",\n taxNumber: \"DE123456789\",\n },\n config: {\n width: \"32%\",\n align: \"right\",\n fields: [\n { key: \"registration\", label: \"Registry\" },\n { key: \"taxNumber\", label: \"Tax no.\" },\n ],\n },\n },\n ],\n },\n ],\n },\n },\n typography: { family: \"Inter\", size: 10 },\n },\n rows: [\n {\n blocks: [\n {\n type: \"image\",\n id: \"logo\",\n src: LOGO_SRC,\n alt: \"PDF UA Kit GmbH logo\",\n config: { width: \"58%\", maxHeight: 28 },\n },\n {\n type: \"key-value\",\n id: \"invoice-meta\",\n values: {\n invoiceNumber: \"RE-2026-001234\",\n issueDate: \"2026-02-17\",\n dueDate: \"2026-03-19\",\n currency: \"EUR\",\n },\n config: {\n width: \"42%\",\n align: \"right\",\n fields: [\n { key: \"invoiceNumber\", label: \"Invoice number\" },\n { key: \"issueDate\", label: \"Issue date\" },\n { key: \"dueDate\", label: \"Due date\" },\n { key: \"currency\", label: \"Currency\" },\n ],\n },\n },\n ],\n },\n {\n blocks: [\n {\n type: \"heading\",\n id: \"title\",\n text: \"Invoice\",\n config: { level: 1 },\n },\n ],\n },\n {\n blocks: [\n {\n type: \"key-value\",\n id: \"seller\",\n values: {\n name: \"PDF UA Kit GmbH\",\n address: \"Musterstraße 1, 10115 Berlin, DE\",\n contact: \"Max Mustermann\",\n email: \"billing@pdfua-kit.example\",\n vatId: \"DE123456789\",\n },\n config: {\n width: \"50%\",\n fields: [\n { key: \"name\", label: \"Seller\" },\n { key: \"address\", label: \"Address\" },\n { key: \"contact\", label: \"Contact\" },\n { key: \"email\", label: \"Email\" },\n { key: \"vatId\", label: \"VAT ID\" },\n ],\n },\n },\n {\n type: \"key-value\",\n id: \"buyer\",\n values: {\n name: \"Musterkunde AG\",\n address: \"Käuferweg 2, 80331 München, DE\",\n email: \"invoice@musterkunde.example\",\n reference: \"04011000-12345-67\",\n },\n config: {\n width: \"50%\",\n fields: [\n { key: \"name\", label: \"Buyer\" },\n { key: \"address\", label: \"Address\" },\n { key: \"email\", label: \"Email\" },\n { key: \"reference\", label: \"Buyer reference\" },\n ],\n },\n },\n ],\n },\n {\n blocks: [{ type: \"divider\", id: \"address-rule\" }],\n },\n {\n blocks: [\n {\n type: \"table\",\n id: \"lineItems\",\n config: {\n style: \"striped\",\n numberRows: true,\n columns: [\n { key: \"description\", label: \"Description\", align: \"left\", width: \"38%\" },\n { key: \"quantity\", label: \"Qty\", align: \"right\", width: \"12%\" },\n { key: \"unitPrice\", label: \"Unit price\", align: \"right\", width: \"16%\" },\n { key: \"vatRate\", label: \"VAT\", align: \"right\", width: \"11%\" },\n { key: \"total\", label: \"Total\", align: \"right\", width: \"16%\" },\n ],\n },\n },\n ],\n },\n {\n blocks: [\n {\n type: \"table\",\n id: \"vat-breakdown\",\n config: {\n style: \"minimal\",\n width: \"54%\",\n numberRows: false,\n columns: [\n { key: \"vatCategory\", label: \"VAT category\", align: \"left\" },\n { key: \"rate\", label: \"Rate\", align: \"right\" },\n { key: \"taxableAmount\", label: \"Taxable amount\", align: \"right\" },\n { key: \"vatAmount\", label: \"VAT amount\", align: \"right\" },\n ],\n },\n },\n {\n type: \"key-value\",\n id: \"totals\",\n values: {\n netAmount: \"6.120,00 €\",\n vatAmount: \"1.162,80 €\",\n grandTotal: \"7.282,80 €\",\n amountDue: \"7.282,80 €\",\n },\n config: {\n width: \"46%\",\n align: \"right\",\n fields: [\n { key: \"netAmount\", label: \"Net amount\" },\n { key: \"vatAmount\", label: \"VAT 19%\" },\n { key: \"grandTotal\", label: \"Grand total\" },\n { key: \"amountDue\", label: \"Amount due\" },\n ],\n },\n },\n ],\n },\n {\n blocks: [\n {\n type: \"text\",\n id: \"notice\",\n text: \"Please transfer the amount due within 30 days. Include the invoice number as the payment reference.\",\n config: { width: \"54%\", spacing: { top: 4 } },\n },\n {\n type: \"key-value\",\n id: \"payment\",\n values: {\n bank: \"Musterbank Berlin\",\n iban: \"DE89370400440532013000\",\n bic: \"COBADEFFXXX\",\n reference: \"RE-2026-001234\",\n },\n config: {\n width: \"46%\",\n align: \"right\",\n fields: [\n { key: \"bank\", label: \"Bank\" },\n { key: \"iban\", label: \"IBAN\" },\n { key: \"bic\", label: \"BIC\" },\n { key: \"reference\", label: \"Payment reference\" },\n ],\n },\n },\n ],\n },\n ],\n };\n\n const data: InvoiceData = {\n lineItems: [\n {\n description: \"Accessible PDF template implementation\",\n quantity: \"40\",\n unitPrice: \"95,00 €\",\n vatRate: \"19%\",\n total: \"3.800,00 €\",\n },\n {\n description: \"Document structure and tagging review\",\n quantity: \"16\",\n unitPrice: \"85,00 €\",\n vatRate: \"19%\",\n total: \"1.360,00 €\",\n },\n {\n description: \"Project management and acceptance testing\",\n quantity: \"8\",\n unitPrice: \"90,00 €\",\n vatRate: \"19%\",\n total: \"720,00 €\",\n },\n {\n description: \"Annual hosting and maintenance package\",\n quantity: \"1\",\n unitPrice: \"240,00 €\",\n vatRate: \"19%\",\n total: \"240,00 €\",\n },\n ],\n \"vat-breakdown\": [\n {\n vatCategory: \"Standard rate\",\n rate: \"19%\",\n taxableAmount: \"6.120,00 €\",\n vatAmount: \"1.162,80 €\",\n },\n ],\n };\n\n return { template, data };\n}\n","import type { Template } from \"../types/generated/template\";\n\nexport interface ParsedTemplate {\n template: Template | null;\n error: string | null;\n}\n\nexport function parseTemplate(text: string): ParsedTemplate {\n if (text.trim() === \"\") {\n return { template: null, error: null };\n }\n\n let parsed: unknown;\n try {\n parsed = JSON.parse(text);\n } catch (cause) {\n return { template: null, error: cause instanceof Error ? cause.message : String(cause) };\n }\n\n if (typeof parsed !== \"object\" || parsed === null || Array.isArray(parsed)) {\n return { template: null, error: \"Template must be a JSON object.\" };\n }\n\n return { template: parsed as Template, error: null };\n}\n","import {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useRef,\n useState,\n type ReactNode,\n} from \"react\";\nimport { resolveDefaultApiUrl } from \"../api/pdfUaApi\";\nimport type { Template } from \"../types/generated/template\";\nimport type { TemplateData } from \"../types/template\";\nimport { RenderProvider, type RenderContextValue } from \"../render/RenderContext\";\nimport { usePdfUaApi } from \"../render/usePdfUaApi\";\nimport { parseTemplate } from \"./parseTemplate\";\n\nexport interface TemplateEditorProviderProps {\n apiUrl?: string;\n initialTemplate?: Template;\n data?: TemplateData;\n onChange?: (template: Template | null, text: string) => void;\n onRendered?: (pdf: Blob) => void;\n children: ReactNode;\n}\n\nexport interface TemplateEditorContextValue {\n text: string;\n setText: (text: string) => void;\n template: Template | null;\n error: string | null;\n data: TemplateData;\n}\n\nconst emptyTemplate: Template = { version: 1 };\nconst emptyData: TemplateData = {};\n\nconst TemplateEditorContext = createContext<TemplateEditorContextValue | null>(null);\n\nexport function useTemplateEditor(): TemplateEditorContextValue {\n const value = useContext(TemplateEditorContext);\n\n if (!value) {\n throw new Error(\"useTemplateEditor must be used within a <TemplateEditorProvider>.\");\n }\n\n return value;\n}\n\nexport function TemplateEditorProvider({\n apiUrl: apiUrlProp,\n initialTemplate,\n data = emptyData,\n onChange,\n onRendered,\n children,\n}: TemplateEditorProviderProps) {\n const apiUrl = resolveDefaultApiUrl(apiUrlProp);\n const [text, setText] = useState(() =>\n JSON.stringify(initialTemplate ?? emptyTemplate, null, 2),\n );\n\n const { template, error } = useMemo(() => parseTemplate(text), [text]);\n\n const onChangeRef = useRef(onChange);\n useEffect(() => {\n onChangeRef.current = onChange;\n }, [onChange]);\n const skipNextChange = useRef(true);\n useEffect(() => {\n if (skipNextChange.current) {\n skipNextChange.current = false;\n return;\n }\n onChangeRef.current?.(template, text);\n }, [text, template]);\n\n const {\n pdfUrl,\n pdfLoading,\n error: renderError,\n renderPdf: renderPdfRequest,\n } = usePdfUaApi({ initialApiUrl: apiUrl, apiUrl, onRendered, loadSchemaOnMount: false });\n\n const templateRef = useRef(template);\n templateRef.current = template;\n const dataRef = useRef(data);\n dataRef.current = data;\n\n const renderPdf = useCallback(() => {\n if (templateRef.current) {\n void renderPdfRequest(templateRef.current, dataRef.current);\n }\n }, [renderPdfRequest]);\n\n const editorValue = useMemo<TemplateEditorContextValue>(\n () => ({ text, setText, template, error, data }),\n [text, template, error, data],\n );\n\n const renderValue = useMemo<RenderContextValue>(\n () => ({\n template,\n data,\n pdfUrl,\n pdfLoading,\n error: renderError,\n renderPdf,\n renderDisabled: template === null || pdfLoading,\n }),\n [template, data, pdfUrl, pdfLoading, renderError, renderPdf],\n );\n\n return (\n <TemplateEditorContext.Provider value={editorValue}>\n <RenderProvider value={renderValue}>{children}</RenderProvider>\n </TemplateEditorContext.Provider>\n );\n}\n","import { HighlightStyle, syntaxHighlighting } from \"@codemirror/language\";\nimport { EditorView } from \"@codemirror/view\";\nimport { tags as t } from \"@lezer/highlight\";\n\nconst baseTheme = EditorView.theme({\n \"&\": {\n color: \"var(--pdfua-fg)\",\n backgroundColor: \"var(--pdfua-surface)\",\n fontSize: \"13px\",\n height: \"100%\",\n },\n \"&.cm-focused\": { outline: \"none\" },\n \".cm-content\": {\n fontFamily: \"ui-monospace, SFMono-Regular, Menlo, Consolas, monospace\",\n caretColor: \"var(--pdfua-fg)\",\n },\n \".cm-gutters\": {\n backgroundColor: \"var(--pdfua-surface)\",\n color: \"var(--pdfua-fg-subtle)\",\n border: \"none\",\n },\n \".cm-activeLine\": {\n backgroundColor: \"color-mix(in oklab, var(--pdfua-accent) 8%, transparent)\",\n },\n \".cm-activeLineGutter\": { backgroundColor: \"transparent\" },\n \".cm-cursor, .cm-dropCursor\": { borderLeftColor: \"var(--pdfua-fg)\" },\n \"&.cm-focused .cm-selectionBackground, .cm-selectionBackground, .cm-content ::selection\": {\n backgroundColor: \"color-mix(in oklab, var(--pdfua-accent) 22%, transparent)\",\n },\n});\n\nconst highlightStyle = HighlightStyle.define([\n { tag: t.propertyName, color: \"var(--pdfua-accent)\" },\n { tag: t.string, color: \"var(--pdfua-fg)\" },\n { tag: [t.number, t.bool, t.null], color: \"var(--pdfua-accent)\" },\n { tag: [t.punctuation, t.separator], color: \"var(--pdfua-fg-muted)\" },\n]);\n\n// Editor styling driven by the --pdfua-* tokens, so it follows light/dark automatically.\nexport const editorTheme = [baseTheme, syntaxHighlighting(highlightStyle)];\n","{\n \"$schema\": \"https://json-schema.org/draft/2020-12/schema\",\n \"$id\": \"https://pdf-ua-api.com/schemas/template-v1.json\",\n \"title\": \"Template\",\n \"type\": \"object\",\n \"properties\": {\n \"version\": {\n \"type\": \"integer\",\n \"const\": 1\n },\n \"config\": {\n \"$ref\": \"#/$defs/templateConfig\"\n },\n \"fonts\": {\n \"type\": \"object\",\n \"description\": \"External fonts keyed by font family name.\",\n \"additionalProperties\": {\n \"$ref\": \"#/$defs/fontFace\"\n }\n },\n \"attachments\": {\n \"type\": \"array\",\n \"items\": {\n \"$ref\": \"#/$defs/fileAttachment\"\n }\n },\n \"rows\": {\n \"type\": \"array\",\n \"items\": {\n \"$ref\": \"#/$defs/row\"\n }\n }\n },\n \"additionalProperties\": false,\n \"required\": [\n \"version\"\n ],\n \"$defs\": {\n \"align\": {\n \"type\": \"string\",\n \"title\": \"Align\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\"\n ]\n },\n \"pageFormat\": {\n \"type\": \"string\",\n \"title\": \"PageFormat\",\n \"enum\": [\n \"A3\",\n \"A4\",\n \"A5\",\n \"A6\",\n \"Letter\",\n \"Legal\",\n \"Tabloid\"\n ]\n },\n \"orientation\": {\n \"type\": \"string\",\n \"title\": \"Orientation\",\n \"enum\": [\n \"portrait\",\n \"landscape\"\n ]\n },\n \"dividerStyle\": {\n \"type\": \"string\",\n \"title\": \"DividerStyle\",\n \"enum\": [\n \"solid\",\n \"dashed\",\n \"dotted\",\n \"double\",\n \"none\"\n ]\n },\n \"tableStyle\": {\n \"type\": \"string\",\n \"title\": \"TableStyle\",\n \"enum\": [\n \"striped\",\n \"bordered\",\n \"minimal\"\n ]\n },\n \"pageBackgroundType\": {\n \"type\": \"string\",\n \"title\": \"PageBackgroundType\",\n \"enum\": [\n \"auto\",\n \"image\",\n \"pdf\"\n ]\n },\n \"typographyConfig\": {\n \"type\": \"object\",\n \"title\": \"TypographyConfig\",\n \"properties\": {\n \"family\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Bundled or external font family key.\"\n },\n \"size\": {\n \"type\": [\n \"integer\",\n \"null\"\n ],\n \"description\": \"Font size in points.\",\n \"minimum\": 1.0\n },\n \"weight\": {\n \"type\": [\n \"integer\",\n \"null\"\n ],\n \"description\": \"Numeric font weight.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Text alignment for this typography scope.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"color\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS color value used for text.\"\n }\n },\n \"additionalProperties\": false\n },\n \"spacingConfig\": {\n \"type\": \"object\",\n \"title\": \"SpacingConfig\",\n \"properties\": {\n \"top\": {\n \"type\": [\n \"integer\",\n \"null\"\n ],\n \"description\": \"Top spacing in millimetres.\",\n \"minimum\": 0.0\n },\n \"right\": {\n \"type\": [\n \"integer\",\n \"null\"\n ],\n \"description\": \"Right spacing in millimetres.\",\n \"minimum\": 0.0\n },\n \"bottom\": {\n \"type\": [\n \"integer\",\n \"null\"\n ],\n \"description\": \"Bottom spacing in millimetres.\",\n \"minimum\": 0.0\n },\n \"left\": {\n \"type\": [\n \"integer\",\n \"null\"\n ],\n \"description\": \"Left spacing in millimetres.\",\n \"minimum\": 0.0\n }\n },\n \"additionalProperties\": false\n },\n \"blockConfig\": {\n \"type\": \"object\",\n \"title\": \"BlockConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"{ typography?: TypographyConfig; spacing?: SpacingConfig; width?: string | null; align?: Align | null }\"\n },\n \"headingConfig\": {\n \"type\": \"object\",\n \"title\": \"HeadingConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"level\": {\n \"type\": \"integer\",\n \"default\": 2,\n \"minimum\": 1.0,\n \"maximum\": 6.0\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"BlockConfig & { level?: number }\"\n },\n \"imageConfig\": {\n \"type\": \"object\",\n \"title\": \"ImageConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"maxHeight\": {\n \"type\": \"integer\",\n \"default\": 60,\n \"minimum\": 1.0\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"BlockConfig & { maxHeight?: number }\"\n },\n \"keyValueField\": {\n \"type\": \"object\",\n \"title\": \"KeyValueField\",\n \"properties\": {\n \"key\": {\n \"type\": \"string\",\n \"pattern\": \"^[A-Za-z][A-Za-z0-9_]*$\"\n },\n \"label\": {\n \"type\": \"string\"\n }\n },\n \"required\": [\n \"key\",\n \"label\"\n ],\n \"additionalProperties\": false,\n \"tsType\": \"{ key: string; label: string }\"\n },\n \"keyValueConfig\": {\n \"type\": \"object\",\n \"title\": \"KeyValueConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"labelWidth\": {\n \"type\": \"string\",\n \"default\": \"30mm\"\n },\n \"fields\": {\n \"type\": \"array\",\n \"items\": {\n \"$ref\": \"#/$defs/keyValueField\"\n }\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"BlockConfig & { labelWidth?: string; fields?: KeyValueField[] }\"\n },\n \"spacerConfig\": {\n \"type\": \"object\",\n \"title\": \"SpacerConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"height\": {\n \"type\": \"integer\",\n \"default\": 5,\n \"minimum\": 0.0\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"BlockConfig & { height?: number }\"\n },\n \"dividerConfig\": {\n \"type\": \"object\",\n \"title\": \"DividerConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"thickness\": {\n \"type\": \"integer\",\n \"default\": 1,\n \"minimum\": 0.0\n },\n \"lineColor\": {\n \"type\": \"string\",\n \"default\": \"#d1d5db\",\n \"pattern\": \"^#[0-9A-Fa-f]{3,8}$\"\n },\n \"style\": {\n \"$ref\": \"#/$defs/dividerStyle\"\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"BlockConfig & { thickness?: number; lineColor?: string; style?: DividerStyle }\"\n },\n \"tableColumn\": {\n \"type\": \"object\",\n \"title\": \"TableColumn\",\n \"properties\": {\n \"key\": {\n \"type\": \"string\",\n \"description\": \"Runtime data key used for this table column.\",\n \"pattern\": \"^[A-Za-z][A-Za-z0-9_]*$\"\n },\n \"label\": {\n \"type\": \"string\",\n \"description\": \"Header label rendered for this table column.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Text alignment for this table column.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Column width as a CSS width value, such as 20mm or 15%.\"\n }\n },\n \"required\": [\n \"key\",\n \"label\"\n ],\n \"additionalProperties\": false\n },\n \"tableConfig\": {\n \"type\": \"object\",\n \"title\": \"TableConfig\",\n \"properties\": {\n \"typography\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/typographyConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"spacing\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"width\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"CSS width for this block, such as 50%, 80mm, or auto.\"\n },\n \"align\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Horizontal placement of this block within its row cell.\",\n \"enum\": [\n \"left\",\n \"center\",\n \"right\",\n null\n ]\n },\n \"numberRows\": {\n \"type\": \"boolean\",\n \"default\": false\n },\n \"columns\": {\n \"type\": \"array\",\n \"items\": {\n \"$ref\": \"#/$defs/tableColumn\"\n }\n },\n \"style\": {\n \"$ref\": \"#/$defs/tableStyle\"\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"BlockConfig & { numberRows?: boolean; columns?: TableColumn[]; style?: TableStyle }\"\n },\n \"textBlock\": {\n \"type\": \"object\",\n \"title\": \"TextBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"text\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"text\": {\n \"type\": \"string\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/blockConfig\"\n }\n },\n \"required\": [\n \"type\",\n \"text\"\n ],\n \"additionalProperties\": false\n },\n \"htmlBlock\": {\n \"type\": \"object\",\n \"title\": \"HtmlBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"html\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"html\": {\n \"type\": \"string\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/blockConfig\"\n }\n },\n \"required\": [\n \"type\",\n \"html\"\n ],\n \"additionalProperties\": false\n },\n \"headingBlock\": {\n \"type\": \"object\",\n \"title\": \"HeadingBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"heading\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"text\": {\n \"type\": \"string\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/headingConfig\"\n }\n },\n \"required\": [\n \"type\",\n \"text\"\n ],\n \"additionalProperties\": false\n },\n \"imageBlock\": {\n \"type\": \"object\",\n \"title\": \"ImageBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"image\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"src\": {\n \"type\": \"string\",\n \"description\": \"Public image URL, SVG markup, or uploaded image data URL.\"\n },\n \"alt\": {\n \"type\": \"string\",\n \"description\": \"Alternative text for screen readers and PDF accessibility.\",\n \"default\": \"\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/imageConfig\"\n }\n },\n \"required\": [\n \"type\",\n \"src\"\n ],\n \"additionalProperties\": false\n },\n \"keyValueBlock\": {\n \"type\": \"object\",\n \"title\": \"KeyValueBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"key-value\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"values\": {\n \"type\": \"object\",\n \"title\": \"KeyValueValues\",\n \"propertyNames\": {\n \"type\": \"string\",\n \"pattern\": \"^[A-Za-z][A-Za-z0-9_]*$\"\n },\n \"additionalProperties\": {\n \"type\": [\n \"string\",\n \"null\"\n ]\n }\n },\n \"config\": {\n \"$ref\": \"#/$defs/keyValueConfig\"\n }\n },\n \"required\": [\n \"type\"\n ],\n \"additionalProperties\": false\n },\n \"spacerBlock\": {\n \"type\": \"object\",\n \"title\": \"SpacerBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"spacer\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/spacerConfig\"\n }\n },\n \"required\": [\n \"type\"\n ],\n \"additionalProperties\": false\n },\n \"dividerBlock\": {\n \"type\": \"object\",\n \"title\": \"DividerBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"divider\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/dividerConfig\"\n }\n },\n \"required\": [\n \"type\"\n ],\n \"additionalProperties\": false\n },\n \"tableBlock\": {\n \"type\": \"object\",\n \"title\": \"TableBlock\",\n \"properties\": {\n \"type\": {\n \"type\": \"string\",\n \"const\": \"table\"\n },\n \"id\": {\n \"type\": [\n \"string\",\n \"null\"\n ],\n \"description\": \"Stable block identifier used for runtime data overrides.\"\n },\n \"config\": {\n \"$ref\": \"#/$defs/tableConfig\"\n }\n },\n \"required\": [\n \"type\"\n ],\n \"additionalProperties\": false\n },\n \"block\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/textBlock\"\n },\n {\n \"$ref\": \"#/$defs/htmlBlock\"\n },\n {\n \"$ref\": \"#/$defs/headingBlock\"\n },\n {\n \"$ref\": \"#/$defs/imageBlock\"\n },\n {\n \"$ref\": \"#/$defs/keyValueBlock\"\n },\n {\n \"$ref\": \"#/$defs/spacerBlock\"\n },\n {\n \"$ref\": \"#/$defs/dividerBlock\"\n },\n {\n \"$ref\": \"#/$defs/tableBlock\"\n }\n ]\n },\n \"row\": {\n \"type\": \"object\",\n \"title\": \"Row\",\n \"properties\": {\n \"blocks\": {\n \"type\": \"array\",\n \"items\": {\n \"$ref\": \"#/$defs/block\"\n }\n }\n },\n \"required\": [\n \"blocks\"\n ],\n \"additionalProperties\": false\n },\n \"presetPageSize\": {\n \"type\": \"object\",\n \"title\": \"PresetPageSize\",\n \"properties\": {\n \"format\": {\n \"$ref\": \"#/$defs/pageFormat\"\n },\n \"orientation\": {\n \"$ref\": \"#/$defs/orientation\"\n }\n },\n \"additionalProperties\": false\n },\n \"customPageSize\": {\n \"type\": \"object\",\n \"title\": \"CustomPageSize\",\n \"properties\": {\n \"width\": {\n \"type\": \"integer\",\n \"minimum\": 1.0\n },\n \"height\": {\n \"type\": \"integer\",\n \"minimum\": 1.0\n }\n },\n \"required\": [\n \"width\",\n \"height\"\n ],\n \"additionalProperties\": false\n },\n \"pageSize\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/presetPageSize\"\n },\n {\n \"$ref\": \"#/$defs/customPageSize\"\n }\n ],\n \"title\": \"PageSize\"\n },\n \"pageNumbersConfig\": {\n \"type\": \"object\",\n \"title\": \"PageNumbersConfig\",\n \"properties\": {\n \"enabled\": {\n \"type\": \"boolean\",\n \"default\": false\n },\n \"position\": {\n \"$ref\": \"#/$defs/align\"\n }\n },\n \"additionalProperties\": false\n },\n \"pageBackgroundConfig\": {\n \"type\": \"object\",\n \"title\": \"PageBackgroundConfig\",\n \"properties\": {\n \"src\": {\n \"type\": \"string\",\n \"description\": \"HTTP, HTTPS, or base64 data URI for an image or PDF page background.\",\n \"minLength\": 1\n },\n \"type\": {\n \"$ref\": \"#/$defs/pageBackgroundType\"\n }\n },\n \"required\": [\n \"src\"\n ],\n \"additionalProperties\": false\n },\n \"pageFooterConfig\": {\n \"type\": \"object\",\n \"title\": \"PageFooterConfig\",\n \"properties\": {\n \"repeat\": {\n \"type\": \"boolean\",\n \"default\": true\n },\n \"rows\": {\n \"type\": \"array\",\n \"items\": {\n \"$ref\": \"#/$defs/row\"\n }\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"{ repeat?: boolean; rows?: Row[] }\"\n },\n \"pageConfig\": {\n \"type\": \"object\",\n \"title\": \"PageConfig\",\n \"properties\": {\n \"size\": {\n \"$ref\": \"#/$defs/pageSize\"\n },\n \"locale\": {\n \"type\": \"string\",\n \"default\": \"de_DE\"\n },\n \"margins\": {\n \"$ref\": \"#/$defs/spacingConfig\"\n },\n \"pageNumbers\": {\n \"$ref\": \"#/$defs/pageNumbersConfig\"\n },\n \"background\": {\n \"oneOf\": [\n {\n \"$ref\": \"#/$defs/pageBackgroundConfig\"\n },\n {\n \"type\": \"null\"\n }\n ]\n },\n \"footer\": {\n \"$ref\": \"#/$defs/pageFooterConfig\"\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"{ size?: PageSize; locale?: string; margins?: SpacingConfig; pageNumbers?: PageNumbersConfig; background?: PageBackgroundConfig | null; footer?: PageFooterConfig }\"\n },\n \"templateConfig\": {\n \"type\": \"object\",\n \"title\": \"TemplateConfig\",\n \"properties\": {\n \"page\": {\n \"$ref\": \"#/$defs/pageConfig\"\n },\n \"typography\": {\n \"$ref\": \"#/$defs/typographyConfig\"\n }\n },\n \"additionalProperties\": false,\n \"tsType\": \"{ page?: PageConfig; typography?: TypographyConfig }\"\n },\n \"fontFace\": {\n \"type\": \"object\",\n \"title\": \"FontFace\",\n \"properties\": {\n \"src\": {\n \"type\": \"string\"\n },\n \"weight\": {\n \"type\": \"integer\",\n \"default\": 400\n },\n \"style\": {\n \"type\": \"string\",\n \"default\": \"normal\"\n }\n },\n \"required\": [\n \"src\"\n ],\n \"additionalProperties\": false\n },\n \"fileAttachment\": {\n \"type\": \"object\",\n \"title\": \"FileAttachment\",\n \"properties\": {\n \"name\": {\n \"type\": \"string\"\n },\n \"content\": {\n \"type\": \"string\",\n \"description\": \"Base64-encoded file content.\"\n },\n \"mimeType\": {\n \"type\": \"string\",\n \"default\": \"application/octet-stream\"\n },\n \"description\": {\n \"type\": [\n \"string\",\n \"null\"\n ]\n },\n \"relationship\": {\n \"type\": \"string\",\n \"default\": \"Alternative\"\n }\n },\n \"required\": [\n \"name\",\n \"content\"\n ],\n \"additionalProperties\": false\n }\n },\n \"x-pdfUa\": {\n \"kind\": \"template\",\n \"templateVersion\": 1,\n \"renderEndpoint\": \"/render/template\",\n \"templateFields\": [\n \"version\",\n \"config\",\n \"fonts\",\n \"attachments\",\n \"rows\"\n ],\n \"attachmentFields\": [\n \"name\",\n \"content\",\n \"mimeType\",\n \"description\",\n \"relationship\"\n ],\n \"externalFontFields\": [\n \"src\",\n \"weight\",\n \"style\"\n ],\n \"bundledFonts\": [\n \"Fira Code\",\n \"Inter\",\n \"JetBrains Mono\",\n \"Lato\",\n \"Liberation Mono\",\n \"Liberation Sans\",\n \"Liberation Serif\",\n \"Merriweather\",\n \"Montserrat\",\n \"Noto Sans\",\n \"Noto Sans Mono\",\n \"Noto Serif\",\n \"Nunito\",\n \"Open Sans\",\n \"Oswald\",\n \"Playfair Display\",\n \"Poppins\",\n \"Raleway\",\n \"Roboto\",\n \"Rubik\"\n ],\n \"blockOrder\": [\n \"text\",\n \"html\",\n \"heading\",\n \"image\",\n \"key-value\",\n \"spacer\",\n \"divider\",\n \"table\"\n ],\n \"pageFormats\": [\n {\n \"name\": \"A3\",\n \"widthMm\": 297.0,\n \"heightMm\": 420.0\n },\n {\n \"name\": \"A4\",\n \"widthMm\": 210.0,\n \"heightMm\": 297.0\n },\n {\n \"name\": \"A5\",\n \"widthMm\": 148.0,\n \"heightMm\": 210.0\n },\n {\n \"name\": \"A6\",\n \"widthMm\": 105.0,\n \"heightMm\": 148.0\n },\n {\n \"name\": \"Letter\",\n \"widthMm\": 215.9,\n \"heightMm\": 279.4\n },\n {\n \"name\": \"Legal\",\n \"widthMm\": 215.9,\n \"heightMm\": 355.6\n },\n {\n \"name\": \"Tabloid\",\n \"widthMm\": 279.4,\n \"heightMm\": 431.8\n }\n ]\n }\n}","import schema from \"../../schemas/template.schema.json\";\n\n// The validation contract shipped with the package, fed to codemirror-json-schema.\nexport const templateSchema = schema as Record<string, unknown>;\n","import { json } from \"@codemirror/lang-json\";\nimport ReactCodeMirror from \"@uiw/react-codemirror\";\nimport { jsonSchema } from \"codemirror-json-schema\";\nimport { useMemo } from \"react\";\nimport { useTemplateEditor } from \"./TemplateEditorContext\";\nimport { editorTheme } from \"./editorTheme\";\nimport { templateSchema } from \"./templateSchema\";\n\nexport interface CodeEditorProps {\n className?: string;\n}\n\nexport function CodeEditor({ className }: CodeEditorProps = {}) {\n const { text, setText } = useTemplateEditor();\n\n const extensions = useMemo(\n () => [json(), jsonSchema(templateSchema as Parameters<typeof jsonSchema>[0]), ...editorTheme],\n [],\n );\n\n return (\n <div\n className={`pdfua-template-builder min-w-0 min-h-0 overflow-auto bg-surface${className ? ` ${className}` : \"\"}`}\n aria-label=\"Template JSON editor\"\n >\n <ReactCodeMirror\n value={text}\n onChange={setText}\n extensions={extensions}\n theme=\"none\"\n height=\"100%\"\n basicSetup={{ foldGutter: true, highlightActiveLine: true, autocompletion: true }}\n />\n </div>\n );\n}\n","import type { Template } from \"../types/generated/template\";\nimport type { TemplateData } from \"../types/template\";\nimport { Preview } from \"../render/Preview\";\nimport { CodeEditor } from \"./CodeEditor\";\nimport { TemplateEditorProvider } from \"./TemplateEditorContext\";\n\nexport interface TemplateEditorProps {\n /** Base URL of a running pdf-ua-api instance. Defaults to \"\" (relative URLs / proxy). */\n apiUrl?: string;\n /** Template seeded into the editor as pretty-printed JSON on first render. */\n initialTemplate?: Template;\n /** Runtime data keyed by block id, passed through to render. */\n data?: TemplateData;\n /** Fires on every edit; `template` is null while the JSON is invalid. */\n onChange?: (template: Template | null, text: string) => void;\n /** Fires after a successful render with the produced PDF blob. */\n onRendered?: (pdf: Blob) => void;\n /** Optional className appended to the root element. */\n className?: string;\n}\n\nfunction DefaultLayout({ className }: { className?: string }) {\n return (\n <main\n className={`pdfua-template-builder grid h-screen overflow-hidden bg-app text-fg grid-cols-[minmax(40rem,1.55fr)_minmax(28rem,0.95fr)] max-[1080px]:h-auto max-[1080px]:grid-cols-1 max-[1080px]:overflow-visible${className ? ` ${className}` : \"\"}`}\n >\n <CodeEditor className=\"border-0 border-r border-solid border-border\" />\n <Preview />\n </main>\n );\n}\n\n/**\n * All-in-one preset: a JSON template editor and a rendered-PDF preview side by side.\n *\n * For custom layouts, compose directly with `TemplateEditorProvider`, `CodeEditor`,\n * and `Preview`.\n */\nexport function TemplateEditor({\n apiUrl,\n initialTemplate,\n data,\n onChange,\n onRendered,\n className,\n}: TemplateEditorProps = {}) {\n return (\n <TemplateEditorProvider\n apiUrl={apiUrl}\n initialTemplate={initialTemplate}\n data={data}\n onChange={onChange}\n onRendered={onRendered}\n >\n <DefaultLayout className={className} />\n </TemplateEditorProvider>\n );\n}\n","/** Seed document for the HTML editor: a small, accessible page that renders cleanly to PDF/UA. */\nexport const DEFAULT_HTML = `<!DOCTYPE html>\n<html lang=\"en\">\n <head>\n <meta charset=\"utf-8\" />\n <title>Untitled document</title>\n <style>\n body {\n font-family: system-ui, sans-serif;\n margin: 2rem;\n color: #1a1a1a;\n line-height: 1.5;\n }\n h1 {\n font-size: 1.75rem;\n margin-bottom: 0.5rem;\n }\n </style>\n </head>\n <body>\n <h1>Hello, PDF/UA</h1>\n <p>Edit this HTML and render it to preview the accessible PDF.</p>\n </body>\n</html>\n`;\n","import { useCallback, useEffect, useRef, useState } from \"react\";\nimport { renderHtmlPdf } from \"../api/pdfUaApi\";\n\ninterface UseHtmlPreviewOptions {\n apiUrl: string;\n baseUrl?: string;\n onRendered?: (pdf: Blob) => void;\n}\n\ninterface HtmlPreview {\n pdfUrl: string | null;\n loading: boolean;\n error: string | null;\n render: (html: string) => Promise<void>;\n}\n\nexport function useHtmlPreview({ apiUrl, baseUrl, onRendered }: UseHtmlPreviewOptions): HtmlPreview {\n const [pdfUrl, setPdfUrl] = useState<string | null>(null);\n const [loading, setLoading] = useState(false);\n const [error, setError] = useState<string | null>(null);\n\n const renderRequestId = useRef(0);\n const pdfUrlRef = useRef<string | null>(null);\n const onRenderedRef = useRef(onRendered);\n\n useEffect(() => {\n onRenderedRef.current = onRendered;\n }, [onRendered]);\n\n useEffect(() => {\n return () => {\n revokeObjectUrl(pdfUrlRef.current);\n pdfUrlRef.current = null;\n };\n }, []);\n\n const render = useCallback(\n async (html: string) => {\n const requestId = renderRequestId.current + 1;\n renderRequestId.current = requestId;\n setLoading(true);\n setError(null);\n\n try {\n const pdf = await renderHtmlPdf(apiUrl, { html, baseUrl });\n const nextPdfUrl = URL.createObjectURL(pdf);\n\n if (requestId !== renderRequestId.current) {\n revokeObjectUrl(nextPdfUrl);\n return;\n }\n\n setPdfUrl((currentPdfUrl) => {\n revokeObjectUrl(currentPdfUrl);\n pdfUrlRef.current = nextPdfUrl;\n return nextPdfUrl;\n });\n onRenderedRef.current?.(pdf);\n } catch (cause) {\n if (requestId === renderRequestId.current) {\n setError(errorMessage(cause));\n }\n } finally {\n if (requestId === renderRequestId.current) {\n setLoading(false);\n }\n }\n },\n [apiUrl, baseUrl],\n );\n\n return { pdfUrl, loading, error, render };\n}\n\nfunction revokeObjectUrl(url: string | null): void {\n if (url) {\n URL.revokeObjectURL(url);\n }\n}\n\nfunction errorMessage(cause: unknown): string {\n return cause instanceof Error ? cause.message : String(cause);\n}\n","import {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useMemo,\n useRef,\n useState,\n type ReactNode,\n} from \"react\";\nimport { resolveDefaultApiUrl } from \"../api/pdfUaApi\";\nimport { DEFAULT_HTML } from \"./defaultHtml\";\nimport { useHtmlPreview } from \"./useHtmlPreview\";\n\nexport interface HtmlEditorProviderProps {\n /** Base URL of a running pdf-ua-api instance. Defaults to \"\" (relative URLs / proxy). */\n apiUrl?: string;\n /** Base URL used to resolve relative asset references in the HTML. */\n baseUrl?: string;\n /** HTML seeded into the editor on first render. Defaults to a small accessible document. */\n initialHtml?: string;\n /** Fires on every edit with the current HTML text. */\n onChange?: (html: string) => void;\n /** Fires after a successful render with the produced PDF blob. */\n onRendered?: (pdf: Blob) => void;\n children: ReactNode;\n}\n\nexport interface HtmlEditorContextValue {\n html: string;\n setHtml: (html: string) => void;\n pdfUrl: string | null;\n pdfLoading: boolean;\n error: string | null;\n renderPdf: () => void;\n renderDisabled: boolean;\n}\n\nconst HtmlEditorContext = createContext<HtmlEditorContextValue | null>(null);\n\nexport function useHtmlEditor(): HtmlEditorContextValue {\n const value = useContext(HtmlEditorContext);\n\n if (!value) {\n throw new Error(\"useHtmlEditor must be used within an <HtmlEditorProvider>.\");\n }\n\n return value;\n}\n\nexport function HtmlEditorProvider({\n apiUrl: apiUrlProp,\n baseUrl,\n initialHtml = DEFAULT_HTML,\n onChange,\n onRendered,\n children,\n}: HtmlEditorProviderProps) {\n const apiUrl = resolveDefaultApiUrl(apiUrlProp);\n const [html, setHtml] = useState(initialHtml);\n\n const onChangeRef = useRef(onChange);\n useEffect(() => {\n onChangeRef.current = onChange;\n }, [onChange]);\n const skipNextChange = useRef(true);\n useEffect(() => {\n if (skipNextChange.current) {\n skipNextChange.current = false;\n return;\n }\n onChangeRef.current?.(html);\n }, [html]);\n\n const { pdfUrl, loading, error, render } = useHtmlPreview({ apiUrl, baseUrl, onRendered });\n\n const htmlRef = useRef(html);\n htmlRef.current = html;\n\n const renderPdf = useCallback(() => {\n if (htmlRef.current.trim() !== \"\") {\n void render(htmlRef.current);\n }\n }, [render]);\n\n const value = useMemo<HtmlEditorContextValue>(\n () => ({\n html,\n setHtml,\n pdfUrl,\n pdfLoading: loading,\n error,\n renderPdf,\n renderDisabled: html.trim() === \"\" || loading,\n }),\n [html, pdfUrl, loading, error, renderPdf],\n );\n\n return <HtmlEditorContext.Provider value={value}>{children}</HtmlEditorContext.Provider>;\n}\n","import { html } from \"@codemirror/lang-html\";\nimport ReactCodeMirror from \"@uiw/react-codemirror\";\nimport { useMemo } from \"react\";\nimport { editorTheme } from \"../editor/editorTheme\";\nimport { useHtmlEditor } from \"./HtmlEditorContext\";\n\nexport interface HtmlCodeEditorProps {\n className?: string;\n}\n\nexport function HtmlCodeEditor({ className }: HtmlCodeEditorProps = {}) {\n const { html: text, setHtml } = useHtmlEditor();\n\n const extensions = useMemo(() => [html(), ...editorTheme], []);\n\n return (\n <div\n className={`pdfua-template-builder min-w-0 min-h-0 overflow-auto bg-surface${className ? ` ${className}` : \"\"}`}\n aria-label=\"Template HTML editor\"\n >\n <ReactCodeMirror\n value={text}\n onChange={setHtml}\n extensions={extensions}\n theme=\"none\"\n height=\"100%\"\n basicSetup={{ foldGutter: true, highlightActiveLine: true, autocompletion: true }}\n />\n </div>\n );\n}\n","import { Button } from \"../builder/primitives/Button\";\nimport { PdfView, StatusPill, deriveStatus, statusLabel } from \"../render/previewChrome\";\nimport { useHtmlEditor } from \"./HtmlEditorContext\";\n\nexport interface HtmlPreviewProps {\n className?: string;\n}\n\nexport function HtmlPreview({ className }: HtmlPreviewProps = {}) {\n const { pdfUrl, pdfLoading, error, renderPdf, renderDisabled } = useHtmlEditor();\n const status = deriveStatus(pdfLoading, pdfUrl);\n\n return (\n <aside\n className={`pdfua-template-builder grid min-w-0 min-h-0 grid-rows-[56px_auto_minmax(0,1fr)] bg-canvas${className ? ` ${className}` : \"\"}`}\n aria-label=\"Output\"\n >\n <header className=\"row-start-1 flex items-center justify-between gap-3 border-0 border-b border-solid border-border bg-surface px-4\">\n <span className=\"text-sm font-medium tracking-[0.02em] text-fg\">PDF preview</span>\n <div className=\"flex items-center gap-3\">\n <StatusPill status={status}>{statusLabel(status)}</StatusPill>\n <Button variant=\"primary\" onClick={renderPdf} disabled={renderDisabled}>\n {pdfLoading ? \"Rendering…\" : \"Render PDF\"}\n </Button>\n </div>\n </header>\n\n {error ? (\n <p\n className=\"row-start-2 mx-4 mt-3 mb-0 rounded border-0 border-l-[3px] border-solid border-danger bg-danger-soft px-3 py-3 text-sm text-danger\"\n role=\"alert\"\n >\n {error}\n </p>\n ) : null}\n\n <div className=\"row-start-3 grid min-h-0 p-4\">\n <PdfView\n pdfUrl={pdfUrl}\n loading={pdfLoading}\n loadingLabel=\"Rendering the latest HTML…\"\n emptyLabel=\"Render the HTML to preview the PDF here.\"\n />\n </div>\n </aside>\n );\n}\n","import { HtmlCodeEditor } from \"./HtmlCodeEditor\";\nimport { HtmlEditorProvider } from \"./HtmlEditorContext\";\nimport { HtmlPreview } from \"./HtmlPreview\";\n\nexport interface HtmlEditorProps {\n /** Base URL of a running pdf-ua-api instance. Defaults to \"\" (relative URLs / proxy). */\n apiUrl?: string;\n /** Base URL used to resolve relative asset references in the HTML. */\n baseUrl?: string;\n /** HTML seeded into the editor on first render. Defaults to a small accessible document. */\n initialHtml?: string;\n /** Fires on every edit with the current HTML text. */\n onChange?: (html: string) => void;\n /** Fires after a successful render with the produced PDF blob. */\n onRendered?: (pdf: Blob) => void;\n /** Optional className appended to the root element. */\n className?: string;\n}\n\nfunction DefaultLayout({ className }: { className?: string }) {\n return (\n <main\n className={`pdfua-template-builder grid h-screen overflow-hidden bg-app text-fg grid-cols-[minmax(40rem,1.55fr)_minmax(28rem,0.95fr)] max-[1080px]:h-auto max-[1080px]:grid-cols-1 max-[1080px]:overflow-visible${className ? ` ${className}` : \"\"}`}\n >\n <HtmlCodeEditor className=\"border-0 border-r border-solid border-border\" />\n <HtmlPreview />\n </main>\n );\n}\n\n/**\n * All-in-one preset: a raw-HTML editor and a rendered-PDF preview side by side.\n * The preview is produced by the backend `POST /convert` (HTML → PDF/UA) endpoint.\n *\n * For custom layouts, compose directly with `HtmlEditorProvider`, `HtmlCodeEditor`,\n * and `HtmlPreview`.\n */\nexport function HtmlEditor({\n apiUrl,\n baseUrl,\n initialHtml,\n onChange,\n onRendered,\n className,\n}: HtmlEditorProps = {}) {\n return (\n <HtmlEditorProvider\n apiUrl={apiUrl}\n baseUrl={baseUrl}\n initialHtml={initialHtml}\n onChange={onChange}\n onRendered={onRendered}\n >\n <DefaultLayout className={className} />\n </HtmlEditorProvider>\n );\n}\n"],"mappings":"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"}