.github/skills/excalidraw-mcp-diagramming/SKILL.md
Create and edit diagrams on a live Excalidraw canvas using the Excalidraw MCP server. Use when asked to draw, diagram, sketch, or visualise architectures, workflows, data flows, system designs, flowcharts, mind maps, or sequence diagrams. Trigger phrases include "create an excalidraw", "draw me a diagram", "make a flowchart", "visualise the system", "diagram this architecture", "export to PNG/SVG". Can export to PNG, SVG, .excalidraw file, or a shareable URL. Do NOT use for Draw.io or diagrams.net output (use drawio-mcp-diagramming instead).
npx skillsauth add thomast1906/github-copilot-agent-skills excalidraw-mcp-diagrammingInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Create diagrams on a live Excalidraw canvas that renders in the browser and updates in real time. You are not generating a static file — you are painting onto a shared whiteboard through MCP tools. The canvas persists between calls, so what you put on it in one call is visible in the next screenshot.
drawio-mcp-diagramming for that)..excalidraw file, or a shareable URL.The excalidraw MCP server must be present in .vscode/mcp.json:
{
"servers": {
"excalidraw": {
"type": "http",
"url": "https://mcp.excalidraw.com"
}
}
}
If the MCP tools are not available, tell the user to add the server and reload VS Code, then stop.
Essential — use on every diagram:
| Tool | Purpose | When |
|---|---|---|
| read_diagram_guide | Returns server-side colour palette and sizing rules | First call — before any elements |
| batch_create_elements | Creates multiple shapes and arrows atomically | Main workhorse |
| get_canvas_screenshot | Returns a photo of the current canvas | After every change — verify before continuing |
| clear_canvas | Wipes all content | Start of every new diagram |
| set_viewport | Scrolls and zooms to fit content | After creating elements |
Secondary — use when needed:
| Tool | When |
|---|---|
| create_from_mermaid | Quick drafts — 3–8 node sequential flows |
| update_element | Small corrections (position, colour, text) |
| export_to_image | User requests PNG or SVG file |
| export_scene | User requests editable .excalidraw file |
| export_to_excalidraw_url | User wants a shareable link |
| describe_scene | Audit what is currently on the canvas |
Mermaid path — use for simple sequential flows (3–8 nodes, no zones):
create_from_mermaid(
mermaidDiagram="graph TD; A[Frontend] -->|REST| B[API]; B -->|SQL| C[DB]"
)
Then jump to Step 6 (screenshot). Skip Steps 2–5.
Batch path — use for everything else: layered architectures, data flows, hub-and-spoke, any diagram needing zones or colour-coded roles. Continue below.
read_diagram_guide()
Retrieve the server's current colour palette and sizing rules. This call is mandatory — the values may differ from the defaults in this skill. The server guide takes precedence.
clear_canvas()
get_canvas_screenshot() // must verify the canvas is truly empty
Previous diagrams leave ghost data even after clear_canvas. If any element
is visible in the screenshot, call clear_canvas() again before proceeding.
Do not skip this confirmation — ghost elements silently break arrow bindings
on new diagrams.
Before writing any JSON, decide:
y = row_y - 50, height = box_height + 100.Sketch coordinates to paper or comments before writing the batch payload.
Call batch_create_elements with all elements in one payload. Arrow
binding resolves at batch time — if the target shape and arrow are not in the
same call, the arrow will not connect.
Element order within the elements array matters for render layering:
id to eachstartElementId / endElementIdget_canvas_screenshot()
Inspect the image:
Load references/iterative-refinement.md if the user asks to change, fix, or update an existing diagram. Quick reference:
Audit first:
describe_scene() // returns all element ids, types, labels, positions
Use this before any edit — you need the exact element id to target.
Small corrections (label text, colour, position):
update_element({ id: "box-1", text: "New Label" })
update_element({ id: "box-1", backgroundColor: "#b2f2bb" })
Then get_canvas_screenshot() to verify.
Shape replacement (size, shape type, structural change):
batch_create_elements([{ "type": "delete", "ids": "old-id" }, ...newShapes, ...newArrows])
Arrows bound to the deleted shape must also be deleted and recreated in the same batch — binding cannot be re-attached after the fact.
Decision rule:
| Change | Method |
|---|---|
| Text / colour / opacity / position | update_element |
| Shape size, shape type | delete + redraw in one batch |
| Moving an arrow endpoint | delete + redraw arrow in one batch with target shape |
| Restructuring a zone (adding/removing shapes) | delete zone bg + all children + redraw all in one batch |
set_viewport({ scrollToContent: true })
// PNG or SVG
export_to_image({ format: "png", filePath: "/path/to/output.png" })
// Editable JSON file
export_scene({ filePath: "/path/to/output.excalidraw" })
// Shareable link — no file needed
export_to_excalidraw_url()
Minimum font sizes — never go smaller. Labels that look correct in JSON are frequently unreadable in screenshots at display scale.
| Context | Minimum fontSize | Notes |
|---|---|---|
| Shape labels / body text | 16 | Default for all labeled boxes |
| Diagram title | 24 | Standalone text above the diagram |
| Zone / section heading | 16 | Inside or above zone background |
| Secondary annotations | 14 | Data form notes, layer labels only — use sparingly |
| Absolute minimum | 14 | Never use below 14 under any circumstance |
Camera scale warning: At XXL (1600×1200) the canvas renders at roughly
40% of original size in the chat panel. A fontSize: 14 label renders at
~5px — invisible. Use fontSize: 20+ for any label that must be readable
without the user zooming in. When in doubt, go larger.
{
"type": "rectangle",
"id": "api-server",
"x": 440, "y": 200,
"width": 230, "height": 160,
"backgroundColor": "#d0bfff",
"strokeColor": "#7048e8",
"roughness": 0,
"text": "API Server\nExpress.js"
}
roughness: 0 — crisp professional edges. roughness: 1 — hand-drawn feel.\n in text creates multi-line labels.rectangle, ellipse, diamond, text (standalone).strokeStyle: "dashed" and
opacity: 30.{
"type": "arrow",
"x": 0, "y": 0,
"startElementId": "api-server",
"endElementId": "database",
"strokeColor": "#2f9e44",
"text": "SQL"
}
x, y are approximate — binding to startElementId/endElementId
overrides position. The server auto-routes to the nearest edges.startArrowhead / endArrowhead: "arrow", "dot", "bar", or null.strokeStyle: "dashed" — async or optional flows.strokeStyle: "dotted" — weak dependency.Read references/color-palette.md for the full semantic colour table — load it when choosing colours for a new diagram type.
| Role | Fill | Stroke |
|---|---|---|
| Frontend / UI | #a5d8ff | #1971c2 |
| Backend / API | #d0bfff | #7048e8 |
| Database | #b2f2bb | #2f9e44 |
| Queue / Events | #fff3bf | #fab005 |
| External / Third-party | #ffc9c9 | #e03131 |
| Zone background | #e9ecef + opacity:30 | #868e96 dashed |
Rule: Same architectural role → same colour pair. Limit diagrams to 3–4 fills. More colours add noise, not clarity.
Run through this before calling set_viewport. Fix any failures before
finishing — they compound and are harder to fix after zooming out.
Arrow bindings
startElementId and endElementId referencing
elements that exist in the same batchTypography
fontSize ≥ 16fontSize ≥ 2414 anywhere\n) have enough box height (≥ 60px per line)Colour discipline
opacity: 25–40 and strokeStyle: "dashed"Layout and spacing
Viewport
set_viewport({ scrollToContent: true }) called as the final step| Anti-pattern | Why it causes problems |
|---|---|
| Skip screenshot after clear_canvas | Ghost elements silently corrupt new arrow bindings |
| Place arrows before their target shapes in a batch | Arrow IDs reference non-existent shapes — no binding |
| Column gap under 100px | Arrow labels bleed into adjacent shapes |
| Arrows without startElementId/endElementId | Arrow floats, doesn't update when shapes move |
| Skip set_viewport after creation | Diagram is off-screen and user sees a blank canvas |
| Zone backgrounds in a separate later batch | Zones render on top of shapes already on the canvas |
| More than 4 fill colours | Diagram becomes unreadable without a legend |
| fontSize below 16 on shape labels | Unreadable at XL/XXL camera scale |
| Editing a bound arrow with update_element | Binding is preserved but endpoints may drift — delete and redraw |
"Draw me a 3-tier web architecture"
read_diagram_guide() — get coloursclear_canvas() → get_canvas_screenshot() — confirm emptybatch_create_elements([zones..., shapes..., arrows..., titles...])get_canvas_screenshot() — verifyset_viewport({ scrollToContent: true })"Quick flowchart of user login"
create_from_mermaid("graph TD; A[Enter email] --> B[Verify] --> C[Dashboard]")get_canvas_screenshot() — verifydevelopment
Assess Azure architectures against Well-Architected Framework (WAF) five pillars - Reliability, Security, Cost Optimization, Operational Excellence, and Performance Efficiency. Provide scores and recommendations.
devops
Safe Terraform provider upgrades with automatic resource migration, breaking change detection, and state management using moved blocks. Use when upgrading provider versions, handling removed resources, migrating deprecated syntax, or performing major version upgrades.
development
Comprehensive skills for creating, compiling, debugging, and managing GitHub Agentic Workflows (gh-aw) with best practices and common patterns
tools
Create and edit architecture diagrams using Draw.io MCP (`drawio/create_diagram`) with reliable Azure and AWS icon rendering guidance and troubleshooting. Supports Azure2 and AWS4 icon libraries. Requires Python 3 and internet access to refresh icon catalogs (periodic, not per-run).