@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.
Files changed (292) hide show
  1. package/LICENSE +21 -0
  2. package/dist/{copilotkit-CtXcs1ea.cjs → copilotkit-B4ouY7qC.cjs} +14 -3
  3. package/dist/copilotkit-B4ouY7qC.cjs.map +1 -0
  4. package/dist/copilotkit-BK9CVq9A.d.cts.map +1 -1
  5. package/dist/{copilotkit-CC8DjOiC.mjs → copilotkit-L4mM_JqG.mjs} +14 -3
  6. package/dist/copilotkit-L4mM_JqG.mjs.map +1 -0
  7. package/dist/copilotkit-WlmeVijs.d.mts.map +1 -1
  8. package/dist/index.cjs +3 -77
  9. package/dist/index.cjs.map +1 -1
  10. package/dist/index.d.cts +2 -2
  11. package/dist/index.d.mts +2 -2
  12. package/dist/index.mjs +3 -77
  13. package/dist/index.mjs.map +1 -1
  14. package/dist/index.umd.js +15 -78
  15. package/dist/index.umd.js.map +1 -1
  16. package/dist/v2/headless.cjs +11 -0
  17. package/dist/v2/headless.cjs.map +1 -1
  18. package/dist/v2/headless.d.cts.map +1 -1
  19. package/dist/v2/headless.d.mts.map +1 -1
  20. package/dist/v2/headless.mjs +11 -0
  21. package/dist/v2/headless.mjs.map +1 -1
  22. package/dist/v2/index.cjs +1 -1
  23. package/dist/v2/index.mjs +1 -1
  24. package/dist/v2/index.umd.js +13 -2
  25. package/dist/v2/index.umd.js.map +1 -1
  26. package/package.json +12 -13
  27. package/skills/react-core/SKILL.md +108 -0
  28. package/skills/react-core/references/agent-access.md +288 -0
  29. package/skills/react-core/references/attachments.md +291 -0
  30. package/skills/react-core/references/capabilities.md +138 -0
  31. package/skills/react-core/references/chat-components.md +221 -0
  32. package/skills/react-core/references/client-side-tools.md +358 -0
  33. package/skills/react-core/references/custom-message-renderers.md +226 -0
  34. package/skills/react-core/references/debug-mode.md +153 -0
  35. package/skills/react-core/references/human-in-the-loop.md +312 -0
  36. package/skills/react-core/references/provider-setup.md +326 -0
  37. package/skills/react-core/references/rendering-activity-messages.md +207 -0
  38. package/skills/react-core/references/rendering-tool-calls.md +319 -0
  39. package/skills/react-core/references/suggestions.md +211 -0
  40. package/skills/react-core/references/switching-agents-recipes.md +160 -0
  41. package/skills/react-core/references/switching-agents.md +231 -0
  42. package/skills/react-core/references/threads.md +226 -0
  43. package/.attw.json +0 -3
  44. package/CHANGELOG.md +0 -5043
  45. package/dist/copilotkit-CC8DjOiC.mjs.map +0 -1
  46. package/dist/copilotkit-CtXcs1ea.cjs.map +0 -1
  47. package/scripts/scope-preflight.mjs +0 -100
  48. package/src/components/CopilotListeners.tsx +0 -137
  49. package/src/components/__tests__/CopilotListeners.test.tsx +0 -38
  50. package/src/components/copilot-provider/__tests__/copilot-messages-key.test.tsx +0 -92
  51. package/src/components/copilot-provider/__tests__/copilotkit-error.test.tsx +0 -77
  52. package/src/components/copilot-provider/__tests__/error-visibility-prod.test.tsx +0 -70
  53. package/src/components/copilot-provider/__tests__/v1-explicit-threadid-bridge.test.tsx +0 -107
  54. package/src/components/copilot-provider/copilot-messages.tsx +0 -314
  55. package/src/components/copilot-provider/copilotkit-props.tsx +0 -214
  56. package/src/components/copilot-provider/copilotkit.tsx +0 -853
  57. package/src/components/copilot-provider/index.ts +0 -3
  58. package/src/components/dev-console/console-trigger.tsx +0 -283
  59. package/src/components/dev-console/developer-console-modal.tsx +0 -1016
  60. package/src/components/dev-console/icons.tsx +0 -106
  61. package/src/components/error-boundary/error-boundary.tsx +0 -99
  62. package/src/components/error-boundary/error-utils.tsx +0 -105
  63. package/src/components/index.ts +0 -1
  64. package/src/components/toast/exclamation-mark-icon.tsx +0 -27
  65. package/src/components/toast/toast-provider.tsx +0 -448
  66. package/src/components/usage-banner.tsx +0 -266
  67. package/src/context/__tests__/threads-context.test.tsx +0 -141
  68. package/src/context/coagent-state-renders-context.tsx +0 -89
  69. package/src/context/copilot-context.tsx +0 -365
  70. package/src/context/copilot-messages-context.tsx +0 -35
  71. package/src/context/index.ts +0 -22
  72. package/src/context/threads-context.tsx +0 -69
  73. package/src/hooks/__tests__/use-coagent-config.test.ts +0 -352
  74. package/src/hooks/__tests__/use-coagent-state-render-bridge.helpers.test.ts +0 -107
  75. package/src/hooks/__tests__/use-coagent-state-render.e2e.test.tsx +0 -1209
  76. package/src/hooks/__tests__/use-coagent-state-render.test.tsx +0 -356
  77. package/src/hooks/__tests__/use-copilot-chat-internal-connect.test.tsx +0 -241
  78. package/src/hooks/__tests__/use-frontend-tool-available.test.tsx +0 -72
  79. package/src/hooks/__tests__/use-frontend-tool-remount.e2e.test.tsx +0 -102
  80. package/src/hooks/index.ts +0 -33
  81. package/src/hooks/use-agent-nodename.ts +0 -33
  82. package/src/hooks/use-coagent-state-render-bridge.helpers.ts +0 -345
  83. package/src/hooks/use-coagent-state-render-bridge.tsx +0 -222
  84. package/src/hooks/use-coagent-state-render-registry.ts +0 -230
  85. package/src/hooks/use-coagent-state-render.ts +0 -163
  86. package/src/hooks/use-coagent.ts +0 -377
  87. package/src/hooks/use-configure-chat-suggestions.tsx +0 -96
  88. package/src/hooks/use-copilot-action.ts +0 -245
  89. package/src/hooks/use-copilot-additional-instructions.ts +0 -98
  90. package/src/hooks/use-copilot-authenticated-action.ts +0 -73
  91. package/src/hooks/use-copilot-chat-headless_c.ts +0 -264
  92. package/src/hooks/use-copilot-chat-suggestions.tsx +0 -134
  93. package/src/hooks/use-copilot-chat.ts +0 -132
  94. package/src/hooks/use-copilot-chat_internal.ts +0 -875
  95. package/src/hooks/use-copilot-readable.ts +0 -135
  96. package/src/hooks/use-copilot-runtime-client.ts +0 -178
  97. package/src/hooks/use-default-tool.ts +0 -13
  98. package/src/hooks/use-flat-category-store.ts +0 -109
  99. package/src/hooks/use-frontend-tool.ts +0 -113
  100. package/src/hooks/use-human-in-the-loop.ts +0 -138
  101. package/src/hooks/use-langgraph-interrupt.ts +0 -103
  102. package/src/hooks/use-lazy-tool-renderer.tsx +0 -30
  103. package/src/hooks/use-make-copilot-document-readable.ts +0 -30
  104. package/src/hooks/use-render-tool-call.ts +0 -89
  105. package/src/hooks/use-tree.ts +0 -222
  106. package/src/index.tsx +0 -7
  107. package/src/lib/copilot-task.ts +0 -215
  108. package/src/lib/index.ts +0 -1
  109. package/src/lib/status-checker.ts +0 -67
  110. package/src/setupTests.ts +0 -37
  111. package/src/test-helpers/copilot-context.ts +0 -91
  112. package/src/types/chat-suggestion-configuration.ts +0 -23
  113. package/src/types/coagent-action.ts +0 -35
  114. package/src/types/coagent-state.ts +0 -13
  115. package/src/types/crew.ts +0 -89
  116. package/src/types/document-pointer.ts +0 -7
  117. package/src/types/frontend-action.ts +0 -213
  118. package/src/types/index.ts +0 -17
  119. package/src/types/interrupt-action.ts +0 -58
  120. package/src/types/system-message.ts +0 -4
  121. package/src/utils/dev-console.ts +0 -19
  122. package/src/utils/index.ts +0 -2
  123. package/src/utils/suggestions-constants.ts +0 -8
  124. package/src/utils/utils.test.ts +0 -7
  125. package/src/utils/utils.ts +0 -6
  126. package/src/v2/__tests__/A2UIMessageRenderer.test.tsx +0 -240
  127. package/src/v2/__tests__/globalSetup.ts +0 -14
  128. package/src/v2/__tests__/setup.ts +0 -93
  129. package/src/v2/__tests__/utils/test-helpers.tsx +0 -570
  130. package/src/v2/a2ui/A2UICatalogContext.tsx +0 -79
  131. package/src/v2/a2ui/A2UIMessageRenderer.tsx +0 -294
  132. package/src/v2/a2ui/A2UIToolCallRenderer.tsx +0 -290
  133. package/src/v2/components/CopilotKitInspector.tsx +0 -52
  134. package/src/v2/components/MCPAppsActivityRenderer.tsx +0 -815
  135. package/src/v2/components/OpenGenerativeUIRenderer.tsx +0 -598
  136. package/src/v2/components/WildcardToolCallRender.tsx +0 -86
  137. package/src/v2/components/__tests__/OpenGenerativeUIRenderer.test.tsx +0 -665
  138. package/src/v2/components/chat/CopilotChat.tsx +0 -664
  139. package/src/v2/components/chat/CopilotChatAssistantMessage.tsx +0 -393
  140. package/src/v2/components/chat/CopilotChatAttachmentQueue.tsx +0 -374
  141. package/src/v2/components/chat/CopilotChatAttachmentRenderer.tsx +0 -159
  142. package/src/v2/components/chat/CopilotChatAudioRecorder.tsx +0 -350
  143. package/src/v2/components/chat/CopilotChatInput.tsx +0 -1412
  144. package/src/v2/components/chat/CopilotChatMessageView.tsx +0 -716
  145. package/src/v2/components/chat/CopilotChatReasoningMessage.tsx +0 -265
  146. package/src/v2/components/chat/CopilotChatSuggestionPill.tsx +0 -59
  147. package/src/v2/components/chat/CopilotChatSuggestionView.tsx +0 -134
  148. package/src/v2/components/chat/CopilotChatToggleButton.tsx +0 -171
  149. package/src/v2/components/chat/CopilotChatToolCallsView.tsx +0 -40
  150. package/src/v2/components/chat/CopilotChatUserMessage.tsx +0 -445
  151. package/src/v2/components/chat/CopilotChatView.tsx +0 -890
  152. package/src/v2/components/chat/CopilotModalHeader.tsx +0 -129
  153. package/src/v2/components/chat/CopilotPopup.tsx +0 -81
  154. package/src/v2/components/chat/CopilotPopupView.tsx +0 -317
  155. package/src/v2/components/chat/CopilotSidebar.tsx +0 -80
  156. package/src/v2/components/chat/CopilotSidebarView.tsx +0 -269
  157. package/src/v2/components/chat/Lightbox.tsx +0 -103
  158. package/src/v2/components/chat/__tests__/CopilotChat.absentThreadConnect.test.tsx +0 -66
  159. package/src/v2/components/chat/__tests__/CopilotChat.attachments.test.tsx +0 -168
  160. package/src/v2/components/chat/__tests__/CopilotChat.e2e.test.tsx +0 -1239
  161. package/src/v2/components/chat/__tests__/CopilotChat.onError.test.tsx +0 -73
  162. package/src/v2/components/chat/__tests__/CopilotChat.slots.e2e.test.tsx +0 -432
  163. package/src/v2/components/chat/__tests__/CopilotChat.suggestionsAlways.test.tsx +0 -183
  164. package/src/v2/components/chat/__tests__/CopilotChat.welcomeGate.test.tsx +0 -184
  165. package/src/v2/components/chat/__tests__/CopilotChatActivityRendering.e2e.test.tsx +0 -649
  166. package/src/v2/components/chat/__tests__/CopilotChatAssistantMessage.slots.e2e.test.tsx +0 -624
  167. package/src/v2/components/chat/__tests__/CopilotChatAssistantMessage.test.tsx +0 -702
  168. package/src/v2/components/chat/__tests__/CopilotChatAssistantMessage.thumbs.test.tsx +0 -72
  169. package/src/v2/components/chat/__tests__/CopilotChatCopyButton.clipboard.test.tsx +0 -241
  170. package/src/v2/components/chat/__tests__/CopilotChatCssClasses.test.tsx +0 -107
  171. package/src/v2/components/chat/__tests__/CopilotChatInput.slots.e2e.test.tsx +0 -929
  172. package/src/v2/components/chat/__tests__/CopilotChatInput.test.tsx +0 -1567
  173. package/src/v2/components/chat/__tests__/CopilotChatMessageView.slots.e2e.test.tsx +0 -1004
  174. package/src/v2/components/chat/__tests__/CopilotChatMessageView.test.tsx +0 -279
  175. package/src/v2/components/chat/__tests__/CopilotChatPerf.e2e.test.tsx +0 -336
  176. package/src/v2/components/chat/__tests__/CopilotChatPropsRerender.e2e.test.tsx +0 -249
  177. package/src/v2/components/chat/__tests__/CopilotChatSuggestionView.slots.e2e.test.tsx +0 -530
  178. package/src/v2/components/chat/__tests__/CopilotChatToolRendering.e2e.test.tsx +0 -785
  179. package/src/v2/components/chat/__tests__/CopilotChatToolRerenders.e2e.test.tsx +0 -2416
  180. package/src/v2/components/chat/__tests__/CopilotChatUserMessage.slots.e2e.test.tsx +0 -621
  181. package/src/v2/components/chat/__tests__/CopilotChatView.connectingGate.test.tsx +0 -56
  182. package/src/v2/components/chat/__tests__/CopilotChatView.inputOverlay.test.tsx +0 -264
  183. package/src/v2/components/chat/__tests__/CopilotChatView.onClick.e2e.test.tsx +0 -853
  184. package/src/v2/components/chat/__tests__/CopilotChatView.pinToSend.test.tsx +0 -94
  185. package/src/v2/components/chat/__tests__/CopilotChatView.slots.e2e.test.tsx +0 -1050
  186. package/src/v2/components/chat/__tests__/CopilotModalHeader.slots.e2e.test.tsx +0 -484
  187. package/src/v2/components/chat/__tests__/CopilotPopupView.slots.e2e.test.tsx +0 -612
  188. package/src/v2/components/chat/__tests__/CopilotSidebarView.position.test.tsx +0 -159
  189. package/src/v2/components/chat/__tests__/CopilotSidebarView.slots.e2e.test.tsx +0 -502
  190. package/src/v2/components/chat/__tests__/MCPAppsActivityRenderer.e2e.test.tsx +0 -1068
  191. package/src/v2/components/chat/__tests__/MCPAppsProxy.e2e.test.tsx +0 -589
  192. package/src/v2/components/chat/__tests__/MCPAppsUiMessage.e2e.test.tsx +0 -403
  193. package/src/v2/components/chat/__tests__/copilot-chat-throttle.test.tsx +0 -137
  194. package/src/v2/components/chat/__tests__/normalize-auto-scroll.test.ts +0 -37
  195. package/src/v2/components/chat/__tests__/setup.ts +0 -1
  196. package/src/v2/components/chat/index.ts +0 -90
  197. package/src/v2/components/chat/last-user-message-context.ts +0 -21
  198. package/src/v2/components/chat/normalize-auto-scroll.ts +0 -17
  199. package/src/v2/components/chat/scroll-element-context.ts +0 -13
  200. package/src/v2/components/index.ts +0 -8
  201. package/src/v2/components/intelligence-indicator/IntelligenceIndicator.tsx +0 -286
  202. package/src/v2/components/intelligence-indicator/__tests__/IntelligenceIndicator.e2e.test.tsx +0 -464
  203. package/src/v2/components/intelligence-indicator/index.ts +0 -2
  204. package/src/v2/components/license-warning-banner.tsx +0 -217
  205. package/src/v2/components/ui/button.tsx +0 -124
  206. package/src/v2/components/ui/dropdown-menu.tsx +0 -258
  207. package/src/v2/components/ui/tooltip.tsx +0 -60
  208. package/src/v2/context.ts +0 -62
  209. package/src/v2/headless.ts +0 -64
  210. package/src/v2/hooks/__tests__/standard-schema-types.test.tsx +0 -152
  211. package/src/v2/hooks/__tests__/standard-schema.test.tsx +0 -282
  212. package/src/v2/hooks/__tests__/use-agent-context-timing.e2e.test.tsx +0 -140
  213. package/src/v2/hooks/__tests__/use-agent-context.test.tsx +0 -401
  214. package/src/v2/hooks/__tests__/use-agent-error-state.test.tsx +0 -44
  215. package/src/v2/hooks/__tests__/use-agent-stability.test.tsx +0 -211
  216. package/src/v2/hooks/__tests__/use-agent-throttle.test.tsx +0 -1029
  217. package/src/v2/hooks/__tests__/use-agent.e2e.test.tsx +0 -159
  218. package/src/v2/hooks/__tests__/use-attachments.test.tsx +0 -169
  219. package/src/v2/hooks/__tests__/use-capabilities.test.tsx +0 -76
  220. package/src/v2/hooks/__tests__/use-component.test.tsx +0 -126
  221. package/src/v2/hooks/__tests__/use-configure-suggestions.e2e.test.tsx +0 -696
  222. package/src/v2/hooks/__tests__/use-default-render-tool.test.tsx +0 -153
  223. package/src/v2/hooks/__tests__/use-frontend-tool-available.test.tsx +0 -167
  224. package/src/v2/hooks/__tests__/use-frontend-tool.e2e.test.tsx +0 -2148
  225. package/src/v2/hooks/__tests__/use-human-in-the-loop.e2e.test.tsx +0 -1261
  226. package/src/v2/hooks/__tests__/use-interrupt.test.tsx +0 -397
  227. package/src/v2/hooks/__tests__/use-katex-styles.test.tsx +0 -56
  228. package/src/v2/hooks/__tests__/use-keyboard-height.test.tsx +0 -192
  229. package/src/v2/hooks/__tests__/use-pin-to-send.test.tsx +0 -219
  230. package/src/v2/hooks/__tests__/use-render-custom-messages.test.tsx +0 -55
  231. package/src/v2/hooks/__tests__/use-render-tool.test.tsx +0 -259
  232. package/src/v2/hooks/__tests__/use-suggestions.e2e.test.tsx +0 -524
  233. package/src/v2/hooks/__tests__/use-threads.test.tsx +0 -757
  234. package/src/v2/hooks/__tests__/zod-regression.test.tsx +0 -311
  235. package/src/v2/hooks/index.ts +0 -24
  236. package/src/v2/hooks/use-agent-context.tsx +0 -45
  237. package/src/v2/hooks/use-agent.tsx +0 -227
  238. package/src/v2/hooks/use-attachments.tsx +0 -269
  239. package/src/v2/hooks/use-capabilities.tsx +0 -25
  240. package/src/v2/hooks/use-component.tsx +0 -91
  241. package/src/v2/hooks/use-configure-suggestions.tsx +0 -236
  242. package/src/v2/hooks/use-default-render-tool.tsx +0 -271
  243. package/src/v2/hooks/use-frontend-tool.tsx +0 -46
  244. package/src/v2/hooks/use-human-in-the-loop.tsx +0 -81
  245. package/src/v2/hooks/use-interrupt.tsx +0 -305
  246. package/src/v2/hooks/use-keyboard-height.tsx +0 -67
  247. package/src/v2/hooks/use-pin-to-send.ts +0 -94
  248. package/src/v2/hooks/use-render-activity-message.tsx +0 -72
  249. package/src/v2/hooks/use-render-custom-messages.tsx +0 -93
  250. package/src/v2/hooks/use-render-tool-call.tsx +0 -208
  251. package/src/v2/hooks/use-render-tool.tsx +0 -184
  252. package/src/v2/hooks/use-suggestions.tsx +0 -91
  253. package/src/v2/hooks/use-threads.tsx +0 -325
  254. package/src/v2/hooks/useKatexStyles.ts +0 -27
  255. package/src/v2/index.css +0 -1
  256. package/src/v2/index.ts +0 -27
  257. package/src/v2/lib/__tests__/completePartialMarkdown.test.ts +0 -495
  258. package/src/v2/lib/__tests__/processPartialHtml.test.ts +0 -112
  259. package/src/v2/lib/__tests__/renderSlot.test.tsx +0 -588
  260. package/src/v2/lib/__tests__/slots.test.ts +0 -56
  261. package/src/v2/lib/processPartialHtml.ts +0 -45
  262. package/src/v2/lib/react-core.ts +0 -156
  263. package/src/v2/lib/slots.tsx +0 -184
  264. package/src/v2/lib/transcription-client.ts +0 -184
  265. package/src/v2/lib/utils.ts +0 -8
  266. package/src/v2/providers/CopilotChatConfigurationProvider.tsx +0 -196
  267. package/src/v2/providers/CopilotKitProvider.tsx +0 -800
  268. package/src/v2/providers/SandboxFunctionsContext.ts +0 -10
  269. package/src/v2/providers/__tests__/CopilotChatConfigurationProvider.test.tsx +0 -652
  270. package/src/v2/providers/__tests__/CopilotKitProvider.license.test.tsx +0 -101
  271. package/src/v2/providers/__tests__/CopilotKitProvider.onError.test.tsx +0 -69
  272. package/src/v2/providers/__tests__/CopilotKitProvider.renderCustomMessages.e2e.test.tsx +0 -881
  273. package/src/v2/providers/__tests__/CopilotKitProvider.sandboxFunctions.test.tsx +0 -198
  274. package/src/v2/providers/__tests__/CopilotKitProvider.stability.test.tsx +0 -740
  275. package/src/v2/providers/__tests__/CopilotKitProvider.test.tsx +0 -713
  276. package/src/v2/providers/__tests__/CopilotKitProvider.wildcard.test.tsx +0 -294
  277. package/src/v2/providers/index.ts +0 -21
  278. package/src/v2/styles/globals.css +0 -349
  279. package/src/v2/types/__tests__/defineToolCallRenderer.test.tsx +0 -525
  280. package/src/v2/types/defineToolCallRenderer.ts +0 -68
  281. package/src/v2/types/frontend-tool.ts +0 -8
  282. package/src/v2/types/human-in-the-loop.ts +0 -33
  283. package/src/v2/types/index.ts +0 -8
  284. package/src/v2/types/interrupt.ts +0 -15
  285. package/src/v2/types/react-activity-message-renderer.ts +0 -27
  286. package/src/v2/types/react-custom-message-renderer.ts +0 -17
  287. package/src/v2/types/react-tool-call-renderer.ts +0 -35
  288. package/src/v2/types/sandbox-function.ts +0 -11
  289. package/tsconfig.json +0 -8
  290. package/tsdown.config.ts +0 -193
  291. package/typedoc.json +0 -4
  292. 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.