@drawio/mcp 1.2.0 → 1.2.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +21 -0
- package/package.json +3 -2
- package/src/index.js +35 -22
- package/src/xml-reference.md +253 -0
package/README.md
CHANGED
|
@@ -59,6 +59,25 @@ Add to your Claude Desktop configuration file:
|
|
|
59
59
|
}
|
|
60
60
|
```
|
|
61
61
|
|
|
62
|
+
### Claude Code
|
|
63
|
+
|
|
64
|
+
```bash
|
|
65
|
+
claude mcp add drawio -- npx -y @drawio/mcp
|
|
66
|
+
```
|
|
67
|
+
|
|
68
|
+
Or manually in `.claude/settings.json`:
|
|
69
|
+
|
|
70
|
+
```json
|
|
71
|
+
{
|
|
72
|
+
"mcpServers": {
|
|
73
|
+
"drawio": {
|
|
74
|
+
"command": "npx",
|
|
75
|
+
"args": ["-y", "@drawio/mcp"]
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
```
|
|
80
|
+
|
|
62
81
|
### Other MCP Clients
|
|
63
82
|
|
|
64
83
|
Configure your MCP client to run the server via stdio:
|
|
@@ -116,6 +135,8 @@ Opens the draw.io editor with a Mermaid.js diagram.
|
|
|
116
135
|
4. The URL is returned to the LLM, which can present it to the user
|
|
117
136
|
5. Opening the URL loads draw.io with the diagram ready to view/edit
|
|
118
137
|
|
|
138
|
+
The `open_drawio_xml` tool description includes the full XML generation reference (edge routing, containers, layers, tags, metadata, dark mode, etc.) loaded from [`shared/xml-reference.md`](../shared/xml-reference.md) — the single source of truth for all draw.io MCP prompts. A `prepack` script bundles this file into the npm package so it works after `npm install`.
|
|
139
|
+
|
|
119
140
|
## Related Resources
|
|
120
141
|
|
|
121
142
|
- [draw.io](https://www.draw.io) - Free online diagram editor
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@drawio/mcp",
|
|
3
|
-
"version": "1.2.
|
|
3
|
+
"version": "1.2.2",
|
|
4
4
|
"description": "Official draw.io MCP server for LLMs - Open diagrams in draw.io editor",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "src/index.js",
|
|
@@ -8,7 +8,8 @@
|
|
|
8
8
|
"drawio-mcp": "./src/index.js"
|
|
9
9
|
},
|
|
10
10
|
"scripts": {
|
|
11
|
-
"start": "node src/index.js"
|
|
11
|
+
"start": "node src/index.js",
|
|
12
|
+
"prepack": "cp ../shared/xml-reference.md src/xml-reference.md"
|
|
12
13
|
},
|
|
13
14
|
"keywords": [
|
|
14
15
|
"mcp",
|
package/src/index.js
CHANGED
|
@@ -5,12 +5,24 @@ import { StdioServerTransport } from "@modelcontextprotocol/sdk/server/stdio.js"
|
|
|
5
5
|
import { ListToolsRequestSchema, CallToolRequestSchema } from "@modelcontextprotocol/sdk/types.js";
|
|
6
6
|
import pako from "pako";
|
|
7
7
|
import { spawn } from "child_process";
|
|
8
|
-
import { writeFileSync, unlinkSync } from "fs";
|
|
9
|
-
import { join } from "path";
|
|
8
|
+
import { existsSync, readFileSync, writeFileSync, unlinkSync } from "fs";
|
|
9
|
+
import { join, dirname } from "path";
|
|
10
|
+
import { fileURLToPath } from "url";
|
|
10
11
|
import { tmpdir } from "os";
|
|
11
12
|
|
|
13
|
+
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
12
14
|
const DRAWIO_BASE_URL = "https://app.diagrams.net/";
|
|
13
15
|
|
|
16
|
+
// Read the shared XML reference once at startup (single source of truth).
|
|
17
|
+
// In the repo: read from shared/. When installed via npm: read from the
|
|
18
|
+
// local copy created by the prepack script.
|
|
19
|
+
const sharedPath = join(__dirname, "..", "..", "shared", "xml-reference.md");
|
|
20
|
+
const localPath = join(__dirname, "xml-reference.md");
|
|
21
|
+
const xmlReference = readFileSync(
|
|
22
|
+
existsSync(sharedPath) ? sharedPath : localPath,
|
|
23
|
+
"utf-8"
|
|
24
|
+
);
|
|
25
|
+
|
|
14
26
|
/**
|
|
15
27
|
* Opens a URL in the default browser (cross-platform)
|
|
16
28
|
*/
|
|
@@ -121,26 +133,8 @@ const tools =
|
|
|
121
133
|
description:
|
|
122
134
|
"Opens the draw.io editor with a diagram from XML content. " +
|
|
123
135
|
"Use this to view, edit, or create diagrams in draw.io format. " +
|
|
124
|
-
"The XML should be valid draw.io/mxGraph XML format
|
|
125
|
-
|
|
126
|
-
"EDGE GEOMETRY: Every edge mxCell MUST contain a <mxGeometry relative=\"1\" as=\"geometry\" /> child element, even when there are no waypoints. Self-closing edge cells (<mxCell ... edge=\"1\" ... />) are invalid and will not render correctly. " +
|
|
127
|
-
"EDGE ROUTING: Use edgeStyle=orthogonalEdgeStyle for right-angle connectors. " +
|
|
128
|
-
"Space nodes at least 60px apart to avoid overlapping edges. " +
|
|
129
|
-
"Use exitX/exitY/entryX/entryY (0-1) to control which side of a node an edge connects to, spreading connections across different sides. " +
|
|
130
|
-
"Add explicit waypoints via <Array as=\"points\"><mxPoint x=\"...\" y=\"...\"/></Array> inside mxGeometry when edges would overlap. " +
|
|
131
|
-
"ARROWHEAD CLEARANCE: The final straight segment of an edge (between the last bend and the target, or source and first bend) must be long enough to fit the arrowhead (default size 6, configurable via startSize/endSize). If too short, the arrowhead overlaps the bend. Ensure at least 20px of straight segment. The orthogonal auto-router can place bends too close to shapes when nodes are nearly aligned - fix by increasing spacing or adding explicit waypoints. " +
|
|
132
|
-
"CONTAINERS: For architecture diagrams and any diagram with nested elements, use proper parent-child containment (set parent=\"containerId\" on children, use relative coordinates). " +
|
|
133
|
-
"Container types: (1) group style (style=\"group;\") for invisible containers with no connections - includes pointerEvents=0 so child connections are not captured by the container; " +
|
|
134
|
-
"(2) swimlane style (style=\"swimlane;startSize=30;\") for labeled containers with a title bar - use when the container needs visual borders/headers or when the container itself has connections; " +
|
|
135
|
-
"(3) any shape can be a container by adding container=1 to its style, but also add pointerEvents=0 unless the container itself needs to be connectable. " +
|
|
136
|
-
"Always use pointerEvents=0 on container styles that should not capture connections being rewired between children. " +
|
|
137
|
-
"EDGE LABELS: Do NOT wrap edge labels in HTML markup to reduce font size. The default font size for edge labels is already 11px (vs 12px for vertices), so they are already smaller. Just set the value attribute directly. " +
|
|
138
|
-
"LAYOUT: Align nodes to a grid (multiples of 10). Use consistent spacing (e.g., 200px horizontal, 120px vertical between nodes). " +
|
|
139
|
-
"DARK MODE COLORS: To enable dark mode color adaptation, the mxGraphModel element must include adaptiveColors=\"auto\". " +
|
|
140
|
-
"strokeColor, fillColor, and fontColor default to 'default', which renders as black in light theme and white in dark theme. " +
|
|
141
|
-
"Explicit colors (e.g. fillColor=#DAE8FC) specify the light-mode color; the dark-mode color is computed automatically by inverting RGB values and rotating the hue 180 degrees. " +
|
|
142
|
-
"To specify both colors explicitly, use light-dark(lightColor,darkColor) in the style string, e.g. fontColor=light-dark(#7EA6E0,#FF0000). " +
|
|
143
|
-
"See https://www.drawio.com/doc/faq/drawio-style-reference.html for the complete style reference.",
|
|
136
|
+
"The XML should be valid draw.io/mxGraph XML format.\n\n" +
|
|
137
|
+
xmlReference,
|
|
144
138
|
inputSchema:
|
|
145
139
|
{
|
|
146
140
|
type: "object",
|
|
@@ -283,6 +277,25 @@ server.setRequestHandler(CallToolRequestSchema, async (request) =>
|
|
|
283
277
|
};
|
|
284
278
|
}
|
|
285
279
|
|
|
280
|
+
if (typeof inputContent !== "string")
|
|
281
|
+
{
|
|
282
|
+
const actualType = typeof inputContent;
|
|
283
|
+
const preview = JSON.stringify(inputContent).substring(0, 200);
|
|
284
|
+
|
|
285
|
+
return {
|
|
286
|
+
content:
|
|
287
|
+
[
|
|
288
|
+
{
|
|
289
|
+
type: "text",
|
|
290
|
+
text: `Error: content parameter must be a string, but received ${actualType}: ${preview}\n\n` +
|
|
291
|
+
"Common mistake: passing a JSON object or nested structure instead of a plain string. " +
|
|
292
|
+
"Make sure the diagram content (XML, CSV, or Mermaid) is passed directly as a string value.",
|
|
293
|
+
},
|
|
294
|
+
],
|
|
295
|
+
isError: true,
|
|
296
|
+
};
|
|
297
|
+
}
|
|
298
|
+
|
|
286
299
|
content = inputContent;
|
|
287
300
|
|
|
288
301
|
switch (name)
|
|
@@ -0,0 +1,253 @@
|
|
|
1
|
+
# draw.io XML Reference
|
|
2
|
+
|
|
3
|
+
Detailed reference for styles, edge routing, containers, layers, tags, metadata, and dark mode. Consult this when generating draw.io XML diagrams.
|
|
4
|
+
|
|
5
|
+
## General principles
|
|
6
|
+
|
|
7
|
+
- **Use proper draw.io shapes and connectors** — choose the semantically correct shape for each element (e.g., `shape=cylinder3` for databases and tanks, `rhombus` for decisions, `shape=mxgraph.pid2valves.*` for valves in P&IDs). draw.io has extensive shape libraries; prefer domain-appropriate shapes over generic rectangles.
|
|
8
|
+
- **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_shapes`** for 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. **Use `search_shapes`** when 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.
|
|
9
|
+
- **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.
|
|
10
|
+
|
|
11
|
+
## Common styles
|
|
12
|
+
|
|
13
|
+
**Rounded rectangle:**
|
|
14
|
+
```xml
|
|
15
|
+
<mxCell id="2" value="Label" style="rounded=1;whiteSpace=wrap;" vertex="1" parent="1">
|
|
16
|
+
<mxGeometry x="100" y="100" width="120" height="60" as="geometry"/>
|
|
17
|
+
</mxCell>
|
|
18
|
+
```
|
|
19
|
+
|
|
20
|
+
**Diamond (decision):**
|
|
21
|
+
```xml
|
|
22
|
+
<mxCell id="3" value="Condition?" style="rhombus;whiteSpace=wrap;" vertex="1" parent="1">
|
|
23
|
+
<mxGeometry x="100" y="200" width="120" height="80" as="geometry"/>
|
|
24
|
+
</mxCell>
|
|
25
|
+
```
|
|
26
|
+
|
|
27
|
+
**Arrow (edge):**
|
|
28
|
+
```xml
|
|
29
|
+
<mxCell id="4" value="" style="edgeStyle=orthogonalEdgeStyle;" edge="1" source="2" target="3" parent="1">
|
|
30
|
+
<mxGeometry relative="1" as="geometry"/>
|
|
31
|
+
</mxCell>
|
|
32
|
+
```
|
|
33
|
+
|
|
34
|
+
**Labeled arrow:**
|
|
35
|
+
```xml
|
|
36
|
+
<mxCell id="5" value="Yes" style="edgeStyle=orthogonalEdgeStyle;" edge="1" source="3" target="6" parent="1">
|
|
37
|
+
<mxGeometry relative="1" as="geometry"/>
|
|
38
|
+
</mxCell>
|
|
39
|
+
```
|
|
40
|
+
|
|
41
|
+
## Style properties
|
|
42
|
+
|
|
43
|
+
| Property | Values | Use for |
|
|
44
|
+
|----------|--------|---------|
|
|
45
|
+
| `rounded=1` | 0 or 1 | Rounded corners |
|
|
46
|
+
| `whiteSpace=wrap` | wrap | Text wrapping |
|
|
47
|
+
| `fillColor=#dae8fc` | Hex color | Background color |
|
|
48
|
+
| `strokeColor=#6c8ebf` | Hex color | Border color |
|
|
49
|
+
| `fontColor=#333333` | Hex color | Text color |
|
|
50
|
+
| `shape=cylinder3` | shape name | Database cylinders |
|
|
51
|
+
| `shape=mxgraph.flowchart.document` | shape name | Document shapes |
|
|
52
|
+
| `ellipse` | style keyword | Circles/ovals |
|
|
53
|
+
| `rhombus` | style keyword | Diamonds |
|
|
54
|
+
| `edgeStyle=orthogonalEdgeStyle` | style keyword | Right-angle connectors |
|
|
55
|
+
| `edgeStyle=elbowEdgeStyle` | style keyword | Elbow connectors |
|
|
56
|
+
| `dashed=1` | 0 or 1 | Dashed lines |
|
|
57
|
+
| `swimlane` | style keyword | Swimlane containers |
|
|
58
|
+
| `group` | style keyword | Invisible container (pointerEvents=0) |
|
|
59
|
+
| `container=1` | 0 or 1 | Enable container behavior on any shape |
|
|
60
|
+
| `pointerEvents=0` | 0 or 1 | Prevent container from capturing child connections |
|
|
61
|
+
|
|
62
|
+
## Edge routing
|
|
63
|
+
|
|
64
|
+
**CRITICAL: Every edge `mxCell` must contain a `<mxGeometry relative="1" as="geometry" />` child element**, even when there are no waypoints. Self-closing edge cells (e.g. `<mxCell ... edge="1" ... />`) are invalid and will not render correctly. Always use the expanded form:
|
|
65
|
+
```xml
|
|
66
|
+
<mxCell id="e1" edge="1" parent="1" source="a" target="b" style="...">
|
|
67
|
+
<mxGeometry relative="1" as="geometry" />
|
|
68
|
+
</mxCell>
|
|
69
|
+
```
|
|
70
|
+
|
|
71
|
+
draw.io does **not** have built-in collision detection for edges. Plan layout and routing carefully:
|
|
72
|
+
|
|
73
|
+
- Use `edgeStyle=orthogonalEdgeStyle` for right-angle connectors (most common)
|
|
74
|
+
- **Space nodes generously** — at least 60px apart, prefer 200px horizontal / 120px vertical gaps
|
|
75
|
+
- Use `exitX`/`exitY` and `entryX`/`entryY` (values 0-1) to control which side of a node an edge connects to. Spread connections across different sides to prevent overlap
|
|
76
|
+
- **Leave room for arrowheads**: The final straight segment of an edge (between the last bend and the target shape, or between the source shape and the first bend) must be long enough to fit the arrowhead. The default arrow size is 6px (configurable via `startSize`/`endSize` styles). If the final segment is too short, the arrowhead overlaps the bend and looks broken. Ensure at least 20px of straight segment before the target and after the source when placing waypoints or positioning nodes
|
|
77
|
+
- When using `orthogonalEdgeStyle`, the auto-router places bends automatically — if source and target are close together or nearly aligned on one axis, the router may place a bend very close to a shape, leaving no room for the arrow. Fix this by either increasing node spacing or adding explicit waypoints that keep the final segment long enough
|
|
78
|
+
- Add explicit **waypoints** when edges would overlap:
|
|
79
|
+
```xml
|
|
80
|
+
<mxCell id="e1" style="edgeStyle=orthogonalEdgeStyle;" edge="1" parent="1" source="a" target="b">
|
|
81
|
+
<mxGeometry relative="1" as="geometry">
|
|
82
|
+
<Array as="points">
|
|
83
|
+
<mxPoint x="300" y="150"/>
|
|
84
|
+
<mxPoint x="300" y="250"/>
|
|
85
|
+
</Array>
|
|
86
|
+
</mxGeometry>
|
|
87
|
+
</mxCell>
|
|
88
|
+
```
|
|
89
|
+
- Use `rounded=1` on edges for cleaner bends
|
|
90
|
+
- Use `jettySize=auto` for better port spacing on orthogonal edges
|
|
91
|
+
- Align all nodes to a grid (multiples of 10)
|
|
92
|
+
- **Edge labels**: Do NOT wrap edge labels in HTML markup to reduce font size. The default font size for edge labels is already 11px (vs 12px for vertices), so they are already smaller. Just set the `value` attribute directly.
|
|
93
|
+
|
|
94
|
+
## Containers and groups
|
|
95
|
+
|
|
96
|
+
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.
|
|
97
|
+
|
|
98
|
+
### How containment works
|
|
99
|
+
|
|
100
|
+
Set `parent="containerId"` on child cells. Children use **relative coordinates** within the container.
|
|
101
|
+
|
|
102
|
+
### Container types
|
|
103
|
+
|
|
104
|
+
| Type | Style | When to use |
|
|
105
|
+
|------|-------|-------------|
|
|
106
|
+
| **Group** (invisible) | `group;` | No visual border needed, container has no connections. Includes `pointerEvents=0` so child connections are not captured |
|
|
107
|
+
| **Swimlane** (titled) | `swimlane;startSize=30;` | Container needs a visible title bar/header, or the container itself has connections |
|
|
108
|
+
| **Custom container** | Add `container=1;pointerEvents=0;` to any shape style | Any shape acting as a container without its own connections |
|
|
109
|
+
|
|
110
|
+
### Key rules
|
|
111
|
+
|
|
112
|
+
- **Always add `pointerEvents=0;`** to container styles that should not capture connections being rewired between children
|
|
113
|
+
- Only omit `pointerEvents=0` when the container itself needs to be connectable — in that case, use `swimlane` style which handles this correctly (the client area is transparent for mouse events while the header remains connectable)
|
|
114
|
+
- Children must set `parent="containerId"` and use coordinates **relative to the container**
|
|
115
|
+
|
|
116
|
+
### Example: Architecture container with swimlane
|
|
117
|
+
|
|
118
|
+
```xml
|
|
119
|
+
<mxCell id="svc1" value="User Service" style="swimlane;startSize=30;fillColor=#dae8fc;strokeColor=#6c8ebf;" vertex="1" parent="1">
|
|
120
|
+
<mxGeometry x="100" y="100" width="300" height="200" as="geometry"/>
|
|
121
|
+
</mxCell>
|
|
122
|
+
<mxCell id="api1" value="REST API" style="rounded=1;whiteSpace=wrap;" vertex="1" parent="svc1">
|
|
123
|
+
<mxGeometry x="20" y="40" width="120" height="60" as="geometry"/>
|
|
124
|
+
</mxCell>
|
|
125
|
+
<mxCell id="db1" value="Database" style="shape=cylinder3;whiteSpace=wrap;" vertex="1" parent="svc1">
|
|
126
|
+
<mxGeometry x="160" y="40" width="120" height="60" as="geometry"/>
|
|
127
|
+
</mxCell>
|
|
128
|
+
```
|
|
129
|
+
|
|
130
|
+
### Example: Invisible group container
|
|
131
|
+
|
|
132
|
+
```xml
|
|
133
|
+
<mxCell id="grp1" value="" style="group;" vertex="1" parent="1">
|
|
134
|
+
<mxGeometry x="100" y="100" width="300" height="200" as="geometry"/>
|
|
135
|
+
</mxCell>
|
|
136
|
+
<mxCell id="c1" value="Component A" style="rounded=1;whiteSpace=wrap;" vertex="1" parent="grp1">
|
|
137
|
+
<mxGeometry x="10" y="10" width="120" height="60" as="geometry"/>
|
|
138
|
+
</mxCell>
|
|
139
|
+
```
|
|
140
|
+
|
|
141
|
+
## Layers
|
|
142
|
+
|
|
143
|
+
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").
|
|
144
|
+
|
|
145
|
+
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"`:
|
|
146
|
+
|
|
147
|
+
```xml
|
|
148
|
+
<mxGraphModel>
|
|
149
|
+
<root>
|
|
150
|
+
<mxCell id="0"/>
|
|
151
|
+
<mxCell id="1" parent="0"/>
|
|
152
|
+
<mxCell id="2" value="Annotations" parent="0"/>
|
|
153
|
+
<mxCell id="10" value="Server" style="rounded=1;" vertex="1" parent="1">
|
|
154
|
+
<mxGeometry x="100" y="100" width="120" height="60" as="geometry"/>
|
|
155
|
+
</mxCell>
|
|
156
|
+
<mxCell id="20" value="Note: deprecated" style="text;" vertex="1" parent="2">
|
|
157
|
+
<mxGeometry x="100" y="170" width="120" height="30" as="geometry"/>
|
|
158
|
+
</mxCell>
|
|
159
|
+
</root>
|
|
160
|
+
</mxGraphModel>
|
|
161
|
+
```
|
|
162
|
+
|
|
163
|
+
- A layer is an `mxCell` with `parent="0"` and no `vertex` or `edge` attribute
|
|
164
|
+
- Assign shapes to a layer by setting `parent` to the layer's id
|
|
165
|
+
- Later layers render on top of earlier layers (higher z-order)
|
|
166
|
+
- Add `visible="0"` as an attribute on the layer cell to hide it by default
|
|
167
|
+
- Use layers when the diagram has distinct conceptual groupings that viewers may want to toggle independently
|
|
168
|
+
|
|
169
|
+
## Tags
|
|
170
|
+
|
|
171
|
+
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").
|
|
172
|
+
|
|
173
|
+
Tags require wrapping `mxCell` in an `<object>` element. Tags are assigned via the `tags` attribute as a space-separated string:
|
|
174
|
+
|
|
175
|
+
```xml
|
|
176
|
+
<mxGraphModel>
|
|
177
|
+
<root>
|
|
178
|
+
<mxCell id="0"/>
|
|
179
|
+
<mxCell id="1" parent="0"/>
|
|
180
|
+
<object id="2" label="Auth Service" tags="critical v2">
|
|
181
|
+
<mxCell style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
|
|
182
|
+
<mxGeometry x="100" y="100" width="120" height="60" as="geometry"/>
|
|
183
|
+
</mxCell>
|
|
184
|
+
</object>
|
|
185
|
+
<object id="3" label="Legacy API" tags="critical deprecated">
|
|
186
|
+
<mxCell style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
|
|
187
|
+
<mxGeometry x="300" y="100" width="120" height="60" as="geometry"/>
|
|
188
|
+
</mxCell>
|
|
189
|
+
</object>
|
|
190
|
+
</root>
|
|
191
|
+
</mxGraphModel>
|
|
192
|
+
```
|
|
193
|
+
|
|
194
|
+
- Tags require the `<object>` wrapper — a plain `mxCell` cannot have tags
|
|
195
|
+
- The `label` attribute on `<object>` replaces `value` on `mxCell`
|
|
196
|
+
- Tags are space-separated in the `tags` attribute
|
|
197
|
+
- Viewers filter the diagram by selecting tags in the draw.io UI (Edit > Tags)
|
|
198
|
+
- Tags do not affect z-order or structural grouping — they are purely a visibility filter
|
|
199
|
+
|
|
200
|
+
## Metadata and placeholders
|
|
201
|
+
|
|
202
|
+
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.
|
|
203
|
+
|
|
204
|
+
Set `placeholders="1"` on the `<object>` to enable `%propertyName%` substitution in the `label`:
|
|
205
|
+
|
|
206
|
+
```xml
|
|
207
|
+
<mxGraphModel>
|
|
208
|
+
<root>
|
|
209
|
+
<mxCell id="0"/>
|
|
210
|
+
<mxCell id="1" parent="0"/>
|
|
211
|
+
<object id="2" label="<b>%component%</b><br>Owner: %owner%<br>Status: %status%"
|
|
212
|
+
placeholders="1" component="Auth Service" owner="Team Backend" status="Active">
|
|
213
|
+
<mxCell style="rounded=1;whiteSpace=wrap;html=1;" vertex="1" parent="1">
|
|
214
|
+
<mxGeometry x="100" y="100" width="160" height="80" as="geometry"/>
|
|
215
|
+
</mxCell>
|
|
216
|
+
</object>
|
|
217
|
+
</root>
|
|
218
|
+
</mxGraphModel>
|
|
219
|
+
```
|
|
220
|
+
|
|
221
|
+
- Custom properties are plain XML attributes on `<object>` (e.g., `component="Auth Service"`)
|
|
222
|
+
- Set `placeholders="1"` to enable `%key%` substitution in the label and tooltip
|
|
223
|
+
- The label must use `html=1` style when using HTML formatting with placeholders
|
|
224
|
+
- Placeholders resolve by walking up the containment hierarchy: shape attributes first, then parent container, then layer, then root — first match wins
|
|
225
|
+
- Predefined placeholders work without custom properties: `%id%`, `%width%`, `%height%`, `%date%`, `%time%`, `%timestamp%`, `%page%`, `%pagenumber%`, `%pagecount%`, `%filename%`
|
|
226
|
+
- Use `%%` for a literal percent sign in labels
|
|
227
|
+
- Tags, metadata, and placeholders can all be combined on the same `<object>` element
|
|
228
|
+
- Use metadata when shapes represent data records (servers, services, components) and you want to attach structured information beyond the visible label
|
|
229
|
+
|
|
230
|
+
## Dark mode colors
|
|
231
|
+
|
|
232
|
+
draw.io supports automatic dark mode rendering. How colors behave depends on the property:
|
|
233
|
+
|
|
234
|
+
- **`strokeColor`, `fillColor`, `fontColor`** default to `"default"`, which renders as black in light theme and white in dark theme. When no explicit color is set, colors adapt automatically.
|
|
235
|
+
- **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° (via `mxUtils.getInverseColor`).
|
|
236
|
+
- **`light-dark()` function** — To specify both colors explicitly, use `light-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.
|
|
237
|
+
|
|
238
|
+
To enable dark mode color adaptation, the `mxGraphModel` element must include `adaptiveColors="auto"`.
|
|
239
|
+
|
|
240
|
+
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.
|
|
241
|
+
|
|
242
|
+
## Style reference
|
|
243
|
+
|
|
244
|
+
Complete style reference (all shape types, style properties, color palettes, HTML labels, and more): https://github.com/jgraph/drawio-mcp/blob/main/shared/drawio-style-reference.md
|
|
245
|
+
|
|
246
|
+
XML Schema (XSD): https://github.com/jgraph/drawio-mcp/blob/main/shared/mxfile.xsd
|
|
247
|
+
|
|
248
|
+
## CRITICAL: XML well-formedness
|
|
249
|
+
|
|
250
|
+
When generating draw.io XML, the output **must** be well-formed XML:
|
|
251
|
+
- **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.
|
|
252
|
+
- Escape special characters in attribute values: `&`, `<`, `>`, `"`
|
|
253
|
+
- Always use unique `id` values for each `mxCell`
|