UNPKG

@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
# CopilotKit Capabilities (React) 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. ## Setup ```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>; } ``` ## Core Patterns ### Scope to a specific agent ```tsx const caps = useCapabilities("research"); ``` ### Feature-gate tools UI ```tsx const caps = useCapabilities("default"); if (caps === undefined) return <ToolsSkeleton />; if (caps.tools?.supported === false) return null; return <ToolsPanel />; ``` ### Narrow optional fields defensively `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"; ``` ## Common Mistakes ### HIGH — Treating `undefined` as "no capabilities" 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` ### MEDIUM — Non-null assertion on optional fields 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` ### MEDIUM — Expecting deep merge from server-side `capabilities` 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`