loading…
Search for a command to run...
loading…
A React single-page application served as an MCP server that renders different UI pages based on the invoked tool name from the host context. It enables embeddi
A React single-page application served as an MCP server that renders different UI pages based on the invoked tool name from the host context. It enables embedding interactive React components directly within MCP clients like Claude Desktop through tool calls.
A minimal React single-page app that is bundled into one HTML blob and served by an MCP Apps server. The SPA doesn't use URL path routing — it picks which page to render by reading the current tool name from the MCP host context.
vite-plugin-singlefile.@modelcontextprotocol/ext-apps/server. Registers tools (show-home, show-counter, show-profile) that all point to the same ui:// resource.hostContext and toolResult, so pages can be exercised without an MCP host.Requires Node.js ≥ 20 and pnpm ≥ 9.
pnpm install
pnpm run dev:ui
Open the printed URL (e.g. http://localhost:5173). The SPA detects that there's no MCP host and shows a route picker so you can preview each page.
pnpm run dev:playground
Opens on http://localhost:5174. The playground reuses the SPA's real route renderer (RouteRenderer from packages/ui/src/Router.tsx) and wraps it in a mock chat UI. The header has two JSON textareas — one for the hostContext (drives which page renders via toolInfo.tool.name) and one for the toolResult passed to the page. Preset buttons load ready-made context+result pairs for each registered tool; edits to either textarea re-render the assistant's tool-output bubble live.
pnpm run build # builds packages/ui → dist/index.html, then packages/mcp
pnpm run serve:mcp # starts Streamable HTTP server on http://localhost:3001/mcp
For stdio transport:
pnpm --filter @react-mcp-spa/mcp run serve:stdio
Some MCP clients (including hosted ones) can't reach localhost. Use cloudflared to open a quick tunnel that assigns a public HTTPS URL.
Install cloudflared via your OS package manager:
# macOS (Homebrew)
brew install cloudflared
# Linux (Debian/Ubuntu)
# See https://pkg.cloudflare.com/ for the apt repo, or grab the .deb from
# https://github.com/cloudflare/cloudflared/releases
# Windows (winget)
winget install --id Cloudflare.cloudflared
Start the server locally:
pnpm run serve:mcp
In another terminal, run:
cloudflared tunnel --url http://localhost:3001
# or: pnpm run tunnel
cloudflared prints a URL like https://<random>.trycloudflare.com. Append /mcp and use it as your MCP server URL:
https://<random>.trycloudflare.com/mcp
The tunnel stays up until you kill cloudflared. For a stable hostname, configure a named Cloudflare tunnel instead of a quick tunnel.
.mcpb)The repo ships a packer that produces an installable MCP Bundle for Claude Desktop. The extension runs the server over stdio — no tunnel required.
pnpm run pack:mcpb
This builds both packages, stages the compiled server + UI HTML + prod node_modules under build/mcpb-staging/, and invokes mcpb pack to produce:
build/react-mcp-spa.mcpb
To install it, double-click the .mcpb in Finder/Explorer (or drag it onto Claude Desktop). Claude validates the manifest and registers the server; after installation the three tools (show-home, show-counter, show-profile) are available and each renders its page as an inline React UI.
The bundle manifest lives at mcpb/manifest.json — bump version there (and in packages/mcp/package.json) when cutting a new extension release.
The SPA never reads window.location. Instead, packages/ui/src/App.tsx uses useApp() from @modelcontextprotocol/ext-apps/react and:
app.getHostContext().toolInfo.tool.name to know which tool the host invoked — this is the "route".app.ontoolresult to receive the CallToolResult from the server and pass it to the page as data.app.onhostcontextchanged so theme / safe-area / locale updates re-render correctly.Adding a new page is a two-step change:
_meta.ui.resourceUri pointing at the shared ui://react-mcp-spa/app.html resource.renderRoute() inside packages/ui/src/App.tsx.Добавь это в claude_desktop_config.json и перезапусти Claude Desktop.
{
"mcpServers": {
"react-mcp-spa": {
"command": "npx",
"args": []
}
}
}