@atom63/resume 0.2.1 → 0.2.2

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/CHANGELOG.md CHANGED
@@ -2,6 +2,14 @@
2
2
 
3
3
  All notable changes to `@atom63/resume`. Versions are `0.x` — the API may shift before `1.0`.
4
4
 
5
+ ## 0.2.2
6
+
7
+ - **Fix: final oversized blocks no longer create a trailing blank page.**
8
+ - **Fix: custom MDX components now merge over the defaults.** Consumers can pass
9
+ `components={{ SkillBar }}` without spreading the built-in resume components.
10
+ - **Fix: dev write-back always restores the file watcher** and returns JSON for
11
+ write failures instead of leaving the watcher unregistered.
12
+
5
13
  ## 0.2.1
6
14
 
7
15
  Standalone-rendering fixes. If you saw oversized, off-center pages or loose
@@ -64,7 +64,8 @@ function packColumn(blocks, pageUsableHeight, gap) {
64
64
  pageOf[b.id] = page;
65
65
  used += (used > 0 ? gap : 0) + b.height;
66
66
  }
67
- return { pageOf, pages: page + 1, oversized };
67
+ const pages = blocks.length === 0 ? 1 : page + (used > 0 ? 1 : 0);
68
+ return { pageOf, pages, oversized };
68
69
  }
69
70
  function packIntoPages(input) {
70
71
  const gap = input.blockGap ?? 0;
@@ -405,5 +406,5 @@ function PageCountReporter({ count, report }) {
405
406
  }
406
407
 
407
408
  export { BLOCK_GAP_PX, Columns, Entry, Footer, Group, Header, Main, PAGE, PaginatedResume, PaginationReportContext, ResumeFontFamilyContext, ResumePageSizeContext, Rule, Section, Sidebar, WRAPPER_PADDING_X, getPageGeometry, packIntoPages, resumeMdxComponents, usePaginationReport, useResumeFontFamily, useResumePageSize };
408
- //# sourceMappingURL=chunk-FI5DMK4W.js.map
409
- //# sourceMappingURL=chunk-FI5DMK4W.js.map
409
+ //# sourceMappingURL=chunk-V7BVTGX3.js.map
410
+ //# sourceMappingURL=chunk-V7BVTGX3.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/geometry.ts","../src/pagination/pagination.ts","../src/document/pagination-context.tsx","../src/mdx/base-components.tsx","../src/mdx/external-href.ts","../src/document/components.tsx","../src/document/paginated-resume.tsx"],"names":["jsx","jsxs","_"],"mappings":"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","file":"chunk-V7BVTGX3.js","sourcesContent":["// packages/resume/src/geometry.ts\n// Numeric page geometry — the single source of truth shared by the JS packer\n// and the rendered page frames. Print-fixed px at 96dpi for PDF fidelity.\n\nconst DPI = 96\nconst MM = DPI / 25.4\n\nexport type PageSize = 'letter' | 'a4'\n\nexport interface PageGeometry {\n widthPx: number\n heightPx: number\n padXPx: number\n padYPx: number\n /** Screen-only gap between page frames. */\n gapPx: number\n contentWidthPx: number\n usableHeightPx: number\n /** Value for the CSS `@page { size: … }` print rule. */\n cssPageSize: 'letter' | 'A4'\n}\n\n// Both sizes keep the same physical 0.7in x 0.6in margins.\nconst PAD_X_PX = 0.7 * DPI\nconst PAD_Y_PX = 0.6 * DPI\nconst GAP_PX = 32\n\nconst DIMENSIONS: Record<\n PageSize,\n { widthPx: number; heightPx: number; cssPageSize: 'letter' | 'A4' }\n> = {\n letter: { widthPx: 8.5 * DPI, heightPx: 11 * DPI, cssPageSize: 'letter' },\n a4: { widthPx: 210 * MM, heightPx: 297 * MM, cssPageSize: 'A4' },\n}\n\n/** Resolve the print-fixed geometry for a page size (defaults to US Letter). */\nexport function getPageGeometry(size: PageSize = 'letter'): PageGeometry {\n const d = DIMENSIONS[size]\n return {\n widthPx: d.widthPx,\n heightPx: d.heightPx,\n padXPx: PAD_X_PX,\n padYPx: PAD_Y_PX,\n gapPx: GAP_PX,\n contentWidthPx: d.widthPx - 2 * PAD_X_PX,\n usableHeightPx: d.heightPx - 2 * PAD_Y_PX,\n cssPageSize: d.cssPageSize,\n }\n}\n\n/** Back-compat default geometry (US Letter). */\nexport const PAGE = getPageGeometry('letter')\n\n// Kept for back-compat; equal to the Letter geometry's derived values.\nexport const CONTENT_WIDTH_PX = PAGE.contentWidthPx\nexport const USABLE_HEIGHT_PX = PAGE.usableHeightPx\n\n// Vertical gap between blocks within a column — must match --doc-block-gap\n// (0.375rem = 6px) in styles/tokens.css and the column gap in document.css.\nexport const BLOCK_GAP_PX = 6\n\n// Outer horizontal padding the viewer reserves around the page in its scroll area.\nexport const WRAPPER_PADDING_X = 32\n","// packages/resume/src/pagination/pagination.ts\n\nexport interface Block {\n id: string\n /** Measured px (offsetHeight + vertical margins). */\n height: number\n /** Section headings: don't leave them orphaned at a page bottom. */\n keepWithNext: boolean\n}\n\nexport type ColumnKey = 'sidebar' | 'main'\n\nexport interface PackInput {\n columns: Record<ColumnKey, Block[]>\n /** Usable content height for a given page index; index 0 may be shorter (header band). */\n pageUsableHeight: (pageIndex: number) => number\n /**\n * Vertical gap rendered between consecutive blocks in a column (the flex\n * `gap`). Block heights are measured individually, so the packer must add\n * this back or pages overflow once they hold many blocks. Defaults to 0.\n */\n blockGap?: number\n}\n\nexport interface PackResult {\n pageCount: number\n pageOf: Record<ColumnKey, Record<string, number>>\n /** Blocks taller than a full page — placed alone, allowed to overflow. */\n oversized: string[]\n}\n\nfunction packColumn(\n blocks: Block[],\n pageUsableHeight: (p: number) => number,\n gap: number\n): { pageOf: Record<string, number>; pages: number; oversized: string[] } {\n const pageOf: Record<string, number> = {}\n const oversized: string[] = []\n let page = 0\n let used = 0\n\n for (let i = 0; i < blocks.length; i++) {\n const b = blocks[i]\n let cap = pageUsableHeight(page)\n // A gap precedes every block except the first on its page.\n const gapBefore = used > 0 ? gap : 0\n\n // Move to a fresh page if this block (plus its leading gap) overflows.\n if (used > 0 && used + gapBefore + b.height > cap) {\n page++\n used = 0\n cap = pageUsableHeight(page)\n } else if (\n // Keep-with-next: heading fits alone but heading + gap + next block\n // doesn't — push the heading forward so it isn't orphaned.\n used > 0 &&\n b.keepWithNext &&\n used + gapBefore + b.height <= cap &&\n i + 1 < blocks.length &&\n used + gapBefore + b.height + gap + blocks[i + 1].height > cap\n ) {\n page++\n used = 0\n cap = pageUsableHeight(page)\n }\n\n // Oversized: taller than a whole (now-empty) page. Place alone, allow overflow.\n if (b.height > cap && used === 0) {\n oversized.push(b.id)\n pageOf[b.id] = page\n page++\n used = 0\n continue\n }\n\n pageOf[b.id] = page\n used += (used > 0 ? gap : 0) + b.height\n }\n\n const pages = blocks.length === 0 ? 1 : page + (used > 0 ? 1 : 0)\n return { pageOf, pages, oversized }\n}\n\n/** Assign measured blocks to pages per column. Pure: no DOM. */\nexport function packIntoPages(input: PackInput): PackResult {\n const gap = input.blockGap ?? 0\n const sidebar = packColumn(input.columns.sidebar, input.pageUsableHeight, gap)\n const main = packColumn(input.columns.main, input.pageUsableHeight, gap)\n return {\n pageCount: Math.max(sidebar.pages, main.pages, 1),\n pageOf: { sidebar: sidebar.pageOf, main: main.pageOf },\n oversized: [...sidebar.oversized, ...main.oversized],\n }\n}\n","// packages/resume/src/document/pagination-context.tsx\nimport { createContext, useContext } from 'react'\nimport type { PageSize } from '../geometry'\n\n/** PaginatedResume reports its computed page count up to the host (ResumeViewer). */\nexport const PaginationReportContext = createContext<((pageCount: number) => void) | null>(null)\n\nexport function usePaginationReport(): (pageCount: number) => void {\n const report = useContext(PaginationReportContext)\n return report ?? (() => {})\n}\n\n/** Optional font-family hint — changes trigger re-measure/re-pagination. */\nexport const ResumeFontFamilyContext = createContext<string | undefined>(undefined)\nexport const useResumeFontFamily = () => useContext(ResumeFontFamilyContext)\n\n/** Page size for the document — a change re-measures/re-paginates. */\nexport const ResumePageSizeContext = createContext<PageSize>('letter')\nexport const useResumePageSize = () => useContext(ResumePageSizeContext)\n","import type React from 'react'\n\n/**\n * Vendored, dependency-free base prose component map.\n *\n * This is a minimal fallback map that the resume spreads over\n * (`...baseMdxComponents`) before defining its own `h1/h3/h4/p/ul/li/strong/em/a/hr`.\n * It deliberately has NO dependency on `@atom63/ui`, lightbox, article blocks,\n * example containers, or any other `@atom63/*` package — every entry is a plain\n * HTML-element component so the resume package stays self-contained.\n *\n * It only needs to provide sane defaults for elements the resume does NOT override.\n */\nexport const baseMdxComponents = {\n h2: (props: React.HTMLAttributes<HTMLHeadingElement>) => <h2 {...props} />,\n h5: (props: React.HTMLAttributes<HTMLHeadingElement>) => <h5 {...props} />,\n h6: (props: React.HTMLAttributes<HTMLHeadingElement>) => <h6 {...props} />,\n ol: (props: React.HTMLAttributes<HTMLOListElement>) => <ol {...props} />,\n blockquote: (props: React.HTMLAttributes<HTMLElement>) => <blockquote {...props} />,\n code: (props: React.HTMLAttributes<HTMLElement>) => <code {...props} />,\n pre: (props: React.HTMLAttributes<HTMLPreElement>) => <pre {...props} />,\n img: (props: React.ImgHTMLAttributes<HTMLImageElement>) => (\n // biome-ignore lint/a11y/useAltText: alt is forwarded via spread props\n <img {...props} />\n ),\n} as const\n","const MDX_SPECIAL_PROTOCOL = /^(mailto|tel|javascript):/i\n\n/**\n * True when `href` should open in a new tab: off-origin http(s), protocol-relative,\n * or absolute http(s) during SSR. Same-origin URLs, in-app paths (`/…`, `#…`),\n * relative paths, and `mailto:` / `tel:` / `javascript:` stay in-tab / default behavior.\n */\nexport function isExternalMdxHref(href: string | undefined): boolean {\n if (!href) {\n return false\n }\n const t = href.trim()\n if (t.startsWith('#') || t.startsWith('/')) {\n return false\n }\n if (MDX_SPECIAL_PROTOCOL.test(t)) {\n return false\n }\n if (t.startsWith('//')) {\n return true\n }\n if (typeof globalThis.window !== 'undefined') {\n try {\n const u = new URL(t, globalThis.window.location.href)\n if (u.protocol !== 'http:' && u.protocol !== 'https:') {\n return false\n }\n return u.origin !== globalThis.window.location.origin\n } catch {\n return false\n }\n }\n try {\n const u = new URL(t)\n return u.protocol === 'http:' || u.protocol === 'https:'\n } catch {\n return false\n }\n}\n","import clsx from 'clsx'\nimport { baseMdxComponents } from '../mdx/base-components'\nimport { isExternalMdxHref } from '../mdx/external-href'\nimport { PaginatedResume } from './paginated-resume'\n\n// Styling for these primitives lives in resume-document.css as a small token\n// system (--doc-*) + semantic classes (.doc-*). Components only compose those —\n// no hardcoded sizes/colors/grid widths — so the resume and CV share one source\n// of truth. See paginated-resume.tsx for where the .css is imported and where\n// the [data-resume-document] token scope is applied.\n\nexport function Header({\n children,\n meta = false,\n}: {\n children: React.ReactNode\n /**\n * Use the universal meta grid (left label column + content), the same track\n * widths as <Section> and <Entry>, so the header aligns with everything\n * below it. The default is the resume's sidebar|main split.\n */\n meta?: boolean\n}) {\n return (\n <div\n className={clsx('doc-header', meta ? 'doc-meta-grid' : 'doc-resume-grid')}\n data-resume-header\n >\n {children}\n </div>\n )\n}\n\nfunction HeaderLeft({ children }: { children: React.ReactNode }) {\n return <div className=\"doc-header-col doc-header-left\">{children}</div>\n}\n\nfunction HeaderRight({ children }: { children: React.ReactNode }) {\n return <div className=\"doc-header-col doc-header-right\">{children}</div>\n}\n\nfunction Links({ children }: { children: React.ReactNode }) {\n return <div className=\"doc-group doc-links\">{children}</div>\n}\n\nexport function Group({ children }: { children: React.ReactNode }) {\n return <div className=\"doc-group\">{children}</div>\n}\n\n/**\n * A record row (award, media mention, talk, …): a `year | role` meta column on\n * the left and the linked title on the right. Each Entry is a single pagination\n * block, so render long CV lists as a flat sequence of Entry elements (NOT one\n * <ul>) so pages can break between them.\n */\nexport function Entry({\n year,\n role,\n children,\n}: {\n year?: string\n role?: string\n children: React.ReactNode\n}) {\n return (\n <div className=\"doc-meta-grid\">\n <span className=\"doc-meta\">{role ? `${year} | ${role}` : year}</span>\n <div className=\"doc-body doc-entry-body\">{children}</div>\n </div>\n )\n}\n\n/** A thin full-width divider — the CV uses one under the header band. */\nexport function Rule() {\n return <div className=\"doc-rule\" />\n}\n\n/**\n * Editorial section: the label sits in the left meta column (aligned with the\n * `year | role` column of Entry rows) and the content fills the right column,\n * so prose sections share one left edge with the record rows below. The\n * .doc-section margin keeps every section on the same vertical rhythm as the\n * record-section headings (h3). One pagination block — keep a body under a page.\n */\nexport function Section({ label, children }: { label: string; children: React.ReactNode }) {\n return (\n <div className=\"doc-section doc-meta-grid\">\n <h3 className=\"doc-heading\">{label}</h3>\n <div className=\"doc-col\">{children}</div>\n </div>\n )\n}\n\nexport function Footer({ children }: { children: React.ReactNode }) {\n return (\n <div className=\"doc-footer\" data-footer>\n {children}\n </div>\n )\n}\n\nexport function Columns({ children }: { children: React.ReactNode }) {\n return <div className=\"doc-resume-grid\">{children}</div>\n}\n\nexport function Sidebar({ children }: { children: React.ReactNode }) {\n return (\n <div className=\"doc-col doc-col-sidebar\" data-resume-col=\"sidebar\">\n {children}\n </div>\n )\n}\n\nexport function Main({ children }: { children: React.ReactNode }) {\n return (\n <div className=\"doc-col doc-col-main\" data-resume-col=\"main\">\n {children}\n </div>\n )\n}\n\nexport const resumeMdxComponents = {\n ...baseMdxComponents,\n\n ResumeDocument: PaginatedResume,\n\n Header,\n HeaderLeft,\n HeaderRight,\n Links,\n Group,\n Entry,\n Rule,\n Section,\n Footer,\n Columns,\n Sidebar,\n Main,\n\n h1: ({ children, ...props }: React.HTMLAttributes<HTMLHeadingElement>) => (\n <h1 className=\"doc-title\" {...props}>\n {children}\n </h1>\n ),\n\n h3: ({ children, ...props }: React.HTMLAttributes<HTMLHeadingElement>) => (\n <h3 className=\"doc-section doc-heading\" data-resume-keep-with-next {...props}>\n {children}\n </h3>\n ),\n\n h4: ({ children, ...props }: React.HTMLAttributes<HTMLHeadingElement>) => (\n <h4 className=\"doc-subheading\" {...props}>\n {children}\n </h4>\n ),\n\n p: ({ children, ...props }: React.HTMLAttributes<HTMLParagraphElement>) => (\n <p className=\"doc-body\" {...props}>\n {children}\n </p>\n ),\n\n ul: ({ children, ...props }: React.HTMLAttributes<HTMLUListElement>) => (\n <ul className=\"doc-list\" {...props}>\n {children}\n </ul>\n ),\n\n li: ({ children, ...props }: React.HTMLAttributes<HTMLLIElement>) => (\n <li className=\"doc-body\" {...props}>\n {children}\n </li>\n ),\n\n strong: ({ children, ...props }: React.HTMLAttributes<HTMLElement>) => (\n <strong className=\"doc-strong\" {...props}>\n {children}\n </strong>\n ),\n\n em: ({ children, ...props }: React.HTMLAttributes<HTMLElement>) => (\n <em className=\"doc-meta doc-em\" {...props}>\n {children}\n </em>\n ),\n\n a: ({ children, href, target, rel, ...props }: React.AnchorHTMLAttributes<HTMLAnchorElement>) => {\n const external = isExternalMdxHref(href)\n return (\n <a\n className=\"doc-link\"\n href={href}\n {...props}\n rel={external ? (rel ?? 'noopener noreferrer') : rel}\n target={external ? (target ?? '_blank') : target}\n >\n {children}\n </a>\n )\n },\n\n hr: (_props: React.HTMLAttributes<HTMLHRElement>) => <div className=\"doc-hr\" />,\n}\n","import {\n Children,\n Fragment,\n isValidElement,\n type ReactNode,\n useEffect,\n useLayoutEffect,\n useRef,\n useState,\n} from 'react'\nimport { BLOCK_GAP_PX, getPageGeometry } from '../geometry'\nimport { type Block, type ColumnKey, packIntoPages } from '../pagination/pagination'\nimport { Columns, Footer, Header, Main, Sidebar } from './components'\nimport { usePaginationReport, useResumeFontFamily, useResumePageSize } from './pagination-context'\n\n// Styles ship via the `@atom63/resume/styles` entry (import it once in your app)\n// — NOT self-imported here, so a consumer's token overrides always win the\n// cascade instead of racing a re-injected copy of the package CSS.\n\ntype Columned = Record<ColumnKey, ReactNode[]>\n\n/**\n * Flatten one level, descending through Fragments. MDX wraps top-level content\n * (and component children) in Fragments, so `Children.toArray` alone yields the\n * Fragment, not the Header/Columns/Sidebar elements inside it.\n */\nfunction flattenChildren(children: ReactNode): ReactNode[] {\n const out: ReactNode[] = []\n for (const child of Children.toArray(children)) {\n if (isValidElement(child) && child.type === Fragment) {\n out.push(...flattenChildren((child.props as { children: ReactNode }).children))\n } else {\n out.push(child)\n }\n }\n return out\n}\n\n/** Pull Header / Columns(Sidebar,Main) / Footer out of the MDX wrapper children. */\nfunction extract(children: ReactNode): {\n header: ReactNode | null\n footer: ReactNode | null\n blocks: Columned\n} {\n let header: ReactNode | null = null\n let footer: ReactNode | null = null\n const blocks: Columned = { sidebar: [], main: [] }\n\n for (const child of flattenChildren(children)) {\n if (!isValidElement(child)) continue\n if (child.type === Header) header = child\n else if (child.type === Footer) footer = child\n else if (child.type === Columns) {\n for (const col of flattenChildren((child.props as { children: ReactNode }).children)) {\n if (!isValidElement(col)) continue\n const key: ColumnKey | null =\n col.type === Sidebar ? 'sidebar' : col.type === Main ? 'main' : null\n if (key) {\n blocks[key] = flattenChildren((col.props as { children: ReactNode }).children)\n }\n }\n }\n }\n return { header, footer, blocks }\n}\n\n/** Read a block element's full vertical extent (border box + margins). */\nfunction blockHeight(el: HTMLElement): number {\n const cs = getComputedStyle(el)\n const mt = Number.parseFloat(cs.marginTop) || 0\n const mb = Number.parseFloat(cs.marginBottom) || 0\n return el.getBoundingClientRect().height + mt + mb\n}\n\n/**\n * Mounted as the MDX `wrapper`, so it receives [Header, Columns, Footer].\n * Measures each column's blocks in a hidden layer, packs them into pages, and\n * renders physical page frames. Reports pageCount up to ResumeApp.\n */\nexport function PaginatedResume({ children }: { children: ReactNode }) {\n const { header, footer, blocks } = extract(children)\n const fontFamily = useResumeFontFamily()\n const pageSize = useResumePageSize()\n const geo = getPageGeometry(pageSize)\n const report = usePaginationReport()\n\n const measureRef = useRef<HTMLDivElement>(null)\n const [heights, setHeights] = useState<Record<ColumnKey, number[]> | null>(null)\n const [headerH, setHeaderH] = useState(0)\n\n // contentKey changes whenever the rendered content could change height.\n // Print-fixed (px) sizes: only the inherited font *family* and the block\n // counts can change measured heights — the OS typography scale is ignored.\n const contentKey = `${fontFamily}:${pageSize}:${blocks.sidebar.length}:${blocks.main.length}`\n\n // contentKey is the intentional re-measure trigger: a change in font family /\n // block counts must force a fresh measure even though it isn't read in the body.\n // biome-ignore lint/correctness/useExhaustiveDependencies: deliberate re-run trigger\n useLayoutEffect(() => {\n const root = measureRef.current\n if (!root) return\n const measure = () => {\n const cols: Record<ColumnKey, number[]> = { sidebar: [], main: [] }\n for (const key of ['sidebar', 'main'] as ColumnKey[]) {\n const colEl = root.querySelector(`[data-resume-col=\"${key}\"]`)\n if (colEl) {\n cols[key] = Array.from(colEl.children).map(c => blockHeight(c as HTMLElement))\n }\n }\n const headerEl = root.querySelector('[data-resume-header]') as HTMLElement | null\n setHeaderH(headerEl ? blockHeight(headerEl) : 0)\n setHeights(cols)\n }\n measure()\n const ro = new ResizeObserver(measure)\n ro.observe(root)\n\n // Re-measure on window (os63 window) resize — e.g. maximize/restore. The\n // measure root is fixed-width, so its own observer never catches this;\n // watch the enclosing scroll viewport, which tracks the window size. A\n // double rAF lets layout settle so a mid-transition read doesn't stick\n // (which collapsed multi-page docs to a single clipped page).\n const viewport = root.closest('[data-slot=\"scroll-area-viewport\"]')\n // Debounce to a TRAILING measure: the os63 window's maximize/restore is a\n // spring animation, so reading mid-transition lands on a wrong height that\n // then sticks. Wait for the resize to stop, then measure the settled layout.\n let settleTimer: ReturnType<typeof setTimeout> | undefined\n const scheduleMeasure = () => {\n clearTimeout(settleTimer)\n settleTimer = setTimeout(measure, 250)\n }\n const vpObserver = viewport ? new ResizeObserver(scheduleMeasure) : null\n vpObserver?.observe(viewport as Element)\n\n if (typeof document !== 'undefined' && 'fonts' in document) {\n document.fonts.ready.then(measure).catch(() => {})\n }\n return () => {\n ro.disconnect()\n vpObserver?.disconnect()\n clearTimeout(settleTimer)\n }\n }, [contentKey])\n\n // The hidden measuring layer renders the real structure at content width.\n // A 0x0 overflow-hidden clip wrapper keeps it out of the scroll area (a large\n // negative offset would inflate the viewport and spawn a phantom horizontal\n // scrollbar, drifting the page off-center). The INNER element keeps its\n // natural size so block heights stay measurable and its ResizeObserver still\n // fires on content changes.\n const measureLayer = (\n <div\n aria-hidden\n style={{\n position: 'absolute',\n top: 0,\n left: 0,\n width: 0,\n height: 0,\n overflow: 'hidden',\n pointerEvents: 'none',\n }}\n >\n <div\n data-resume-measure\n ref={measureRef}\n style={{ width: geo.contentWidthPx, visibility: 'hidden' }}\n >\n {header}\n {blocks.sidebar.length === 0 ? (\n // Single-column docs (e.g. the CV) put everything in Main and render\n // full-width. Measure at full width too so block heights match the\n // full-width page render below (otherwise they'd be measured at the\n // narrower 2/3 main-column width and mis-paginate).\n <div className=\"doc-col\" data-resume-col=\"main\">\n {blocks.main}\n </div>\n ) : (\n <Columns>\n <Sidebar>{blocks.sidebar}</Sidebar>\n <Main>{blocks.main}</Main>\n </Columns>\n )}\n </div>\n </div>\n )\n\n if (!heights) {\n return <div data-resume-document>{measureLayer}</div>\n }\n\n const toBlocks = (key: ColumnKey): Block[] =>\n blocks[key].map((node, i) => ({\n id: `${key}-${i}`,\n height: heights[key][i] ?? 0,\n keepWithNext:\n isValidElement(node) &&\n Boolean((node.props as Record<string, unknown>)['data-resume-keep-with-next']),\n }))\n\n const sidebarBlocks = toBlocks('sidebar')\n const mainBlocks = toBlocks('main')\n\n const pack = packIntoPages({\n columns: { sidebar: sidebarBlocks, main: mainBlocks },\n pageUsableHeight: p =>\n p === 0 ? Math.max(0, geo.usableHeightPx - headerH) : geo.usableHeightPx,\n blockGap: BLOCK_GAP_PX,\n })\n\n return (\n <div data-resume-document>\n {measureLayer}\n <div className=\"doc-pages\" style={{ gap: geo.gapPx }}>\n {Array.from({ length: pack.pageCount }, (_, p) => {\n const sideOnPage = blocks.sidebar.filter(\n (_, i) => pack.pageOf.sidebar[`sidebar-${i}`] === p\n )\n const mainOnPage = blocks.main.filter((_, i) => pack.pageOf.main[`main-${i}`] === p)\n // Key the layout on the WHOLE document, not the current page: a\n // single-column doc (the CV — no sidebar at all) renders full\n // width, but a two-column doc keeps its grid on every page,\n // including continuation pages where the sidebar has run out, so\n // the main column stays in its right-hand track instead of\n // jumping to full width (which broke the 2-col layout on page 2+).\n const isTwoColumn = blocks.sidebar.length > 0\n return (\n <div\n className=\"doc-page\"\n data-resume-page\n key={`page-${String(p)}`}\n style={{\n width: geo.widthPx,\n height: geo.heightPx,\n overflow: 'hidden',\n padding: `${geo.padYPx}px ${geo.padXPx}px`,\n }}\n >\n {p === 0 && header}\n {isTwoColumn ? (\n <div className=\"doc-resume-grid\">\n <div className=\"doc-col doc-col-sidebar\">{sideOnPage}</div>\n <div className=\"doc-col doc-col-main\">{mainOnPage}</div>\n </div>\n ) : (\n <div className=\"doc-col\">{mainOnPage}</div>\n )}\n {p === pack.pageCount - 1 && footer}\n </div>\n )\n })}\n </div>\n <PageCountReporter count={pack.pageCount} report={report} />\n </div>\n )\n}\n\n/** Reports pageCount via an effect (no setState-during-render). */\nfunction PageCountReporter({ count, report }: { count: number; report: (n: number) => void }) {\n useEffect(() => {\n report(count)\n }, [count, report])\n return null\n}\n"]}
@@ -1,4 +1,4 @@
1
- import { resumeMdxComponents } from '../chunk-FI5DMK4W.js';
1
+ import { resumeMdxComponents } from '../chunk-V7BVTGX3.js';
2
2
  import { MDXProvider } from '@mdx-js/react';
3
3
  import { useEffect, useState, useRef, Component } from 'react';
4
4
  import { jsx, jsxs } from 'react/jsx-runtime';
package/dist/index.js CHANGED
@@ -1,5 +1,5 @@
1
- import { getPageGeometry, ResumePageSizeContext, ResumeFontFamilyContext, PaginationReportContext, resumeMdxComponents, WRAPPER_PADDING_X } from './chunk-FI5DMK4W.js';
2
- export { BLOCK_GAP_PX, Columns, Entry, Footer, Group, Header, Main, PAGE, PaginatedResume, PaginationReportContext, ResumeFontFamilyContext, ResumePageSizeContext, Rule, Section, Sidebar, getPageGeometry, packIntoPages, resumeMdxComponents, usePaginationReport, useResumeFontFamily, useResumePageSize } from './chunk-FI5DMK4W.js';
1
+ import { getPageGeometry, resumeMdxComponents, ResumePageSizeContext, ResumeFontFamilyContext, PaginationReportContext, WRAPPER_PADDING_X } from './chunk-V7BVTGX3.js';
2
+ export { BLOCK_GAP_PX, Columns, Entry, Footer, Group, Header, Main, PAGE, PaginatedResume, PaginationReportContext, ResumeFontFamilyContext, ResumePageSizeContext, Rule, Section, Sidebar, getPageGeometry, packIntoPages, resumeMdxComponents, usePaginationReport, useResumeFontFamily, useResumePageSize } from './chunk-V7BVTGX3.js';
3
3
  import { MDXProvider } from '@mdx-js/react';
4
4
  import clsx from 'clsx';
5
5
  import { Maximize2, Minus, Plus, ClipboardCopy } from 'lucide-react';
@@ -478,6 +478,7 @@ function ResumeViewer({
478
478
  const isMobile = useIsMobile();
479
479
  const scrollRef = useRef(null);
480
480
  const contentRef = useRef(null);
481
+ const mdxComponents = { ...resumeMdxComponents, ...components };
481
482
  const {
482
483
  pageCount,
483
484
  setPageCount,
@@ -619,7 +620,7 @@ function ResumeViewer({
619
620
  height: totalHeight,
620
621
  transform: effectiveScale === 1 ? void 0 : `scale(${effectiveScale})`
621
622
  },
622
- children: /* @__PURE__ */ jsx(ResumePageSizeContext.Provider, { value: pageSize, children: /* @__PURE__ */ jsx(ResumeFontFamilyContext.Provider, { value: fontFamily, children: /* @__PURE__ */ jsx(PaginationReportContext.Provider, { value: setPageCount, children: /* @__PURE__ */ jsx(MDXProvider, { components: components ?? resumeMdxComponents, children: /* @__PURE__ */ jsx(
623
+ children: /* @__PURE__ */ jsx(ResumePageSizeContext.Provider, { value: pageSize, children: /* @__PURE__ */ jsx(ResumeFontFamilyContext.Provider, { value: fontFamily, children: /* @__PURE__ */ jsx(PaginationReportContext.Provider, { value: setPageCount, children: /* @__PURE__ */ jsx(MDXProvider, { components: mdxComponents, children: /* @__PURE__ */ jsx(
623
624
  "div",
624
625
  {
625
626
  className: "resume-viewer-content",
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/viewer/use-is-mobile.ts","../src/viewer/print.ts","../src/viewer/viewport.ts","../src/viewer/use-resume-viewport.ts","../src/viewer/resume-viewer.tsx"],"names":["useState","useEffect","useRef"],"mappings":"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","file":"index.js","sourcesContent":["import { useEffect, useState } from 'react'\n\n/** True when the viewport is narrow (<= 768px). SSR-safe (defaults false). */\nexport function useIsMobile(breakpointPx = 768): boolean {\n const [isMobile, setIsMobile] = useState(false)\n useEffect(() => {\n const mql = window.matchMedia(`(max-width: ${breakpointPx}px)`)\n const update = () => setIsMobile(mql.matches)\n update()\n mql.addEventListener('change', update)\n return () => mql.removeEventListener('change', update)\n }, [breakpointPx])\n return isMobile\n}\n","/**\n * Build a print-ready clone of the resume content element. The pagination\n * engine already renders real 8.5x11 `[data-resume-page]` frames, so the clone\n * is a plain wrapper holding those frames (no extra page sizing/padding); the\n * print stylesheet maps each frame to a physical page. The hidden measuring\n * layer is dropped so only the real pages print.\n */\nexport function createPrintClone(source: HTMLElement): HTMLElement {\n const clone = source.cloneNode(true) as HTMLElement\n clone.id = 'resume-print-clone'\n clone.style.cssText = 'background:white;color:black'\n clone.querySelector('[data-resume-measure]')?.remove()\n return clone\n}\n\n/** Hide all non-essential body children for printing; returns the hidden elements. */\nexport function hideBodyChildren(): HTMLElement[] {\n const hidden: HTMLElement[] = []\n const preserveTags = new Set(['STYLE', 'LINK', 'SCRIPT'])\n for (const child of Array.from(document.body.children) as HTMLElement[]) {\n if (!preserveTags.has(child.tagName)) {\n child.style.setProperty('display', 'none', 'important')\n hidden.push(child)\n }\n }\n return hidden\n}\n\nexport interface PrintControllerOptions {\n /** The element holding the rendered resume page frames (contentRef.current). */\n getTarget: () => HTMLElement | null\n /** Suggested PDF filename — becomes document.title during print. */\n getFilename: () => string\n /** Whether THIS viewer should own the print (e.g. it requested print, or it's the active/focused view). */\n shouldOwnPrint: () => boolean\n /** Optional: called after print is torn down (e.g. to force a re-render). */\n onAfterPrint?: () => void\n}\n\n/**\n * Register beforeprint/afterprint handlers implementing the clone-and-hide\n * print strategy. Returns an unsubscribe function that removes the listeners.\n */\nexport function registerPrintController(opts: PrintControllerOptions): () => void {\n let hiddenElements: HTMLElement[] = []\n let savedTitle = ''\n\n const beforePrint = () => {\n // Own the print only when this viewer should (its Save-to-PDF button /\n // shortcut requested it, or it's the active/focused view). Otherwise leave\n // it alone so another app isn't cloned/printed underneath, and so this\n // viewer isn't skipped when focus hasn't flushed yet.\n if (!opts.shouldOwnPrint()) return\n const target = opts.getTarget()\n if (!target) return\n savedTitle = document.title\n document.title = opts.getFilename()\n const clone = createPrintClone(target)\n hiddenElements = hideBodyChildren()\n document.body.appendChild(clone)\n }\n\n const afterPrint = () => {\n const clone = document.getElementById('resume-print-clone')\n if (clone) clone.remove()\n document.title = savedTitle\n for (const el of hiddenElements) {\n el.style.removeProperty('display')\n }\n hiddenElements = []\n opts.onAfterPrint?.()\n }\n\n window.addEventListener('beforeprint', beforePrint)\n window.addEventListener('afterprint', afterPrint)\n return () => {\n window.removeEventListener('beforeprint', beforePrint)\n window.removeEventListener('afterprint', afterPrint)\n }\n}\n","export const INTERACTIVE_PAN_SELECTOR = 'a, button, [role=\"button\"], input, textarea, select'\n\nexport const PAN_THRESHOLD_PX = 6\n\nexport function clampZoom(value: number, min: number, max: number): number {\n return Math.max(min, Math.min(value, max))\n}\n\nexport function isInteractivePanTarget(target: EventTarget | null): boolean {\n return target instanceof HTMLElement && Boolean(target.closest(INTERACTIVE_PAN_SELECTOR))\n}\n\nexport function getTouchDistance(touches: TouchList): number {\n if (touches.length < 2) {\n return 0\n }\n\n const first = touches[0]\n const second = touches[1]\n if (!first || !second) {\n return 0\n }\n\n const deltaX = first.clientX - second.clientX\n const deltaY = first.clientY - second.clientY\n return Math.hypot(deltaX, deltaY)\n}\n\nexport function clampScrollPosition(element: HTMLElement): void {\n const maxScrollLeft = Math.max(0, element.scrollWidth - element.clientWidth)\n const maxScrollTop = Math.max(0, element.scrollHeight - element.clientHeight)\n element.scrollLeft = Math.min(Math.max(0, element.scrollLeft), maxScrollLeft)\n element.scrollTop = Math.min(Math.max(0, element.scrollTop), maxScrollTop)\n}\n\nexport function hasExceededPanThreshold(\n startX: number,\n startY: number,\n currentX: number,\n currentY: number,\n threshold = PAN_THRESHOLD_PX\n): boolean {\n return Math.abs(currentX - startX) > threshold || Math.abs(currentY - startY) > threshold\n}\n","// packages/resume/src/viewer/use-resume-viewport.ts\n//\n// Headless viewport engine for a paginated resume: zoom (buttons / Ctrl+wheel /\n// pinch), pan (mouse drag / touch), fit-to-width, focus-aware print + keyboard\n// shortcuts. The single source of truth for this behaviour — both the built-in\n// `ResumeViewer` and bespoke hosts (e.g. an OS window with its own toolbar\n// chrome) consume this hook so the logic is never duplicated.\n//\n// The host owns the markup and supplies the scroll + content refs; the hook\n// wires every listener to them and returns the derived scale, page count, and\n// the toolbar action handlers.\nimport { type RefObject, useCallback, useEffect, useLayoutEffect, useRef, useState } from 'react'\nimport { getPageGeometry, type PageGeometry, type PageSize, WRAPPER_PADDING_X } from '../geometry'\nimport { registerPrintController } from './print'\nimport {\n clampScrollPosition,\n clampZoom,\n getTouchDistance,\n hasExceededPanThreshold,\n isInteractivePanTarget,\n} from './viewport'\n\nexport const SCALE_STEP = 0.1\nexport const MIN_SCALE = 0.25\nexport const MAX_SCALE = 2\nconst WHEEL_ZOOM_SENSITIVITY = 0.01\nconst WHEEL_ZOOM_CAP = 0.03\n\nconst clampResumeZoom = (value: number) => clampZoom(value, MIN_SCALE, MAX_SCALE)\n\nfunction isInputFocused(): boolean {\n const tag = document.activeElement?.tagName\n return tag === 'INPUT' || tag === 'TEXTAREA'\n}\n\nexport interface UseResumeViewportOptions {\n /** Scroll viewport element (the host attaches this to its scrollable region). */\n scrollRef: RefObject<HTMLDivElement | null>\n /** Element holding the rendered page frames (the print/copy target). */\n contentRef: RefObject<HTMLDivElement | null>\n /** Narrow-viewport flag — drives touch pan/pinch vs mouse drag. */\n isMobile: boolean\n /** Whether this viewport owns global print + keyboard shortcuts. */\n isActive: boolean\n /** Paper size driving the page dimensions + fit-to-width. Default 'letter'. */\n pageSize?: PageSize\n /** Suggested PDF filename (becomes document.title during print). */\n pdfFilename?: string\n /** Called after a successful copy-to-clipboard. */\n onCopy?: () => void\n /** Called when copy-to-clipboard fails. */\n onError?: (error: unknown) => void\n}\n\nexport interface UseResumeViewport {\n /** Current page count (reported by the pagination engine via setPageCount). */\n pageCount: number\n /** Reporter wired into PaginationReportContext. */\n setPageCount: (n: number) => void\n /** The scale actually applied to the page stage (fit-to-width or manual zoom). */\n effectiveScale: number\n /** Whether fit-to-width mode is active. */\n isFitToWidth: boolean\n /** Total stacked page height (unscaled px) — for sizing the page box. */\n totalHeight: number\n /** Resolved geometry for the active page size — for sizing the scaled box. */\n geo: PageGeometry\n handleZoomIn: () => void\n handleZoomOut: () => void\n handleActualSize: () => void\n handleFitToWidth: () => void\n requestPrint: () => void\n handleCopyText: () => Promise<void>\n /** Lower/upper zoom bounds (for disabling +/- controls). */\n minScale: number\n maxScale: number\n}\n\n/**\n * Wire the resume viewport behaviour to host-supplied refs and return the\n * derived state + action handlers. The host renders the scaled page stage; the\n * pagination engine inside it reports its page count through `setPageCount`.\n */\nexport function useResumeViewport({\n scrollRef,\n contentRef,\n isMobile,\n isActive,\n pageSize = 'letter',\n pdfFilename,\n onCopy,\n onError,\n}: UseResumeViewportOptions): UseResumeViewport {\n const geo = getPageGeometry(pageSize)\n\n // Keep the latest active state readable from global listeners (print/keyboard)\n // without re-registering them, so this viewport only hijacks printing when\n // it's the active view (avoids clashing with other apps' print handlers).\n const isActiveRef = useRef(isActive)\n isActiveRef.current = isActive\n\n // Set synchronously right before window.print() so the global beforeprint\n // listener knows THIS viewport initiated the print (active state can lag a click).\n const printRequestedRef = useRef(false)\n const requestPrint = useCallback(() => {\n printRequestedRef.current = true\n window.print()\n }, [])\n\n const zoomAnchorRef = useRef<{\n pointX: number\n pointY: number\n viewportX: number\n viewportY: number\n prevScale: number\n } | null>(null)\n const [pageCount, setPageCount] = useState(1)\n const [zoom, setZoom] = useState(1)\n const [fitScale, setFitScale] = useState(1)\n const [isFitToWidth, setIsFitToWidth] = useState(true)\n const [, forceRender] = useState(0)\n const dragRef = useRef<{\n startX: number\n startY: number\n scrollX: number\n scrollY: number\n } | null>(null)\n const touchPanRef = useRef<{\n scrollX: number\n scrollY: number\n startX: number\n startY: number\n } | null>(null)\n const pinchRef = useRef<{\n startDistance: number\n startScale: number\n } | null>(null)\n\n const effectiveScale = isFitToWidth ? fitScale : zoom\n\n // Refs for stable closures — avoids tearing down/re-registering listeners on every zoom tick\n const effectiveScaleRef = useRef(effectiveScale)\n const isFitToWidthRef = useRef(isFitToWidth)\n const fitScaleRef = useRef(fitScale)\n effectiveScaleRef.current = effectiveScale\n isFitToWidthRef.current = isFitToWidth\n fitScaleRef.current = fitScale\n\n // Keep zoom synced to fitScale while in fit-to-width mode\n useEffect(() => {\n if (isFitToWidth) setZoom(fitScale)\n }, [isFitToWidth, fitScale])\n\n // Responsive scaling — compute fit-to-width from the scroll viewport.\n useEffect(() => {\n const viewport = scrollRef.current\n if (!viewport) return\n\n const updateFitScale = () => {\n const available = Math.max(0, viewport.clientWidth - WRAPPER_PADDING_X)\n setFitScale(clampResumeZoom(Math.min(1, available / geo.widthPx)))\n }\n\n updateFitScale()\n const observer = new ResizeObserver(updateFitScale)\n observer.observe(viewport)\n window.visualViewport?.addEventListener('resize', updateFitScale)\n\n return () => {\n observer.disconnect()\n window.visualViewport?.removeEventListener('resize', updateFitScale)\n }\n }, [scrollRef, geo.widthPx])\n\n const captureViewportCenter = useCallback(() => {\n const el = scrollRef.current\n if (!el) return\n zoomAnchorRef.current = {\n pointX: el.scrollLeft + el.clientWidth / 2,\n pointY: el.scrollTop + el.clientHeight / 2,\n viewportX: el.clientWidth / 2,\n viewportY: el.clientHeight / 2,\n prevScale: effectiveScaleRef.current,\n }\n }, [scrollRef])\n\n const handleZoomIn = useCallback(() => {\n captureViewportCenter()\n setIsFitToWidth(false)\n setZoom(prev => clampResumeZoom(prev + SCALE_STEP))\n }, [captureViewportCenter])\n\n const handleZoomOut = useCallback(() => {\n captureViewportCenter()\n setIsFitToWidth(false)\n setZoom(prev => clampResumeZoom(prev - SCALE_STEP))\n }, [captureViewportCenter])\n\n const handleActualSize = useCallback(() => {\n captureViewportCenter()\n setIsFitToWidth(false)\n setZoom(1)\n }, [captureViewportCenter])\n\n const handleFitToWidth = useCallback(() => {\n setIsFitToWidth(true)\n }, [])\n\n // Ctrl+wheel zoom — rAF-batched for smooth trackpad pinch\n useEffect(() => {\n const el = scrollRef.current\n if (!el) return\n\n let pendingDelta = 0\n let rafId = 0\n let lastCursorX = 0\n let lastCursorY = 0\n\n const flush = () => {\n rafId = 0\n const delta = pendingDelta\n pendingDelta = 0\n const cursorX = lastCursorX\n const cursorY = lastCursorY\n\n setZoom(prev => {\n const next = clampResumeZoom(prev + delta)\n if (next === prev) return prev\n zoomAnchorRef.current = {\n pointX: el.scrollLeft + cursorX,\n pointY: el.scrollTop + cursorY,\n viewportX: cursorX,\n viewportY: cursorY,\n prevScale: isFitToWidthRef.current ? fitScaleRef.current : prev,\n }\n return next\n })\n setIsFitToWidth(false)\n }\n\n const onWheel = (e: WheelEvent) => {\n if (!e.ctrlKey) return\n e.preventDefault()\n\n pendingDelta = Math.max(\n -WHEEL_ZOOM_CAP,\n Math.min(pendingDelta + -e.deltaY * WHEEL_ZOOM_SENSITIVITY, WHEEL_ZOOM_CAP)\n )\n const rect = el.getBoundingClientRect()\n lastCursorX = e.clientX - rect.left\n lastCursorY = e.clientY - rect.top\n\n if (rafId === 0) {\n rafId = requestAnimationFrame(flush)\n }\n }\n\n el.addEventListener('wheel', onWheel, { passive: false })\n return () => {\n el.removeEventListener('wheel', onWheel)\n if (rafId !== 0) cancelAnimationFrame(rafId)\n }\n }, [scrollRef])\n\n // Scroll anchoring — adjust scroll position after zoom so the focal point stays put\n useLayoutEffect(() => {\n const anchor = zoomAnchorRef.current\n if (!anchor) return\n const el = scrollRef.current\n if (!el) return\n\n const ratio = effectiveScale / anchor.prevScale\n el.scrollLeft = anchor.pointX * ratio - anchor.viewportX\n el.scrollTop = anchor.pointY * ratio - anchor.viewportY\n zoomAnchorRef.current = null\n }, [effectiveScale, scrollRef])\n\n // Print handling — clone-and-hide strategy, gated on this viewport owning the print.\n useEffect(\n () =>\n registerPrintController({\n getTarget: () => contentRef.current,\n getFilename: () => pdfFilename ?? 'resume',\n shouldOwnPrint: () => printRequestedRef.current || isActiveRef.current,\n onAfterPrint: () => {\n printRequestedRef.current = false\n forceRender(n => n + 1)\n },\n }),\n [pdfFilename, contentRef]\n )\n\n // Keyboard shortcuts\n useEffect(() => {\n const keyActions: Record<string, () => void> = {\n '+': handleZoomIn,\n '=': handleZoomIn,\n '-': handleZoomOut,\n _: handleZoomOut,\n }\n const modKeyActions: Record<string, () => void> = {\n '0': handleFitToWidth,\n '1': handleActualSize,\n p: requestPrint,\n }\n\n const handleKeyDown = (e: KeyboardEvent) => {\n if (!isActiveRef.current || isInputFocused()) return\n const mod = e.ctrlKey || e.metaKey\n const action = mod ? modKeyActions[e.key] : keyActions[e.key]\n if (action) {\n e.preventDefault()\n action()\n }\n }\n\n globalThis.addEventListener('keydown', handleKeyDown)\n return () => globalThis.removeEventListener('keydown', handleKeyDown)\n }, [handleZoomIn, handleZoomOut, handleFitToWidth, handleActualSize, requestPrint])\n\n // Click-and-drag panning (hand tool). Mobile uses touch pan + pinch-to-zoom.\n useEffect(() => {\n const el = scrollRef.current\n if (!el) return\n\n const setPanActive = (active: boolean) => {\n if (isMobile) {\n el.dataset.panActive = active ? '' : undefined\n }\n }\n\n const moveDrag = (clientX: number, clientY: number) => {\n const drag = dragRef.current\n if (!drag) return\n el.scrollLeft = drag.scrollX - (clientX - drag.startX)\n el.scrollTop = drag.scrollY - (clientY - drag.startY)\n clampScrollPosition(el)\n }\n\n const beginDrag = (clientX: number, clientY: number) => {\n dragRef.current = {\n startX: clientX,\n startY: clientY,\n scrollX: el.scrollLeft,\n scrollY: el.scrollTop,\n }\n setPanActive(true)\n if (!isMobile) {\n el.classList.replace('resume-viewer-scroll-grab', 'resume-viewer-scroll-grabbing')\n el.classList.replace('cursor-grab', 'cursor-grabbing')\n }\n }\n\n const endDrag = () => {\n touchPanRef.current = null\n pinchRef.current = null\n if (!dragRef.current) return\n dragRef.current = null\n setPanActive(false)\n if (!isMobile) {\n el.classList.replace('resume-viewer-scroll-grabbing', 'resume-viewer-scroll-grab')\n el.classList.replace('cursor-grabbing', 'cursor-grab')\n }\n }\n\n const captureZoomAnchor = (clientX: number, clientY: number) => {\n const rect = el.getBoundingClientRect()\n zoomAnchorRef.current = {\n pointX: el.scrollLeft + clientX - rect.left,\n pointY: el.scrollTop + clientY - rect.top,\n viewportX: clientX - rect.left,\n viewportY: clientY - rect.top,\n prevScale: effectiveScaleRef.current,\n }\n }\n\n const onMouseDown = (e: MouseEvent) => {\n if (isMobile || e.button !== 0) return\n if (isInteractivePanTarget(e.target)) return\n\n beginDrag(e.clientX, e.clientY)\n e.preventDefault()\n }\n\n const onMouseMove = (e: MouseEvent) => {\n if (isMobile) return\n moveDrag(e.clientX, e.clientY)\n }\n\n const onTouchStart = (e: TouchEvent) => {\n if (!isMobile) return\n\n if (e.touches.length === 2) {\n touchPanRef.current = null\n dragRef.current = null\n const distance = getTouchDistance(e.touches)\n if (distance <= 0) return\n pinchRef.current = {\n startDistance: distance,\n startScale: effectiveScaleRef.current,\n }\n setIsFitToWidth(false)\n return\n }\n\n if (e.touches.length !== 1 || pinchRef.current) return\n if (isInteractivePanTarget(e.target)) return\n\n const touch = e.touches[0]\n if (!touch) return\n touchPanRef.current = {\n startX: touch.clientX,\n startY: touch.clientY,\n scrollX: el.scrollLeft,\n scrollY: el.scrollTop,\n }\n }\n\n const onTouchMove = (e: TouchEvent) => {\n if (!isMobile) return\n\n if (pinchRef.current && e.touches.length === 2) {\n const distance = getTouchDistance(e.touches)\n if (distance <= 0 || pinchRef.current.startDistance <= 0) return\n\n const first = e.touches[0]\n const second = e.touches[1]\n if (!first || !second) return\n\n const centerX = (first.clientX + second.clientX) / 2\n const centerY = (first.clientY + second.clientY) / 2\n const nextScale = clampResumeZoom(\n pinchRef.current.startScale * (distance / pinchRef.current.startDistance)\n )\n\n captureZoomAnchor(centerX, centerY)\n setZoom(nextScale)\n setIsFitToWidth(false)\n e.preventDefault()\n return\n }\n\n const pendingPan = touchPanRef.current\n const touch = e.touches[0]\n if (!pendingPan || !touch) return\n\n if (!dragRef.current) {\n if (\n !hasExceededPanThreshold(\n pendingPan.startX,\n pendingPan.startY,\n touch.clientX,\n touch.clientY\n )\n ) {\n return\n }\n\n dragRef.current = {\n startX: pendingPan.startX,\n startY: pendingPan.startY,\n scrollX: pendingPan.scrollX,\n scrollY: pendingPan.scrollY,\n }\n setPanActive(true)\n }\n\n moveDrag(touch.clientX, touch.clientY)\n e.preventDefault()\n }\n\n el.addEventListener('mousedown', onMouseDown)\n window.addEventListener('mousemove', onMouseMove)\n window.addEventListener('mouseup', endDrag)\n\n if (isMobile) {\n el.addEventListener('touchstart', onTouchStart, { passive: true })\n el.addEventListener('touchmove', onTouchMove, { passive: false })\n el.addEventListener('touchend', endDrag)\n el.addEventListener('touchcancel', endDrag)\n }\n\n return () => {\n el.removeEventListener('mousedown', onMouseDown)\n window.removeEventListener('mousemove', onMouseMove)\n window.removeEventListener('mouseup', endDrag)\n delete el.dataset.panActive\n if (isMobile) {\n el.removeEventListener('touchstart', onTouchStart)\n el.removeEventListener('touchmove', onTouchMove)\n el.removeEventListener('touchend', endDrag)\n el.removeEventListener('touchcancel', endDrag)\n }\n }\n }, [isMobile, scrollRef])\n\n const handleCopyText = useCallback(async () => {\n const content = contentRef.current\n if (!content) return\n try {\n await navigator.clipboard.writeText(content.innerText)\n onCopy?.()\n } catch (error) {\n onError?.(error)\n }\n }, [contentRef, onCopy, onError])\n\n const totalHeight = pageCount * geo.heightPx + Math.max(0, pageCount - 1) * geo.gapPx\n\n return {\n pageCount,\n setPageCount,\n effectiveScale,\n isFitToWidth,\n totalHeight,\n geo,\n handleZoomIn,\n handleZoomOut,\n handleActualSize,\n handleFitToWidth,\n requestPrint,\n handleCopyText,\n minScale: MIN_SCALE,\n maxScale: MAX_SCALE,\n }\n}\n","import { MDXProvider } from '@mdx-js/react'\nimport clsx from 'clsx'\nimport { ClipboardCopy, Maximize2, Minus, Plus } from 'lucide-react'\nimport { useRef } from 'react'\nimport { resumeMdxComponents } from '../document/components'\nimport {\n PaginationReportContext,\n ResumeFontFamilyContext,\n ResumePageSizeContext,\n} from '../document/pagination-context'\n// Styles ship via the `@atom63/resume/styles` entry (import it once in your app)\n// — NOT self-imported here, so a consumer's token overrides reliably win the\n// cascade instead of racing a re-injected copy of the package CSS.\nimport type { ResumeViewerProps } from './resume-viewer.types'\nimport { useIsMobile } from './use-is-mobile'\nimport { useResumeViewport } from './use-resume-viewport'\n\nexport function ResumeViewer({\n Content,\n components,\n fontFamily,\n isActive = true,\n pageSize = 'letter',\n pdfFilename,\n onCopy,\n onError,\n className,\n toolbarStart,\n toolbarEnd,\n}: ResumeViewerProps) {\n const isMobile = useIsMobile()\n const scrollRef = useRef<HTMLDivElement>(null)\n const contentRef = useRef<HTMLDivElement>(null)\n\n const {\n pageCount,\n setPageCount,\n effectiveScale,\n isFitToWidth,\n totalHeight,\n geo,\n handleZoomIn,\n handleZoomOut,\n handleActualSize,\n handleFitToWidth,\n requestPrint,\n handleCopyText,\n minScale,\n maxScale,\n } = useResumeViewport({\n scrollRef,\n contentRef,\n isMobile,\n isActive,\n pageSize,\n pdfFilename,\n onCopy,\n onError,\n })\n\n return (\n <div className=\"resume-viewer\">\n {/* Drives the printed paper size; @page can't read a CSS custom property. */}\n <style>{`@page { size: ${geo.cssPageSize}; margin: 0; }`}</style>\n <div className=\"resume-viewer-toolbar\">\n {toolbarStart}\n <span className=\"resume-viewer-pagecount\">\n {pageCount} page{pageCount > 1 ? 's' : ''}\n </span>\n <div className=\"resume-viewer-controls\">\n <button\n aria-label=\"Fit resume to width\"\n aria-pressed={isFitToWidth}\n className={clsx('resume-viewer-btn', isFitToWidth && 'resume-viewer-btn-active')}\n onClick={handleFitToWidth}\n title=\"Fit to width\"\n type=\"button\"\n >\n <Maximize2 className=\"resume-viewer-icon\" />\n </button>\n <div className=\"resume-viewer-btn-group\">\n <button\n aria-label=\"Zoom out\"\n className=\"resume-viewer-btn\"\n disabled={!isFitToWidth && effectiveScale <= minScale}\n onClick={handleZoomOut}\n title=\"Zoom out\"\n type=\"button\"\n >\n <Minus className=\"resume-viewer-icon\" />\n </button>\n <button\n className=\"resume-viewer-zoom-label\"\n onClick={handleActualSize}\n title=\"Actual size (100%)\"\n type=\"button\"\n >\n {Math.round(effectiveScale * 100)}%\n </button>\n <button\n aria-label=\"Zoom in\"\n className=\"resume-viewer-btn\"\n disabled={effectiveScale >= maxScale}\n onClick={handleZoomIn}\n title=\"Zoom in\"\n type=\"button\"\n >\n <Plus className=\"resume-viewer-icon\" />\n </button>\n </div>\n <button\n aria-label=\"Copy resume text\"\n className=\"resume-viewer-btn\"\n onClick={handleCopyText}\n title=\"Copy resume text\"\n type=\"button\"\n >\n <ClipboardCopy className=\"resume-viewer-icon\" />\n </button>\n <button\n className=\"resume-viewer-btn resume-viewer-btn-primary\"\n onClick={requestPrint}\n type=\"button\"\n >\n Save to PDF\n </button>\n {toolbarEnd}\n </div>\n </div>\n\n <div\n className={clsx(\n 'resume-viewer-scroll',\n isMobile ? 'resume-viewer-scroll-mobile' : 'resume-viewer-scroll-grab',\n className\n )}\n ref={scrollRef}\n >\n <section\n aria-label={\n isMobile\n ? 'Resume preview. Drag with one finger to pan, pinch to zoom.'\n : 'Resume preview. Drag to pan, Ctrl+scroll to zoom.'\n }\n className=\"resume-viewer-stage\"\n >\n <div\n className=\"resume-viewer-page-box\"\n style={{\n width: geo.widthPx * effectiveScale,\n height: totalHeight * effectiveScale,\n }}\n >\n <div\n className=\"resume-viewer-scale-root\"\n data-resume-scale-root=\"\"\n style={{\n width: geo.widthPx,\n height: totalHeight,\n transform: effectiveScale === 1 ? undefined : `scale(${effectiveScale})`,\n }}\n >\n <ResumePageSizeContext.Provider value={pageSize}>\n <ResumeFontFamilyContext.Provider value={fontFamily}>\n <PaginationReportContext.Provider value={setPageCount}>\n <MDXProvider components={components ?? resumeMdxComponents}>\n <div\n className=\"resume-viewer-content\"\n id=\"resume-print-target\"\n ref={contentRef}\n style={{ zIndex: 1 }}\n >\n <Content />\n </div>\n </MDXProvider>\n </PaginationReportContext.Provider>\n </ResumeFontFamilyContext.Provider>\n </ResumePageSizeContext.Provider>\n </div>\n </div>\n </section>\n </div>\n </div>\n )\n}\n"]}
1
+ {"version":3,"sources":["../src/viewer/use-is-mobile.ts","../src/viewer/print.ts","../src/viewer/viewport.ts","../src/viewer/use-resume-viewport.ts","../src/viewer/resume-viewer.tsx"],"names":["useState","useEffect","useRef"],"mappings":"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","file":"index.js","sourcesContent":["import { useEffect, useState } from 'react'\n\n/** True when the viewport is narrow (<= 768px). SSR-safe (defaults false). */\nexport function useIsMobile(breakpointPx = 768): boolean {\n const [isMobile, setIsMobile] = useState(false)\n useEffect(() => {\n const mql = window.matchMedia(`(max-width: ${breakpointPx}px)`)\n const update = () => setIsMobile(mql.matches)\n update()\n mql.addEventListener('change', update)\n return () => mql.removeEventListener('change', update)\n }, [breakpointPx])\n return isMobile\n}\n","/**\n * Build a print-ready clone of the resume content element. The pagination\n * engine already renders real 8.5x11 `[data-resume-page]` frames, so the clone\n * is a plain wrapper holding those frames (no extra page sizing/padding); the\n * print stylesheet maps each frame to a physical page. The hidden measuring\n * layer is dropped so only the real pages print.\n */\nexport function createPrintClone(source: HTMLElement): HTMLElement {\n const clone = source.cloneNode(true) as HTMLElement\n clone.id = 'resume-print-clone'\n clone.style.cssText = 'background:white;color:black'\n clone.querySelector('[data-resume-measure]')?.remove()\n return clone\n}\n\n/** Hide all non-essential body children for printing; returns the hidden elements. */\nexport function hideBodyChildren(): HTMLElement[] {\n const hidden: HTMLElement[] = []\n const preserveTags = new Set(['STYLE', 'LINK', 'SCRIPT'])\n for (const child of Array.from(document.body.children) as HTMLElement[]) {\n if (!preserveTags.has(child.tagName)) {\n child.style.setProperty('display', 'none', 'important')\n hidden.push(child)\n }\n }\n return hidden\n}\n\nexport interface PrintControllerOptions {\n /** The element holding the rendered resume page frames (contentRef.current). */\n getTarget: () => HTMLElement | null\n /** Suggested PDF filename — becomes document.title during print. */\n getFilename: () => string\n /** Whether THIS viewer should own the print (e.g. it requested print, or it's the active/focused view). */\n shouldOwnPrint: () => boolean\n /** Optional: called after print is torn down (e.g. to force a re-render). */\n onAfterPrint?: () => void\n}\n\n/**\n * Register beforeprint/afterprint handlers implementing the clone-and-hide\n * print strategy. Returns an unsubscribe function that removes the listeners.\n */\nexport function registerPrintController(opts: PrintControllerOptions): () => void {\n let hiddenElements: HTMLElement[] = []\n let savedTitle = ''\n\n const beforePrint = () => {\n // Own the print only when this viewer should (its Save-to-PDF button /\n // shortcut requested it, or it's the active/focused view). Otherwise leave\n // it alone so another app isn't cloned/printed underneath, and so this\n // viewer isn't skipped when focus hasn't flushed yet.\n if (!opts.shouldOwnPrint()) return\n const target = opts.getTarget()\n if (!target) return\n savedTitle = document.title\n document.title = opts.getFilename()\n const clone = createPrintClone(target)\n hiddenElements = hideBodyChildren()\n document.body.appendChild(clone)\n }\n\n const afterPrint = () => {\n const clone = document.getElementById('resume-print-clone')\n if (clone) clone.remove()\n document.title = savedTitle\n for (const el of hiddenElements) {\n el.style.removeProperty('display')\n }\n hiddenElements = []\n opts.onAfterPrint?.()\n }\n\n window.addEventListener('beforeprint', beforePrint)\n window.addEventListener('afterprint', afterPrint)\n return () => {\n window.removeEventListener('beforeprint', beforePrint)\n window.removeEventListener('afterprint', afterPrint)\n }\n}\n","export const INTERACTIVE_PAN_SELECTOR = 'a, button, [role=\"button\"], input, textarea, select'\n\nexport const PAN_THRESHOLD_PX = 6\n\nexport function clampZoom(value: number, min: number, max: number): number {\n return Math.max(min, Math.min(value, max))\n}\n\nexport function isInteractivePanTarget(target: EventTarget | null): boolean {\n return target instanceof HTMLElement && Boolean(target.closest(INTERACTIVE_PAN_SELECTOR))\n}\n\nexport function getTouchDistance(touches: TouchList): number {\n if (touches.length < 2) {\n return 0\n }\n\n const first = touches[0]\n const second = touches[1]\n if (!first || !second) {\n return 0\n }\n\n const deltaX = first.clientX - second.clientX\n const deltaY = first.clientY - second.clientY\n return Math.hypot(deltaX, deltaY)\n}\n\nexport function clampScrollPosition(element: HTMLElement): void {\n const maxScrollLeft = Math.max(0, element.scrollWidth - element.clientWidth)\n const maxScrollTop = Math.max(0, element.scrollHeight - element.clientHeight)\n element.scrollLeft = Math.min(Math.max(0, element.scrollLeft), maxScrollLeft)\n element.scrollTop = Math.min(Math.max(0, element.scrollTop), maxScrollTop)\n}\n\nexport function hasExceededPanThreshold(\n startX: number,\n startY: number,\n currentX: number,\n currentY: number,\n threshold = PAN_THRESHOLD_PX\n): boolean {\n return Math.abs(currentX - startX) > threshold || Math.abs(currentY - startY) > threshold\n}\n","// packages/resume/src/viewer/use-resume-viewport.ts\n//\n// Headless viewport engine for a paginated resume: zoom (buttons / Ctrl+wheel /\n// pinch), pan (mouse drag / touch), fit-to-width, focus-aware print + keyboard\n// shortcuts. The single source of truth for this behaviour — both the built-in\n// `ResumeViewer` and bespoke hosts (e.g. an OS window with its own toolbar\n// chrome) consume this hook so the logic is never duplicated.\n//\n// The host owns the markup and supplies the scroll + content refs; the hook\n// wires every listener to them and returns the derived scale, page count, and\n// the toolbar action handlers.\nimport { type RefObject, useCallback, useEffect, useLayoutEffect, useRef, useState } from 'react'\nimport { getPageGeometry, type PageGeometry, type PageSize, WRAPPER_PADDING_X } from '../geometry'\nimport { registerPrintController } from './print'\nimport {\n clampScrollPosition,\n clampZoom,\n getTouchDistance,\n hasExceededPanThreshold,\n isInteractivePanTarget,\n} from './viewport'\n\nexport const SCALE_STEP = 0.1\nexport const MIN_SCALE = 0.25\nexport const MAX_SCALE = 2\nconst WHEEL_ZOOM_SENSITIVITY = 0.01\nconst WHEEL_ZOOM_CAP = 0.03\n\nconst clampResumeZoom = (value: number) => clampZoom(value, MIN_SCALE, MAX_SCALE)\n\nfunction isInputFocused(): boolean {\n const tag = document.activeElement?.tagName\n return tag === 'INPUT' || tag === 'TEXTAREA'\n}\n\nexport interface UseResumeViewportOptions {\n /** Scroll viewport element (the host attaches this to its scrollable region). */\n scrollRef: RefObject<HTMLDivElement | null>\n /** Element holding the rendered page frames (the print/copy target). */\n contentRef: RefObject<HTMLDivElement | null>\n /** Narrow-viewport flag — drives touch pan/pinch vs mouse drag. */\n isMobile: boolean\n /** Whether this viewport owns global print + keyboard shortcuts. */\n isActive: boolean\n /** Paper size driving the page dimensions + fit-to-width. Default 'letter'. */\n pageSize?: PageSize\n /** Suggested PDF filename (becomes document.title during print). */\n pdfFilename?: string\n /** Called after a successful copy-to-clipboard. */\n onCopy?: () => void\n /** Called when copy-to-clipboard fails. */\n onError?: (error: unknown) => void\n}\n\nexport interface UseResumeViewport {\n /** Current page count (reported by the pagination engine via setPageCount). */\n pageCount: number\n /** Reporter wired into PaginationReportContext. */\n setPageCount: (n: number) => void\n /** The scale actually applied to the page stage (fit-to-width or manual zoom). */\n effectiveScale: number\n /** Whether fit-to-width mode is active. */\n isFitToWidth: boolean\n /** Total stacked page height (unscaled px) — for sizing the page box. */\n totalHeight: number\n /** Resolved geometry for the active page size — for sizing the scaled box. */\n geo: PageGeometry\n handleZoomIn: () => void\n handleZoomOut: () => void\n handleActualSize: () => void\n handleFitToWidth: () => void\n requestPrint: () => void\n handleCopyText: () => Promise<void>\n /** Lower/upper zoom bounds (for disabling +/- controls). */\n minScale: number\n maxScale: number\n}\n\n/**\n * Wire the resume viewport behaviour to host-supplied refs and return the\n * derived state + action handlers. The host renders the scaled page stage; the\n * pagination engine inside it reports its page count through `setPageCount`.\n */\nexport function useResumeViewport({\n scrollRef,\n contentRef,\n isMobile,\n isActive,\n pageSize = 'letter',\n pdfFilename,\n onCopy,\n onError,\n}: UseResumeViewportOptions): UseResumeViewport {\n const geo = getPageGeometry(pageSize)\n\n // Keep the latest active state readable from global listeners (print/keyboard)\n // without re-registering them, so this viewport only hijacks printing when\n // it's the active view (avoids clashing with other apps' print handlers).\n const isActiveRef = useRef(isActive)\n isActiveRef.current = isActive\n\n // Set synchronously right before window.print() so the global beforeprint\n // listener knows THIS viewport initiated the print (active state can lag a click).\n const printRequestedRef = useRef(false)\n const requestPrint = useCallback(() => {\n printRequestedRef.current = true\n window.print()\n }, [])\n\n const zoomAnchorRef = useRef<{\n pointX: number\n pointY: number\n viewportX: number\n viewportY: number\n prevScale: number\n } | null>(null)\n const [pageCount, setPageCount] = useState(1)\n const [zoom, setZoom] = useState(1)\n const [fitScale, setFitScale] = useState(1)\n const [isFitToWidth, setIsFitToWidth] = useState(true)\n const [, forceRender] = useState(0)\n const dragRef = useRef<{\n startX: number\n startY: number\n scrollX: number\n scrollY: number\n } | null>(null)\n const touchPanRef = useRef<{\n scrollX: number\n scrollY: number\n startX: number\n startY: number\n } | null>(null)\n const pinchRef = useRef<{\n startDistance: number\n startScale: number\n } | null>(null)\n\n const effectiveScale = isFitToWidth ? fitScale : zoom\n\n // Refs for stable closures — avoids tearing down/re-registering listeners on every zoom tick\n const effectiveScaleRef = useRef(effectiveScale)\n const isFitToWidthRef = useRef(isFitToWidth)\n const fitScaleRef = useRef(fitScale)\n effectiveScaleRef.current = effectiveScale\n isFitToWidthRef.current = isFitToWidth\n fitScaleRef.current = fitScale\n\n // Keep zoom synced to fitScale while in fit-to-width mode\n useEffect(() => {\n if (isFitToWidth) setZoom(fitScale)\n }, [isFitToWidth, fitScale])\n\n // Responsive scaling — compute fit-to-width from the scroll viewport.\n useEffect(() => {\n const viewport = scrollRef.current\n if (!viewport) return\n\n const updateFitScale = () => {\n const available = Math.max(0, viewport.clientWidth - WRAPPER_PADDING_X)\n setFitScale(clampResumeZoom(Math.min(1, available / geo.widthPx)))\n }\n\n updateFitScale()\n const observer = new ResizeObserver(updateFitScale)\n observer.observe(viewport)\n window.visualViewport?.addEventListener('resize', updateFitScale)\n\n return () => {\n observer.disconnect()\n window.visualViewport?.removeEventListener('resize', updateFitScale)\n }\n }, [scrollRef, geo.widthPx])\n\n const captureViewportCenter = useCallback(() => {\n const el = scrollRef.current\n if (!el) return\n zoomAnchorRef.current = {\n pointX: el.scrollLeft + el.clientWidth / 2,\n pointY: el.scrollTop + el.clientHeight / 2,\n viewportX: el.clientWidth / 2,\n viewportY: el.clientHeight / 2,\n prevScale: effectiveScaleRef.current,\n }\n }, [scrollRef])\n\n const handleZoomIn = useCallback(() => {\n captureViewportCenter()\n setIsFitToWidth(false)\n setZoom(prev => clampResumeZoom(prev + SCALE_STEP))\n }, [captureViewportCenter])\n\n const handleZoomOut = useCallback(() => {\n captureViewportCenter()\n setIsFitToWidth(false)\n setZoom(prev => clampResumeZoom(prev - SCALE_STEP))\n }, [captureViewportCenter])\n\n const handleActualSize = useCallback(() => {\n captureViewportCenter()\n setIsFitToWidth(false)\n setZoom(1)\n }, [captureViewportCenter])\n\n const handleFitToWidth = useCallback(() => {\n setIsFitToWidth(true)\n }, [])\n\n // Ctrl+wheel zoom — rAF-batched for smooth trackpad pinch\n useEffect(() => {\n const el = scrollRef.current\n if (!el) return\n\n let pendingDelta = 0\n let rafId = 0\n let lastCursorX = 0\n let lastCursorY = 0\n\n const flush = () => {\n rafId = 0\n const delta = pendingDelta\n pendingDelta = 0\n const cursorX = lastCursorX\n const cursorY = lastCursorY\n\n setZoom(prev => {\n const next = clampResumeZoom(prev + delta)\n if (next === prev) return prev\n zoomAnchorRef.current = {\n pointX: el.scrollLeft + cursorX,\n pointY: el.scrollTop + cursorY,\n viewportX: cursorX,\n viewportY: cursorY,\n prevScale: isFitToWidthRef.current ? fitScaleRef.current : prev,\n }\n return next\n })\n setIsFitToWidth(false)\n }\n\n const onWheel = (e: WheelEvent) => {\n if (!e.ctrlKey) return\n e.preventDefault()\n\n pendingDelta = Math.max(\n -WHEEL_ZOOM_CAP,\n Math.min(pendingDelta + -e.deltaY * WHEEL_ZOOM_SENSITIVITY, WHEEL_ZOOM_CAP)\n )\n const rect = el.getBoundingClientRect()\n lastCursorX = e.clientX - rect.left\n lastCursorY = e.clientY - rect.top\n\n if (rafId === 0) {\n rafId = requestAnimationFrame(flush)\n }\n }\n\n el.addEventListener('wheel', onWheel, { passive: false })\n return () => {\n el.removeEventListener('wheel', onWheel)\n if (rafId !== 0) cancelAnimationFrame(rafId)\n }\n }, [scrollRef])\n\n // Scroll anchoring — adjust scroll position after zoom so the focal point stays put\n useLayoutEffect(() => {\n const anchor = zoomAnchorRef.current\n if (!anchor) return\n const el = scrollRef.current\n if (!el) return\n\n const ratio = effectiveScale / anchor.prevScale\n el.scrollLeft = anchor.pointX * ratio - anchor.viewportX\n el.scrollTop = anchor.pointY * ratio - anchor.viewportY\n zoomAnchorRef.current = null\n }, [effectiveScale, scrollRef])\n\n // Print handling — clone-and-hide strategy, gated on this viewport owning the print.\n useEffect(\n () =>\n registerPrintController({\n getTarget: () => contentRef.current,\n getFilename: () => pdfFilename ?? 'resume',\n shouldOwnPrint: () => printRequestedRef.current || isActiveRef.current,\n onAfterPrint: () => {\n printRequestedRef.current = false\n forceRender(n => n + 1)\n },\n }),\n [pdfFilename, contentRef]\n )\n\n // Keyboard shortcuts\n useEffect(() => {\n const keyActions: Record<string, () => void> = {\n '+': handleZoomIn,\n '=': handleZoomIn,\n '-': handleZoomOut,\n _: handleZoomOut,\n }\n const modKeyActions: Record<string, () => void> = {\n '0': handleFitToWidth,\n '1': handleActualSize,\n p: requestPrint,\n }\n\n const handleKeyDown = (e: KeyboardEvent) => {\n if (!isActiveRef.current || isInputFocused()) return\n const mod = e.ctrlKey || e.metaKey\n const action = mod ? modKeyActions[e.key] : keyActions[e.key]\n if (action) {\n e.preventDefault()\n action()\n }\n }\n\n globalThis.addEventListener('keydown', handleKeyDown)\n return () => globalThis.removeEventListener('keydown', handleKeyDown)\n }, [handleZoomIn, handleZoomOut, handleFitToWidth, handleActualSize, requestPrint])\n\n // Click-and-drag panning (hand tool). Mobile uses touch pan + pinch-to-zoom.\n useEffect(() => {\n const el = scrollRef.current\n if (!el) return\n\n const setPanActive = (active: boolean) => {\n if (isMobile) {\n el.dataset.panActive = active ? '' : undefined\n }\n }\n\n const moveDrag = (clientX: number, clientY: number) => {\n const drag = dragRef.current\n if (!drag) return\n el.scrollLeft = drag.scrollX - (clientX - drag.startX)\n el.scrollTop = drag.scrollY - (clientY - drag.startY)\n clampScrollPosition(el)\n }\n\n const beginDrag = (clientX: number, clientY: number) => {\n dragRef.current = {\n startX: clientX,\n startY: clientY,\n scrollX: el.scrollLeft,\n scrollY: el.scrollTop,\n }\n setPanActive(true)\n if (!isMobile) {\n el.classList.replace('resume-viewer-scroll-grab', 'resume-viewer-scroll-grabbing')\n el.classList.replace('cursor-grab', 'cursor-grabbing')\n }\n }\n\n const endDrag = () => {\n touchPanRef.current = null\n pinchRef.current = null\n if (!dragRef.current) return\n dragRef.current = null\n setPanActive(false)\n if (!isMobile) {\n el.classList.replace('resume-viewer-scroll-grabbing', 'resume-viewer-scroll-grab')\n el.classList.replace('cursor-grabbing', 'cursor-grab')\n }\n }\n\n const captureZoomAnchor = (clientX: number, clientY: number) => {\n const rect = el.getBoundingClientRect()\n zoomAnchorRef.current = {\n pointX: el.scrollLeft + clientX - rect.left,\n pointY: el.scrollTop + clientY - rect.top,\n viewportX: clientX - rect.left,\n viewportY: clientY - rect.top,\n prevScale: effectiveScaleRef.current,\n }\n }\n\n const onMouseDown = (e: MouseEvent) => {\n if (isMobile || e.button !== 0) return\n if (isInteractivePanTarget(e.target)) return\n\n beginDrag(e.clientX, e.clientY)\n e.preventDefault()\n }\n\n const onMouseMove = (e: MouseEvent) => {\n if (isMobile) return\n moveDrag(e.clientX, e.clientY)\n }\n\n const onTouchStart = (e: TouchEvent) => {\n if (!isMobile) return\n\n if (e.touches.length === 2) {\n touchPanRef.current = null\n dragRef.current = null\n const distance = getTouchDistance(e.touches)\n if (distance <= 0) return\n pinchRef.current = {\n startDistance: distance,\n startScale: effectiveScaleRef.current,\n }\n setIsFitToWidth(false)\n return\n }\n\n if (e.touches.length !== 1 || pinchRef.current) return\n if (isInteractivePanTarget(e.target)) return\n\n const touch = e.touches[0]\n if (!touch) return\n touchPanRef.current = {\n startX: touch.clientX,\n startY: touch.clientY,\n scrollX: el.scrollLeft,\n scrollY: el.scrollTop,\n }\n }\n\n const onTouchMove = (e: TouchEvent) => {\n if (!isMobile) return\n\n if (pinchRef.current && e.touches.length === 2) {\n const distance = getTouchDistance(e.touches)\n if (distance <= 0 || pinchRef.current.startDistance <= 0) return\n\n const first = e.touches[0]\n const second = e.touches[1]\n if (!first || !second) return\n\n const centerX = (first.clientX + second.clientX) / 2\n const centerY = (first.clientY + second.clientY) / 2\n const nextScale = clampResumeZoom(\n pinchRef.current.startScale * (distance / pinchRef.current.startDistance)\n )\n\n captureZoomAnchor(centerX, centerY)\n setZoom(nextScale)\n setIsFitToWidth(false)\n e.preventDefault()\n return\n }\n\n const pendingPan = touchPanRef.current\n const touch = e.touches[0]\n if (!pendingPan || !touch) return\n\n if (!dragRef.current) {\n if (\n !hasExceededPanThreshold(\n pendingPan.startX,\n pendingPan.startY,\n touch.clientX,\n touch.clientY\n )\n ) {\n return\n }\n\n dragRef.current = {\n startX: pendingPan.startX,\n startY: pendingPan.startY,\n scrollX: pendingPan.scrollX,\n scrollY: pendingPan.scrollY,\n }\n setPanActive(true)\n }\n\n moveDrag(touch.clientX, touch.clientY)\n e.preventDefault()\n }\n\n el.addEventListener('mousedown', onMouseDown)\n window.addEventListener('mousemove', onMouseMove)\n window.addEventListener('mouseup', endDrag)\n\n if (isMobile) {\n el.addEventListener('touchstart', onTouchStart, { passive: true })\n el.addEventListener('touchmove', onTouchMove, { passive: false })\n el.addEventListener('touchend', endDrag)\n el.addEventListener('touchcancel', endDrag)\n }\n\n return () => {\n el.removeEventListener('mousedown', onMouseDown)\n window.removeEventListener('mousemove', onMouseMove)\n window.removeEventListener('mouseup', endDrag)\n delete el.dataset.panActive\n if (isMobile) {\n el.removeEventListener('touchstart', onTouchStart)\n el.removeEventListener('touchmove', onTouchMove)\n el.removeEventListener('touchend', endDrag)\n el.removeEventListener('touchcancel', endDrag)\n }\n }\n }, [isMobile, scrollRef])\n\n const handleCopyText = useCallback(async () => {\n const content = contentRef.current\n if (!content) return\n try {\n await navigator.clipboard.writeText(content.innerText)\n onCopy?.()\n } catch (error) {\n onError?.(error)\n }\n }, [contentRef, onCopy, onError])\n\n const totalHeight = pageCount * geo.heightPx + Math.max(0, pageCount - 1) * geo.gapPx\n\n return {\n pageCount,\n setPageCount,\n effectiveScale,\n isFitToWidth,\n totalHeight,\n geo,\n handleZoomIn,\n handleZoomOut,\n handleActualSize,\n handleFitToWidth,\n requestPrint,\n handleCopyText,\n minScale: MIN_SCALE,\n maxScale: MAX_SCALE,\n }\n}\n","import { MDXProvider } from '@mdx-js/react'\nimport clsx from 'clsx'\nimport { ClipboardCopy, Maximize2, Minus, Plus } from 'lucide-react'\nimport { useRef } from 'react'\nimport { resumeMdxComponents } from '../document/components'\nimport {\n PaginationReportContext,\n ResumeFontFamilyContext,\n ResumePageSizeContext,\n} from '../document/pagination-context'\n// Styles ship via the `@atom63/resume/styles` entry (import it once in your app)\n// — NOT self-imported here, so a consumer's token overrides reliably win the\n// cascade instead of racing a re-injected copy of the package CSS.\nimport type { ResumeViewerProps } from './resume-viewer.types'\nimport { useIsMobile } from './use-is-mobile'\nimport { useResumeViewport } from './use-resume-viewport'\n\nexport function ResumeViewer({\n Content,\n components,\n fontFamily,\n isActive = true,\n pageSize = 'letter',\n pdfFilename,\n onCopy,\n onError,\n className,\n toolbarStart,\n toolbarEnd,\n}: ResumeViewerProps) {\n const isMobile = useIsMobile()\n const scrollRef = useRef<HTMLDivElement>(null)\n const contentRef = useRef<HTMLDivElement>(null)\n const mdxComponents = { ...resumeMdxComponents, ...components }\n\n const {\n pageCount,\n setPageCount,\n effectiveScale,\n isFitToWidth,\n totalHeight,\n geo,\n handleZoomIn,\n handleZoomOut,\n handleActualSize,\n handleFitToWidth,\n requestPrint,\n handleCopyText,\n minScale,\n maxScale,\n } = useResumeViewport({\n scrollRef,\n contentRef,\n isMobile,\n isActive,\n pageSize,\n pdfFilename,\n onCopy,\n onError,\n })\n\n return (\n <div className=\"resume-viewer\">\n {/* Drives the printed paper size; @page can't read a CSS custom property. */}\n <style>{`@page { size: ${geo.cssPageSize}; margin: 0; }`}</style>\n <div className=\"resume-viewer-toolbar\">\n {toolbarStart}\n <span className=\"resume-viewer-pagecount\">\n {pageCount} page{pageCount > 1 ? 's' : ''}\n </span>\n <div className=\"resume-viewer-controls\">\n <button\n aria-label=\"Fit resume to width\"\n aria-pressed={isFitToWidth}\n className={clsx('resume-viewer-btn', isFitToWidth && 'resume-viewer-btn-active')}\n onClick={handleFitToWidth}\n title=\"Fit to width\"\n type=\"button\"\n >\n <Maximize2 className=\"resume-viewer-icon\" />\n </button>\n <div className=\"resume-viewer-btn-group\">\n <button\n aria-label=\"Zoom out\"\n className=\"resume-viewer-btn\"\n disabled={!isFitToWidth && effectiveScale <= minScale}\n onClick={handleZoomOut}\n title=\"Zoom out\"\n type=\"button\"\n >\n <Minus className=\"resume-viewer-icon\" />\n </button>\n <button\n className=\"resume-viewer-zoom-label\"\n onClick={handleActualSize}\n title=\"Actual size (100%)\"\n type=\"button\"\n >\n {Math.round(effectiveScale * 100)}%\n </button>\n <button\n aria-label=\"Zoom in\"\n className=\"resume-viewer-btn\"\n disabled={effectiveScale >= maxScale}\n onClick={handleZoomIn}\n title=\"Zoom in\"\n type=\"button\"\n >\n <Plus className=\"resume-viewer-icon\" />\n </button>\n </div>\n <button\n aria-label=\"Copy resume text\"\n className=\"resume-viewer-btn\"\n onClick={handleCopyText}\n title=\"Copy resume text\"\n type=\"button\"\n >\n <ClipboardCopy className=\"resume-viewer-icon\" />\n </button>\n <button\n className=\"resume-viewer-btn resume-viewer-btn-primary\"\n onClick={requestPrint}\n type=\"button\"\n >\n Save to PDF\n </button>\n {toolbarEnd}\n </div>\n </div>\n\n <div\n className={clsx(\n 'resume-viewer-scroll',\n isMobile ? 'resume-viewer-scroll-mobile' : 'resume-viewer-scroll-grab',\n className\n )}\n ref={scrollRef}\n >\n <section\n aria-label={\n isMobile\n ? 'Resume preview. Drag with one finger to pan, pinch to zoom.'\n : 'Resume preview. Drag to pan, Ctrl+scroll to zoom.'\n }\n className=\"resume-viewer-stage\"\n >\n <div\n className=\"resume-viewer-page-box\"\n style={{\n width: geo.widthPx * effectiveScale,\n height: totalHeight * effectiveScale,\n }}\n >\n <div\n className=\"resume-viewer-scale-root\"\n data-resume-scale-root=\"\"\n style={{\n width: geo.widthPx,\n height: totalHeight,\n transform: effectiveScale === 1 ? undefined : `scale(${effectiveScale})`,\n }}\n >\n <ResumePageSizeContext.Provider value={pageSize}>\n <ResumeFontFamilyContext.Provider value={fontFamily}>\n <PaginationReportContext.Provider value={setPageCount}>\n <MDXProvider components={mdxComponents}>\n <div\n className=\"resume-viewer-content\"\n id=\"resume-print-target\"\n ref={contentRef}\n style={{ zIndex: 1 }}\n >\n <Content />\n </div>\n </MDXProvider>\n </PaginationReportContext.Provider>\n </ResumeFontFamilyContext.Provider>\n </ResumePageSizeContext.Provider>\n </div>\n </div>\n </section>\n </div>\n </div>\n )\n}\n"]}
@@ -58,11 +58,23 @@ function resumeWriteBackPlugin(options = {}) {
58
58
  }
59
59
  }
60
60
  server.watcher.unwatch(resumeAbs);
61
- const result = await handleWriteBack({ resumePath, root: server.config.root }, body);
62
- setTimeout(() => server.watcher.add(resumeAbs), 250);
63
- res.statusCode = result.ok ? 200 : 400;
64
- res.setHeader("content-type", "application/json");
65
- res.end(JSON.stringify(result));
61
+ try {
62
+ const result = await handleWriteBack({ resumePath, root: server.config.root }, body);
63
+ res.statusCode = result.ok ? 200 : 400;
64
+ res.setHeader("content-type", "application/json");
65
+ res.end(JSON.stringify(result));
66
+ } catch (error) {
67
+ res.statusCode = 500;
68
+ res.setHeader("content-type", "application/json");
69
+ res.end(
70
+ JSON.stringify({
71
+ ok: false,
72
+ error: error instanceof Error ? error.message : "Failed to write resume"
73
+ })
74
+ );
75
+ } finally {
76
+ setTimeout(() => server.watcher.add(resumeAbs), 250);
77
+ }
66
78
  });
67
79
  }
68
80
  };
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/vite/mdx-raw-plugin.ts","../../src/vite/write-back.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["import fs from 'node:fs/promises'\nimport type { Plugin } from 'vite'\n\nconst RAW_MDX = /\\.mdx\\?raw$/\nconst VIRTUAL_PREFIX = '\\0mdx-raw:'\nconst VIRTUAL_SUFFIX = '.raw'\n\n/**\n * Serve raw `.mdx?raw` imports as plain strings.\n *\n * Vite's built-in `?raw` works for most files, but `@mdx-js/rollup` strips the\n * query and compiles every `.mdx` to a component — so `?raw` would otherwise\n * yield the compiled module, not the source. This `pre` plugin resolves\n * `*.mdx?raw` to a virtual id ending in `.raw` (not `.mdx`), which MDX's\n * extension check skips, then loads the file text for it.\n *\n * Standalone resume consumers need this so `import source from './resume.mdx?raw'`\n * returns the raw MDX text (e.g. to feed the in-app editor / source view).\n */\nexport function mdxRawPlugin(): Plugin {\n return {\n name: 'mdx-raw',\n enforce: 'pre',\n async resolveId(id, importer) {\n if (!RAW_MDX.test(id)) {\n return null\n }\n const clean = id.replace(/\\?raw$/, '')\n const resolved = await this.resolve(clean, importer, { skipSelf: true })\n const filePath = resolved?.id ?? clean\n return `${VIRTUAL_PREFIX}${filePath}${VIRTUAL_SUFFIX}`\n },\n async load(id) {\n if (!id.startsWith(VIRTUAL_PREFIX)) {\n return null\n }\n const filePath = id.slice(VIRTUAL_PREFIX.length, id.length - VIRTUAL_SUFFIX.length)\n const code = await fs.readFile(filePath, 'utf-8')\n return { code: `export default ${JSON.stringify(code)}`, map: null }\n },\n }\n}\n","import { writeFile } from 'node:fs/promises'\nimport { isAbsolute, relative, resolve } from 'node:path'\nimport type { Plugin } from 'vite'\n\nexport interface ResumeWriteBackOptions {\n /** Path (absolute or relative to Vite root) of the resume file to persist to. */\n resumePath?: string\n /** POST endpoint. Default '/__write-resume'. */\n endpoint?: string\n}\n\n/** @internal Not part of the public API — use `resumeWriteBackPlugin` instead. */\nexport async function handleWriteBack(\n cfg: { resumePath: string; root: string },\n body: string\n): Promise<{ ok: boolean; error?: string }> {\n const abs = isAbsolute(cfg.resumePath) ? cfg.resumePath : resolve(cfg.root, cfg.resumePath)\n const rel = relative(cfg.root, abs)\n if (rel.startsWith('..') || isAbsolute(rel)) {\n return { ok: false, error: 'resume path escapes project root' }\n }\n await writeFile(abs, body, 'utf8')\n return { ok: true }\n}\n\nconst MAX_BODY = 5_000_000\n\n/** DEV-ONLY plugin: POST <endpoint> writes the request body to the resume file. */\nexport function resumeWriteBackPlugin(options: ResumeWriteBackOptions = {}): Plugin {\n const endpoint = options.endpoint ?? '/__write-resume'\n const resumePath = options.resumePath ?? 'src/resume.mdx'\n\n return {\n name: '@atom63/resume:resume-write-back',\n apply: 'serve',\n configureServer(server) {\n const resumeAbs = isAbsolute(resumePath)\n ? resumePath\n : resolve(server.config.root, resumePath)\n server.middlewares.use(async (req, res, next) => {\n if (req.method !== 'POST' || req.url?.split('?')[0] !== endpoint) return next()\n let body = ''\n for await (const chunk of req) {\n body += chunk\n if (body.length > MAX_BODY) {\n res.statusCode = 413\n res.end()\n return\n }\n }\n // A GUI Save writes the resume file, which Vite would otherwise full-reload\n // — dropping you out of the editor (the editor already holds this content\n // in state, so the reload is pure disruption). Unwatch the resume file\n // around our own write so the change event never fires, then re-watch\n // shortly after so EXTERNAL edits (you / your agent editing resume.mdx)\n // still hot-reload normally.\n server.watcher.unwatch(resumeAbs)\n const result = await handleWriteBack({ resumePath, root: server.config.root }, body)\n setTimeout(() => server.watcher.add(resumeAbs), 250)\n res.statusCode = result.ok ? 200 : 400\n res.setHeader('content-type', 'application/json')\n res.end(JSON.stringify(result))\n })\n },\n }\n}\n"]}
1
+ {"version":3,"sources":["../../src/vite/mdx-raw-plugin.ts","../../src/vite/write-back.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["import fs from 'node:fs/promises'\nimport type { Plugin } from 'vite'\n\nconst RAW_MDX = /\\.mdx\\?raw$/\nconst VIRTUAL_PREFIX = '\\0mdx-raw:'\nconst VIRTUAL_SUFFIX = '.raw'\n\n/**\n * Serve raw `.mdx?raw` imports as plain strings.\n *\n * Vite's built-in `?raw` works for most files, but `@mdx-js/rollup` strips the\n * query and compiles every `.mdx` to a component — so `?raw` would otherwise\n * yield the compiled module, not the source. This `pre` plugin resolves\n * `*.mdx?raw` to a virtual id ending in `.raw` (not `.mdx`), which MDX's\n * extension check skips, then loads the file text for it.\n *\n * Standalone resume consumers need this so `import source from './resume.mdx?raw'`\n * returns the raw MDX text (e.g. to feed the in-app editor / source view).\n */\nexport function mdxRawPlugin(): Plugin {\n return {\n name: 'mdx-raw',\n enforce: 'pre',\n async resolveId(id, importer) {\n if (!RAW_MDX.test(id)) {\n return null\n }\n const clean = id.replace(/\\?raw$/, '')\n const resolved = await this.resolve(clean, importer, { skipSelf: true })\n const filePath = resolved?.id ?? clean\n return `${VIRTUAL_PREFIX}${filePath}${VIRTUAL_SUFFIX}`\n },\n async load(id) {\n if (!id.startsWith(VIRTUAL_PREFIX)) {\n return null\n }\n const filePath = id.slice(VIRTUAL_PREFIX.length, id.length - VIRTUAL_SUFFIX.length)\n const code = await fs.readFile(filePath, 'utf-8')\n return { code: `export default ${JSON.stringify(code)}`, map: null }\n },\n }\n}\n","import { writeFile } from 'node:fs/promises'\nimport { isAbsolute, relative, resolve } from 'node:path'\nimport type { Plugin } from 'vite'\n\nexport interface ResumeWriteBackOptions {\n /** Path (absolute or relative to Vite root) of the resume file to persist to. */\n resumePath?: string\n /** POST endpoint. Default '/__write-resume'. */\n endpoint?: string\n}\n\n/** @internal Not part of the public API — use `resumeWriteBackPlugin` instead. */\nexport async function handleWriteBack(\n cfg: { resumePath: string; root: string },\n body: string\n): Promise<{ ok: boolean; error?: string }> {\n const abs = isAbsolute(cfg.resumePath) ? cfg.resumePath : resolve(cfg.root, cfg.resumePath)\n const rel = relative(cfg.root, abs)\n if (rel.startsWith('..') || isAbsolute(rel)) {\n return { ok: false, error: 'resume path escapes project root' }\n }\n await writeFile(abs, body, 'utf8')\n return { ok: true }\n}\n\nconst MAX_BODY = 5_000_000\n\n/** DEV-ONLY plugin: POST <endpoint> writes the request body to the resume file. */\nexport function resumeWriteBackPlugin(options: ResumeWriteBackOptions = {}): Plugin {\n const endpoint = options.endpoint ?? '/__write-resume'\n const resumePath = options.resumePath ?? 'src/resume.mdx'\n\n return {\n name: '@atom63/resume:resume-write-back',\n apply: 'serve',\n configureServer(server) {\n const resumeAbs = isAbsolute(resumePath)\n ? resumePath\n : resolve(server.config.root, resumePath)\n server.middlewares.use(async (req, res, next) => {\n if (req.method !== 'POST' || req.url?.split('?')[0] !== endpoint) return next()\n let body = ''\n for await (const chunk of req) {\n body += chunk\n if (body.length > MAX_BODY) {\n res.statusCode = 413\n res.end()\n return\n }\n }\n // A GUI Save writes the resume file, which Vite would otherwise full-reload\n // — dropping you out of the editor (the editor already holds this content\n // in state, so the reload is pure disruption). Unwatch the resume file\n // around our own write so the change event never fires, then re-watch\n // shortly after so EXTERNAL edits (you / your agent editing resume.mdx)\n // still hot-reload normally.\n server.watcher.unwatch(resumeAbs)\n try {\n const result = await handleWriteBack({ resumePath, root: server.config.root }, body)\n res.statusCode = result.ok ? 200 : 400\n res.setHeader('content-type', 'application/json')\n res.end(JSON.stringify(result))\n } catch (error) {\n res.statusCode = 500\n res.setHeader('content-type', 'application/json')\n res.end(\n JSON.stringify({\n ok: false,\n error: error instanceof Error ? error.message : 'Failed to write resume',\n })\n )\n } finally {\n setTimeout(() => server.watcher.add(resumeAbs), 250)\n }\n })\n },\n }\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@atom63/resume",
3
- "version": "0.2.1",
3
+ "version": "0.2.2",
4
4
  "license": "MIT",
5
5
  "type": "module",
6
6
  "description": "Write resumes and CVs as MDX — a paginated, print-ready document engine + viewer for React, styled with a self-contained paper/document token system.",
@@ -65,6 +65,13 @@
65
65
  "src/styles",
66
66
  "CHANGELOG.md"
67
67
  ],
68
+ "scripts": {
69
+ "build": "tsup",
70
+ "dev": "tsup --watch",
71
+ "test": "vitest run",
72
+ "typecheck": "tsc --noEmit",
73
+ "lint": "biome check ."
74
+ },
68
75
  "dependencies": {
69
76
  "@mdx-js/mdx": "^3.1.1",
70
77
  "@mdx-js/react": "^3.1.1",
@@ -82,6 +89,8 @@
82
89
  }
83
90
  },
84
91
  "devDependencies": {
92
+ "@atom63/biome-config": "workspace:*",
93
+ "@atom63/tsconfig": "workspace:*",
85
94
  "@biomejs/biome": "2.4.6",
86
95
  "@testing-library/dom": "10.4.1",
87
96
  "@testing-library/jest-dom": "6.9.1",
@@ -95,15 +104,6 @@
95
104
  "react-dom": "^19.1.1",
96
105
  "tsup": "^8.3.5",
97
106
  "typescript": "~5.8.3",
98
- "vitest": "^4.1.8",
99
- "@atom63/biome-config": "0.1.0",
100
- "@atom63/tsconfig": "0.1.0"
101
- },
102
- "scripts": {
103
- "build": "tsup",
104
- "dev": "tsup --watch",
105
- "test": "vitest run",
106
- "typecheck": "tsc --noEmit",
107
- "lint": "biome check ."
107
+ "vitest": "^4.1.8"
108
108
  }
109
- }
109
+ }
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/geometry.ts","../src/pagination/pagination.ts","../src/document/pagination-context.tsx","../src/mdx/base-components.tsx","../src/mdx/external-href.ts","../src/document/components.tsx","../src/document/paginated-resume.tsx"],"names":["jsx","jsxs","_"],"mappings":"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","file":"chunk-FI5DMK4W.js","sourcesContent":["// packages/resume/src/geometry.ts\n// Numeric page geometry — the single source of truth shared by the JS packer\n// and the rendered page frames. Print-fixed px at 96dpi for PDF fidelity.\n\nconst DPI = 96\nconst MM = DPI / 25.4\n\nexport type PageSize = 'letter' | 'a4'\n\nexport interface PageGeometry {\n widthPx: number\n heightPx: number\n padXPx: number\n padYPx: number\n /** Screen-only gap between page frames. */\n gapPx: number\n contentWidthPx: number\n usableHeightPx: number\n /** Value for the CSS `@page { size: … }` print rule. */\n cssPageSize: 'letter' | 'A4'\n}\n\n// Both sizes keep the same physical 0.7in x 0.6in margins.\nconst PAD_X_PX = 0.7 * DPI\nconst PAD_Y_PX = 0.6 * DPI\nconst GAP_PX = 32\n\nconst DIMENSIONS: Record<\n PageSize,\n { widthPx: number; heightPx: number; cssPageSize: 'letter' | 'A4' }\n> = {\n letter: { widthPx: 8.5 * DPI, heightPx: 11 * DPI, cssPageSize: 'letter' },\n a4: { widthPx: 210 * MM, heightPx: 297 * MM, cssPageSize: 'A4' },\n}\n\n/** Resolve the print-fixed geometry for a page size (defaults to US Letter). */\nexport function getPageGeometry(size: PageSize = 'letter'): PageGeometry {\n const d = DIMENSIONS[size]\n return {\n widthPx: d.widthPx,\n heightPx: d.heightPx,\n padXPx: PAD_X_PX,\n padYPx: PAD_Y_PX,\n gapPx: GAP_PX,\n contentWidthPx: d.widthPx - 2 * PAD_X_PX,\n usableHeightPx: d.heightPx - 2 * PAD_Y_PX,\n cssPageSize: d.cssPageSize,\n }\n}\n\n/** Back-compat default geometry (US Letter). */\nexport const PAGE = getPageGeometry('letter')\n\n// Kept for back-compat; equal to the Letter geometry's derived values.\nexport const CONTENT_WIDTH_PX = PAGE.contentWidthPx\nexport const USABLE_HEIGHT_PX = PAGE.usableHeightPx\n\n// Vertical gap between blocks within a column — must match --doc-block-gap\n// (0.375rem = 6px) in styles/tokens.css and the column gap in document.css.\nexport const BLOCK_GAP_PX = 6\n\n// Outer horizontal padding the viewer reserves around the page in its scroll area.\nexport const WRAPPER_PADDING_X = 32\n","// packages/resume/src/pagination/pagination.ts\n\nexport interface Block {\n id: string\n /** Measured px (offsetHeight + vertical margins). */\n height: number\n /** Section headings: don't leave them orphaned at a page bottom. */\n keepWithNext: boolean\n}\n\nexport type ColumnKey = 'sidebar' | 'main'\n\nexport interface PackInput {\n columns: Record<ColumnKey, Block[]>\n /** Usable content height for a given page index; index 0 may be shorter (header band). */\n pageUsableHeight: (pageIndex: number) => number\n /**\n * Vertical gap rendered between consecutive blocks in a column (the flex\n * `gap`). Block heights are measured individually, so the packer must add\n * this back or pages overflow once they hold many blocks. Defaults to 0.\n */\n blockGap?: number\n}\n\nexport interface PackResult {\n pageCount: number\n pageOf: Record<ColumnKey, Record<string, number>>\n /** Blocks taller than a full page — placed alone, allowed to overflow. */\n oversized: string[]\n}\n\nfunction packColumn(\n blocks: Block[],\n pageUsableHeight: (p: number) => number,\n gap: number\n): { pageOf: Record<string, number>; pages: number; oversized: string[] } {\n const pageOf: Record<string, number> = {}\n const oversized: string[] = []\n let page = 0\n let used = 0\n\n for (let i = 0; i < blocks.length; i++) {\n const b = blocks[i]\n let cap = pageUsableHeight(page)\n // A gap precedes every block except the first on its page.\n const gapBefore = used > 0 ? gap : 0\n\n // Move to a fresh page if this block (plus its leading gap) overflows.\n if (used > 0 && used + gapBefore + b.height > cap) {\n page++\n used = 0\n cap = pageUsableHeight(page)\n } else if (\n // Keep-with-next: heading fits alone but heading + gap + next block\n // doesn't — push the heading forward so it isn't orphaned.\n used > 0 &&\n b.keepWithNext &&\n used + gapBefore + b.height <= cap &&\n i + 1 < blocks.length &&\n used + gapBefore + b.height + gap + blocks[i + 1].height > cap\n ) {\n page++\n used = 0\n cap = pageUsableHeight(page)\n }\n\n // Oversized: taller than a whole (now-empty) page. Place alone, allow overflow.\n if (b.height > cap && used === 0) {\n oversized.push(b.id)\n pageOf[b.id] = page\n page++\n used = 0\n continue\n }\n\n pageOf[b.id] = page\n used += (used > 0 ? gap : 0) + b.height\n }\n\n return { pageOf, pages: page + 1, oversized }\n}\n\n/** Assign measured blocks to pages per column. Pure: no DOM. */\nexport function packIntoPages(input: PackInput): PackResult {\n const gap = input.blockGap ?? 0\n const sidebar = packColumn(input.columns.sidebar, input.pageUsableHeight, gap)\n const main = packColumn(input.columns.main, input.pageUsableHeight, gap)\n return {\n pageCount: Math.max(sidebar.pages, main.pages, 1),\n pageOf: { sidebar: sidebar.pageOf, main: main.pageOf },\n oversized: [...sidebar.oversized, ...main.oversized],\n }\n}\n","// packages/resume/src/document/pagination-context.tsx\nimport { createContext, useContext } from 'react'\nimport type { PageSize } from '../geometry'\n\n/** PaginatedResume reports its computed page count up to the host (ResumeViewer). */\nexport const PaginationReportContext = createContext<((pageCount: number) => void) | null>(null)\n\nexport function usePaginationReport(): (pageCount: number) => void {\n const report = useContext(PaginationReportContext)\n return report ?? (() => {})\n}\n\n/** Optional font-family hint — changes trigger re-measure/re-pagination. */\nexport const ResumeFontFamilyContext = createContext<string | undefined>(undefined)\nexport const useResumeFontFamily = () => useContext(ResumeFontFamilyContext)\n\n/** Page size for the document — a change re-measures/re-paginates. */\nexport const ResumePageSizeContext = createContext<PageSize>('letter')\nexport const useResumePageSize = () => useContext(ResumePageSizeContext)\n","import type React from 'react'\n\n/**\n * Vendored, dependency-free base prose component map.\n *\n * This is a minimal fallback map that the resume spreads over\n * (`...baseMdxComponents`) before defining its own `h1/h3/h4/p/ul/li/strong/em/a/hr`.\n * It deliberately has NO dependency on `@atom63/ui`, lightbox, article blocks,\n * example containers, or any other `@atom63/*` package — every entry is a plain\n * HTML-element component so the resume package stays self-contained.\n *\n * It only needs to provide sane defaults for elements the resume does NOT override.\n */\nexport const baseMdxComponents = {\n h2: (props: React.HTMLAttributes<HTMLHeadingElement>) => <h2 {...props} />,\n h5: (props: React.HTMLAttributes<HTMLHeadingElement>) => <h5 {...props} />,\n h6: (props: React.HTMLAttributes<HTMLHeadingElement>) => <h6 {...props} />,\n ol: (props: React.HTMLAttributes<HTMLOListElement>) => <ol {...props} />,\n blockquote: (props: React.HTMLAttributes<HTMLElement>) => <blockquote {...props} />,\n code: (props: React.HTMLAttributes<HTMLElement>) => <code {...props} />,\n pre: (props: React.HTMLAttributes<HTMLPreElement>) => <pre {...props} />,\n img: (props: React.ImgHTMLAttributes<HTMLImageElement>) => (\n // biome-ignore lint/a11y/useAltText: alt is forwarded via spread props\n <img {...props} />\n ),\n} as const\n","const MDX_SPECIAL_PROTOCOL = /^(mailto|tel|javascript):/i\n\n/**\n * True when `href` should open in a new tab: off-origin http(s), protocol-relative,\n * or absolute http(s) during SSR. Same-origin URLs, in-app paths (`/…`, `#…`),\n * relative paths, and `mailto:` / `tel:` / `javascript:` stay in-tab / default behavior.\n */\nexport function isExternalMdxHref(href: string | undefined): boolean {\n if (!href) {\n return false\n }\n const t = href.trim()\n if (t.startsWith('#') || t.startsWith('/')) {\n return false\n }\n if (MDX_SPECIAL_PROTOCOL.test(t)) {\n return false\n }\n if (t.startsWith('//')) {\n return true\n }\n if (typeof globalThis.window !== 'undefined') {\n try {\n const u = new URL(t, globalThis.window.location.href)\n if (u.protocol !== 'http:' && u.protocol !== 'https:') {\n return false\n }\n return u.origin !== globalThis.window.location.origin\n } catch {\n return false\n }\n }\n try {\n const u = new URL(t)\n return u.protocol === 'http:' || u.protocol === 'https:'\n } catch {\n return false\n }\n}\n","import clsx from 'clsx'\nimport { baseMdxComponents } from '../mdx/base-components'\nimport { isExternalMdxHref } from '../mdx/external-href'\nimport { PaginatedResume } from './paginated-resume'\n\n// Styling for these primitives lives in resume-document.css as a small token\n// system (--doc-*) + semantic classes (.doc-*). Components only compose those —\n// no hardcoded sizes/colors/grid widths — so the resume and CV share one source\n// of truth. See paginated-resume.tsx for where the .css is imported and where\n// the [data-resume-document] token scope is applied.\n\nexport function Header({\n children,\n meta = false,\n}: {\n children: React.ReactNode\n /**\n * Use the universal meta grid (left label column + content), the same track\n * widths as <Section> and <Entry>, so the header aligns with everything\n * below it. The default is the resume's sidebar|main split.\n */\n meta?: boolean\n}) {\n return (\n <div\n className={clsx('doc-header', meta ? 'doc-meta-grid' : 'doc-resume-grid')}\n data-resume-header\n >\n {children}\n </div>\n )\n}\n\nfunction HeaderLeft({ children }: { children: React.ReactNode }) {\n return <div className=\"doc-header-col doc-header-left\">{children}</div>\n}\n\nfunction HeaderRight({ children }: { children: React.ReactNode }) {\n return <div className=\"doc-header-col doc-header-right\">{children}</div>\n}\n\nfunction Links({ children }: { children: React.ReactNode }) {\n return <div className=\"doc-group doc-links\">{children}</div>\n}\n\nexport function Group({ children }: { children: React.ReactNode }) {\n return <div className=\"doc-group\">{children}</div>\n}\n\n/**\n * A record row (award, media mention, talk, …): a `year | role` meta column on\n * the left and the linked title on the right. Each Entry is a single pagination\n * block, so render long CV lists as a flat sequence of Entry elements (NOT one\n * <ul>) so pages can break between them.\n */\nexport function Entry({\n year,\n role,\n children,\n}: {\n year?: string\n role?: string\n children: React.ReactNode\n}) {\n return (\n <div className=\"doc-meta-grid\">\n <span className=\"doc-meta\">{role ? `${year} | ${role}` : year}</span>\n <div className=\"doc-body doc-entry-body\">{children}</div>\n </div>\n )\n}\n\n/** A thin full-width divider — the CV uses one under the header band. */\nexport function Rule() {\n return <div className=\"doc-rule\" />\n}\n\n/**\n * Editorial section: the label sits in the left meta column (aligned with the\n * `year | role` column of Entry rows) and the content fills the right column,\n * so prose sections share one left edge with the record rows below. The\n * .doc-section margin keeps every section on the same vertical rhythm as the\n * record-section headings (h3). One pagination block — keep a body under a page.\n */\nexport function Section({ label, children }: { label: string; children: React.ReactNode }) {\n return (\n <div className=\"doc-section doc-meta-grid\">\n <h3 className=\"doc-heading\">{label}</h3>\n <div className=\"doc-col\">{children}</div>\n </div>\n )\n}\n\nexport function Footer({ children }: { children: React.ReactNode }) {\n return (\n <div className=\"doc-footer\" data-footer>\n {children}\n </div>\n )\n}\n\nexport function Columns({ children }: { children: React.ReactNode }) {\n return <div className=\"doc-resume-grid\">{children}</div>\n}\n\nexport function Sidebar({ children }: { children: React.ReactNode }) {\n return (\n <div className=\"doc-col doc-col-sidebar\" data-resume-col=\"sidebar\">\n {children}\n </div>\n )\n}\n\nexport function Main({ children }: { children: React.ReactNode }) {\n return (\n <div className=\"doc-col doc-col-main\" data-resume-col=\"main\">\n {children}\n </div>\n )\n}\n\nexport const resumeMdxComponents = {\n ...baseMdxComponents,\n\n ResumeDocument: PaginatedResume,\n\n Header,\n HeaderLeft,\n HeaderRight,\n Links,\n Group,\n Entry,\n Rule,\n Section,\n Footer,\n Columns,\n Sidebar,\n Main,\n\n h1: ({ children, ...props }: React.HTMLAttributes<HTMLHeadingElement>) => (\n <h1 className=\"doc-title\" {...props}>\n {children}\n </h1>\n ),\n\n h3: ({ children, ...props }: React.HTMLAttributes<HTMLHeadingElement>) => (\n <h3 className=\"doc-section doc-heading\" data-resume-keep-with-next {...props}>\n {children}\n </h3>\n ),\n\n h4: ({ children, ...props }: React.HTMLAttributes<HTMLHeadingElement>) => (\n <h4 className=\"doc-subheading\" {...props}>\n {children}\n </h4>\n ),\n\n p: ({ children, ...props }: React.HTMLAttributes<HTMLParagraphElement>) => (\n <p className=\"doc-body\" {...props}>\n {children}\n </p>\n ),\n\n ul: ({ children, ...props }: React.HTMLAttributes<HTMLUListElement>) => (\n <ul className=\"doc-list\" {...props}>\n {children}\n </ul>\n ),\n\n li: ({ children, ...props }: React.HTMLAttributes<HTMLLIElement>) => (\n <li className=\"doc-body\" {...props}>\n {children}\n </li>\n ),\n\n strong: ({ children, ...props }: React.HTMLAttributes<HTMLElement>) => (\n <strong className=\"doc-strong\" {...props}>\n {children}\n </strong>\n ),\n\n em: ({ children, ...props }: React.HTMLAttributes<HTMLElement>) => (\n <em className=\"doc-meta doc-em\" {...props}>\n {children}\n </em>\n ),\n\n a: ({ children, href, target, rel, ...props }: React.AnchorHTMLAttributes<HTMLAnchorElement>) => {\n const external = isExternalMdxHref(href)\n return (\n <a\n className=\"doc-link\"\n href={href}\n {...props}\n rel={external ? (rel ?? 'noopener noreferrer') : rel}\n target={external ? (target ?? '_blank') : target}\n >\n {children}\n </a>\n )\n },\n\n hr: (_props: React.HTMLAttributes<HTMLHRElement>) => <div className=\"doc-hr\" />,\n}\n","import {\n Children,\n Fragment,\n isValidElement,\n type ReactNode,\n useEffect,\n useLayoutEffect,\n useRef,\n useState,\n} from 'react'\nimport { BLOCK_GAP_PX, getPageGeometry } from '../geometry'\nimport { type Block, type ColumnKey, packIntoPages } from '../pagination/pagination'\nimport { Columns, Footer, Header, Main, Sidebar } from './components'\nimport { usePaginationReport, useResumeFontFamily, useResumePageSize } from './pagination-context'\n\n// Styles ship via the `@atom63/resume/styles` entry (import it once in your app)\n// — NOT self-imported here, so a consumer's token overrides always win the\n// cascade instead of racing a re-injected copy of the package CSS.\n\ntype Columned = Record<ColumnKey, ReactNode[]>\n\n/**\n * Flatten one level, descending through Fragments. MDX wraps top-level content\n * (and component children) in Fragments, so `Children.toArray` alone yields the\n * Fragment, not the Header/Columns/Sidebar elements inside it.\n */\nfunction flattenChildren(children: ReactNode): ReactNode[] {\n const out: ReactNode[] = []\n for (const child of Children.toArray(children)) {\n if (isValidElement(child) && child.type === Fragment) {\n out.push(...flattenChildren((child.props as { children: ReactNode }).children))\n } else {\n out.push(child)\n }\n }\n return out\n}\n\n/** Pull Header / Columns(Sidebar,Main) / Footer out of the MDX wrapper children. */\nfunction extract(children: ReactNode): {\n header: ReactNode | null\n footer: ReactNode | null\n blocks: Columned\n} {\n let header: ReactNode | null = null\n let footer: ReactNode | null = null\n const blocks: Columned = { sidebar: [], main: [] }\n\n for (const child of flattenChildren(children)) {\n if (!isValidElement(child)) continue\n if (child.type === Header) header = child\n else if (child.type === Footer) footer = child\n else if (child.type === Columns) {\n for (const col of flattenChildren((child.props as { children: ReactNode }).children)) {\n if (!isValidElement(col)) continue\n const key: ColumnKey | null =\n col.type === Sidebar ? 'sidebar' : col.type === Main ? 'main' : null\n if (key) {\n blocks[key] = flattenChildren((col.props as { children: ReactNode }).children)\n }\n }\n }\n }\n return { header, footer, blocks }\n}\n\n/** Read a block element's full vertical extent (border box + margins). */\nfunction blockHeight(el: HTMLElement): number {\n const cs = getComputedStyle(el)\n const mt = Number.parseFloat(cs.marginTop) || 0\n const mb = Number.parseFloat(cs.marginBottom) || 0\n return el.getBoundingClientRect().height + mt + mb\n}\n\n/**\n * Mounted as the MDX `wrapper`, so it receives [Header, Columns, Footer].\n * Measures each column's blocks in a hidden layer, packs them into pages, and\n * renders physical page frames. Reports pageCount up to ResumeApp.\n */\nexport function PaginatedResume({ children }: { children: ReactNode }) {\n const { header, footer, blocks } = extract(children)\n const fontFamily = useResumeFontFamily()\n const pageSize = useResumePageSize()\n const geo = getPageGeometry(pageSize)\n const report = usePaginationReport()\n\n const measureRef = useRef<HTMLDivElement>(null)\n const [heights, setHeights] = useState<Record<ColumnKey, number[]> | null>(null)\n const [headerH, setHeaderH] = useState(0)\n\n // contentKey changes whenever the rendered content could change height.\n // Print-fixed (px) sizes: only the inherited font *family* and the block\n // counts can change measured heights — the OS typography scale is ignored.\n const contentKey = `${fontFamily}:${pageSize}:${blocks.sidebar.length}:${blocks.main.length}`\n\n // contentKey is the intentional re-measure trigger: a change in font family /\n // block counts must force a fresh measure even though it isn't read in the body.\n // biome-ignore lint/correctness/useExhaustiveDependencies: deliberate re-run trigger\n useLayoutEffect(() => {\n const root = measureRef.current\n if (!root) return\n const measure = () => {\n const cols: Record<ColumnKey, number[]> = { sidebar: [], main: [] }\n for (const key of ['sidebar', 'main'] as ColumnKey[]) {\n const colEl = root.querySelector(`[data-resume-col=\"${key}\"]`)\n if (colEl) {\n cols[key] = Array.from(colEl.children).map(c => blockHeight(c as HTMLElement))\n }\n }\n const headerEl = root.querySelector('[data-resume-header]') as HTMLElement | null\n setHeaderH(headerEl ? blockHeight(headerEl) : 0)\n setHeights(cols)\n }\n measure()\n const ro = new ResizeObserver(measure)\n ro.observe(root)\n\n // Re-measure on window (os63 window) resize — e.g. maximize/restore. The\n // measure root is fixed-width, so its own observer never catches this;\n // watch the enclosing scroll viewport, which tracks the window size. A\n // double rAF lets layout settle so a mid-transition read doesn't stick\n // (which collapsed multi-page docs to a single clipped page).\n const viewport = root.closest('[data-slot=\"scroll-area-viewport\"]')\n // Debounce to a TRAILING measure: the os63 window's maximize/restore is a\n // spring animation, so reading mid-transition lands on a wrong height that\n // then sticks. Wait for the resize to stop, then measure the settled layout.\n let settleTimer: ReturnType<typeof setTimeout> | undefined\n const scheduleMeasure = () => {\n clearTimeout(settleTimer)\n settleTimer = setTimeout(measure, 250)\n }\n const vpObserver = viewport ? new ResizeObserver(scheduleMeasure) : null\n vpObserver?.observe(viewport as Element)\n\n if (typeof document !== 'undefined' && 'fonts' in document) {\n document.fonts.ready.then(measure).catch(() => {})\n }\n return () => {\n ro.disconnect()\n vpObserver?.disconnect()\n clearTimeout(settleTimer)\n }\n }, [contentKey])\n\n // The hidden measuring layer renders the real structure at content width.\n // A 0x0 overflow-hidden clip wrapper keeps it out of the scroll area (a large\n // negative offset would inflate the viewport and spawn a phantom horizontal\n // scrollbar, drifting the page off-center). The INNER element keeps its\n // natural size so block heights stay measurable and its ResizeObserver still\n // fires on content changes.\n const measureLayer = (\n <div\n aria-hidden\n style={{\n position: 'absolute',\n top: 0,\n left: 0,\n width: 0,\n height: 0,\n overflow: 'hidden',\n pointerEvents: 'none',\n }}\n >\n <div\n data-resume-measure\n ref={measureRef}\n style={{ width: geo.contentWidthPx, visibility: 'hidden' }}\n >\n {header}\n {blocks.sidebar.length === 0 ? (\n // Single-column docs (e.g. the CV) put everything in Main and render\n // full-width. Measure at full width too so block heights match the\n // full-width page render below (otherwise they'd be measured at the\n // narrower 2/3 main-column width and mis-paginate).\n <div className=\"doc-col\" data-resume-col=\"main\">\n {blocks.main}\n </div>\n ) : (\n <Columns>\n <Sidebar>{blocks.sidebar}</Sidebar>\n <Main>{blocks.main}</Main>\n </Columns>\n )}\n </div>\n </div>\n )\n\n if (!heights) {\n return <div data-resume-document>{measureLayer}</div>\n }\n\n const toBlocks = (key: ColumnKey): Block[] =>\n blocks[key].map((node, i) => ({\n id: `${key}-${i}`,\n height: heights[key][i] ?? 0,\n keepWithNext:\n isValidElement(node) &&\n Boolean((node.props as Record<string, unknown>)['data-resume-keep-with-next']),\n }))\n\n const sidebarBlocks = toBlocks('sidebar')\n const mainBlocks = toBlocks('main')\n\n const pack = packIntoPages({\n columns: { sidebar: sidebarBlocks, main: mainBlocks },\n pageUsableHeight: p =>\n p === 0 ? Math.max(0, geo.usableHeightPx - headerH) : geo.usableHeightPx,\n blockGap: BLOCK_GAP_PX,\n })\n\n return (\n <div data-resume-document>\n {measureLayer}\n <div className=\"doc-pages\" style={{ gap: geo.gapPx }}>\n {Array.from({ length: pack.pageCount }, (_, p) => {\n const sideOnPage = blocks.sidebar.filter(\n (_, i) => pack.pageOf.sidebar[`sidebar-${i}`] === p\n )\n const mainOnPage = blocks.main.filter((_, i) => pack.pageOf.main[`main-${i}`] === p)\n // Key the layout on the WHOLE document, not the current page: a\n // single-column doc (the CV — no sidebar at all) renders full\n // width, but a two-column doc keeps its grid on every page,\n // including continuation pages where the sidebar has run out, so\n // the main column stays in its right-hand track instead of\n // jumping to full width (which broke the 2-col layout on page 2+).\n const isTwoColumn = blocks.sidebar.length > 0\n return (\n <div\n className=\"doc-page\"\n data-resume-page\n key={`page-${String(p)}`}\n style={{\n width: geo.widthPx,\n height: geo.heightPx,\n overflow: 'hidden',\n padding: `${geo.padYPx}px ${geo.padXPx}px`,\n }}\n >\n {p === 0 && header}\n {isTwoColumn ? (\n <div className=\"doc-resume-grid\">\n <div className=\"doc-col doc-col-sidebar\">{sideOnPage}</div>\n <div className=\"doc-col doc-col-main\">{mainOnPage}</div>\n </div>\n ) : (\n <div className=\"doc-col\">{mainOnPage}</div>\n )}\n {p === pack.pageCount - 1 && footer}\n </div>\n )\n })}\n </div>\n <PageCountReporter count={pack.pageCount} report={report} />\n </div>\n )\n}\n\n/** Reports pageCount via an effect (no setState-during-render). */\nfunction PageCountReporter({ count, report }: { count: number; report: (n: number) => void }) {\n useEffect(() => {\n report(count)\n }, [count, report])\n return null\n}\n"]}