dev-docs/architecture/setup-react.md
This guide shows how to set up CopilotKit in a React app — from minimal to fully configured.
graph LR
subgraph Your React App
Provider["<b>CopilotKitProvider</b>
<i>Wraps your app</i>"]
Chat["<b>CopilotChat</b>
<i>Chat UI component</i>"]
Hook1["useFrontendTool()"]
Hook2["useAgentContext()"]
Hook3["useAgent()"]
end
subgraph Under the Hood
Core["CopilotKitCore
<i>Orchestrator</i>"]
Proxy["ProxiedAgent
<i>HTTP client</i>"]
end
subgraph Your Server
Runtime["CopilotRuntime
<i>Express / Hono</i>"]
end
Provider -->|creates| Core
Chat -->|uses| Hook3
Hook1 -->|registers tool in| Core
Hook2 -->|registers context in| Core
Hook3 -->|gets agent from| Core
Core -->|creates| Proxy
Proxy -->|HTTP POST + SSE| Runtime
npm install @copilotkit/react-core @copilotkit/react-ui
// app.tsx
import { CopilotKit } from "@copilotkit/react-core";
import "@copilotkit/react-ui/styles.css";
export default function App() {
return (
<CopilotKit runtimeUrl="/api/copilotkit">
<YourApp />
</CopilotKit>
);
}
// components/chat.tsx
import { CopilotPopup } from "@copilotkit/react-ui";
export function ChatWidget() {
return (
<CopilotPopup
labels={{ title: "AI Assistant", initial: "How can I help?" }}
/>
);
}
That's it — you now have a working AI chat. The provider connects to your runtime, fetches available agents, and the popup gives users a chat interface.
sequenceDiagram
participant App as React App
participant Provider as CopilotKit Provider
participant Runtime as Your Server
App->>Provider: Mounts with runtimeUrl
Provider->>Runtime: GET /info
Runtime-->>Provider: Available agents
Note over Provider: Ready for chat
If you're building new features and want the V2 API directly:
npm install @copilotkit/react
import { CopilotKitProvider, CopilotChat } from "@copilotkit/react-core";
export default function App() {
return (
<CopilotKitProvider runtimeUrl="/api/copilotkit">
<CopilotChat />
</CopilotKitProvider>
);
}
V1's
<CopilotKit>wraps V2's<CopilotKitProvider>under the hood, so both work the same way.
Tools are functions the AI agent can call. They run in the browser.
import { useFrontendTool } from "@copilotkit/react-core";
// or: import { useCopilotAction } from "@copilotkit/react-core"; (V1 equivalent)
import { z } from "zod";
function ProductPage({ products }) {
// The agent can now call "addToCart" during a conversation
useFrontendTool({
name: "addToCart",
description: "Add a product to the user's shopping cart",
parameters: z.object({
productId: z.string().describe("The product ID to add"),
quantity: z.number().default(1).describe("How many to add"),
}),
handler: async ({ productId, quantity }) => {
await cartApi.add(productId, quantity);
return `Added ${quantity} item(s) to cart`;
},
});
return <div></div>;
}
sequenceDiagram
participant Agent as AI Agent
participant Runtime as CopilotRuntime
participant Core as CopilotKitCore
participant Tool as addToCart handler
Agent->>Runtime: TOOL_CALL_START { name: "addToCart" }
Agent->>Runtime: TOOL_CALL_ARGS { productId: "abc", quantity: 2 }
Runtime->>Core: SSE events
Core->>Tool: Execute handler({ productId: "abc", quantity: 2 })
Tool-->>Core: "Added 2 item(s) to cart"
Core->>Runtime: TOOL_CALL_RESULT
Runtime->>Agent: Agent continues with result
Context tells the agent about what the user currently sees.
import { useAgentContext } from "@copilotkit/react-core";
// or: import { useCopilotReadable } from "@copilotkit/react-core"; (V1 equivalent)
function Dashboard({ user, metrics }) {
// The agent now knows about the current user and their metrics
useAgentContext("Current user and dashboard metrics", {
user: { name: user.name, role: user.role },
metrics: { revenue: metrics.revenue, activeUsers: metrics.activeUsers },
});
return <div></div>;
}
Show custom UI while a tool is being called:
import { useRenderToolCall } from "@copilotkit/react-core";
import { z } from "zod";
function App() {
useRenderToolCall({
name: "searchProducts",
args: z.object({ query: z.string() }),
render: ({ args, status, result }) => {
if (status === "in-progress") {
return <div>Searching for "{args.query}"...</div>;
}
if (status === "executing") {
return <Spinner>Running search...</Spinner>;
}
// status === "complete"
return <div>Found results: {result}</div>;
},
});
}
graph LR
subgraph Tool Call Lifecycle
IP["in-progress
<i>Args streaming in</i>"]
EX["executing
<i>Handler running</i>"]
CO["complete
<i>Result available</i>"]
IP --> EX --> CO
end
Require user approval before a tool executes:
import { useHumanInTheLoop } from "@copilotkit/react-core";
import { z } from "zod";
function App() {
useHumanInTheLoop({
name: "deleteAccount",
description: "Permanently delete a user account",
parameters: z.object({ userId: z.string() }),
render: ({ args, status, respond }) => {
if (status === "executing") {
return (
<div>
<p>Delete account {args.userId}?</p>
<button onClick={() => respond("approved")}>Approve</button>
<button onClick={() => respond("denied")}>Deny</button>
</div>
);
}
if (status === "complete") {
return <div>Action completed</div>;
}
return <div>Preparing...</div>;
},
});
}
Auto-generate prompt suggestions for users:
import { useConfigureSuggestions } from "@copilotkit/react-core";
function App() {
useConfigureSuggestions({
instructions: "Suggest questions about the user's dashboard data",
minSuggestions: 2,
maxSuggestions: 4,
available: "always", // "before-first-message" | "after-first-message" | "always" | "disabled"
});
}
<CopilotKitProvider
// Required
runtimeUrl="/api/copilotkit"
// Authentication
headers={{ Authorization: "Bearer token" }}
credentials="include" // Forward cookies
publicApiKey="ck_..." // CopilotKit Cloud key
// Custom properties forwarded to agents
properties={{ userId: "123", plan: "pro" }}
// Tools & rendering (can also use hooks instead)
frontendTools={
[
/* ... */
]
}
renderToolCalls={
[
/* ... */
]
}
renderActivityMessages={
[
/* ... */
]
}
renderCustomMessages={
[
/* ... */
]
}
humanInTheLoop={
[
/* ... */
]
}
// Dev tools
showDevConsole="auto" // true | false | "auto"
// Advanced: local agents for development
agents__unsafe_dev_only={{ test: myTestAgent }}
/>
graph TB
subgraph "CopilotKitProvider Props"
direction TB
subgraph Required
URL["runtimeUrl"]
end
subgraph "Optional: Auth"
H["headers"]
C["credentials"]
K["publicApiKey"]
end
subgraph "Optional: Tools & Rendering"
FT["frontendTools"]
RTC["renderToolCalls"]
RAM["renderActivityMessages"]
RCM["renderCustomMessages"]
HIL["humanInTheLoop"]
end
subgraph "Optional: Other"
P["properties"]
DC["showDevConsole"]
AG["agents__unsafe_dev_only"]
end
end
import {
CopilotChat, // Inline chat, fills its container
CopilotPopup, // Floating popup button + chat
CopilotSidebar, // Side panel
CopilotPanel, // Inline panel
} from "@copilotkit/react-ui";
// All accept the same core props:
<CopilotChat
agentId="research" // Which agent to talk to (default: "default")
labels={{
title: "Research Assistant",
initial: "What would you like to research?",
placeholder: "Ask me anything...",
}}
/>;
import { CopilotKit } from "@copilotkit/react-core";
import { CopilotSidebar } from "@copilotkit/react-ui";
import "@copilotkit/react-ui/styles.css";
import { z } from "zod";
export default function App() {
return (
<CopilotKit
runtimeUrl="/api/copilotkit"
headers={{ Authorization: `Bearer ${getToken()}` }}
>
<CopilotSidebar labels={{ title: "Shopping Assistant" }}>
<ProductCatalog />
</CopilotSidebar>
</CopilotKit>
);
}
function ProductCatalog() {
const [products] = useProducts();
const [cart, setCart] = useCart();
// Context: tell the agent what the user sees
useAgentContext("Product catalog the user is browsing", {
products: products.map((p) => ({ id: p.id, name: p.name, price: p.price })),
cartTotal: cart.total,
cartItems: cart.items.length,
});
// Tool: agent can add items to cart
useFrontendTool({
name: "addToCart",
description: "Add a product to the shopping cart",
parameters: z.object({
productId: z.string(),
quantity: z.number().default(1),
}),
handler: async ({ productId, quantity }) => {
setCart((prev) => addItem(prev, productId, quantity));
return "Added to cart";
},
});
// Tool: agent can search products
useFrontendTool({
name: "searchProducts",
description: "Search for products by name or category",
parameters: z.object({ query: z.string() }),
handler: async ({ query }) => {
const results = products.filter((p) =>
p.name.toLowerCase().includes(query.toLowerCase()),
);
return JSON.stringify(
results.map((p) => ({ id: p.id, name: p.name, price: p.price })),
);
},
});
// Suggestions
useConfigureSuggestions({
instructions:
"Suggest shopping-related questions based on the product catalog",
maxSuggestions: 3,
available: "always",
});
return <div></div>;
}