deck-contract
MCP server for AI-native editable presentations: surgically edit slide components by stable ID, with local rendering and self-checking overflow.
README
deck-contract
An open standard + reference MCP server for AI-native editable artifacts. An AI edits one component of a deck surgically (by stable id) instead of regenerating the whole slide. No font drift (style is theme-token-only), no overflow (constraints + a render-and-check loop). First domain: presentations. Designed to extend to docs / resume / landing / LaTeX through the same contract.
Inference runs on the user's own AI subscription via MCP (the tool pays $0 for tokens); rendering is local (Chromium, $0 COGS).
Architecture (functional core / imperative shell)
template package (data)
-> deck.json immutable tree, stable string ids
-> apply(deck, ops[]) pure, transactional (all-or-nothing), inverse for free
-> compileHtml(deck) PURE, zero I/O -> deterministic HTML/CSS
-> render (Day 2) Playwright -> PNG + machine-readable issues[]
-> loop: model -> apply_patch -> render -> until issues = []
-> export PDF
MCP, Playwright and git are swappable adapters at the edge; the core knows nothing about transport.
Packages
| package | role |
|---|---|
@deck/contract |
Zod schemas, contract version, issues[] enum. Zero deps. |
@deck/core |
immutable tree, stable id allocator, transactional apply+inverse, token validation |
@deck/compile |
compileHtml(deck) — pure, deterministic deck → HTML/CSS |
@deck/renderer |
LocalChromiumRenderer — headless Chromium → PNG + issues[] (overflow/out-of-bounds) |
@deck/loop |
render-and-check loop with monotone gate + anti-oscillation + bounded iters |
@deck/loader |
FileTemplateSource — resolve self-contained template packages from a folder/git |
@deck/mcp-server |
10 MCP tools over stdio (thin shell); handle-based state behind a DeckStore |
@deck/viewer |
local interactive viewer — click to select, edit text, mark an "AI target"; live reload |
Dependency rule: everything depends on contract; contract depends on nothing. The
domain ("slide", "title") must never leak into core/contract.
Load-bearing invariants (expensive to change later)
- Stable string ids are identity — ops address nodes by id, never by position.
- Token-only style — node styles are token refs only; raw hex/px are rejected at the schema boundary, so font drift has nothing to drift to.
compileHtmlis pure — no I/O, time, or randomness; same deck → byte-identical HTML.- Versioned format —
spec: "deck-contract/1", tolerant reader, open nodetype.
Develop
pnpm install
pnpm --filter @deck/renderer exec playwright install chromium # one-time
pnpm test # 38 tests: id stability, atomicity, inverse, token-only, determinism, render, convergence
pnpm preview # compile the sample deck -> examples/out/preview.html
pnpm render # surgical edit -> headless Chromium -> examples/out/slide-*.png
pnpm loop # run the fix loop on an overflowing deck -> monotone descent + before/after PNGs
pnpm agent # drive the MCP tools as a model would -> examples/out/deck-*.png + deck.pdf
pnpm viewer # open the interactive viewer on a working deck (http://localhost:4570)
Two ways to edit — you and the AI, one deck.json
The point is not full AI generation. The source of truth is one deck.json, and both
you and the model edit it through the same id-addressed ops:
- You, directly —
pnpm viewerrenders the live deck. Click any element to see its stable id; double-click a title/heading to edit text in place; edits write back and reload live. - You point, the AI acts — click an element and hit "Set as AI target". The model
(via
get_selection) reads exactly what you pointed at, thenapply_patchchanges it. Withopen_deckthe model edits the very file the viewer shows, so its changes appear live in front of you.
Use it in Claude Code / Cursor
A project-scoped .mcp.json is included. From a client that supports MCP, point it at
the stdio server:
{
"mcpServers": {
"deck-contract": { "command": "npx", "args": ["tsx", "packages/mcp-server/src/bin.ts"] }
}
}
Then the model has 10 tools: list_templates, load_template, open_deck,
get_selection, get_outline, get_node, apply_patch, insert_block, render,
export_deck. A typical turn: load_template (or open_deck to share the viewer's
file) → get_selection/get_outline → apply_patch (surgical edit) → render (see
the PNG + any overflow issues) → fix → export_deck (PDF). Inference runs on the
client's own model; this server only edits structure and renders locally.
Status
-
[x] Day 1 — contract, core (tree/ids/ops), pure compile, tests without a browser
-
[x] Day 2 —
LocalChromiumRenderer: warm browser, network-blocked, fonts awaited, deterministic overflow / out-of-bounds measurement, PNG screenshots -
[x] Day 3 — render-and-check loop (monotone gate, anti-oscillation, bounded iters); proven to converge on golden fixtures with a deterministic solver (406→192→48→0px)
-
[x] Day 4 — 8 MCP tools over stdio, PDF export, template loader + first
minimal-darkpackage, handle-based session state; full model-flow integration test green -
[x] Day 5 — interactive viewer (click-select, inline text edit, "AI target");
get_selection+open_deckso user and model edit one shared deck.json live
MVP complete: an AI can load a template, surgically edit one component, render and self-check for overflow, and export a PDF — and the user can see the deck, edit elements directly, or point the AI at exactly the element they want changed — all on the client's own model.
Recommended Servers
playwright-mcp
A Model Context Protocol server that enables LLMs to interact with web pages through structured accessibility snapshots without requiring vision models or screenshots.
Magic Component Platform (MCP)
An AI-powered tool that generates modern UI components from natural language descriptions, integrating with popular IDEs to streamline UI development workflow.
Audiense Insights MCP Server
Enables interaction with Audiense Insights accounts via the Model Context Protocol, facilitating the extraction and analysis of marketing insights and audience data including demographics, behavior, and influencer engagement.
VeyraX MCP
Single MCP tool to connect all your favorite tools: Gmail, Calendar and 40 more.
graphlit-mcp-server
The Model Context Protocol (MCP) Server enables integration between MCP clients and the Graphlit service. Ingest anything from Slack to Gmail to podcast feeds, in addition to web crawling, into a Graphlit project - and then retrieve relevant contents from the MCP client.
Kagi MCP Server
An MCP server that integrates Kagi search capabilities with Claude AI, enabling Claude to perform real-time web searches when answering questions that require up-to-date information.
E2B
Using MCP to run code via e2b.
Neon Database
MCP server for interacting with Neon Management API and databases
Exa Search
A Model Context Protocol (MCP) server lets AI assistants like Claude use the Exa AI Search API for web searches. This setup allows AI models to get real-time web information in a safe and controlled way.
Qdrant Server
This repository is an example of how to create a MCP server for Qdrant, a vector search engine.