Show Inline draw.io
show_inline_drawioCreate interactive diagrams from Mermaid or draw.io XML and display them inline for instant visualization.
Instructions
Creates and displays an interactive draw.io diagram. Accepts either draw.io XML or Mermaid.js syntax — provide exactly one.
Format decision — this is the first thing to settle before you write anything: if the diagram type appears on the Mermaid list below, use mermaid. Only use xml when the diagram type isn't on that list (UI mockups, floorplans, cloud/network/electrical architecture with stencils, hand-placed UML, etc.) or when the user has explicitly asked for draw.io XML.
Use Mermaid for the following diagram types (all rendered natively, no upstream mermaid runtime):
flowchart / graph (TD, LR, …)
sequenceDiagram
classDiagram
stateDiagram / stateDiagram-v2
erDiagram
gantt
pie
journey (user-journey)
gitGraph
mindmap
timeline
quadrantChart
xychart-beta
sankey-beta
requirementDiagram
C4Context / C4Container / C4Component
block-beta
architecture-beta
packet-beta
kanban
radar-beta
treemap-beta
treeview-beta (draw.io-specific)
venn (draw.io-specific) — syntax:
vennthenset A ["Label"]for each set,union A,Bfor declared overlaps (informational), andtext A/text A,Bfollowed by["Region label"]for text inside a region. Do NOT useA AND B[...]orA["..."]shorthand — those lines are ignored.ishikawa (draw.io-specific)
zenuml
Strong default: use Mermaid for every diagram type on that list above. Mermaid is simpler, more reliable, and the native Mermaid layout handles positioning and routing for you. For a flowchart, state diagram, sequence, ER, class, gantt, gitGraph, mindmap, etc. — reach for the mermaid parameter, not xml. Do not default to XML for flowcharts.
Use XML when the diagram type isn't on the Mermaid list above OR when the user explicitly asks for XML / draw.io format. Typical cases where XML is the right choice:
UI mockups / wireframes / screen designs — buttons, form fields, sidebars, modal dialogs (
shape=mxgraph.bootstrap.*,shape=mxgraph.ios.*,shape=mxgraph.android.*)Floor plans / seating charts / room layouts — rooms, doors, furniture (
shape=mxgraph.floorplan.*)Cloud architecture with AWS / Azure / GCP / Kubernetes icons (
shape=mxgraph.aws4.*,shape=mxgraph.azure.*,shape=mxgraph.gcp2.*,shape=mxgraph.kubernetes.*)Network topology with Cisco / Rack / networking shapes (
shape=mxgraph.cisco*.*,shape=mxgraph.rack.*,shape=mxgraph.networking.*)P&ID / electrical / engineering schematics (
shape=mxgraph.pid2.*,shape=mxgraph.electrical.*,shape=mxgraph.mscae.*)Swimlanes / pools with custom colors and hand-placed contents
UML class / component / deployment diagrams where positioning carries meaning
Venn diagrams, quadrant charts, concept maps with custom regions — anything where hand-placed geometry is the point
Any diagram requiring specific colors, fonts, stencils, or layouts that Mermaid can't control precisely
Call search_shapes first when you need industry icons (AWS / Azure / Cisco / P&ID / Kubernetes / floorplan / mockup / electrical) to find the correct style string for each shape.
User preference override — XML only. If the user expresses a preference for draw.io XML over Mermaid in any phrasing (examples: "no mermaid", "skip mermaid", "use xml", "I want drawio format", "stop using mermaid", "give me the xml", "native drawio only", etc.), from that point onward in the conversation you MUST use the xml parameter exclusively and MUST NOT use the mermaid parameter, even for diagram types where Mermaid would normally be preferable. This preference persists for the remainder of the conversation unless the user clearly reverses it (e.g. "mermaid is fine again"). When the preference is active, translate any diagram request — including flowcharts, sequence diagrams, ER diagrams, etc. — directly to well-formed mxGraphModel XML.
When using XML: IMPORTANT — the XML must be well-formed. Do NOT include ANY XML comments (<!-- -->) in the output.
draw.io XML Reference
Detailed reference for styles, edge routing, containers, layers, tags, metadata, and dark mode. Consult this when generating draw.io XML diagrams.
Reasoning budget (read this first)
Your job is to declare the logical structure of the diagram — what nodes exist, what edges connect them, what labels they carry, what lane/container groups them. draw.io's edge router and (when available) a post-layout pass handle routing and placement; you do not need to do layout math.
Do NOT in your reasoning:
Do NOT debate the topic. The user asked for a flowchart / architecture / sequence / etc. — pick one concrete scenario on your first impulse and commit. Never write "Actually, let me think of something else…" or pitch alternatives.
Do NOT debate flat-lanes vs nested-pools, horizontal vs vertical orientation, one vs multiple variations. Pick the first reasonable option (almost always: flat swimlanes, top-down or left-right based on what fits the content). Do not flip-flop.
Do NOT compute x/y coordinates in prose. No "column spacings of 160px totaling 1840px width — that's too wide, let me tighten to 1700…" loops. Use the rigid grid below; do the arithmetic in your head and write the XML.
Do NOT re-derive drawio mechanics (
horizontal=0,startSize=110, nested-lane coordinates). Use the templates below as-is.Do NOT enumerate columns ("customer lane columns 0-10, web app 1-7"). Place a node, move on.
Do NOT add
<Array as="points">waypoints. Edges are routed automatically.Do NOT set
exitX/exitY/entryX/entryYconnection-point overrides unless you have specific geometric intent.Do NOT verify, re-check, or adjust coordinates after placing a node.
Do NOT narrate "building the diagram / finalizing the XML / now let me…". Just emit XML.
Do NOT write out lists of node positions as planning text. Emit them as
<mxCell>elements directly.
Do in your reasoning:
Identify the diagram type + actors/stages (1-2 short sentences).
Identify any grouping (swimlanes? containers? none?).
Go straight to XML.
Rigid grid — use for every XML diagram:
Column x =
col_index * 180 + 40(col 0 = 40, col 1 = 220, col 2 = 400, …)Row y =
row_index * 120 + 40(row 0 = 40, row 1 = 160, row 2 = 280, …)Node size: rectangles
140×60, diamonds140×80, circles60×60, documents120×80, cylinders100×70
Pick a (col, row) for each node. Don't think about centers, gaps, or overlap — ELK handles routing between rough positions. Slight misalignment is invisible in the result.
General principles
Use proper draw.io shapes and connectors — choose the semantically correct shape for each element (e.g.,
shape=cylinder3for databases and tanks,rhombusfor decisions,shape=mxgraph.pid2valves.*for valves in P&IDs). draw.io has extensive shape libraries; prefer domain-appropriate shapes over generic rectangles.Decide whether to search for shapes — before generating a diagram, decide if it needs domain-specific shapes from draw.io's extended libraries. Skip
search_shapesfor standard diagram types that use basic geometric shapes: flowcharts, UML (class, sequence, state, activity), ERD, org charts, mind maps, Venn diagrams, timelines, wireframes, and any diagram using only rectangles, diamonds, circles, cylinders, and arrows. Also skip if the user explicitly asks to use basic/simple shapes or says not to search. Usesearch_shapeswhen the diagram requires industry-specific or branded icons: cloud architecture (AWS, Azure, GCP), network topology (Cisco, rack equipment), P&ID (valves, instruments, vessels), electrical/circuit diagrams, Kubernetes, BPMN with specific task types, or any domain where the user expects realistic/standardized symbols rather than labeled boxes.Match the language of labels to the user's language — if the user writes in German, French, Japanese, etc., all diagram labels, titles, and annotations should be in that same language.
Group related nodes, and surface a hub when edges converge — put nodes that belong together inside a container or swimlane, and keep external actors (users, files, third-party systems) outside implementation containers. When many edges converge on one area or cross several groups, route them through a single hub/gateway node (a registry, broker, event log, …) instead of drawing every low-level dependency across the canvas — fewer crossings, clearer contract.
Encode secondary detail in node text, not edges — draw an edge only when the relationship itself carries meaning; push incidental detail into the node label so the connector layer stays readable.
Common styles
Rounded rectangle:
<mxCell id="2" value="Label" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="100" y="100" width="120" height="60" as="geometry"/>
</mxCell>Diamond (decision):
<mxCell id="3" value="Condition?" style="rhombus;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="100" y="200" width="120" height="80" as="geometry"/>
</mxCell>Arrow (edge):
<mxCell id="4" value="" style="edgeStyle=orthogonalEdgeStyle;html=1;" edge="1" source="2" target="3" parent="1">
<mxGeometry relative="1" as="geometry"/>
</mxCell>Labeled arrow:
<mxCell id="5" value="Yes" style="edgeStyle=orthogonalEdgeStyle;html=1;" edge="1" source="3" target="6" parent="1">
<mxGeometry relative="1" as="geometry"/>
</mxCell>Style properties
Property | Values | Use for |
| 0 or 1 | Rounded corners |
| wrap | Text wrapping |
| Hex color | Background color |
| Hex color | Border color |
| Hex color | Text color |
| shape name | Database cylinders |
| shape name | Document shapes |
| style keyword | Circles/ovals |
| style keyword | Diamonds |
| style keyword | Right-angle connectors |
| style keyword | Elbow connectors |
| 0 or 1 | Dashed lines |
| style keyword | Swimlane containers |
| style keyword | Invisible container (pointerEvents=0) |
| 0 or 1 | Enable container behavior on any shape |
| 0 or 1 | Prevent container from capturing child connections |
| 0 or 1 | Enable HTML rendering in labels (required for |
| shape | UML sequence diagram lifeline (size = header height) |
HTML labels
Always include html=1 in the style when the value attribute contains any HTML tags (<b>, <br>, <font>, <i>, <u>, <hr>, <p>, <table>, etc.). Without html=1, HTML tags are displayed as literal text instead of being rendered.
HTML in attribute values must be XML-escaped: < → <, > → >, & → &, " → "
<mxCell value="<b>Title</b><br>Description"
style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="100" y="100" width="120" height="60" as="geometry"/>
</mxCell>Line breaks: Use
(works with both html=1 and html=0) or <br> (requires html=1) for line breaks — never use \n, which renders as literal backslash-n text instead of a newline.
Best practice: Always include html=1 in every cell style. This ensures labels render correctly whether they contain HTML or plain text — plain text is unaffected by the flag.
Bold/italic/underline: Use fontStyle in the style string when the entire label should be bold (fontStyle=1), italic (fontStyle=2), or underline (fontStyle=4). Values can be combined via bitwise OR (e.g., fontStyle=3 = bold+italic). Use HTML tags (<b>, <i>, <u>) only when formatting part of the label (e.g., bold title with normal description). Never combine fontStyle with HTML tags for the same effect — this is redundant and causes visible raw tags if html=1 is missing.
Edges
CRITICAL: Every edge mxCell must contain a <mxGeometry relative="1" as="geometry" /> child element. Self-closing edge cells (e.g. <mxCell ... edge="1" ... />) are invalid and will not render correctly. Always use the expanded form:
<mxCell id="e1" edge="1" parent="1" source="a" target="b" style="...">
<mxGeometry relative="1" as="geometry" />
</mxCell>Don't hand-route edges. Just declare source and target. You do not need to:
Add
<mxPoint>waypointsSet
exitX/exitY/entryX/entryYRoute around obstacles
Worry about edge-vertex collisions or parallel edge spacing
draw.io's built-in router is basic: it draws each edge as a straight line or a simple right-angle path between source and target, with no awareness of other shapes — a wire will run straight across any box that sits between its endpoints. That's fine when connected nodes have open space between them. When edges would otherwise cross over shapes, or you want consistently clean orthogonal wires that route around the boxes, set routing: "libavoid" on create_diagram; for a full re-layout use postLayout: "elk" (see Edge routing & layout passes below). Both compute the waypoints for you — you never add them by hand either way.
What you still choose: the edge style. The style determines the overall look (orthogonal angles, curves, straight lines) — the router honors the style family.
Style | Syntax | Best for |
Orthogonal |
| Flowcharts, architecture, network diagrams, BPMN — any diagram with right-angle connectors |
Straight | no | UML class/sequence diagrams, direct point-to-point connections. For sequence diagram messages use |
Entity Relation |
| ER diagrams — creates perpendicular stubs at both ends |
Curved |
| Mind maps, informal diagrams |
Elbow |
| Rarely needed — |
Use a consistent edge style within each diagram. Pick one based on diagram type and apply it to all edges: ER → entityRelationEdgeStyle; UML class → straight; mind maps → curved; flowcharts/architecture/network → orthogonalEdgeStyle.
Useful edge style attributes that apply regardless of routing:
rounded=1— rounded corners at bend points (recommended for orthogonal)endArrow=classic/endArrow=none— arrow headsdashed=1— dashed linestrokeColor=#...,strokeWidth=2— color/widthEdge labels: set
valuedirectly on the edge cell
Keep edge labels short and meaningful — one to three words (Yes, async, reads). Drop labels that merely restate an obvious action (call, register); move longer explanations into node text or a small legend node.
Visual semantics — stay consistent, add a legend when mixing styles. Within one diagram apply dashed=1, strokeColor, and strokeWidth consistently for one chosen meaning (e.g. dashed = optional / async / inferred relationship). Don't mix several dashed meanings without a small legend explaining them.
Containers and groups
For architecture diagrams or any diagram with nested elements, use draw.io's proper parent-child containment — do not just place shapes on top of larger shapes.
How containment works
Set parent="containerId" on child cells. Children use relative coordinates within the container.
Container types
Type | Style | When to use |
Group (invisible) |
| No visual border needed, container has no connections. Includes |
Swimlane (titled) |
| Container needs a visible title bar/header, or the container itself has connections |
Custom container | Add | Any shape acting as a container without its own connections |
Key rules
Edges to children inside containers naturally cross the container boundary — this is correct and expected. Do not add extra waypoints or complex routing to avoid a parent container when connecting to shapes inside it.
Always add
pointerEvents=0;to container styles that should not capture connections being rewired between childrenOnly omit
pointerEvents=0when the container itself needs to be connectable — in that case, useswimlanestyle which handles this correctly (the client area is transparent for mouse events while the header remains connectable)Children must set
parent="containerId"and use coordinates relative to the container
Example: Architecture container with swimlane
<mxCell id="svc1" value="User Service" style="swimlane;startSize=30;fillColor=#dae8fc;strokeColor=#6c8ebf;html=1;" vertex="1" parent="1">
<mxGeometry x="100" y="100" width="300" height="200" as="geometry"/>
</mxCell>
<mxCell id="api1" value="REST API" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="svc1">
<mxGeometry x="20" y="40" width="120" height="60" as="geometry"/>
</mxCell>
<mxCell id="db1" value="Database" style="shape=cylinder3;whiteSpace=wrap;html=1;" vertex="1" parent="svc1">
<mxGeometry x="160" y="40" width="120" height="60" as="geometry"/>
</mxCell>Example: Invisible group container
<mxCell id="grp1" value="" style="group;" vertex="1" parent="1">
<mxGeometry x="100" y="100" width="300" height="200" as="geometry"/>
</mxCell>
<mxCell id="c1" value="Component A" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="grp1">
<mxGeometry x="10" y="10" width="120" height="60" as="geometry"/>
</mxCell>Swimlanes for grouped actors (BPMN-style flowcharts)
Use flat swimlanes at parent="1", stacked vertically. One row of nodes per lane.
Fixed values — do not compute or debate:
Lane size:
x=0, y=lane_index*150, width=CANVAS_W, height=150Lane style:
swimlane;horizontal=0;startSize=110;fillColor=<pastel>;html=1;Child nodes inside a lane:
parent="<lane_id>",x = 120 + col*180,y = 45(always 45), size 140×60 (or 140×80 for diamonds)Cross-lane edges:
parent="1"(not inside a lane)
Pick CANVAS_W = max_col * 180 + 300. Choose lane colors from #f5f5f5, #e8f4f8, #fff0e6, #e8f5e9, #fff9e6, #fce4ec in that order.
<mxCell id="lane1" value="Customer" style="swimlane;horizontal=0;startSize=110;fillColor=#f5f5f5;html=1;" vertex="1" parent="1">
<mxGeometry x="0" y="0" width="1800" height="150" as="geometry"/>
</mxCell>
<mxCell id="n1" value="Place Order" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="lane1">
<mxGeometry x="120" y="45" width="140" height="60" as="geometry"/>
</mxCell>
<mxCell id="lane2" value="System" style="swimlane;horizontal=0;startSize=110;fillColor=#e8f4f8;html=1;" vertex="1" parent="1">
<mxGeometry x="0" y="150" width="1800" height="150" as="geometry"/>
</mxCell>
<mxCell id="n2" value="Validate" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="lane2">
<mxGeometry x="300" y="45" width="140" height="60" as="geometry"/>
</mxCell>
<mxCell id="e1" edge="1" parent="1" source="n1" target="n2" style="edgeStyle=orthogonalEdgeStyle;rounded=1;html=1;">
<mxGeometry relative="1" as="geometry"/>
</mxCell>Do NOT nest lanes inside a pool. Do NOT vary lane heights. Do NOT compute title-area offset — it is always 110, children start at x=120 to clear it.
Nested architecture containers (cloud, infra, network topologies)
For diagrams with nested groupings — VPC → Availability Zone → EC2 instance, Datacenter → Rack → Server, Region → Environment → Service — use nested swimlanes. This is where the AI most often flattens hierarchy that should be nested. Treat each level as a swimlane container.
Rules:
Every container is a
swimlanewithstartSize=24(title area at the top).Child cells set
parent="<container_id>"and use coordinates relative to their parent (origin 0,0 is the parent's top-left, below the title).Edges between cells in different containers must have
parent="1"(not a container) — otherwise they render inside the container and get clipped.For industry-specific icons (AWS/Azure/GCP logos, Cisco equipment, etc.), call
search_shapesto get the exactstylestring and substitute it into a regular vertex — the container structure stays the same.
<mxCell id="vpc" value="VPC" style="swimlane;startSize=24;fillColor=#dae8fc;strokeColor=#6c8ebf;html=1;" vertex="1" parent="1">
<mxGeometry x="0" y="0" width="720" height="360" as="geometry"/>
</mxCell>
<mxCell id="az1" value="AZ us-east-1a" style="swimlane;startSize=24;fillColor=#fff2cc;strokeColor=#d6b656;html=1;" vertex="1" parent="vpc">
<mxGeometry x="20" y="36" width="320" height="300" as="geometry"/>
</mxCell>
<mxCell id="web1" value="web-1" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="az1">
<mxGeometry x="30" y="40" width="120" height="60" as="geometry"/>
</mxCell>
<mxCell id="db1" value="db-1" style="shape=cylinder3;whiteSpace=wrap;html=1;" vertex="1" parent="az1">
<mxGeometry x="180" y="40" width="100" height="70" as="geometry"/>
</mxCell>
<mxCell id="az2" value="AZ us-east-1b" style="swimlane;startSize=24;fillColor=#fff2cc;strokeColor=#d6b656;html=1;" vertex="1" parent="vpc">
<mxGeometry x="360" y="36" width="340" height="300" as="geometry"/>
</mxCell>
<mxCell id="web2" value="web-2" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="az2">
<mxGeometry x="30" y="40" width="120" height="60" as="geometry"/>
</mxCell>
<mxCell id="e1" edge="1" parent="1" source="web1" target="web2" style="edgeStyle=orthogonalEdgeStyle;rounded=1;html=1;">
<mxGeometry relative="1" as="geometry"/>
</mxCell>Cross-functional flowcharts (actor × phase grid, as a table)
Cross-functional flowcharts show a process across two axes at once — actors (rows) and phases (columns). Use drawio's table shape, which auto-arranges cells into a grid via childLayout=tableLayout. This is the canonical draw.io pattern and is distinct from plain swimlanes (which only group on one axis).
Structure:
Outer container:
shape=table;childLayout=tableLayout;startSize=0;collapsible=0;fillColor=none;Rows are children of the table:
shape=tableRow;horizontal=0;startSize=0;collapsible=0;Cells are children of rows — regular vertices, one per (actor, phase) intersection
Row heights and cell widths are set via
mxGeometry; they tile automaticallyFirst row = phase headers; first cell of every other row = actor label
Process nodes go INSIDE the appropriate cell (parent = cell id) at coordinates relative to the cell
Cross-cell edges must use
parent="1"(same rule as containers)
<mxCell id="tbl" style="shape=table;childLayout=tableLayout;startSize=0;collapsible=0;fillColor=none;" vertex="1" parent="1">
<mxGeometry x="0" y="0" width="900" height="320" as="geometry"/>
</mxCell>
<mxCell id="r0" style="shape=tableRow;horizontal=0;startSize=0;collapsible=0;" vertex="1" parent="tbl">
<mxGeometry width="900" height="40" as="geometry"/>
</mxCell>
<mxCell id="h0" style="text;html=1;" vertex="1" parent="r0">
<mxGeometry width="140" height="40" as="geometry"/>
</mxCell>
<mxCell id="h1" value="Order" style="text;align=center;fontStyle=1;fillColor=#e8e8e8;" vertex="1" parent="r0">
<mxGeometry x="140" width="380" height="40" as="geometry"/>
</mxCell>
<mxCell id="h2" value="Fulfill" style="text;align=center;fontStyle=1;fillColor=#e8e8e8;" vertex="1" parent="r0">
<mxGeometry x="520" width="380" height="40" as="geometry"/>
</mxCell>
<mxCell id="r1" style="shape=tableRow;horizontal=0;startSize=0;collapsible=0;" vertex="1" parent="tbl">
<mxGeometry y="40" width="900" height="140" as="geometry"/>
</mxCell>
<mxCell id="a1" value="Customer" style="fillColor=#dae8fc;fontStyle=1;" vertex="1" parent="r1">
<mxGeometry width="140" height="140" as="geometry"/>
</mxCell>
<mxCell id="c_cust_order" style="fillColor=none;" vertex="1" parent="r1">
<mxGeometry x="140" width="380" height="140" as="geometry"/>
</mxCell>
<mxCell id="t_place" value="Place Order" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="c_cust_order">
<mxGeometry x="120" y="40" width="140" height="60" as="geometry"/>
</mxCell>
<mxCell id="c_cust_fulfill" style="fillColor=none;" vertex="1" parent="r1">
<mxGeometry x="520" width="380" height="140" as="geometry"/>
</mxCell>
<mxCell id="r2" style="shape=tableRow;horizontal=0;startSize=0;collapsible=0;" vertex="1" parent="tbl">
<mxGeometry y="180" width="900" height="140" as="geometry"/>
</mxCell>
<mxCell id="a2" value="System" style="fillColor=#d5e8d4;fontStyle=1;" vertex="1" parent="r2">
<mxGeometry width="140" height="140" as="geometry"/>
</mxCell>
<mxCell id="c_sys_order" style="fillColor=none;" vertex="1" parent="r2">
<mxGeometry x="140" width="380" height="140" as="geometry"/>
</mxCell>
<mxCell id="t_validate" value="Validate" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="c_sys_order">
<mxGeometry x="120" y="40" width="140" height="60" as="geometry"/>
</mxCell>
<mxCell id="c_sys_fulfill" style="fillColor=none;" vertex="1" parent="r2">
<mxGeometry x="520" width="380" height="140" as="geometry"/>
</mxCell>
<mxCell id="t_ship" value="Ship" style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="c_sys_fulfill">
<mxGeometry x="120" y="40" width="140" height="60" as="geometry"/>
</mxCell>
<mxCell id="e1" edge="1" parent="1" source="t_place" target="t_validate" style="edgeStyle=orthogonalEdgeStyle;rounded=1;html=1;">
<mxGeometry relative="1" as="geometry"/>
</mxCell>
<mxCell id="e2" edge="1" parent="1" source="t_validate" target="t_ship" style="edgeStyle=orthogonalEdgeStyle;rounded=1;html=1;">
<mxGeometry relative="1" as="geometry"/>
</mxCell>When to use cross-functional tables vs flat swimlanes:
Flat swimlanes — one-dimensional (actors only, or phases only). Simpler. Use this when you just need to show who does what in sequence.
Cross-functional table — two-dimensional (actors AND phases). Use this when both the actor and the process stage matter, and every step belongs to a specific (actor, phase) cell.
Do NOT nest swimlanes inside a table row, do NOT set startSize on rows or cells (columns tile from x=0), and do NOT rely on the AI to produce exact widths that sum to the table width — close-enough totals are fine, the tableLayout normalizes them.
Layers
Layers control visibility and z-order. Every cell belongs to exactly one layer. Use layers to manage diagram complexity — viewers can toggle layer visibility to show or hide groups of elements (e.g., "Physical Infrastructure" vs "Logical Network" vs "Security Zones").
Cell id="0" is the root and cell id="1" is the default layer — both always exist. Additional layers are mxCell elements with parent="0":
<mxGraphModel>
<root>
<mxCell id="0"/>
<mxCell id="1" parent="0"/>
<mxCell id="2" value="Annotations" parent="0"/>
<mxCell id="10" value="Server" style="rounded=1;html=1;" vertex="1" parent="1">
<mxGeometry x="100" y="100" width="120" height="60" as="geometry"/>
</mxCell>
<mxCell id="20" value="Note: deprecated" style="text;" vertex="1" parent="2">
<mxGeometry x="100" y="170" width="120" height="30" as="geometry"/>
</mxCell>
</root>
</mxGraphModel>A layer is an
mxCellwithparent="0"and novertexoredgeattributeAssign shapes to a layer by setting
parentto the layer's idLater layers render on top of earlier layers (higher z-order)
Add
visible="0"as an attribute on the layer cell to hide it by defaultUse layers when the diagram has distinct conceptual groupings that viewers may want to toggle independently
Tags
Tags are visual filters that let viewers show or hide elements by category. Unlike layers, a single element can have multiple tags, making tags ideal for cross-cutting concerns (e.g., tagging shapes as "critical", "v2", or "backend").
Tags require wrapping mxCell in an <object> element. Tags are assigned via the tags attribute as a space-separated string:
<mxGraphModel>
<root>
<mxCell id="0"/>
<mxCell id="1" parent="0"/>
<object id="2" label="Auth Service" tags="critical v2">
<mxCell style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="100" y="100" width="120" height="60" as="geometry"/>
</mxCell>
</object>
<object id="3" label="Legacy API" tags="critical deprecated">
<mxCell style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="300" y="100" width="120" height="60" as="geometry"/>
</mxCell>
</object>
</root>
</mxGraphModel>Tags require the
<object>wrapper — a plainmxCellcannot have tagsThe
labelattribute on<object>replacesvalueonmxCellTags are space-separated in the
tagsattributeViewers filter the diagram by selecting tags in the draw.io UI (Edit > Tags)
Tags do not affect z-order or structural grouping — they are purely a visibility filter
Metadata and placeholders
Metadata stores custom key-value properties on shapes as additional attributes on the <object> wrapper element. Combined with placeholders, metadata values can be displayed in labels — useful for data-driven diagrams showing status, owner, IP addresses, or versions on each shape.
Set placeholders="1" on the <object> to enable %propertyName% substitution in the label:
<mxGraphModel>
<root>
<mxCell id="0"/>
<mxCell id="1" parent="0"/>
<object id="2" label="<b>%component%</b><br>Owner: %owner%<br>Status: %status%"
placeholders="1" component="Auth Service" owner="Team Backend" status="Active">
<mxCell style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
<mxGeometry x="100" y="100" width="160" height="80" as="geometry"/>
</mxCell>
</object>
</root>
</mxGraphModel>Custom properties are plain XML attributes on
<object>(e.g.,component="Auth Service")Set
placeholders="1"to enable%key%substitution in the label and tooltipThe label must use
html=1style when using HTML formatting with placeholdersPlaceholders resolve by walking up the containment hierarchy: shape attributes first, then parent container, then layer, then root — first match wins
Predefined placeholders work without custom properties:
%id%,%width%,%height%,%date%,%time%,%timestamp%,%page%,%pagenumber%,%pagecount%,%filename%Use
%%for a literal percent sign in labelsTags, metadata, and placeholders can all be combined on the same
<object>elementUse metadata when shapes represent data records (servers, services, components) and you want to attach structured information beyond the visible label
Dark mode colors
draw.io supports automatic dark mode rendering. How colors behave depends on the property:
strokeColor,fillColor,fontColordefault to"default", which renders as black in light theme and white in dark theme. When no explicit color is set, colors adapt automatically.Explicit colors (e.g.
fillColor=#DAE8FC) specify the light-mode color. The dark-mode color is computed automatically by inverting the RGB values (blending toward the inverse at 93%) and rotating the hue by 180° (viamxUtils.getInverseColor).light-dark()function — To specify both colors explicitly, uselight-dark(lightColor,darkColor)in the style string, e.g.fontColor=light-dark(#7EA6E0,#FF0000). The first argument is used in light mode, the second in dark mode.
To enable dark mode color adaptation, the mxGraphModel element must include adaptiveColors="auto".
When generating diagrams, you generally do not need to specify dark-mode colors — the automatic inversion handles most cases. Use light-dark() only when the automatic inverse color is unsatisfactory.
Edge routing & layout passes
By default, edges are drawn by draw.io's built-in router, which is intentionally basic: each edge is a straight line or a simple right-angle path between its endpoints, with no obstacle avoidance — a connector runs straight through any shape lying between its source and target. There is no server-side post-processing. Two opt-in passes on create_diagram upgrade this; they are independent and combine freely, run client-side after the diagram renders, and the exported XML (copy/clipboard, "Open in draw.io") reflects the final routed result.
routing: "libavoid"(XML only) — obstacle-avoiding orthogonal edge routing. Vertices stay exactly where you placed them; only the connectors are recomputed, so they run in clean right-angle segments that route around the boxes (and spread apart when parallel) instead of cutting across them. Use it for diagrams you laid out deliberately — architecture, network topology, deployment, swimlanes, UML, floor plans — where you want tidy wires without disturbing your layout.postLayout: "elk"— a full re-layout (ELKlayeredflow). Vertices animate (morph) from your positions to canonical hierarchical positions, and the edges are routed as part of that. Best for flowcharts, process/state diagrams, decision flows, pipelines, and other directional/hierarchical diagrams. (You should rarely hand-write these as XML — prefer Mermaid.) Flow direction: on XML set the optionaldirectionfield ("vertical"(default) /"horizontal"); on Mermaid it is read from the flowchart code (flowchart TD/TBvsLR/RL) anddirectionis ignored.
The four combinations:
|
| Result |
— | — | basic built-in router (straight / simple right-angle, no obstacle avoidance); your positions kept |
— |
| your positions kept; wires re-routed orthogonally around the shapes |
| — | ELK places the vertices and routes the edges (decent routing built in) |
|
| rarely worth it — ELK already routes; only add |
Pick ONE — they are essentially alternatives, not a stack:
Neither — fine when connected nodes sit in clear rows/columns with open space between them, so the basic router's straight/right-angle lines won't cross another shape. Simplest and lightest; do this by default for sparse layouts.
routing: "libavoid"— keep your hand-placed layout but clean up the wires: use whenever an edge would otherwise cut across a box, or you want consistently clean orthogonal wires routed around shapes (architecture, network topology, deployment, UML, floor plans — anything densely connected).postLayout: "elk"— when you want a canonical re-layout (vertices moved). ELK routes the edges itself as part of the layout, so do not also setrouting— the combination is redundant in almost all cases. Adddirection: "horizontal"for left-to-right flow.
For Mermaid diagrams: see the postLayout parameter description for when to set it. Complex Mermaid flowcharts (≥ ~20 nodes, ≥ 3 decision diamonds, feedback edges, or ≥ 3 endpoints) need postLayout: "elk" because the native parser's layout goes cramped or unbalanced past that threshold — the direction follows the flowchart code, so no direction is needed. Simple flowcharts and all non-flowchart Mermaid types (sequence, class, ER, sankey, …) need no postLayout.
When NOT to use (XML):
The user has asked for specific positions (swim lanes with exact lanes, architecture diagrams with meaningful spatial arrangement).
The diagram relies on containers/grouping where spatial layout encodes information.
CRITICAL: XML well-formedness
When generating draw.io XML, the output must be well-formed XML:
NEVER include ANY XML comments (
<!-- -->) in the output. XML comments are strictly forbidden — they waste tokens, can cause parse errors, and serve no purpose in diagram XML.Escape special characters in attribute values:
&,<,>,"Always use unique
idvalues for eachmxCell
Mermaid Reference
Short hints for generating Mermaid diagrams that render correctly in draw.io. draw.io's Mermaid parser covers 28 diagram types — the header keyword on the first non-directive line selects the type.
Canonical list & dialog/ELK docs: https://github.com/jgraph/drawio/discussions/5643.
General rules
Pick the type keyword carefully.
graph/flowchart,classDiagram,stateDiagram-v2,erDiagram,sequenceDiagram,gitGraph,journey,pie,gantt,mindmap,timeline,quadrantChart,requirementDiagram,sankey-beta,xychart-beta,block-beta,c4Context/C4Container/C4Component,architecture-beta,radar-beta,packet-beta,venn-beta,treemap-beta,treeView-beta,ishikawa-beta,kanban,zenuml,wardley-beta,eventmodeling. Misspelling the header yields a blank diagram.No trailing punctuation on node IDs. IDs are identifiers (
myNode,node_1,A) — spaces, hyphens (in some contexts), and reserved words (end,class,subgraph) break the parse. Put display text in brackets or quotes instead:A["User's Account"].One statement per line. Separate statements with newlines;
;works as a delimiter in flowchart but not everywhere.Quote labels with special characters (
:,-, parentheses, non-ASCII). Use"not'.HTML in labels: only
<br>,<b>,<i>,<u>are reliable across types. Use#for hex colors in styles, neverrgb().Diagrams can take a title block for some types:
--- title: My Diagram --- flowchart TDMatch the language of labels to the user's language — if the user writes in German, French, etc., the diagram labels should be in that language too.
Flowchart (most common)
flowchart TD
A[Start] --> B{Decision?}
B -->|Yes| C[Do thing]
B -->|No| D[Skip]
C --> E((End))
D --> EDirection:
TD/TB(top-down),BT,LR,RL.Node shapes by bracket:
[rect],(rounded),([stadium]),[[subroutine]],[(cylinder)],((circle)),{rhombus},{{hexagon}},[/parallelogram/],[\parallelogram alt\],[/trapezoid\],>asymmetric].Edges:
-->arrow,---no arrow,-.->dotted,==>thick,<-->bidirectional. Inline label:A -- text --> BorA -->|text| B.Subgraphs:
subgraph Frontend A --> B end
Styling & colors
Three ways — pick one, don't mix for the same node:
1. Inline per-node (style):
flowchart LR
A[Start] --> B[End]
style A fill:#f9f,stroke:#333,stroke-width:2px,color:#fff
style B fill:#bbf,stroke:#f66,stroke-dasharray:5 52. Reusable classes (classDef + :::):
flowchart LR
A:::happy --> B:::sad
classDef happy fill:#dfd,stroke:#0a0
classDef sad fill:#fdd,stroke:#a00Or apply to many: class A,B,C happy.
3. Link styling (edges):
linkStyle 0 stroke:#f00,stroke-width:3px
linkStyle default stroke:#9990 = first edge in order defined; default targets unstyled edges.
Style properties that work: fill, stroke, stroke-width, stroke-dasharray, color (font color).
Sequence diagram
sequenceDiagram
participant U as User
participant S as Server
U->>S: Request
S-->>U: Response
Note right of S: LoggedArrows:
->(no head),->>(arrow),-->>(dashed),-x(X end),--x(dashed X).Activate/deactivate:
activate S/deactivate SorS->>+S2: call/S2-->>-S: return.Blocks:
alt/else/end,opt/end,loop/end,par/and/end,critical/option/end.Notes:
Note left of A,Note over A,B: text.Optional
autonumberafter header numbers the messages.
Class diagram
classDiagram
class Animal {
+String name
+int age
+eat() void
}
class Dog
Animal <|-- Dog : inherits
Dog "1" --> "*" Bone : hasRelations:
<|--inherit,*--composition,o--aggregation,-->association,..>dependency,..|>realize,<-->bidirectional.Visibility:
+public,-private,#protected,~package.Annotations:
<<interface>>,<<abstract>>,<<enumeration>>inside the class block or viaAnimal <<interface>>.Cardinality: quoted strings flanking the arrow (
"1","0..*","*").
State diagram
stateDiagram-v2
[*] --> Idle
Idle --> Running : start
Running --> Idle : stop
Running --> [*]
state Running {
[*] --> Working
Working --> Waiting : block
Waiting --> Working : unblock
}Use
stateDiagram-v2, notstateDiagram(v1 is legacy).[*]= start (source) or end (target) depending on direction.state X { ... }nests a compound state;state fork1 <<fork>>,<<join>>,<<choice>>mark junction nodes.Transition labels:
A --> B : event [guard] / action.
ER diagram
erDiagram
CUSTOMER ||--o{ ORDER : places
ORDER ||--|{ LINE-ITEM : contains
CUSTOMER {
string name
string email PK
}Cardinality symbols:
|ozero-or-one,||exactly-one,}ozero-or-many,}|one-or-many. Mirror on both sides (e.g.,||--o{).Attribute blocks list
type name [PK|FK|UK]plus optional comment in quotes.Entity names are typically UPPERCASE by convention.
Journey
journey
title Morning routine
section Wake up
Coffee: 5: Me
Read news: 3: Me
section Commute
Drive: 2: Me, TrafficEach task: Name: score(1-5): Actor[, Actor...]. Section headers group tasks.
Pie
pie showData title Browser share
"Chrome" : 60
"Firefox" : 20
"Safari" : 20showData is optional (renders the numbers). Quotes on labels, colon, numeric value.
Gantt
gantt
title Project timeline
dateFormat YYYY-MM-DD
section Phase 1
Design : a1, 2025-01-01, 7d
Build : after a1, 14d
section Phase 2
Test : 2025-01-25, 5ddateFormatis mandatory.Task line:
Name : [id,] [after id | YYYY-MM-DD], duration[d/w].Status tags:
done,active,critbefore the id (crit a1).
Gitgraph
gitGraph
commit
branch develop
checkout develop
commit
commit
checkout main
merge developCommands: commit [id: "x"] [tag: "v1"], branch name, checkout name, merge name, cherry-pick id: "x".
Mindmap
mindmap
root((Project))
Frontend
React
CSS
Backend
Node
DBIndentation (2-space increments) defines hierarchy.
Root shape:
((circle)),[rect],(rounded),))cloud((,)hexagon(,{{hexagon}}.No edges — they are implied by nesting.
Timeline
timeline
title Company history
section 2020s
2021 : Founded
2022 : Series A
: Launched product
section 2030s
2030 : IPOColon separates year/label; multiple : lines under one year add sub-events.
Quadrant chart
quadrantChart
title Reach vs Engagement
x-axis Low --> High
y-axis Low --> High
quadrant-1 Stars
quadrant-2 Question Marks
quadrant-3 Dogs
quadrant-4 Cash Cows
Campaign A: [0.3, 0.6]
Campaign B: [0.75, 0.85]Point coords are [0..1, 0..1].
Requirement diagram
requirementDiagram
requirement req1 {
id: "1"
text: "The system shall..."
risk: high
verifymethod: test
}
element user_story {
type: "story"
}
user_story - satisfies -> req1Requirement types: requirement, functionalRequirement, performanceRequirement, interfaceRequirement, physicalRequirement, designConstraint. Relations: contains, copies, derives, satisfies, verifies, refines, traces.
Sankey
sankey-beta
Source,Intermediate,10
Source,Direct,5
Intermediate,Sink,10CSV-style: source,target,value. No header. No title (use frontmatter).
XY chart
xychart-beta
title "Revenue"
x-axis [jan, feb, mar, apr]
y-axis "USD" 0 --> 10000
bar [2500, 5000, 7500, 9000]
line [3000, 4500, 6500, 8500]bar [...] and line [...] can stack; order matters (later overlays earlier).
Block
block-beta
columns 3
A B C
D["Wide"]:2 E
A --> Dcolumns N sets grid width. Name:N spans N columns. Edges use flowchart arrow syntax.
C4
C4Context
Person(user, "User")
System(app, "App", "Does things")
Rel(user, app, "Uses")Variants:
C4Context,C4Container,C4Component,C4Dynamic,C4Deployment.Element helpers:
Person,System,System_Ext,Container,ComponentDb,Boundary(id, "label", "type"), etc. Arguments are positional:(id, label, [type/tech], [description]).UpdateElementStyle(tag, $bgColor="#…")andAddElementTagtweak appearance.
Architecture
architecture-beta
group cloud(cloud)[Cloud]
service api(server)[API] in cloud
service db(database)[DB] in cloud
api:R --> L:dbBuilt-in icons:
cloud,server,database,disk,internet. Suffix edge ends with:T,:B,:L,:Rto pick the side.group id(icon)[Label]thenin groupIdon services places nodes.
Radar
radar-beta
title Skills
axis js["JS"], py["Python"], go["Go"]
curve alice["Alice"]{80, 60, 70}
curve bob["Bob"]{50, 90, 65}Axes and curves are positionally aligned — list values in axis order, 0–100.
Packet
packet-beta
0-15: "Source Port"
16-31: "Dest Port"
32-63: "Seq Number"start-end (bit ranges) or single-bit N. Use a title frontmatter.
Venn
venn-beta
set A ["Set A"]
set B ["Set B"]
union A,B
text A ["only A"]
text A,B ["shared"]Define every union combination whose region you plan to label. text A,B [...] places text in intersections.
Treemap
treemap-beta
"Category"
"Leaf 1": 40
"Leaf 2": 60Numbers are values (area-weighted). Indent (2+ spaces) for hierarchy.
Tree view
treeView-beta
"Root"
"Child 1"
"Grandchild"
"Child 2"Pure indentation hierarchy, no numbers.
Ishikawa (fishbone)
ishikawa-beta
Main Problem
Category
Cause
Sub-cause
Another Category
CauseFirst line after header is the problem; top-level indents are categories (Materials, Methods, Machinery, etc. — use whatever makes sense).
Kanban
kanban
todo[To Do]
task1[Write spec]@{ assigned: "Alice", priority: "High" }
doing[In progress]
task2[Build feature]
done[Done]Columns are id[Label] at indent 0; cards are id[Label]@{ metadata } inside. Metadata keys: assigned, priority (Very Low/Low/Medium/High/Very High), ticket.
ZenUML
zenuml
@Actor User
@Boundary Web
@Control Service
User -> Web: request
Web -> Service: process()
Service -> Web: resultParticipant roles: @Actor, @Boundary, @Control, @Entity, @Database. Messages use -> with a colon-separated label. Supports if/else, while, par blocks like sequence diagrams.
Wardley map
wardley-beta
title Tea Shop
anchor Business [0.95, 0.63]
component Cup of Tea [0.79, 0.61]
component Kettle [0.43, 0.35] (inertia)
Business -> Cup of Tea
Cup of Tea -> Kettle
evolve Kettle 0.62Header
wardleyorwardley-beta;titleoptional.anchor/component Name [visibility, evolution]— coords are[0..1, 0..1](y = value-chain visibility, x = evolution from Genesis to Commodity).Component evolution markers in parens:
(inertia),(build),(buy),(outsource),(market).Links:
A -> Bdependency,A +> Bflow.evolve Name <x>adds an evolution target;evolution Genesis -> Custom -> Product -> Commodityrelabels the x-axis stages.Extras:
note "text" [x,y],annotation N,[x,y] "text",accelerator/deaccelerator "text" [x,y].
Event Modeling
eventmodeling
tf 01 ui CartUI
tf 02 cmd AddItem
tf 03 evt ItemAdded
tf 04 rmo CartEach
tf <id> <type> <Name>is a time-frame (column). Types:ui/pcr(processor),cmd/command,rmo/readmodel,evt/event— placed on the UI/Automation, Command/Read-Model, and Events swimlanes.Wire frames with
->>:tf 04 evt ItemChanged ->> 02 ->> 03links frame 04 back to 02 and 03.Namespace.Namegroups frames into slices (e.g.Order.ChangeOrder).data <id> { ... }blocks attach payloads, referenced inline with[[id]]:tf 02 cmd AddItem [[AddItem01]].
When to prefer XML over Mermaid
Precise positions / custom coordinates.
draw.io-native shapes (AWS, Azure, GCP, P&ID, Cisco, electrical).
Mixed shape libraries or complex multi-layer diagrams.
Anything that needs exact colors per element with many variations — Mermaid's styling works but at scale XML is easier to reason about.
Default to Mermaid for the standard types above; reach for XML only when Mermaid's syntax clearly can't express what's needed.
Input Schema
| Name | Required | Description | Default |
|---|---|---|---|
| xml | No | draw.io XML content in mxGraphModel format. Must be well-formed XML: no XML comments (<!-- -->), no unescaped special characters in attribute values. Mutually exclusive with 'mermaid'. | |
| mermaid | No | Mermaid.js diagram definition (e.g. 'graph TD\n A-->B'). Supports 26 diagram types — see the tool description for the full list. The diagram is parsed and laid out natively (no upstream mermaid runtime) and converted to draw.io format. Mutually exclusive with 'xml'. | |
| postLayout | No | Optional client-side ELK (Eclipse Layout Kernel) layered-flow pass applied after the diagram renders. The only value is `"elk"`. Vertices animate (morph) from the positions you supplied to the layered layout — they are **replaced**, so only your edge topology survives. Set it for directional/hierarchical diagrams: flowcharts, process diagrams, state diagrams, decision flows, pipelines, ER/class diagrams with clear parent→child direction. Flow direction (top-down vs left-to-right): for **Mermaid** it is taken from the flowchart code (`flowchart TD/TB` → top-down, `LR/RL` → left-to-right) — do not try to set it here; for **XML** set the optional `direction` field (defaults to top-down). **Omit** for diagrams whose layout carries meaning you hand-crafted: swimlanes/pools, containers, architecture / deployment / network topology with grouped regions, P&ID or circuit schematics, floor plans, UML diagrams with deliberate placement. **For Mermaid flowcharts**, the native parser does its own layout, but it produces cramped or unbalanced output once the diagram has any structural complexity. Request `postLayout: "elk"` whenever ANY of the following holds: ≥ ~20 nodes, OR ≥ 3 decision diamonds (`{...}` shapes), OR any feedback/back-edges (an edge that points to an earlier node, e.g. an error path looping back to a retry), OR ≥ 3 distinct endpoints — the flow direction follows the flowchart code, so you never pass `direction` for Mermaid. Skip for simple flowcharts (linear chains, < 20 nodes, no branching/back-edges) and for non-flowchart Mermaid types (sequence, class, ER, sankey, etc. — postLayout doesn't apply). | |
| direction | No | **XML only** — the flow direction for `postLayout: "elk"` on XML diagrams: `vertical` (top-down) or `horizontal` (left-to-right). Defaults to `vertical`. **Ignored for Mermaid**, where direction is read from the flowchart code (`flowchart TD/TB` vs `LR/RL`). Only meaningful together with `postLayout: "elk"`. | |
| routing | No | **XML only** — optional obstacle-avoiding orthogonal **edge-routing** pass (libavoid). The only value is `"libavoid"`. Unlike `postLayout`, this does **not** move any vertices — it keeps your hand-placed coordinates and only recomputes each edge's path so wires run in clean right-angle segments that route *around* the boxes instead of cutting through them. Set it for XML diagrams where you have placed nodes deliberately (architecture, network topology, deployment, swimlanes, UML, floor plans) and want tidy connectors without overlaps — exactly the diagrams where you'd OMIT `postLayout`. draw.io's default router is basic (straight or simple right-angle lines with **no obstacle avoidance** — wires cut straight through any box between the endpoints), so reach for `"libavoid"` whenever edges would otherwise cross shapes or you want uniformly clean orthogonal wiring on a layout you placed yourself. Think of `routing` and `postLayout` as **alternatives**: use `routing: "libavoid"` to keep your positions and only fix the wires, or `postLayout: "elk"` to re-lay-out the vertices — and note ELK already routes its edges decently, so when you use `postLayout` you normally should **not** also set `routing` (the combination is redundant in almost all cases). Do not add `<Array as="points">` waypoints yourself when routing is set; libavoid computes them. **Ignored for Mermaid** (its native/ELK layout already routes). |