| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798 |
- #!/usr/bin/env node
- /**
- * export_deck_pdf.mjs — 把多文件 slide deck 导出为单个矢量 PDF
- *
- * 用法:
- * node export_deck_pdf.mjs --slides <dir> --out <file.pdf> [--width 1920] [--height 1080]
- *
- * 特点:
- * - 文字保留矢量(可复制、可搜索)
- * - 背景/图形 1:1 保真(Playwright 内嵌 Chromium 渲染)
- * - 不需要对 HTML 做任何改造
- * - 视觉损失 = 0(PDF 就是浏览器打印出来的)
- *
- * trade-off:
- * - PDF 不可再编辑文字(要改回到 HTML 改)
- *
- * 依赖:playwright pdf-lib
- * npm install playwright pdf-lib
- *
- * 会按文件名排序(01-xxx.html → 02-xxx.html → ...)
- */
- import { chromium } from 'playwright';
- import { PDFDocument } from 'pdf-lib';
- import fs from 'fs/promises';
- import path from 'path';
- function parseArgs() {
- const args = { width: 1920, height: 1080 };
- const a = process.argv.slice(2);
- for (let i = 0; i < a.length; i += 2) {
- const k = a[i].replace(/^--/, '');
- args[k] = a[i + 1];
- }
- if (!args.slides || !args.out) {
- console.error('用法: node export_deck_pdf.mjs --slides <dir> --out <file.pdf> [--width 1920] [--height 1080]');
- process.exit(1);
- }
- args.width = parseInt(args.width);
- args.height = parseInt(args.height);
- return args;
- }
- async function main() {
- const { slides, out, width, height } = parseArgs();
- const slidesDir = path.resolve(slides);
- const outFile = path.resolve(out);
- const files = (await fs.readdir(slidesDir))
- .filter(f => f.endsWith('.html'))
- .sort();
- if (!files.length) {
- console.error(`No .html files found in ${slidesDir}`);
- process.exit(1);
- }
- console.log(`Found ${files.length} slides in ${slidesDir}`);
- const browser = await chromium.launch();
- const ctx = await browser.newContext({ viewport: { width, height } });
- // 1) Render each HTML to its own PDF buffer
- const pageBuffers = [];
- for (const f of files) {
- const page = await ctx.newPage();
- const url = 'file://' + path.join(slidesDir, f);
- await page.goto(url, { waitUntil: 'networkidle' }).catch(() => page.goto(url));
- await page.waitForTimeout(1200); // web-font paint
- // emulate "screen" so CSS colors/backgrounds render the same as browser
- await page.emulateMedia({ media: 'screen' });
- const buf = await page.pdf({
- width: `${width}px`,
- height: `${height}px`,
- printBackground: true,
- margin: { top: 0, right: 0, bottom: 0, left: 0 },
- preferCSSPageSize: false,
- });
- pageBuffers.push(buf);
- await page.close();
- console.log(` [${pageBuffers.length}/${files.length}] ${f}`);
- }
- await browser.close();
- // 2) Merge into a single PDF
- const merged = await PDFDocument.create();
- for (const buf of pageBuffers) {
- const src = await PDFDocument.load(buf);
- const copied = await merged.copyPages(src, src.getPageIndices());
- copied.forEach(p => merged.addPage(p));
- }
- const bytes = await merged.save();
- await fs.writeFile(outFile, bytes);
- const kb = (bytes.byteLength / 1024).toFixed(0);
- console.log(`\n✓ Wrote ${outFile} (${kb} KB, ${files.length} pages, vector)`);
- }
- main().catch(e => { console.error(e); process.exit(1); });
|