deck-contract
Click on "Install Server".
Wait a few minutes for the server to deploy. Once ready, it will show a "Started" state.
In the chat, type
@followed by the MCP server name and your instructions, e.g., "@deck-contractedit the title of slide 3 to say 'Q3 Review'"
That's it! The server will respond to your query, and you can continue using it as needed.
Here is a step-by-step guide with screenshots.
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 PDFMCP, Playwright and git are swappable adapters at the edge; the core knows nothing about transport.
Related MCP server: Slideforge
Packages
package | role |
| Zod schemas, contract version, |
| immutable tree, stable id allocator, transactional |
|
|
|
|
| render-and-check loop with monotone gate + anti-oscillation + bounded iters |
|
|
| 10 MCP tools over stdio (thin shell); handle-based state behind a |
| 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
Day 1 — contract, core (tree/ids/ops), pure compile, tests without a browser
Day 2 —
LocalChromiumRenderer: warm browser, network-blocked, fonts awaited, deterministic overflow / out-of-bounds measurement, PNG screenshotsDay 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)
Day 4 — 8 MCP tools over stdio, PDF export, template loader + first
minimal-darkpackage, handle-based session state; full model-flow integration test greenDay 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.
This server cannot be installed
Maintenance
Resources
Unclaimed servers have limited discoverability.
Looking for Admin?
If you are the server author, to access and configure the admin panel.
Latest Blog Posts
MCP directory API
We provide all the information about MCP servers via our MCP API.
curl -X GET 'https://glama.ai/api/mcp/v1/servers/Andersonscat/deck-contract'
If you have feedback or need assistance with the MCP directory API, please join our Discord server