@copilotkit/react-core 1.57.3 → 1.58.0-canary.thread-id-propagation
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/dist/{copilotkit-CtXcs1ea.cjs → copilotkit-B4ouY7qC.cjs} +14 -3
- package/dist/copilotkit-B4ouY7qC.cjs.map +1 -0
- package/dist/copilotkit-BK9CVq9A.d.cts.map +1 -1
- package/dist/{copilotkit-CC8DjOiC.mjs → copilotkit-L4mM_JqG.mjs} +14 -3
- package/dist/copilotkit-L4mM_JqG.mjs.map +1 -0
- package/dist/copilotkit-WlmeVijs.d.mts.map +1 -1
- package/dist/index.cjs +3 -77
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.cts +2 -2
- package/dist/index.d.mts +2 -2
- package/dist/index.mjs +3 -77
- package/dist/index.mjs.map +1 -1
- package/dist/index.umd.js +15 -78
- package/dist/index.umd.js.map +1 -1
- package/dist/v2/headless.cjs +11 -0
- package/dist/v2/headless.cjs.map +1 -1
- package/dist/v2/headless.d.cts.map +1 -1
- package/dist/v2/headless.d.mts.map +1 -1
- package/dist/v2/headless.mjs +11 -0
- package/dist/v2/headless.mjs.map +1 -1
- package/dist/v2/index.cjs +1 -1
- package/dist/v2/index.mjs +1 -1
- package/dist/v2/index.umd.js +13 -2
- package/dist/v2/index.umd.js.map +1 -1
- package/package.json +12 -13
- package/skills/react-core/SKILL.md +108 -0
- package/skills/react-core/references/agent-access.md +288 -0
- package/skills/react-core/references/attachments.md +291 -0
- package/skills/react-core/references/capabilities.md +138 -0
- package/skills/react-core/references/chat-components.md +221 -0
- package/skills/react-core/references/client-side-tools.md +358 -0
- package/skills/react-core/references/custom-message-renderers.md +226 -0
- package/skills/react-core/references/debug-mode.md +153 -0
- package/skills/react-core/references/human-in-the-loop.md +312 -0
- package/skills/react-core/references/provider-setup.md +326 -0
- package/skills/react-core/references/rendering-activity-messages.md +207 -0
- package/skills/react-core/references/rendering-tool-calls.md +319 -0
- package/skills/react-core/references/suggestions.md +211 -0
- package/skills/react-core/references/switching-agents-recipes.md +160 -0
- package/skills/react-core/references/switching-agents.md +231 -0
- package/skills/react-core/references/threads.md +226 -0
- package/.attw.json +0 -3
- package/CHANGELOG.md +0 -5043
- package/dist/copilotkit-CC8DjOiC.mjs.map +0 -1
- package/dist/copilotkit-CtXcs1ea.cjs.map +0 -1
- package/scripts/scope-preflight.mjs +0 -100
- package/src/components/CopilotListeners.tsx +0 -137
- package/src/components/__tests__/CopilotListeners.test.tsx +0 -38
- package/src/components/copilot-provider/__tests__/copilot-messages-key.test.tsx +0 -92
- package/src/components/copilot-provider/__tests__/copilotkit-error.test.tsx +0 -77
- package/src/components/copilot-provider/__tests__/error-visibility-prod.test.tsx +0 -70
- package/src/components/copilot-provider/__tests__/v1-explicit-threadid-bridge.test.tsx +0 -107
- package/src/components/copilot-provider/copilot-messages.tsx +0 -314
- package/src/components/copilot-provider/copilotkit-props.tsx +0 -214
- package/src/components/copilot-provider/copilotkit.tsx +0 -853
- package/src/components/copilot-provider/index.ts +0 -3
- package/src/components/dev-console/console-trigger.tsx +0 -283
- package/src/components/dev-console/developer-console-modal.tsx +0 -1016
- package/src/components/dev-console/icons.tsx +0 -106
- package/src/components/error-boundary/error-boundary.tsx +0 -99
- package/src/components/error-boundary/error-utils.tsx +0 -105
- package/src/components/index.ts +0 -1
- package/src/components/toast/exclamation-mark-icon.tsx +0 -27
- package/src/components/toast/toast-provider.tsx +0 -448
- package/src/components/usage-banner.tsx +0 -266
- package/src/context/__tests__/threads-context.test.tsx +0 -141
- package/src/context/coagent-state-renders-context.tsx +0 -89
- package/src/context/copilot-context.tsx +0 -365
- package/src/context/copilot-messages-context.tsx +0 -35
- package/src/context/index.ts +0 -22
- package/src/context/threads-context.tsx +0 -69
- package/src/hooks/__tests__/use-coagent-config.test.ts +0 -352
- package/src/hooks/__tests__/use-coagent-state-render-bridge.helpers.test.ts +0 -107
- package/src/hooks/__tests__/use-coagent-state-render.e2e.test.tsx +0 -1209
- package/src/hooks/__tests__/use-coagent-state-render.test.tsx +0 -356
- package/src/hooks/__tests__/use-copilot-chat-internal-connect.test.tsx +0 -241
- package/src/hooks/__tests__/use-frontend-tool-available.test.tsx +0 -72
- package/src/hooks/__tests__/use-frontend-tool-remount.e2e.test.tsx +0 -102
- package/src/hooks/index.ts +0 -33
- package/src/hooks/use-agent-nodename.ts +0 -33
- package/src/hooks/use-coagent-state-render-bridge.helpers.ts +0 -345
- package/src/hooks/use-coagent-state-render-bridge.tsx +0 -222
- package/src/hooks/use-coagent-state-render-registry.ts +0 -230
- package/src/hooks/use-coagent-state-render.ts +0 -163
- package/src/hooks/use-coagent.ts +0 -377
- package/src/hooks/use-configure-chat-suggestions.tsx +0 -96
- package/src/hooks/use-copilot-action.ts +0 -245
- package/src/hooks/use-copilot-additional-instructions.ts +0 -98
- package/src/hooks/use-copilot-authenticated-action.ts +0 -73
- package/src/hooks/use-copilot-chat-headless_c.ts +0 -264
- package/src/hooks/use-copilot-chat-suggestions.tsx +0 -134
- package/src/hooks/use-copilot-chat.ts +0 -132
- package/src/hooks/use-copilot-chat_internal.ts +0 -875
- package/src/hooks/use-copilot-readable.ts +0 -135
- package/src/hooks/use-copilot-runtime-client.ts +0 -178
- package/src/hooks/use-default-tool.ts +0 -13
- package/src/hooks/use-flat-category-store.ts +0 -109
- package/src/hooks/use-frontend-tool.ts +0 -113
- package/src/hooks/use-human-in-the-loop.ts +0 -138
- package/src/hooks/use-langgraph-interrupt.ts +0 -103
- package/src/hooks/use-lazy-tool-renderer.tsx +0 -30
- package/src/hooks/use-make-copilot-document-readable.ts +0 -30
- package/src/hooks/use-render-tool-call.ts +0 -89
- package/src/hooks/use-tree.ts +0 -222
- package/src/index.tsx +0 -7
- package/src/lib/copilot-task.ts +0 -215
- package/src/lib/index.ts +0 -1
- package/src/lib/status-checker.ts +0 -67
- package/src/setupTests.ts +0 -37
- package/src/test-helpers/copilot-context.ts +0 -91
- package/src/types/chat-suggestion-configuration.ts +0 -23
- package/src/types/coagent-action.ts +0 -35
- package/src/types/coagent-state.ts +0 -13
- package/src/types/crew.ts +0 -89
- package/src/types/document-pointer.ts +0 -7
- package/src/types/frontend-action.ts +0 -213
- package/src/types/index.ts +0 -17
- package/src/types/interrupt-action.ts +0 -58
- package/src/types/system-message.ts +0 -4
- package/src/utils/dev-console.ts +0 -19
- package/src/utils/index.ts +0 -2
- package/src/utils/suggestions-constants.ts +0 -8
- package/src/utils/utils.test.ts +0 -7
- package/src/utils/utils.ts +0 -6
- package/src/v2/__tests__/A2UIMessageRenderer.test.tsx +0 -240
- package/src/v2/__tests__/globalSetup.ts +0 -14
- package/src/v2/__tests__/setup.ts +0 -93
- package/src/v2/__tests__/utils/test-helpers.tsx +0 -570
- package/src/v2/a2ui/A2UICatalogContext.tsx +0 -79
- package/src/v2/a2ui/A2UIMessageRenderer.tsx +0 -294
- package/src/v2/a2ui/A2UIToolCallRenderer.tsx +0 -290
- package/src/v2/components/CopilotKitInspector.tsx +0 -52
- package/src/v2/components/MCPAppsActivityRenderer.tsx +0 -815
- package/src/v2/components/OpenGenerativeUIRenderer.tsx +0 -598
- package/src/v2/components/WildcardToolCallRender.tsx +0 -86
- package/src/v2/components/__tests__/OpenGenerativeUIRenderer.test.tsx +0 -665
- package/src/v2/components/chat/CopilotChat.tsx +0 -664
- package/src/v2/components/chat/CopilotChatAssistantMessage.tsx +0 -393
- package/src/v2/components/chat/CopilotChatAttachmentQueue.tsx +0 -374
- package/src/v2/components/chat/CopilotChatAttachmentRenderer.tsx +0 -159
- package/src/v2/components/chat/CopilotChatAudioRecorder.tsx +0 -350
- package/src/v2/components/chat/CopilotChatInput.tsx +0 -1412
- package/src/v2/components/chat/CopilotChatMessageView.tsx +0 -716
- package/src/v2/components/chat/CopilotChatReasoningMessage.tsx +0 -265
- package/src/v2/components/chat/CopilotChatSuggestionPill.tsx +0 -59
- package/src/v2/components/chat/CopilotChatSuggestionView.tsx +0 -134
- package/src/v2/components/chat/CopilotChatToggleButton.tsx +0 -171
- package/src/v2/components/chat/CopilotChatToolCallsView.tsx +0 -40
- package/src/v2/components/chat/CopilotChatUserMessage.tsx +0 -445
- package/src/v2/components/chat/CopilotChatView.tsx +0 -890
- package/src/v2/components/chat/CopilotModalHeader.tsx +0 -129
- package/src/v2/components/chat/CopilotPopup.tsx +0 -81
- package/src/v2/components/chat/CopilotPopupView.tsx +0 -317
- package/src/v2/components/chat/CopilotSidebar.tsx +0 -80
- package/src/v2/components/chat/CopilotSidebarView.tsx +0 -269
- package/src/v2/components/chat/Lightbox.tsx +0 -103
- package/src/v2/components/chat/__tests__/CopilotChat.absentThreadConnect.test.tsx +0 -66
- package/src/v2/components/chat/__tests__/CopilotChat.attachments.test.tsx +0 -168
- package/src/v2/components/chat/__tests__/CopilotChat.e2e.test.tsx +0 -1239
- package/src/v2/components/chat/__tests__/CopilotChat.onError.test.tsx +0 -73
- package/src/v2/components/chat/__tests__/CopilotChat.slots.e2e.test.tsx +0 -432
- package/src/v2/components/chat/__tests__/CopilotChat.suggestionsAlways.test.tsx +0 -183
- package/src/v2/components/chat/__tests__/CopilotChat.welcomeGate.test.tsx +0 -184
- package/src/v2/components/chat/__tests__/CopilotChatActivityRendering.e2e.test.tsx +0 -649
- package/src/v2/components/chat/__tests__/CopilotChatAssistantMessage.slots.e2e.test.tsx +0 -624
- package/src/v2/components/chat/__tests__/CopilotChatAssistantMessage.test.tsx +0 -702
- package/src/v2/components/chat/__tests__/CopilotChatAssistantMessage.thumbs.test.tsx +0 -72
- package/src/v2/components/chat/__tests__/CopilotChatCopyButton.clipboard.test.tsx +0 -241
- package/src/v2/components/chat/__tests__/CopilotChatCssClasses.test.tsx +0 -107
- package/src/v2/components/chat/__tests__/CopilotChatInput.slots.e2e.test.tsx +0 -929
- package/src/v2/components/chat/__tests__/CopilotChatInput.test.tsx +0 -1567
- package/src/v2/components/chat/__tests__/CopilotChatMessageView.slots.e2e.test.tsx +0 -1004
- package/src/v2/components/chat/__tests__/CopilotChatMessageView.test.tsx +0 -279
- package/src/v2/components/chat/__tests__/CopilotChatPerf.e2e.test.tsx +0 -336
- package/src/v2/components/chat/__tests__/CopilotChatPropsRerender.e2e.test.tsx +0 -249
- package/src/v2/components/chat/__tests__/CopilotChatSuggestionView.slots.e2e.test.tsx +0 -530
- package/src/v2/components/chat/__tests__/CopilotChatToolRendering.e2e.test.tsx +0 -785
- package/src/v2/components/chat/__tests__/CopilotChatToolRerenders.e2e.test.tsx +0 -2416
- package/src/v2/components/chat/__tests__/CopilotChatUserMessage.slots.e2e.test.tsx +0 -621
- package/src/v2/components/chat/__tests__/CopilotChatView.connectingGate.test.tsx +0 -56
- package/src/v2/components/chat/__tests__/CopilotChatView.inputOverlay.test.tsx +0 -264
- package/src/v2/components/chat/__tests__/CopilotChatView.onClick.e2e.test.tsx +0 -853
- package/src/v2/components/chat/__tests__/CopilotChatView.pinToSend.test.tsx +0 -94
- package/src/v2/components/chat/__tests__/CopilotChatView.slots.e2e.test.tsx +0 -1050
- package/src/v2/components/chat/__tests__/CopilotModalHeader.slots.e2e.test.tsx +0 -484
- package/src/v2/components/chat/__tests__/CopilotPopupView.slots.e2e.test.tsx +0 -612
- package/src/v2/components/chat/__tests__/CopilotSidebarView.position.test.tsx +0 -159
- package/src/v2/components/chat/__tests__/CopilotSidebarView.slots.e2e.test.tsx +0 -502
- package/src/v2/components/chat/__tests__/MCPAppsActivityRenderer.e2e.test.tsx +0 -1068
- package/src/v2/components/chat/__tests__/MCPAppsProxy.e2e.test.tsx +0 -589
- package/src/v2/components/chat/__tests__/MCPAppsUiMessage.e2e.test.tsx +0 -403
- package/src/v2/components/chat/__tests__/copilot-chat-throttle.test.tsx +0 -137
- package/src/v2/components/chat/__tests__/normalize-auto-scroll.test.ts +0 -37
- package/src/v2/components/chat/__tests__/setup.ts +0 -1
- package/src/v2/components/chat/index.ts +0 -90
- package/src/v2/components/chat/last-user-message-context.ts +0 -21
- package/src/v2/components/chat/normalize-auto-scroll.ts +0 -17
- package/src/v2/components/chat/scroll-element-context.ts +0 -13
- package/src/v2/components/index.ts +0 -8
- package/src/v2/components/intelligence-indicator/IntelligenceIndicator.tsx +0 -286
- package/src/v2/components/intelligence-indicator/__tests__/IntelligenceIndicator.e2e.test.tsx +0 -464
- package/src/v2/components/intelligence-indicator/index.ts +0 -2
- package/src/v2/components/license-warning-banner.tsx +0 -217
- package/src/v2/components/ui/button.tsx +0 -124
- package/src/v2/components/ui/dropdown-menu.tsx +0 -258
- package/src/v2/components/ui/tooltip.tsx +0 -60
- package/src/v2/context.ts +0 -62
- package/src/v2/headless.ts +0 -64
- package/src/v2/hooks/__tests__/standard-schema-types.test.tsx +0 -152
- package/src/v2/hooks/__tests__/standard-schema.test.tsx +0 -282
- package/src/v2/hooks/__tests__/use-agent-context-timing.e2e.test.tsx +0 -140
- package/src/v2/hooks/__tests__/use-agent-context.test.tsx +0 -401
- package/src/v2/hooks/__tests__/use-agent-error-state.test.tsx +0 -44
- package/src/v2/hooks/__tests__/use-agent-stability.test.tsx +0 -211
- package/src/v2/hooks/__tests__/use-agent-throttle.test.tsx +0 -1029
- package/src/v2/hooks/__tests__/use-agent.e2e.test.tsx +0 -159
- package/src/v2/hooks/__tests__/use-attachments.test.tsx +0 -169
- package/src/v2/hooks/__tests__/use-capabilities.test.tsx +0 -76
- package/src/v2/hooks/__tests__/use-component.test.tsx +0 -126
- package/src/v2/hooks/__tests__/use-configure-suggestions.e2e.test.tsx +0 -696
- package/src/v2/hooks/__tests__/use-default-render-tool.test.tsx +0 -153
- package/src/v2/hooks/__tests__/use-frontend-tool-available.test.tsx +0 -167
- package/src/v2/hooks/__tests__/use-frontend-tool.e2e.test.tsx +0 -2148
- package/src/v2/hooks/__tests__/use-human-in-the-loop.e2e.test.tsx +0 -1261
- package/src/v2/hooks/__tests__/use-interrupt.test.tsx +0 -397
- package/src/v2/hooks/__tests__/use-katex-styles.test.tsx +0 -56
- package/src/v2/hooks/__tests__/use-keyboard-height.test.tsx +0 -192
- package/src/v2/hooks/__tests__/use-pin-to-send.test.tsx +0 -219
- package/src/v2/hooks/__tests__/use-render-custom-messages.test.tsx +0 -55
- package/src/v2/hooks/__tests__/use-render-tool.test.tsx +0 -259
- package/src/v2/hooks/__tests__/use-suggestions.e2e.test.tsx +0 -524
- package/src/v2/hooks/__tests__/use-threads.test.tsx +0 -757
- package/src/v2/hooks/__tests__/zod-regression.test.tsx +0 -311
- package/src/v2/hooks/index.ts +0 -24
- package/src/v2/hooks/use-agent-context.tsx +0 -45
- package/src/v2/hooks/use-agent.tsx +0 -227
- package/src/v2/hooks/use-attachments.tsx +0 -269
- package/src/v2/hooks/use-capabilities.tsx +0 -25
- package/src/v2/hooks/use-component.tsx +0 -91
- package/src/v2/hooks/use-configure-suggestions.tsx +0 -236
- package/src/v2/hooks/use-default-render-tool.tsx +0 -271
- package/src/v2/hooks/use-frontend-tool.tsx +0 -46
- package/src/v2/hooks/use-human-in-the-loop.tsx +0 -81
- package/src/v2/hooks/use-interrupt.tsx +0 -305
- package/src/v2/hooks/use-keyboard-height.tsx +0 -67
- package/src/v2/hooks/use-pin-to-send.ts +0 -94
- package/src/v2/hooks/use-render-activity-message.tsx +0 -72
- package/src/v2/hooks/use-render-custom-messages.tsx +0 -93
- package/src/v2/hooks/use-render-tool-call.tsx +0 -208
- package/src/v2/hooks/use-render-tool.tsx +0 -184
- package/src/v2/hooks/use-suggestions.tsx +0 -91
- package/src/v2/hooks/use-threads.tsx +0 -325
- package/src/v2/hooks/useKatexStyles.ts +0 -27
- package/src/v2/index.css +0 -1
- package/src/v2/index.ts +0 -27
- package/src/v2/lib/__tests__/completePartialMarkdown.test.ts +0 -495
- package/src/v2/lib/__tests__/processPartialHtml.test.ts +0 -112
- package/src/v2/lib/__tests__/renderSlot.test.tsx +0 -588
- package/src/v2/lib/__tests__/slots.test.ts +0 -56
- package/src/v2/lib/processPartialHtml.ts +0 -45
- package/src/v2/lib/react-core.ts +0 -156
- package/src/v2/lib/slots.tsx +0 -184
- package/src/v2/lib/transcription-client.ts +0 -184
- package/src/v2/lib/utils.ts +0 -8
- package/src/v2/providers/CopilotChatConfigurationProvider.tsx +0 -196
- package/src/v2/providers/CopilotKitProvider.tsx +0 -800
- package/src/v2/providers/SandboxFunctionsContext.ts +0 -10
- package/src/v2/providers/__tests__/CopilotChatConfigurationProvider.test.tsx +0 -652
- package/src/v2/providers/__tests__/CopilotKitProvider.license.test.tsx +0 -101
- package/src/v2/providers/__tests__/CopilotKitProvider.onError.test.tsx +0 -69
- package/src/v2/providers/__tests__/CopilotKitProvider.renderCustomMessages.e2e.test.tsx +0 -881
- package/src/v2/providers/__tests__/CopilotKitProvider.sandboxFunctions.test.tsx +0 -198
- package/src/v2/providers/__tests__/CopilotKitProvider.stability.test.tsx +0 -740
- package/src/v2/providers/__tests__/CopilotKitProvider.test.tsx +0 -713
- package/src/v2/providers/__tests__/CopilotKitProvider.wildcard.test.tsx +0 -294
- package/src/v2/providers/index.ts +0 -21
- package/src/v2/styles/globals.css +0 -349
- package/src/v2/types/__tests__/defineToolCallRenderer.test.tsx +0 -525
- package/src/v2/types/defineToolCallRenderer.ts +0 -68
- package/src/v2/types/frontend-tool.ts +0 -8
- package/src/v2/types/human-in-the-loop.ts +0 -33
- package/src/v2/types/index.ts +0 -8
- package/src/v2/types/interrupt.ts +0 -15
- package/src/v2/types/react-activity-message-renderer.ts +0 -27
- package/src/v2/types/react-custom-message-renderer.ts +0 -17
- package/src/v2/types/react-tool-call-renderer.ts +0 -35
- package/src/v2/types/sandbox-function.ts +0 -11
- package/tsconfig.json +0 -8
- package/tsdown.config.ts +0 -193
- package/typedoc.json +0 -4
- package/vitest.config.mjs +0 -31
|
@@ -0,0 +1,319 @@
|
|
|
1
|
+
# CopilotKit Rendering Tool Calls (React)
|
|
2
|
+
|
|
3
|
+
This skill builds on `copilotkit/provider-setup` and
|
|
4
|
+
`copilotkit/client-side-tools`.
|
|
5
|
+
|
|
6
|
+
Four hooks, distinct roles:
|
|
7
|
+
|
|
8
|
+
| Hook | Role |
|
|
9
|
+
| ---------------------- | ----------------------------------------------------------------- |
|
|
10
|
+
| `useRenderTool` | Primary registration hook for a named tool's progress/result UI |
|
|
11
|
+
| `useComponent` | Register a NEW render-only tool (agent calls it just to render) |
|
|
12
|
+
| `useDefaultRenderTool` | Sanctioned wildcard fallback for tools without a dedicated render |
|
|
13
|
+
| `useRenderToolCall` | Resolver — returns a function. For custom chat surfaces only |
|
|
14
|
+
|
|
15
|
+
Status is camelCase: `"inProgress" | "executing" | "complete"`. The
|
|
16
|
+
`RenderToolProps` discriminated union narrows `parameters` per state.
|
|
17
|
+
|
|
18
|
+
## UI-kit detection rule
|
|
19
|
+
|
|
20
|
+
Before writing raw JSX, check the consumer's `package.json` for shadcn /
|
|
21
|
+
MUI / Chakra / Ant / Mantine and reuse those primitives.
|
|
22
|
+
|
|
23
|
+
## Setup
|
|
24
|
+
|
|
25
|
+
```tsx
|
|
26
|
+
"use client";
|
|
27
|
+
import { useRenderTool } from "@copilotkit/react-core/v2";
|
|
28
|
+
import { z } from "zod";
|
|
29
|
+
import { Card, CardContent } from "@/components/ui/card";
|
|
30
|
+
import { Skeleton } from "@/components/ui/skeleton";
|
|
31
|
+
|
|
32
|
+
export function SearchRenderer() {
|
|
33
|
+
useRenderTool({
|
|
34
|
+
name: "searchDocs",
|
|
35
|
+
parameters: z.object({ query: z.string() }),
|
|
36
|
+
render: ({ status, parameters, result }) => {
|
|
37
|
+
if (status === "inProgress") return <Skeleton className="h-16 w-full" />;
|
|
38
|
+
if (status === "executing") {
|
|
39
|
+
return (
|
|
40
|
+
<Card>
|
|
41
|
+
<CardContent>Searching "{parameters.query}"…</CardContent>
|
|
42
|
+
</Card>
|
|
43
|
+
);
|
|
44
|
+
}
|
|
45
|
+
return (
|
|
46
|
+
<Card>
|
|
47
|
+
<CardContent>{result}</CardContent>
|
|
48
|
+
</Card>
|
|
49
|
+
);
|
|
50
|
+
},
|
|
51
|
+
});
|
|
52
|
+
return null;
|
|
53
|
+
}
|
|
54
|
+
```
|
|
55
|
+
|
|
56
|
+
## Core Patterns
|
|
57
|
+
|
|
58
|
+
### Wildcard fallback with the built-in card
|
|
59
|
+
|
|
60
|
+
```tsx
|
|
61
|
+
import { useDefaultRenderTool } from "@copilotkit/react-core/v2";
|
|
62
|
+
|
|
63
|
+
useDefaultRenderTool(); // renders the built-in expandable tool-call card
|
|
64
|
+
```
|
|
65
|
+
|
|
66
|
+
### Custom wildcard fallback
|
|
67
|
+
|
|
68
|
+
```tsx
|
|
69
|
+
import { useDefaultRenderTool } from "@copilotkit/react-core/v2";
|
|
70
|
+
|
|
71
|
+
useDefaultRenderTool({
|
|
72
|
+
render: ({ name, status, parameters, result }) => {
|
|
73
|
+
// parameters is unknown — narrow by tool name
|
|
74
|
+
if (name === "search") {
|
|
75
|
+
const args = parameters as { q: string };
|
|
76
|
+
return <SearchCard q={args.q} status={status} result={result} />;
|
|
77
|
+
}
|
|
78
|
+
return <GenericCard name={name} status={status} />;
|
|
79
|
+
},
|
|
80
|
+
});
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
### Render-only tool (the agent's only reason to call it is to render)
|
|
84
|
+
|
|
85
|
+
```tsx
|
|
86
|
+
import { useComponent } from "@copilotkit/react-core/v2";
|
|
87
|
+
import { z } from "zod";
|
|
88
|
+
|
|
89
|
+
useComponent({
|
|
90
|
+
name: "productCard",
|
|
91
|
+
parameters: z.object({ productId: z.string() }),
|
|
92
|
+
render: ({ productId }) => <ProductCard id={productId} />,
|
|
93
|
+
});
|
|
94
|
+
// `useComponent` registers a NEW tool called "productCard".
|
|
95
|
+
// The agent calls it to render; there is no handler to run.
|
|
96
|
+
```
|
|
97
|
+
|
|
98
|
+
### Custom chat surface (resolver hook)
|
|
99
|
+
|
|
100
|
+
`useRenderToolCall` is for building your own message list, NOT for
|
|
101
|
+
registering renderers.
|
|
102
|
+
|
|
103
|
+
```tsx
|
|
104
|
+
import { useRenderToolCall } from "@copilotkit/react-core/v2";
|
|
105
|
+
import { useAgent } from "@copilotkit/react-core/v2";
|
|
106
|
+
|
|
107
|
+
export function CustomToolList() {
|
|
108
|
+
const { agent } = useAgent({ agentId: "default" });
|
|
109
|
+
const renderToolCall = useRenderToolCall();
|
|
110
|
+
|
|
111
|
+
const toolCalls = agent.messages.flatMap((m) =>
|
|
112
|
+
"toolCalls" in m ? (m.toolCalls ?? []) : [],
|
|
113
|
+
);
|
|
114
|
+
|
|
115
|
+
return (
|
|
116
|
+
<>
|
|
117
|
+
{toolCalls.map((tc) => (
|
|
118
|
+
<div key={tc.id}>{renderToolCall({ toolCall: tc })}</div>
|
|
119
|
+
))}
|
|
120
|
+
</>
|
|
121
|
+
);
|
|
122
|
+
}
|
|
123
|
+
```
|
|
124
|
+
|
|
125
|
+
## Common Mistakes
|
|
126
|
+
|
|
127
|
+
### CRITICAL — Using `useRenderToolCall` for registration
|
|
128
|
+
|
|
129
|
+
Wrong:
|
|
130
|
+
|
|
131
|
+
```tsx
|
|
132
|
+
useRenderToolCall({
|
|
133
|
+
name: "search",
|
|
134
|
+
args: z.object({ q: z.string() }),
|
|
135
|
+
render: ({ status, args }) => <Card>…</Card>,
|
|
136
|
+
});
|
|
137
|
+
```
|
|
138
|
+
|
|
139
|
+
Correct:
|
|
140
|
+
|
|
141
|
+
```tsx
|
|
142
|
+
useRenderTool({
|
|
143
|
+
name: "search",
|
|
144
|
+
parameters: z.object({ q: z.string() }),
|
|
145
|
+
render: ({ status, parameters }) => <Card>…</Card>,
|
|
146
|
+
});
|
|
147
|
+
```
|
|
148
|
+
|
|
149
|
+
`useRenderToolCall` takes no arguments — it returns a resolver function for
|
|
150
|
+
custom chat surfaces. Passing config to it does nothing. `useRenderTool` is
|
|
151
|
+
the registration hook.
|
|
152
|
+
|
|
153
|
+
Source: `packages/react-core/src/v2/hooks/index.ts:2,7`;
|
|
154
|
+
`packages/react-core/src/v2/hooks/use-render-tool.tsx:37-40`
|
|
155
|
+
|
|
156
|
+
### CRITICAL — Using hyphenated `"in-progress"` status
|
|
157
|
+
|
|
158
|
+
Wrong:
|
|
159
|
+
|
|
160
|
+
```tsx
|
|
161
|
+
render: ({ status, parameters, result }) => {
|
|
162
|
+
if (status === "in-progress") return <Spinner />;
|
|
163
|
+
if (status === "executing") return <RunningCard args={parameters} />;
|
|
164
|
+
return <ResultCard result={result} />;
|
|
165
|
+
};
|
|
166
|
+
```
|
|
167
|
+
|
|
168
|
+
Correct:
|
|
169
|
+
|
|
170
|
+
```tsx
|
|
171
|
+
render: ({ status, parameters, result }) => {
|
|
172
|
+
if (status === "inProgress") return <Spinner />;
|
|
173
|
+
if (status === "executing") return <RunningCard args={parameters} />;
|
|
174
|
+
return <ResultCard result={result} />;
|
|
175
|
+
};
|
|
176
|
+
```
|
|
177
|
+
|
|
178
|
+
Real status values are camelCase: `"inProgress" | "executing" | "complete"`.
|
|
179
|
+
Hyphenated branches never match — users see no progress UI and the fallback
|
|
180
|
+
path fires.
|
|
181
|
+
|
|
182
|
+
Source: `packages/react-core/src/v2/hooks/use-render-tool.tsx:8-35`
|
|
183
|
+
|
|
184
|
+
### CRITICAL — Writing JSX from scratch when the app has a UI kit
|
|
185
|
+
|
|
186
|
+
Wrong:
|
|
187
|
+
|
|
188
|
+
```tsx
|
|
189
|
+
useRenderTool({
|
|
190
|
+
name: "search",
|
|
191
|
+
parameters: z.object({ q: z.string() }),
|
|
192
|
+
render: () => <div className="my-badge">…</div>,
|
|
193
|
+
});
|
|
194
|
+
```
|
|
195
|
+
|
|
196
|
+
Correct:
|
|
197
|
+
|
|
198
|
+
```tsx
|
|
199
|
+
import { Badge } from "@/components/ui/badge";
|
|
200
|
+
|
|
201
|
+
useRenderTool({
|
|
202
|
+
name: "search",
|
|
203
|
+
parameters: z.object({ q: z.string() }),
|
|
204
|
+
render: () => <Badge variant="secondary">…</Badge>,
|
|
205
|
+
});
|
|
206
|
+
```
|
|
207
|
+
|
|
208
|
+
Check consumer `package.json` for shadcn / MUI / Chakra / Ant / Mantine
|
|
209
|
+
first. Raw JSX ignores their design system.
|
|
210
|
+
|
|
211
|
+
Source: maintainer interview (Phase 2c)
|
|
212
|
+
|
|
213
|
+
### HIGH — Dereferencing required fields from `Partial<T>` during `inProgress`
|
|
214
|
+
|
|
215
|
+
Wrong:
|
|
216
|
+
|
|
217
|
+
```tsx
|
|
218
|
+
render: ({ status, parameters }) => (
|
|
219
|
+
<span>{parameters.user.id.toUpperCase()}</span>
|
|
220
|
+
);
|
|
221
|
+
// `parameters` is Partial<T> during inProgress — `parameters.user` may be undefined.
|
|
222
|
+
```
|
|
223
|
+
|
|
224
|
+
Correct:
|
|
225
|
+
|
|
226
|
+
```tsx
|
|
227
|
+
render: ({ status, parameters }) =>
|
|
228
|
+
status === "inProgress" ? (
|
|
229
|
+
<Skeleton />
|
|
230
|
+
) : (
|
|
231
|
+
<span>{parameters.user.id.toUpperCase()}</span>
|
|
232
|
+
);
|
|
233
|
+
```
|
|
234
|
+
|
|
235
|
+
During streaming, `RenderToolInProgressProps` has
|
|
236
|
+
`parameters: Partial<InferSchemaOutput<S>>`. Fields are `undefined` until
|
|
237
|
+
the stream completes. Narrow with `status === "inProgress"` first.
|
|
238
|
+
|
|
239
|
+
Source: `packages/react-core/src/v2/hooks/use-render-tool.tsx:8-14`
|
|
240
|
+
|
|
241
|
+
### HIGH — Using `useComponent` to decorate an existing tool
|
|
242
|
+
|
|
243
|
+
Wrong:
|
|
244
|
+
|
|
245
|
+
```tsx
|
|
246
|
+
useFrontendTool({ name: "search", parameters, handler });
|
|
247
|
+
useComponent({
|
|
248
|
+
name: "search", // creates a SECOND tool named "search" — collision
|
|
249
|
+
parameters: z.object({ q: z.string() }),
|
|
250
|
+
render: ({ q }) => <SearchCard q={q} />,
|
|
251
|
+
});
|
|
252
|
+
```
|
|
253
|
+
|
|
254
|
+
Correct:
|
|
255
|
+
|
|
256
|
+
```tsx
|
|
257
|
+
useFrontendTool({ name: "search", parameters, handler });
|
|
258
|
+
useRenderTool({
|
|
259
|
+
name: "search",
|
|
260
|
+
parameters: z.object({ q: z.string() }),
|
|
261
|
+
render: ({ status, parameters, result }) => {
|
|
262
|
+
if (status === "inProgress") return <Skeleton />;
|
|
263
|
+
if (status === "executing") return <div>Searching {parameters.q}…</div>;
|
|
264
|
+
return <div>{result}</div>;
|
|
265
|
+
},
|
|
266
|
+
});
|
|
267
|
+
|
|
268
|
+
// useComponent is only for render-only tools the agent invokes:
|
|
269
|
+
useComponent({
|
|
270
|
+
name: "productCard",
|
|
271
|
+
parameters: z.object({ productId: z.string() }),
|
|
272
|
+
render: ({ productId }) => <ProductCard id={productId} />,
|
|
273
|
+
});
|
|
274
|
+
```
|
|
275
|
+
|
|
276
|
+
`useComponent` synthesizes a NEW tool whose only job is to render —
|
|
277
|
+
description is auto-prefixed with "Use this tool to display the …
|
|
278
|
+
component". It does NOT decorate an existing tool. The misleading name
|
|
279
|
+
trap: agents read "useComponent" as "register a component for this tool"
|
|
280
|
+
and end up with two tools colliding on the same name.
|
|
281
|
+
|
|
282
|
+
Source: `packages/react-core/src/v2/hooks/use-component.tsx:59-88`
|
|
283
|
+
|
|
284
|
+
### HIGH — Hand-rolling `useRenderTool({ name: "*" })` instead of `useDefaultRenderTool`
|
|
285
|
+
|
|
286
|
+
Wrong:
|
|
287
|
+
|
|
288
|
+
```tsx
|
|
289
|
+
useRenderTool({
|
|
290
|
+
name: "*",
|
|
291
|
+
render: ({ parameters }) => <pre>{JSON.stringify(parameters)}</pre>,
|
|
292
|
+
});
|
|
293
|
+
```
|
|
294
|
+
|
|
295
|
+
Correct:
|
|
296
|
+
|
|
297
|
+
```tsx
|
|
298
|
+
// Use the built-in default card:
|
|
299
|
+
useDefaultRenderTool();
|
|
300
|
+
|
|
301
|
+
// Or customize, with the correct DefaultRenderProps typing (parameters: unknown):
|
|
302
|
+
useDefaultRenderTool({
|
|
303
|
+
render: ({ name, status, parameters, result }) => {
|
|
304
|
+
if (name === "search") {
|
|
305
|
+
const args = parameters as { q: string };
|
|
306
|
+
return <SearchCard q={args.q} status={status} />;
|
|
307
|
+
}
|
|
308
|
+
return <GenericCard name={name} status={status} />;
|
|
309
|
+
},
|
|
310
|
+
});
|
|
311
|
+
```
|
|
312
|
+
|
|
313
|
+
The sanctioned wildcard API is `useDefaultRenderTool`. It wraps
|
|
314
|
+
`useRenderTool({ name: "*" })` with the correct `DefaultRenderProps`
|
|
315
|
+
typing (`parameters: unknown`) and provides a built-in default card when
|
|
316
|
+
no `render` is passed. Hand-rolling loses the default card and invites
|
|
317
|
+
the untyped-args footgun.
|
|
318
|
+
|
|
319
|
+
Source: `packages/react-core/src/v2/hooks/use-default-render-tool.tsx:15-64`
|
|
@@ -0,0 +1,211 @@
|
|
|
1
|
+
# CopilotKit Suggestions (React)
|
|
2
|
+
|
|
3
|
+
This skill builds on `copilotkit/provider-setup` and
|
|
4
|
+
`copilotkit/chat-components`. Suggestions render via
|
|
5
|
+
`CopilotChatSuggestionView` which `<CopilotChat>` mounts automatically.
|
|
6
|
+
|
|
7
|
+
Two sides:
|
|
8
|
+
|
|
9
|
+
- `useConfigureSuggestions(config, deps?)` — register dynamic (LLM) or
|
|
10
|
+
static suggestions.
|
|
11
|
+
- `useSuggestions({ agentId })` — read current suggestions + trigger
|
|
12
|
+
reload / clear.
|
|
13
|
+
|
|
14
|
+
## Setup
|
|
15
|
+
|
|
16
|
+
### Dynamic suggestions (LLM-generated)
|
|
17
|
+
|
|
18
|
+
```tsx
|
|
19
|
+
"use client";
|
|
20
|
+
import { useConfigureSuggestions } from "@copilotkit/react-core/v2";
|
|
21
|
+
import { useMemo } from "react";
|
|
22
|
+
|
|
23
|
+
export function DynamicSuggestionsHost({ page }: { page: string }) {
|
|
24
|
+
const instructions = useMemo(
|
|
25
|
+
() => `Suggest 3 follow-up questions about the "${page}" page.`,
|
|
26
|
+
[page],
|
|
27
|
+
);
|
|
28
|
+
|
|
29
|
+
useConfigureSuggestions(
|
|
30
|
+
{
|
|
31
|
+
instructions,
|
|
32
|
+
minSuggestions: 2,
|
|
33
|
+
maxSuggestions: 4,
|
|
34
|
+
available: "always",
|
|
35
|
+
},
|
|
36
|
+
[page],
|
|
37
|
+
);
|
|
38
|
+
|
|
39
|
+
return null;
|
|
40
|
+
}
|
|
41
|
+
```
|
|
42
|
+
|
|
43
|
+
### Static suggestions
|
|
44
|
+
|
|
45
|
+
```tsx
|
|
46
|
+
"use client";
|
|
47
|
+
import { useConfigureSuggestions } from "@copilotkit/react-core/v2";
|
|
48
|
+
|
|
49
|
+
export function StaticStarters() {
|
|
50
|
+
useConfigureSuggestions({
|
|
51
|
+
suggestions: [
|
|
52
|
+
{ title: "Summarize this page", message: "Summarize the current page." },
|
|
53
|
+
{ title: "Explain like I'm 5", message: "Explain this in simple terms." },
|
|
54
|
+
],
|
|
55
|
+
available: "before-first-message",
|
|
56
|
+
});
|
|
57
|
+
return null;
|
|
58
|
+
}
|
|
59
|
+
```
|
|
60
|
+
|
|
61
|
+
## Core Patterns
|
|
62
|
+
|
|
63
|
+
### Read and refresh suggestions from any component
|
|
64
|
+
|
|
65
|
+
```tsx
|
|
66
|
+
import { useSuggestions } from "@copilotkit/react-core/v2";
|
|
67
|
+
|
|
68
|
+
export function RefreshButton() {
|
|
69
|
+
const { suggestions, reloadSuggestions, clearSuggestions, isLoading } =
|
|
70
|
+
useSuggestions({ agentId: "default" });
|
|
71
|
+
return (
|
|
72
|
+
<div>
|
|
73
|
+
<button onClick={reloadSuggestions} disabled={isLoading}>
|
|
74
|
+
{isLoading ? "Loading…" : "Refresh"}
|
|
75
|
+
</button>
|
|
76
|
+
<button onClick={clearSuggestions}>Clear</button>
|
|
77
|
+
<span>{suggestions.length} suggestions</span>
|
|
78
|
+
</div>
|
|
79
|
+
);
|
|
80
|
+
}
|
|
81
|
+
```
|
|
82
|
+
|
|
83
|
+
### Feature-flag the suggestions config
|
|
84
|
+
|
|
85
|
+
```tsx
|
|
86
|
+
const enabled = useFeatureFlag("suggestions");
|
|
87
|
+
useConfigureSuggestions(
|
|
88
|
+
enabled ? { instructions: "Suggest 3 follow-ups" } : null,
|
|
89
|
+
);
|
|
90
|
+
```
|
|
91
|
+
|
|
92
|
+
### Agent-scoped dynamic suggestions
|
|
93
|
+
|
|
94
|
+
```tsx
|
|
95
|
+
useConfigureSuggestions({
|
|
96
|
+
instructions: "Suggest follow-ups for the research agent.",
|
|
97
|
+
consumerAgentId: "research",
|
|
98
|
+
});
|
|
99
|
+
```
|
|
100
|
+
|
|
101
|
+
## Common Mistakes
|
|
102
|
+
|
|
103
|
+
### MEDIUM — Using `available: "disabled"` expecting reload to still fire
|
|
104
|
+
|
|
105
|
+
Wrong:
|
|
106
|
+
|
|
107
|
+
```tsx
|
|
108
|
+
useConfigureSuggestions({
|
|
109
|
+
instructions: "Suggest 3 follow-ups",
|
|
110
|
+
available: "disabled",
|
|
111
|
+
});
|
|
112
|
+
// Then calling reloadSuggestions() — no-op
|
|
113
|
+
```
|
|
114
|
+
|
|
115
|
+
Correct:
|
|
116
|
+
|
|
117
|
+
```tsx
|
|
118
|
+
const enabled = useFeatureFlag("suggestions");
|
|
119
|
+
useConfigureSuggestions(
|
|
120
|
+
enabled ? { instructions: "Suggest 3 follow-ups" } : null,
|
|
121
|
+
);
|
|
122
|
+
```
|
|
123
|
+
|
|
124
|
+
`available: "disabled"` is normalized to a `null` config — the same as
|
|
125
|
+
passing `null`/`undefined`. Reloads become no-ops. Pass `null` (or gate on
|
|
126
|
+
a condition) when you want to fully disable suggestions.
|
|
127
|
+
|
|
128
|
+
Source: `packages/react-core/src/v2/hooks/use-configure-suggestions.tsx:59-62`
|
|
129
|
+
|
|
130
|
+
### MEDIUM — Inline config object without `deps`
|
|
131
|
+
|
|
132
|
+
Wrong:
|
|
133
|
+
|
|
134
|
+
```tsx
|
|
135
|
+
useConfigureSuggestions({ instructions: `about ${currentPage}` });
|
|
136
|
+
// Config re-serialized every render — reload may or may not fire depending on cache equality
|
|
137
|
+
```
|
|
138
|
+
|
|
139
|
+
Correct:
|
|
140
|
+
|
|
141
|
+
```tsx
|
|
142
|
+
useConfigureSuggestions({ instructions: `about ${currentPage}` }, [
|
|
143
|
+
currentPage,
|
|
144
|
+
]);
|
|
145
|
+
```
|
|
146
|
+
|
|
147
|
+
`useConfigureSuggestions` uses a serialized-config cache keyed off the
|
|
148
|
+
JSON-stringified value. Without `deps`, React invariance + inline objects
|
|
149
|
+
produce unstable identities that thrash the cache. Always pass `deps` that
|
|
150
|
+
cover the values interpolated into the config.
|
|
151
|
+
|
|
152
|
+
Source: `packages/react-core/src/v2/hooks/use-configure-suggestions.tsx:166-171`
|
|
153
|
+
|
|
154
|
+
### MEDIUM — Calling `reloadSuggestions` mid-run
|
|
155
|
+
|
|
156
|
+
Wrong:
|
|
157
|
+
|
|
158
|
+
```tsx
|
|
159
|
+
<button onClick={() => reloadSuggestions()}>Refresh</button>
|
|
160
|
+
// Fires during agent streaming → competes with the running agent
|
|
161
|
+
```
|
|
162
|
+
|
|
163
|
+
Correct:
|
|
164
|
+
|
|
165
|
+
```tsx
|
|
166
|
+
import { useAgent, UseAgentUpdate } from "@copilotkit/react-core/v2";
|
|
167
|
+
|
|
168
|
+
const { agent } = useAgent({
|
|
169
|
+
agentId: "default",
|
|
170
|
+
updates: [UseAgentUpdate.OnRunStatusChanged],
|
|
171
|
+
});
|
|
172
|
+
const isRunning = agent.isRunning;
|
|
173
|
+
<button
|
|
174
|
+
disabled={isRunning}
|
|
175
|
+
onClick={() => {
|
|
176
|
+
if (!isRunning) reloadSuggestions();
|
|
177
|
+
}}
|
|
178
|
+
>
|
|
179
|
+
Refresh
|
|
180
|
+
</button>;
|
|
181
|
+
```
|
|
182
|
+
|
|
183
|
+
The internal auto-reload skips when `isRunning`, but the user-triggered
|
|
184
|
+
`reloadSuggestions()` does not guard itself. Guard the caller, or the
|
|
185
|
+
suggestion generation races the active agent run.
|
|
186
|
+
|
|
187
|
+
Source: `packages/react-core/src/v2/hooks/use-configure-suggestions.tsx:121-124`
|
|
188
|
+
|
|
189
|
+
### MEDIUM — Expecting `maxSuggestions` above 3 without setting it
|
|
190
|
+
|
|
191
|
+
Wrong:
|
|
192
|
+
|
|
193
|
+
```tsx
|
|
194
|
+
useConfigureSuggestions({ instructions: "…" });
|
|
195
|
+
// Then surprised the UI only shows 3 pills even when the LLM returned 8
|
|
196
|
+
```
|
|
197
|
+
|
|
198
|
+
Correct:
|
|
199
|
+
|
|
200
|
+
```tsx
|
|
201
|
+
useConfigureSuggestions({
|
|
202
|
+
instructions: "…",
|
|
203
|
+
minSuggestions: 1,
|
|
204
|
+
maxSuggestions: 6,
|
|
205
|
+
});
|
|
206
|
+
```
|
|
207
|
+
|
|
208
|
+
`minSuggestions` and `maxSuggestions` default to 1 and 3 respectively. Set
|
|
209
|
+
them explicitly when you want a different count.
|
|
210
|
+
|
|
211
|
+
Source: `packages/core/src/types.ts` (DynamicSuggestionsConfig defaults)
|
|
@@ -0,0 +1,160 @@
|
|
|
1
|
+
# Agent Switcher Recipes
|
|
2
|
+
|
|
3
|
+
Three copy-paste patterns for multi-agent UIs. All subscribe to
|
|
4
|
+
`copilotkit.subscribe({ onAgentsChanged })` for live agent discovery — there
|
|
5
|
+
is no `useAgents()` hook.
|
|
6
|
+
|
|
7
|
+
## Recipe 1 — Dropdown switcher
|
|
8
|
+
|
|
9
|
+
```tsx
|
|
10
|
+
"use client";
|
|
11
|
+
import { CopilotChat, useCopilotKit } from "@copilotkit/react-core/v2";
|
|
12
|
+
import { useEffect, useState } from "react";
|
|
13
|
+
|
|
14
|
+
export function DropdownAgentSwitcher() {
|
|
15
|
+
const { copilotkit } = useCopilotKit();
|
|
16
|
+
const [agentIds, setAgentIds] = useState<string[]>(() =>
|
|
17
|
+
Object.keys(copilotkit.agents ?? {}),
|
|
18
|
+
);
|
|
19
|
+
const [activeAgent, setActiveAgent] = useState<string>(
|
|
20
|
+
() => Object.keys(copilotkit.agents ?? {})[0] ?? "default",
|
|
21
|
+
);
|
|
22
|
+
|
|
23
|
+
useEffect(() => {
|
|
24
|
+
const sub = copilotkit.subscribe({
|
|
25
|
+
onAgentsChanged: ({ agents }) => {
|
|
26
|
+
setAgentIds(Object.keys(agents ?? {}));
|
|
27
|
+
},
|
|
28
|
+
});
|
|
29
|
+
return () => sub.unsubscribe();
|
|
30
|
+
}, [copilotkit]);
|
|
31
|
+
|
|
32
|
+
return (
|
|
33
|
+
<div className="flex flex-col gap-3">
|
|
34
|
+
<select
|
|
35
|
+
value={activeAgent}
|
|
36
|
+
onChange={(e) => setActiveAgent(e.target.value)}
|
|
37
|
+
>
|
|
38
|
+
{agentIds.map((id) => (
|
|
39
|
+
<option key={id} value={id}>
|
|
40
|
+
{id}
|
|
41
|
+
</option>
|
|
42
|
+
))}
|
|
43
|
+
</select>
|
|
44
|
+
<CopilotChat key={activeAgent} agentId={activeAgent} />
|
|
45
|
+
</div>
|
|
46
|
+
);
|
|
47
|
+
}
|
|
48
|
+
```
|
|
49
|
+
|
|
50
|
+
## Recipe 2 — Tabs switcher
|
|
51
|
+
|
|
52
|
+
```tsx
|
|
53
|
+
"use client";
|
|
54
|
+
import { CopilotChat, useCopilotKit } from "@copilotkit/react-core/v2";
|
|
55
|
+
import { useEffect, useRef, useState } from "react";
|
|
56
|
+
|
|
57
|
+
export function TabsAgentSwitcher() {
|
|
58
|
+
const { copilotkit } = useCopilotKit();
|
|
59
|
+
const [agentIds, setAgentIds] = useState<string[]>(() =>
|
|
60
|
+
Object.keys(copilotkit.agents ?? {}),
|
|
61
|
+
);
|
|
62
|
+
const [activeAgent, setActiveAgent] = useState<string>(
|
|
63
|
+
() => agentIds[0] ?? "default",
|
|
64
|
+
);
|
|
65
|
+
|
|
66
|
+
// Hold activeAgent in a ref so the subscribe effect only re-binds when
|
|
67
|
+
// `copilotkit` changes. Depending on `activeAgent` would tear down and
|
|
68
|
+
// re-establish the subscription on every tab click.
|
|
69
|
+
const activeAgentRef = useRef(activeAgent);
|
|
70
|
+
useEffect(() => {
|
|
71
|
+
activeAgentRef.current = activeAgent;
|
|
72
|
+
}, [activeAgent]);
|
|
73
|
+
|
|
74
|
+
useEffect(() => {
|
|
75
|
+
const sub = copilotkit.subscribe({
|
|
76
|
+
onAgentsChanged: ({ agents }) => {
|
|
77
|
+
const ids = Object.keys(agents ?? {});
|
|
78
|
+
setAgentIds(ids);
|
|
79
|
+
if (!ids.includes(activeAgentRef.current) && ids.length > 0) {
|
|
80
|
+
setActiveAgent(ids[0]);
|
|
81
|
+
}
|
|
82
|
+
},
|
|
83
|
+
});
|
|
84
|
+
return () => sub.unsubscribe();
|
|
85
|
+
}, [copilotkit]);
|
|
86
|
+
|
|
87
|
+
return (
|
|
88
|
+
<div>
|
|
89
|
+
<div role="tablist" className="flex gap-2 border-b">
|
|
90
|
+
{agentIds.map((id) => (
|
|
91
|
+
<button
|
|
92
|
+
key={id}
|
|
93
|
+
role="tab"
|
|
94
|
+
aria-selected={id === activeAgent}
|
|
95
|
+
onClick={() => setActiveAgent(id)}
|
|
96
|
+
>
|
|
97
|
+
{id}
|
|
98
|
+
</button>
|
|
99
|
+
))}
|
|
100
|
+
</div>
|
|
101
|
+
<CopilotChat key={activeAgent} agentId={activeAgent} />
|
|
102
|
+
</div>
|
|
103
|
+
);
|
|
104
|
+
}
|
|
105
|
+
```
|
|
106
|
+
|
|
107
|
+
## Recipe 3 — Keyboard shortcut switcher
|
|
108
|
+
|
|
109
|
+
Cycles through agents with `Cmd/Ctrl + Shift + A`.
|
|
110
|
+
|
|
111
|
+
```tsx
|
|
112
|
+
"use client";
|
|
113
|
+
import { CopilotChat, useCopilotKit } from "@copilotkit/react-core/v2";
|
|
114
|
+
import { useEffect, useState } from "react";
|
|
115
|
+
|
|
116
|
+
export function KeyboardAgentSwitcher() {
|
|
117
|
+
const { copilotkit } = useCopilotKit();
|
|
118
|
+
const [agentIds, setAgentIds] = useState<string[]>(() =>
|
|
119
|
+
Object.keys(copilotkit.agents ?? {}),
|
|
120
|
+
);
|
|
121
|
+
const [activeAgent, setActiveAgent] = useState<string>(
|
|
122
|
+
() => agentIds[0] ?? "default",
|
|
123
|
+
);
|
|
124
|
+
|
|
125
|
+
useEffect(() => {
|
|
126
|
+
const sub = copilotkit.subscribe({
|
|
127
|
+
onAgentsChanged: ({ agents }) => setAgentIds(Object.keys(agents ?? {})),
|
|
128
|
+
});
|
|
129
|
+
return () => sub.unsubscribe();
|
|
130
|
+
}, [copilotkit]);
|
|
131
|
+
|
|
132
|
+
useEffect(() => {
|
|
133
|
+
function onKey(e: KeyboardEvent) {
|
|
134
|
+
const isCombo = (e.metaKey || e.ctrlKey) && e.shiftKey && e.key === "A";
|
|
135
|
+
if (!isCombo || agentIds.length === 0) return;
|
|
136
|
+
e.preventDefault();
|
|
137
|
+
const idx = agentIds.indexOf(activeAgent);
|
|
138
|
+
const next = agentIds[(idx + 1) % agentIds.length];
|
|
139
|
+
setActiveAgent(next);
|
|
140
|
+
}
|
|
141
|
+
window.addEventListener("keydown", onKey);
|
|
142
|
+
return () => window.removeEventListener("keydown", onKey);
|
|
143
|
+
}, [agentIds, activeAgent]);
|
|
144
|
+
|
|
145
|
+
return (
|
|
146
|
+
<div>
|
|
147
|
+
<div className="text-xs opacity-60">
|
|
148
|
+
Active: {activeAgent} — press ⌘/Ctrl+Shift+A to cycle
|
|
149
|
+
</div>
|
|
150
|
+
<CopilotChat key={activeAgent} agentId={activeAgent} />
|
|
151
|
+
</div>
|
|
152
|
+
);
|
|
153
|
+
}
|
|
154
|
+
```
|
|
155
|
+
|
|
156
|
+
## Key rules across all three recipes
|
|
157
|
+
|
|
158
|
+
- Use `copilotkit.subscribe({ onAgentsChanged })` — there is no `useAgents()` hook.
|
|
159
|
+
- Always `key={activeAgent}` on `<CopilotChat>` so thread state doesn't leak when swapping agents in the same slot.
|
|
160
|
+
- Clean up the subscription with `sub.unsubscribe()` in the effect cleanup.
|