Pure local PowerPoint conversion tooling for Open Presentation Format documents. This repo owns the Phase 3 and Phase 4 toolkit lanes: OPF to PPTX export and PPTX to OPF import.
- Package:
@openpresentation/opf-pptx - Repository:
OpenPresentation/opf-pptx - License: MIT
- Compatibility target:
@openpresentation/opf - Renderer relationship: may use
@openpresentation/opf-renderfor chart rasterization and visual verification - Public export API:
toPptx(opf, opts) - Public import API:
fromPptx(buffer, opts)
The export path validates OPF with @openpresentation/opf, maps slide titles and common content payloads to editable PowerPoint objects through pptxgenjs, then normalizes the generated ZIP for stable entry ordering, fixed timestamps, and reproducible bytes.
import { toPptx } from "@openpresentation/opf-pptx";
const bytes = await toPptx({
$schema: "https://openpresentation.org/schema/opf/v1",
name: "Quarterly Review",
slides: [
{
title: "Revenue grew across all regions",
items: ["North America +18%", "EMEA +14%", "APAC +11%"]
}
]
});
await fs.promises.writeFile("quarterly-review.pptx", bytes);toPptx returns a Uint8Array containing a PowerPoint-openable .pptx. It does not fetch remote assets. Data URI images and local paths can be embedded directly; hosts that need private asset loading should pass imageResolver(src, context). Set strictAssets: true to turn unresolved or remote image assets into structured OPFPptxError failures instead of editable placeholder boxes.
fromPptx parses an existing .pptx buffer locally and returns an OPF document that validates with @openpresentation/opf:
import { fromPptx, toPptx } from "@openpresentation/opf-pptx";
const opf = await fromPptx(await fs.promises.readFile("source.pptx"));
const roundTripBytes = await toPptx(opf);
await fs.promises.writeFile("round-trip.pptx", roundTripBytes);The importer reads core properties, slide order, text boxes, speaker notes, embedded images, tables, and basic cached chart data from the OOXML parts. Slides or objects that do not map cleanly fall back to editable blocks[] payloads; OOXML positions are used for deterministic ordering and title/subtitle detection while keeping the emitted OPF schema-valid.
The first exporter keeps the public API stable while using pptxgenjs internally:
Slide.title,Slide.subtitle, andSlide.tagbecome editable text boxes, not PowerPoint master placeholders.- Root payloads,
blocks[], and promoted region keys become editable slide objects in deterministic regions. Promoted keys use the OPF 3x3 region vocabulary (top,middle,bottom,left,center,right). - Text, lists, metrics, quotes, timelines, code, tables, and inline-data charts are emitted as editable PowerPoint text, table, and chart objects.
- Image assets are embedded only when supplied as data URIs, local paths, or host-resolved bytes/paths. Remote asset URLs are never fetched by the runtime path.
- ZIP entries, generated chart/workbook part names, core-property timestamps, and nested chart workbook timestamps are normalized for reproducible bytes.
This pass did not require an OPF schema change. The deferred full OOXML placeholder mapping from docs/plans/layout-placeholders.md remains a later hand-written OOXML emitter concern.
The first importer is mechanical and schema-compatible:
- Presentation core properties map to OPF
name,description, andauthor. - Slide text placeholders and large top-of-slide text boxes map to
titleandsubtitlewhen recognizable. - Remaining text boxes map to
blocks[]as text or list payloads, sorted by OOXML position. - PowerPoint tables map to OPF table blocks, embedded images map to data URI image blocks, and cached chart series map to basic OPF chart blocks.
- Unknown non-text shapes and unsupported graphic frames become editable text fallback blocks instead of failing the import.
There is no AI classification pass in the OSS runtime. Hosts can run optional cleanup or semantic remapping after fromPptx returns.
The package runtime must stay local and deterministic:
- No hosted service in the critical path
- No telemetry or hidden analytics
- No commercial SDK dependency in the critical path
- No required network calls
- No required AI dependency
- No required AI cleanup or classification pass for PPTX import
- No required LibreOffice dependency in the runtime path; LibreOffice is allowed only as an optional verification tool in CI
- Host applications own auth, storage, queues, analytics, collaboration, branding, and product workflow
npm ci
npm run build
npm run typecheck
npm test
npm run validateLibreOffice is not a runtime dependency. When it is installed in CI or a local verification environment, generated .pptx files can be smoke-opened there as an optional export check.
Public npm package publication is handled by .github/workflows/release.yml with npm provenance.
Required first-publish setup:
- An npm owner for the
@openpresentationscope must run the first publish or reserve/grant the@openpresentation/opf-pptxpackage. - Configure npm Trusted Publishing for GitHub repository
OpenPresentation/opf-pptxand workflow.github/workflows/release.yml. - Publish by creating a GitHub Release or manually running the Release workflow after CI passes.
This repo does not require an npm automation token when Trusted Publishing is configured.