skills/react-core/references/rendering-tool-calls.md
This skill builds on copilotkit/provider-setup and
copilotkit/client-side-tools.
Four hooks, distinct roles:
| Hook | Role |
|---|---|
useRenderTool | Primary registration hook for a named tool's progress/result UI |
useComponent | Register a NEW render-only tool (agent calls it just to render) |
useDefaultRenderTool | Sanctioned wildcard fallback for tools without a dedicated render |
useRenderToolCall | Resolver — returns a function. For custom chat surfaces only |
Status is camelCase: "inProgress" | "executing" | "complete". The
RenderToolProps discriminated union narrows parameters per state.
Before writing raw JSX, check the consumer's package.json for shadcn /
MUI / Chakra / Ant / Mantine and reuse those primitives.
"use client";
import { useRenderTool } from "@copilotkit/react-core/v2";
import { z } from "zod";
import { Card, CardContent } from "@/components/ui/card";
import { Skeleton } from "@/components/ui/skeleton";
export function SearchRenderer() {
useRenderTool({
name: "searchDocs",
parameters: z.object({ query: z.string() }),
render: ({ status, parameters, result }) => {
if (status === "inProgress") return <Skeleton className="h-16 w-full" />;
if (status === "executing") {
return (
<Card>
<CardContent>Searching "{parameters.query}"…</CardContent>
</Card>
);
}
return (
<Card>
<CardContent>{result}</CardContent>
</Card>
);
},
});
return null;
}
import { useDefaultRenderTool } from "@copilotkit/react-core/v2";
useDefaultRenderTool(); // renders the built-in expandable tool-call card
import { useDefaultRenderTool } from "@copilotkit/react-core/v2";
useDefaultRenderTool({
render: ({ name, status, parameters, result }) => {
// parameters is unknown — narrow by tool name
if (name === "search") {
const args = parameters as { q: string };
return <SearchCard q={args.q} status={status} result={result} />;
}
return <GenericCard name={name} status={status} />;
},
});
import { useComponent } from "@copilotkit/react-core/v2";
import { z } from "zod";
useComponent({
name: "productCard",
parameters: z.object({ productId: z.string() }),
render: ({ productId }) => <ProductCard id={productId} />,
});
// `useComponent` registers a NEW tool called "productCard".
// The agent calls it to render; there is no handler to run.
useRenderToolCall is for building your own message list, NOT for
registering renderers.
import { useRenderToolCall } from "@copilotkit/react-core/v2";
import { useAgent } from "@copilotkit/react-core/v2";
export function CustomToolList() {
const { agent } = useAgent({ agentId: "default" });
const renderToolCall = useRenderToolCall();
const toolCalls = agent.messages.flatMap((m) =>
"toolCalls" in m ? (m.toolCalls ?? []) : [],
);
return (
<>
{toolCalls.map((tc) => (
<div key={tc.id}>{renderToolCall({ toolCall: tc })}</div>
))}
</>
);
}
useRenderToolCall for registrationWrong:
useRenderToolCall({
name: "search",
args: z.object({ q: z.string() }),
render: ({ status, args }) => <Card>…</Card>,
});
Correct:
useRenderTool({
name: "search",
parameters: z.object({ q: z.string() }),
render: ({ status, parameters }) => <Card>…</Card>,
});
useRenderToolCall takes no arguments — it returns a resolver function for
custom chat surfaces. Passing config to it does nothing. useRenderTool is
the registration hook.
Source: packages/react-core/src/v2/hooks/index.ts:2,7;
packages/react-core/src/v2/hooks/use-render-tool.tsx:37-40
"in-progress" statusWrong:
render: ({ status, parameters, result }) => {
if (status === "in-progress") return <Spinner />;
if (status === "executing") return <RunningCard args={parameters} />;
return <ResultCard result={result} />;
};
Correct:
render: ({ status, parameters, result }) => {
if (status === "inProgress") return <Spinner />;
if (status === "executing") return <RunningCard args={parameters} />;
return <ResultCard result={result} />;
};
Real status values are camelCase: "inProgress" | "executing" | "complete".
Hyphenated branches never match — users see no progress UI and the fallback
path fires.
Source: packages/react-core/src/v2/hooks/use-render-tool.tsx:8-35
Wrong:
useRenderTool({
name: "search",
parameters: z.object({ q: z.string() }),
render: () => <div className="my-badge">…</div>,
});
Correct:
import { Badge } from "@/components/ui/badge";
useRenderTool({
name: "search",
parameters: z.object({ q: z.string() }),
render: () => <Badge variant="secondary">…</Badge>,
});
Check consumer package.json for shadcn / MUI / Chakra / Ant / Mantine
first. Raw JSX ignores their design system.
Source: maintainer interview (Phase 2c)
Partial<T> during inProgressWrong:
render: ({ status, parameters }) => (
<span>{parameters.user.id.toUpperCase()}</span>
);
// `parameters` is Partial<T> during inProgress — `parameters.user` may be undefined.
Correct:
render: ({ status, parameters }) =>
status === "inProgress" ? (
<Skeleton />
) : (
<span>{parameters.user.id.toUpperCase()}</span>
);
During streaming, RenderToolInProgressProps has
parameters: Partial<InferSchemaOutput<S>>. Fields are undefined until
the stream completes. Narrow with status === "inProgress" first.
Source: packages/react-core/src/v2/hooks/use-render-tool.tsx:8-14
useComponent to decorate an existing toolWrong:
useFrontendTool({ name: "search", parameters, handler });
useComponent({
name: "search", // creates a SECOND tool named "search" — collision
parameters: z.object({ q: z.string() }),
render: ({ q }) => <SearchCard q={q} />,
});
Correct:
useFrontendTool({ name: "search", parameters, handler });
useRenderTool({
name: "search",
parameters: z.object({ q: z.string() }),
render: ({ status, parameters, result }) => {
if (status === "inProgress") return <Skeleton />;
if (status === "executing") return <div>Searching {parameters.q}…</div>;
return <div>{result}</div>;
},
});
// useComponent is only for render-only tools the agent invokes:
useComponent({
name: "productCard",
parameters: z.object({ productId: z.string() }),
render: ({ productId }) => <ProductCard id={productId} />,
});
useComponent synthesizes a NEW tool whose only job is to render —
description is auto-prefixed with "Use this tool to display the …
component". It does NOT decorate an existing tool. The misleading name
trap: agents read "useComponent" as "register a component for this tool"
and end up with two tools colliding on the same name.
Source: packages/react-core/src/v2/hooks/use-component.tsx:59-88
useRenderTool({ name: "*" }) instead of useDefaultRenderToolWrong:
useRenderTool({
name: "*",
render: ({ parameters }) => <pre>{JSON.stringify(parameters)}</pre>,
});
Correct:
// Use the built-in default card:
useDefaultRenderTool();
// Or customize, with the correct DefaultRenderProps typing (parameters: unknown):
useDefaultRenderTool({
render: ({ name, status, parameters, result }) => {
if (name === "search") {
const args = parameters as { q: string };
return <SearchCard q={args.q} status={status} />;
}
return <GenericCard name={name} status={status} />;
},
});
The sanctioned wildcard API is useDefaultRenderTool. It wraps
useRenderTool({ name: "*" }) with the correct DefaultRenderProps
typing (parameters: unknown) and provides a built-in default card when
no render is passed. Hand-rolling loses the default card and invites
the untyped-args footgun.
Source: packages/react-core/src/v2/hooks/use-default-render-tool.tsx:15-64