JimLiu-baoyu-skills/skills/baoyu-slide-deck/scripts/merge-to-pptx.ts

138 lines
3.8 KiB
TypeScript

import { existsSync, readdirSync, readFileSync } from "fs";
import { join, basename, extname } from "path";
import PptxGenJS from "pptxgenjs";
interface SlideInfo {
filename: string;
path: string;
index: number;
promptPath?: string;
}
function parseArgs(): { dir: string; output?: string } {
const args = process.argv.slice(2);
let dir = "";
let output: string | undefined;
for (let i = 0; i < args.length; i++) {
if (args[i] === "--output" || args[i] === "-o") {
output = args[++i];
} else if (!args[i].startsWith("-")) {
dir = args[i];
}
}
if (!dir) {
console.error("Usage: bun merge-to-pptx.ts <slide-deck-dir> [--output filename.pptx]");
process.exit(1);
}
return { dir, output };
}
function findSlideImages(dir: string): SlideInfo[] {
if (!existsSync(dir)) {
console.error(`Directory not found: ${dir}`);
process.exit(1);
}
const files = readdirSync(dir);
const slidePattern = /^(\d+)-slide-.*\.(png|jpg|jpeg)$/i;
const promptsDir = join(dir, "prompts");
const hasPrompts = existsSync(promptsDir);
const slides: SlideInfo[] = files
.filter((f) => slidePattern.test(f))
.map((f) => {
const match = f.match(slidePattern);
const baseName = f.replace(/\.(png|jpg|jpeg)$/i, "");
const promptPath = hasPrompts ? join(promptsDir, `${baseName}.md`) : undefined;
return {
filename: f,
path: join(dir, f),
index: parseInt(match![1], 10),
promptPath: promptPath && existsSync(promptPath) ? promptPath : undefined,
};
})
.sort((a, b) => a.index - b.index);
if (slides.length === 0) {
console.error(`No slide images found in: ${dir}`);
console.error("Expected format: 01-slide-*.png, 02-slide-*.png, etc.");
process.exit(1);
}
return slides;
}
function findBasePrompt(): string | undefined {
const scriptDir = import.meta.dir;
const basePromptPath = join(scriptDir, "..", "references", "base-prompt.md");
if (existsSync(basePromptPath)) {
return readFileSync(basePromptPath, "utf-8");
}
return undefined;
}
async function createPptx(slides: SlideInfo[], outputPath: string) {
const pptx = new PptxGenJS();
pptx.layout = "LAYOUT_16x9";
pptx.author = "baoyu-slide-deck";
pptx.subject = "Generated Slide Deck";
const basePrompt = findBasePrompt();
let notesCount = 0;
for (const slide of slides) {
const s = pptx.addSlide();
const imageData = readFileSync(slide.path);
const base64 = imageData.toString("base64");
const ext = extname(slide.filename).toLowerCase().replace(".", "");
const mimeType = ext === "png" ? "image/png" : "image/jpeg";
s.addImage({
data: `data:${mimeType};base64,${base64}`,
x: 0,
y: 0,
w: "100%",
h: "100%",
sizing: { type: "cover", w: "100%", h: "100%" },
});
if (slide.promptPath) {
const slidePrompt = readFileSync(slide.promptPath, "utf-8");
const fullNotes = basePrompt ? `${basePrompt}\n\n---\n\n${slidePrompt}` : slidePrompt;
s.addNotes(fullNotes);
notesCount++;
}
console.log(`Added: ${slide.filename}${slide.promptPath ? " (with notes)" : ""}`);
}
await pptx.writeFile({ fileName: outputPath });
console.log(`\nCreated: ${outputPath}`);
console.log(`Total slides: ${slides.length}`);
if (notesCount > 0) {
console.log(`Slides with notes: ${notesCount}${basePrompt ? " (includes base prompt)" : ""}`);
}
}
async function main() {
const { dir, output } = parseArgs();
const slides = findSlideImages(dir);
const dirName = basename(dir) === "slide-deck" ? basename(join(dir, "..")) : basename(dir);
const outputPath = output || join(dir, `${dirName}.pptx`);
console.log(`Found ${slides.length} slides in: ${dir}\n`);
await createPptx(slides, outputPath);
}
main().catch((err) => {
console.error("Error:", err.message);
process.exit(1);
});