Inline Visualizer
Render rich visual content — SVG diagrams, HTML interactive widgets, charts — directly inline in a chat conversation. Output streams token-by-token into a sandboxed iframe. The result feels like a natural extension of the conversation, not an attachment.
How it works
You generate raw HTML or SVG fragments. The client renders them in a sandboxed iframe with the design system's CSS variables injected. No <html>, <head>, or <body> tags — just content fragments.
Two output modes:
- SVG mode: Output starts with
<svg>. Client auto-wraps in a card. Best for static diagrams. - HTML mode: Raw HTML fragment. Best for interactive content (sliders, tabs, charts, controls). You can embed
<svg>elements inside HTML mode.
The client detects which mode based on whether the output starts with <svg.
Before generating any visual
Read the design system reference before your first visual in a conversation:
- Always read first:
references/design-system.md— CSS variables, color ramps, typography, core rules - Then read the relevant module:
- Diagrams (flowcharts, structural, illustrative):
references/diagrams.md - Interactive explainers, comparisons, data records:
references/components.md - Charts (Chart.js, data viz):
references/charts.md
- Diagrams (flowcharts, structural, illustrative):
Read the design system file once per conversation. Read module files as needed for each visual type.
Streaming constraints
Output streams token-by-token into the DOM. This dictates structure:
<style>first (keep under 15 lines — inline styles preferred)- Visible HTML/SVG content next (the user sees it building up)
<script>last (executes only after streaming completes)
Because of streaming:
- No gradients, drop shadows, blur, or glow — they flash during DOM diffs. Use flat fills.
- No
display:noneor hidden content — it streams invisibly. - No
<!-- comments -->or/* comments */— waste tokens, break streaming. - Inline
style="..."preferred over<style>blocks so elements look correct mid-stream. - No tabs or carousels that start hidden — show everything stacked vertically during stream. Post-streaming JS can add interactivity.
Iframe sandbox rules
The visual renders in a sandboxed iframe. These are hard constraints:
- No localStorage / sessionStorage — all state must be in-memory
- No position: fixed — the iframe auto-sizes to content height; fixed elements collapse it
- No external fetches — CSP blocks API calls from inside the widget
- CDN allowlist only:
cdnjs.cloudflare.com,esm.sh,cdn.jsdelivr.net,unpkg.com - No DOCTYPE,
<html>,<head>,<body>— just content fragments - Background is transparent — the host provides the container/card styling
- Load libraries via
<script src="https://cdnjs.cloudflare.com/ajax/libs/...">(UMD globals)
The sendPrompt bridge
A global sendPrompt(text) function is available inside the iframe. It sends a message to the chat as if the user typed it. Use it to make visuals conversational — clicking a node in a diagram can trigger a follow-up explanation.
Use sendPrompt when the user's next step benefits from the model thinking. Handle filtering, sorting, toggling, and calculations in local JS instead.
Choosing the right visual type
Route on the verb, not the noun. The same subject gets different treatment depending on what was asked:
| User says | Type | What to build |
|---|---|---|
| "how does X work" | Illustrative diagram | Spatial metaphor showing the mechanism |
| "what are the components of X" | Structural diagram | Labelled boxes showing containment |
| "walk me through the steps" | Flowchart | Sequential boxes and arrows |
| "compare X vs Y" | Comparison layout | Side-by-side cards with metrics |
| "show me the data" | Chart | Chart.js or inline data viz |
| "explain X" (spatial concept) | Interactive explainer | Sliders, controls, live state |
Default to illustrative for "how does X work" — it's the more ambitious choice. Don't retreat to a flowchart because it feels safer.
Multiple visuals per response
Generate multiple visuals in a single response, interleaved with prose:
- Text block (introduce/explain)
- Visual
- Text block (transition)
- Visual (if needed)
Never stack visuals back-to-back without text between them.
Output wrapper
Wrap visual output in a code fence with the language tag visualizer so the client detects and routes it to the iframe renderer:
```visualizer
<svg width="100%" viewBox="0 0 680 400">
...
</svg>
```
The client strips the fence and injects the content into the iframe with theme variables prepended.
Quick reference
| Rule | Value |
|---|---|
| SVG viewBox width | Always 680px |
| Font sizes | 14px labels, 12px subtitles only |
| Stroke width | 0.5px for borders and edges |
| Max colors per diagram | 2-3 ramps |
| Box subtitle length | ≤5 words |
| Corner radius (SVG) | rx="4" default, rx="8" for emphasis |
| Corner radius (HTML) | var(--border-radius-md) or -lg |
| Min font size | 11px |
| Text weights | 400 regular, 500 bold only |
| Heading sizes | h1=22px, h2=18px, h3=16px |