examples/v2/docs/reference/use-frontend-tool.mdx
useFrontendTool is a React hook that dynamically registers tools (functions) that AI agents can invoke in your
application. It enables components to expose interactive capabilities to agents, with optional visual rendering of tool
execution.
The useFrontendTool hook:
import { useFrontendTool } from "@copilotkit/react-core";
import { z } from "zod";
function SearchComponent() {
useFrontendTool({
name: "searchProducts",
description: "Search for products in the catalog",
parameters: z.object({
query: z.string(),
category: z.string().optional(),
}),
handler: async ({ query, category }) => {
const results = await searchAPI(query, category);
return results;
},
});
return <div>Rest of your component...</div>;
}
The hook accepts:
ReactFrontendTool object describing the toolstring (required)
A unique identifier for the tool. This is the name agents will use to request this tool's execution.
useFrontendTool({
name: "calculateTotal",
// ...
});
string (optional)
A description that helps agents understand when and how to use this tool. This is sent to the LLM to guide its decision-making.
useFrontendTool({
name: "fetchUserData",
description:
"Retrieve detailed user profile information including preferences and history",
// ...
});
z.ZodType<T> (optional)
A Zod schema defining the tool's input parameters. Provides type safety and automatic validation.
import { z } from "zod";
useFrontendTool({
name: "updateSettings",
parameters: z.object({
theme: z.enum(["light", "dark", "auto"]),
language: z.string(),
notifications: z.boolean(),
}),
handler: async ({ theme, language, notifications }) => {
// settings is fully typed based on the schema
await updateUserSettings({ theme, language, notifications });
return "Settings updated successfully";
},
});
(args: T, toolCall: ToolCall) => Promise<unknown> (optional)
The async function executed when the agent invokes this tool. Receives validated arguments and metadata about the invocation.
useFrontendTool({
name: "addToCart",
parameters: z.object({
productId: z.string(),
quantity: z.number().min(1),
}),
handler: async ({ productId, quantity }, toolCall) => {
console.log(`Tool called by agent at ${toolCall.id}`);
const result = await cartService.addItem(productId, quantity);
return {
success: true,
cartTotal: result.total,
};
},
});
(props: { name: string; args: T; result?: unknown; status: ToolCallStatus }) => React.ReactNode (optional)
A React component that renders visual feedback when the tool is executed. This appears in the chat interface to show tool execution progress and results.
import { ToolCallStatus } from "@copilotkit/core";
useFrontendTool({
name: "generateChart",
parameters: z.object({
data: z.array(z.number()),
type: z.enum(["bar", "line", "pie"]),
}),
handler: async ({ data, type }) => {
return generateChartData(data, type);
},
render: ({ name, args, result, status }) => (
<div className="tool-execution">
<h3>Generating {args.type} chart...</h3>
{status === ToolCallStatus.InProgress && <Spinner />}
{status === ToolCallStatus.Complete && result && (
<ChartDisplay data={result} />
)}
</div>
),
});
boolean (optional, default: true)
Controls whether the agent should automatically continue after this tool completes. Set to false for final actions.
useFrontendTool({
name: "submitForm",
handler: async (formData) => {
await submitToServer(formData);
return "Form submitted successfully";
},
followUp: false, // Don't continue after submission
});
string (optional)
Restricts this tool to a specific agent. Only the specified agent can invoke this tool.
useFrontendTool({
name: "adminAction",
agentId: "admin-assistant",
handler: async (args) => {
return await performAdminAction(args);
},
});
ReadonlyArray<unknown> (optional)
Additional dependencies that should trigger re-registration of the tool. By default, the hook only depends on the tool name and CopilotKit instance to avoid re-register loops from object identity changes. Pass a dependency array as the second argument when the tool's configuration is derived from changing props or state:
function PriceTool({ currency }: { currency: string }) {
useFrontendTool(
{
name: "convertPrice",
handler: async (args) => convertPrice(args.amount, currency),
},
[currency],
);
return null;
}
Tools are automatically registered when the component mounts:
function DynamicTool() {
// Tool is registered when this component mounts
useFrontendTool({
name: "dynamicAction",
handler: async () => "Action performed",
});
return <div>Tool available</div>;
}
// Usage
function App() {
const [showTool, setShowTool] = useState(false);
return (
<div>
<button onClick={() => setShowTool(!showTool)}>Toggle Tool</button>
{showTool && <DynamicTool />}
</div>
);
}
Tools are automatically removed when components unmount, but their renderers persist to maintain chat history:
function TemporaryTool() {
useFrontendTool({
name: "tempAction",
handler: async () => "Temporary action",
render: ({ status }) => <div>Temp tool: {status}</div>,
});
// When this component unmounts:
// - The tool handler is removed (agent can't call it anymore)
// - The renderer persists (previous executions still visible in chat)
return null;
}
If a tool with the same name already exists, it will be overridden with a console warning:
// First registration
useFrontendTool({
name: "search",
handler: async () => "Search v1",
});
// Second registration (overrides first)
useFrontendTool({
name: "search", // Same name - will override with warning
handler: async () => "Search v2",
});