@copilotkit/react-core
Version:
<img src="https://github.com/user-attachments/assets/0a6b64d9-e193-4940-a3f6-60334ac34084" alt="banner" style="border-radius: 12px; border: 2px solid #d6d4fa;" />
139 lines (99 loc) • 3.38 kB
Markdown
This skill builds on `copilotkit/agent-access`. `useCapabilities` internally
calls `useAgent` and reads the `capabilities` field populated from the
runtime `/info` response.
`AgentCapabilities` is from `@ag-ui/core`. The hook is synchronous — there
is no loading state, but the value is `undefined` until the handshake
completes.
```tsx
"use client";
import { useCapabilities } from "@copilotkit/react-core/v2";
export function VoiceButton() {
const caps = useCapabilities(); // defaults to DEFAULT_AGENT_ID
// Handshake pending — show a placeholder
if (caps === undefined) return <div className="skeleton h-8 w-8" />;
// Handshake complete — feature-gate
if (!caps.transcription) return null;
return <button>Record</button>;
}
```
```tsx
const caps = useCapabilities("research");
```
```tsx
const caps = useCapabilities("default");
if (caps === undefined) return <ToolsSkeleton />;
if (caps.tools?.supported === false) return null;
return <ToolsPanel />;
```
`AgentCapabilities` is a partial declaration — fields may be absent when
the agent opts not to declare them.
```tsx
const caps = useCapabilities();
const maxTokens = caps?.maxOutputTokens ?? "unknown";
```
Wrong:
```tsx
function VoiceButton() {
const caps = useCapabilities();
if (!caps?.transcription) return null; // hides button forever while handshake pending
return <button>Record</button>;
}
```
Correct:
```tsx
function VoiceButton() {
const caps = useCapabilities();
if (caps === undefined) return <div className="skeleton h-8 w-8" />;
if (!caps.transcription) return null;
return <button>Record</button>;
}
```
`useCapabilities` returns `undefined` until the runtime `/info` handshake
completes. Treating `undefined` the same as `{ transcription: false }`
hides features that should be visible post-handshake.
Source: `packages/react-core/src/v2/hooks/use-capabilities.tsx:7-9`
Wrong:
```tsx
const caps = useCapabilities();
return <div>Max tokens: {caps!.maxOutputTokens}</div>;
// Crashes if agent didn't declare capabilities, or didn't declare maxOutputTokens.
```
Correct:
```tsx
const caps = useCapabilities();
return <div>Max tokens: {caps?.maxOutputTokens ?? "unknown"}</div>;
```
`AgentCapabilities` is a partial declaration. Agents opt in to each
capability, so every field is optional. Narrow before deref.
Source: `packages/react-core/src/v2/hooks/use-capabilities.tsx:20-22`
Wrong:
```ts
// Server:
new BuiltInAgent({
// ...
capabilities: { tools: { supported: true } },
});
// Client expects caps.tools.clientProvided to still be set by the default
```
Correct:
```ts
// Server — provide full category:
new BuiltInAgent({
// ...
capabilities: { tools: { supported: true, clientProvided: true } },
});
```
BuiltInAgent shallow-merges capabilities at the category level — providing
`tools: {...}` replaces the whole category, not just the specified fields.
The client then sees exactly what was declared.
Source: `packages/runtime/src/agent/index.ts:821-829,883-887`