App instance from @modelcontextprotocol/ext-apps. See the SDK Quickstart if you haven’t.
Let the host background show through
Three settings on your side keep the transparency intact.Don’t paint a body background
Any opaque background on<html> or <body> hides the chat surface behind it. Explicitly set both to transparent:
Declare color-scheme in your document head
Browsers give iframe documents an opaque canvas backdrop (white in light mode, near-black in dark mode) when the iframe’s color-scheme differs from the embedding page. Declaring both schemes opts your document into whichever mode the host is in, so the browser drops the backdrop and makes the CSS light-dark() values in Claude’s tokens resolve correctly:
Request a borderless frame
SetprefersBorder: false in your UI resource’s _meta.ui object so the host doesn’t wrap your widget in its own bordered card. Claude web’s default is already borderless, but other hosts differ, so being explicit keeps your app portable. Register the resource with registerAppResource:
Apply the host’s style variables
Claude passes ahostContext object to your widget during the connect() handshake. The fields relevant to theming are:
| Field | Contents |
|---|---|
theme | "light" or "dark" |
styles.variables | CSS custom properties: --color-background-*, --color-text-*, --color-border-*, --color-ring-*, --font-*, --border-radius-*, --border-width-* |
styles.css.fonts | @font-face rules for Anthropic Sans, served from https://assets.claude.ai |
Read hostContext and listen for changes
The App class exposes the initial context via getHostContext() once connect() resolves, and delivers subsequent updates (such as the user toggling dark mode) through the hostcontextchanged event. Register the listener before you connect so you don’t miss an early update.
The SDK provides three helpers that do the DOM work for you, plus React hooks that wrap them:
applyDocumentTheme(theme)sets<html data-theme>and the rootcolor-scheme, so[data-theme="dark"]selectors andlight-dark()values resolve correctly.applyHostStyleVariables(variables)writes every entry instyles.variablesonto:rootas a CSS custom property.applyHostFonts(fontCss)injects the host’s@font-facerules once.useApp(options)creates and connects theAppinstance for you in React.useHostStyles(app, hostContext)applies all of the above and re-applies onhostcontextchanged.
<meta name="color-scheme"> tag from the previous section even though applyDocumentTheme also sets color-scheme at runtime. The tag covers the first paint before your script runs and prevents an opaque-backdrop flash.
Reference the variables in your CSS
Once the variables are on:root, reference them directly. Provide fallbacks so the widget is still readable when rendered outside a host:
Claude’s token values use CSS
light-dark(), so once applyDocumentTheme has set the root color-scheme, every --color-* variable resolves to the right variant without any [data-theme] selectors on your side.Allow the host font origin in your CSP
ForapplyHostFonts to load the @font-face files, your resource’s _meta.ui.csp allowlist must include https://assets.claude.ai in resourceDomains (shown in the registerAppResource snippet above). resourceDomains also adds the listed origins to script-src and style-src, so keep it to origins you trust to serve executable code; prefer bundling third-party fonts into your widget rather than allowlisting public CDNs.
Related topics
- Design guidelines: Style variables and Visual design for the full variable palette and usage guidance.
- SDK API reference for
App,McpUiHostContext, andMcpUiResourceMeta.