@brainfish-ai/devdoc 0.1.21

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 (268) hide show
  1. package/LICENSE +33 -0
  2. package/README.md +415 -0
  3. package/bin/devdoc.js +13 -0
  4. package/dist/cli/commands/build.d.ts +5 -0
  5. package/dist/cli/commands/build.js +87 -0
  6. package/dist/cli/commands/check.d.ts +1 -0
  7. package/dist/cli/commands/check.js +143 -0
  8. package/dist/cli/commands/create.d.ts +24 -0
  9. package/dist/cli/commands/create.js +387 -0
  10. package/dist/cli/commands/deploy.d.ts +9 -0
  11. package/dist/cli/commands/deploy.js +433 -0
  12. package/dist/cli/commands/dev.d.ts +6 -0
  13. package/dist/cli/commands/dev.js +139 -0
  14. package/dist/cli/commands/init.d.ts +11 -0
  15. package/dist/cli/commands/init.js +238 -0
  16. package/dist/cli/commands/keys.d.ts +12 -0
  17. package/dist/cli/commands/keys.js +165 -0
  18. package/dist/cli/commands/start.d.ts +5 -0
  19. package/dist/cli/commands/start.js +56 -0
  20. package/dist/cli/commands/upload.d.ts +13 -0
  21. package/dist/cli/commands/upload.js +238 -0
  22. package/dist/cli/commands/whoami.d.ts +8 -0
  23. package/dist/cli/commands/whoami.js +91 -0
  24. package/dist/cli/index.d.ts +1 -0
  25. package/dist/cli/index.js +106 -0
  26. package/dist/config/index.d.ts +80 -0
  27. package/dist/config/index.js +133 -0
  28. package/dist/constants.d.ts +9 -0
  29. package/dist/constants.js +13 -0
  30. package/dist/index.d.ts +7 -0
  31. package/dist/index.js +12 -0
  32. package/dist/utils/logger.d.ts +16 -0
  33. package/dist/utils/logger.js +61 -0
  34. package/dist/utils/paths.d.ts +16 -0
  35. package/dist/utils/paths.js +50 -0
  36. package/package.json +51 -0
  37. package/renderer/app/api/assets/[...path]/route.ts +123 -0
  38. package/renderer/app/api/assets/route.ts +124 -0
  39. package/renderer/app/api/assets/upload/route.ts +177 -0
  40. package/renderer/app/api/auth-schemes/route.ts +77 -0
  41. package/renderer/app/api/chat/route.ts +858 -0
  42. package/renderer/app/api/codegen/route.ts +72 -0
  43. package/renderer/app/api/collections/route.ts +1016 -0
  44. package/renderer/app/api/debug/route.ts +53 -0
  45. package/renderer/app/api/deploy/route.ts +234 -0
  46. package/renderer/app/api/device/route.ts +42 -0
  47. package/renderer/app/api/docs/route.ts +187 -0
  48. package/renderer/app/api/keys/regenerate/route.ts +80 -0
  49. package/renderer/app/api/openapi-spec/route.ts +151 -0
  50. package/renderer/app/api/projects/[slug]/route.ts +153 -0
  51. package/renderer/app/api/projects/[slug]/stats/route.ts +96 -0
  52. package/renderer/app/api/projects/register/route.ts +152 -0
  53. package/renderer/app/api/proxy/route.ts +149 -0
  54. package/renderer/app/api/proxy-stream/route.ts +168 -0
  55. package/renderer/app/api/redirects/route.ts +47 -0
  56. package/renderer/app/api/schema/route.ts +65 -0
  57. package/renderer/app/api/subdomains/check/route.ts +172 -0
  58. package/renderer/app/api/suggestions/route.ts +144 -0
  59. package/renderer/app/favicon.ico +0 -0
  60. package/renderer/app/globals.css +1103 -0
  61. package/renderer/app/layout.tsx +47 -0
  62. package/renderer/app/llms-full.txt/route.ts +346 -0
  63. package/renderer/app/llms.txt/route.ts +279 -0
  64. package/renderer/app/page.tsx +14 -0
  65. package/renderer/app/robots.txt/route.ts +84 -0
  66. package/renderer/app/sitemap.xml/route.ts +199 -0
  67. package/renderer/components/docs/index.ts +12 -0
  68. package/renderer/components/docs/mdx/accordion.tsx +169 -0
  69. package/renderer/components/docs/mdx/badge.tsx +132 -0
  70. package/renderer/components/docs/mdx/callouts.tsx +154 -0
  71. package/renderer/components/docs/mdx/cards.tsx +213 -0
  72. package/renderer/components/docs/mdx/changelog.tsx +120 -0
  73. package/renderer/components/docs/mdx/code-block.tsx +186 -0
  74. package/renderer/components/docs/mdx/code-group.tsx +421 -0
  75. package/renderer/components/docs/mdx/file-embeds.tsx +105 -0
  76. package/renderer/components/docs/mdx/frame.tsx +112 -0
  77. package/renderer/components/docs/mdx/highlight.tsx +151 -0
  78. package/renderer/components/docs/mdx/iframe.tsx +134 -0
  79. package/renderer/components/docs/mdx/image.tsx +235 -0
  80. package/renderer/components/docs/mdx/index.ts +204 -0
  81. package/renderer/components/docs/mdx/mermaid.tsx +240 -0
  82. package/renderer/components/docs/mdx/param-field.tsx +200 -0
  83. package/renderer/components/docs/mdx/steps.tsx +113 -0
  84. package/renderer/components/docs/mdx/tabs.tsx +86 -0
  85. package/renderer/components/docs/mdx-renderer.tsx +100 -0
  86. package/renderer/components/docs/navigation/breadcrumbs.tsx +76 -0
  87. package/renderer/components/docs/navigation/index.ts +8 -0
  88. package/renderer/components/docs/navigation/page-nav.tsx +64 -0
  89. package/renderer/components/docs/navigation/sidebar.tsx +515 -0
  90. package/renderer/components/docs/navigation/toc.tsx +113 -0
  91. package/renderer/components/docs/notice.tsx +105 -0
  92. package/renderer/components/docs-header.tsx +274 -0
  93. package/renderer/components/docs-viewer/agent/agent-chat.tsx +2076 -0
  94. package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +90 -0
  95. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +49 -0
  96. package/renderer/components/docs-viewer/agent/cards/index.tsx +50 -0
  97. package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +212 -0
  98. package/renderer/components/docs-viewer/agent/cards/types.ts +84 -0
  99. package/renderer/components/docs-viewer/agent/chat-message.tsx +17 -0
  100. package/renderer/components/docs-viewer/agent/index.tsx +6 -0
  101. package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +119 -0
  102. package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +46 -0
  103. package/renderer/components/docs-viewer/agent/messages/index.ts +17 -0
  104. package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +721 -0
  105. package/renderer/components/docs-viewer/agent/messages/types.ts +61 -0
  106. package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +24 -0
  107. package/renderer/components/docs-viewer/agent/messages/user-message.tsx +51 -0
  108. package/renderer/components/docs-viewer/code-editor/index.tsx +2 -0
  109. package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +1283 -0
  110. package/renderer/components/docs-viewer/content/changelog-page.tsx +331 -0
  111. package/renderer/components/docs-viewer/content/doc-page.tsx +285 -0
  112. package/renderer/components/docs-viewer/content/documentation-viewer.tsx +17 -0
  113. package/renderer/components/docs-viewer/content/index.tsx +29 -0
  114. package/renderer/components/docs-viewer/content/introduction.tsx +21 -0
  115. package/renderer/components/docs-viewer/content/request-details.tsx +330 -0
  116. package/renderer/components/docs-viewer/content/sections/auth.tsx +69 -0
  117. package/renderer/components/docs-viewer/content/sections/body.tsx +66 -0
  118. package/renderer/components/docs-viewer/content/sections/headers.tsx +43 -0
  119. package/renderer/components/docs-viewer/content/sections/overview.tsx +40 -0
  120. package/renderer/components/docs-viewer/content/sections/parameters.tsx +43 -0
  121. package/renderer/components/docs-viewer/content/sections/responses.tsx +87 -0
  122. package/renderer/components/docs-viewer/global-auth-modal.tsx +352 -0
  123. package/renderer/components/docs-viewer/index.tsx +1466 -0
  124. package/renderer/components/docs-viewer/playground/auth-editor.tsx +280 -0
  125. package/renderer/components/docs-viewer/playground/body-editor.tsx +221 -0
  126. package/renderer/components/docs-viewer/playground/code-editor.tsx +224 -0
  127. package/renderer/components/docs-viewer/playground/code-snippet.tsx +387 -0
  128. package/renderer/components/docs-viewer/playground/graphql-playground.tsx +745 -0
  129. package/renderer/components/docs-viewer/playground/index.tsx +671 -0
  130. package/renderer/components/docs-viewer/playground/key-value-editor.tsx +261 -0
  131. package/renderer/components/docs-viewer/playground/method-selector.tsx +60 -0
  132. package/renderer/components/docs-viewer/playground/request-builder.tsx +179 -0
  133. package/renderer/components/docs-viewer/playground/request-tabs.tsx +237 -0
  134. package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +21 -0
  135. package/renderer/components/docs-viewer/playground/response-cards/index.tsx +93 -0
  136. package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +16 -0
  137. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +23 -0
  138. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +268 -0
  139. package/renderer/components/docs-viewer/playground/response-cards/types.ts +82 -0
  140. package/renderer/components/docs-viewer/playground/response-viewer.tsx +43 -0
  141. package/renderer/components/docs-viewer/search/index.ts +2 -0
  142. package/renderer/components/docs-viewer/search/search-dialog.tsx +331 -0
  143. package/renderer/components/docs-viewer/search/use-search.ts +117 -0
  144. package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +431 -0
  145. package/renderer/components/docs-viewer/shared/method-badge.tsx +41 -0
  146. package/renderer/components/docs-viewer/shared/schema-viewer.tsx +349 -0
  147. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +239 -0
  148. package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +316 -0
  149. package/renderer/components/docs-viewer/sidebar/index.tsx +343 -0
  150. package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +202 -0
  151. package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +118 -0
  152. package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +226 -0
  153. package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +52 -0
  154. package/renderer/components/theme-provider.tsx +11 -0
  155. package/renderer/components/theme-toggle.tsx +76 -0
  156. package/renderer/components/ui/badge.tsx +46 -0
  157. package/renderer/components/ui/button.tsx +59 -0
  158. package/renderer/components/ui/dialog.tsx +118 -0
  159. package/renderer/components/ui/dropdown-menu.tsx +257 -0
  160. package/renderer/components/ui/input.tsx +21 -0
  161. package/renderer/components/ui/label.tsx +24 -0
  162. package/renderer/components/ui/navigation-menu.tsx +168 -0
  163. package/renderer/components/ui/select.tsx +190 -0
  164. package/renderer/components/ui/spinner.tsx +114 -0
  165. package/renderer/components/ui/tabs.tsx +66 -0
  166. package/renderer/components/ui/tooltip.tsx +61 -0
  167. package/renderer/hooks/use-code-copy.ts +88 -0
  168. package/renderer/hooks/use-openapi-title.ts +44 -0
  169. package/renderer/lib/api-docs/agent/index.ts +6 -0
  170. package/renderer/lib/api-docs/agent/indexer.ts +323 -0
  171. package/renderer/lib/api-docs/agent/spec-summary.ts +335 -0
  172. package/renderer/lib/api-docs/agent/types.ts +116 -0
  173. package/renderer/lib/api-docs/auth/auth-context.tsx +225 -0
  174. package/renderer/lib/api-docs/auth/auth-storage.ts +87 -0
  175. package/renderer/lib/api-docs/auth/crypto.ts +89 -0
  176. package/renderer/lib/api-docs/auth/index.ts +4 -0
  177. package/renderer/lib/api-docs/code-editor/db.ts +164 -0
  178. package/renderer/lib/api-docs/code-editor/hooks.ts +266 -0
  179. package/renderer/lib/api-docs/code-editor/index.ts +6 -0
  180. package/renderer/lib/api-docs/code-editor/mode-context.tsx +207 -0
  181. package/renderer/lib/api-docs/code-editor/types.ts +105 -0
  182. package/renderer/lib/api-docs/codegen/definitions.ts +297 -0
  183. package/renderer/lib/api-docs/codegen/har.ts +251 -0
  184. package/renderer/lib/api-docs/codegen/index.ts +159 -0
  185. package/renderer/lib/api-docs/factories.ts +151 -0
  186. package/renderer/lib/api-docs/index.ts +17 -0
  187. package/renderer/lib/api-docs/mobile-context.tsx +112 -0
  188. package/renderer/lib/api-docs/navigation-context.tsx +88 -0
  189. package/renderer/lib/api-docs/parsers/graphql/README.md +129 -0
  190. package/renderer/lib/api-docs/parsers/graphql/index.ts +91 -0
  191. package/renderer/lib/api-docs/parsers/graphql/parser.ts +491 -0
  192. package/renderer/lib/api-docs/parsers/graphql/transformer.ts +246 -0
  193. package/renderer/lib/api-docs/parsers/graphql/types.ts +283 -0
  194. package/renderer/lib/api-docs/parsers/openapi/README.md +32 -0
  195. package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +60 -0
  196. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +574 -0
  197. package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +403 -0
  198. package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +232 -0
  199. package/renderer/lib/api-docs/parsers/openapi/index.ts +171 -0
  200. package/renderer/lib/api-docs/parsers/openapi/transformer.ts +277 -0
  201. package/renderer/lib/api-docs/parsers/openapi/validator.ts +31 -0
  202. package/renderer/lib/api-docs/playground/context.tsx +107 -0
  203. package/renderer/lib/api-docs/playground/navigation-context.tsx +124 -0
  204. package/renderer/lib/api-docs/playground/request-builder.ts +223 -0
  205. package/renderer/lib/api-docs/playground/request-runner.ts +282 -0
  206. package/renderer/lib/api-docs/playground/types.ts +35 -0
  207. package/renderer/lib/api-docs/types.ts +269 -0
  208. package/renderer/lib/api-docs/utils.ts +311 -0
  209. package/renderer/lib/cache.ts +193 -0
  210. package/renderer/lib/docs/config/index.ts +29 -0
  211. package/renderer/lib/docs/config/loader.ts +142 -0
  212. package/renderer/lib/docs/config/schema.ts +298 -0
  213. package/renderer/lib/docs/index.ts +12 -0
  214. package/renderer/lib/docs/mdx/compiler.ts +176 -0
  215. package/renderer/lib/docs/mdx/frontmatter.ts +80 -0
  216. package/renderer/lib/docs/mdx/index.ts +26 -0
  217. package/renderer/lib/docs/navigation/generator.ts +348 -0
  218. package/renderer/lib/docs/navigation/index.ts +12 -0
  219. package/renderer/lib/docs/navigation/types.ts +123 -0
  220. package/renderer/lib/docs-navigation-context.tsx +80 -0
  221. package/renderer/lib/multi-tenant/context.ts +105 -0
  222. package/renderer/lib/storage/blob.ts +845 -0
  223. package/renderer/lib/utils.ts +6 -0
  224. package/renderer/next.config.ts +76 -0
  225. package/renderer/package.json +66 -0
  226. package/renderer/postcss.config.mjs +5 -0
  227. package/renderer/public/assets/images/screenshot.png +0 -0
  228. package/renderer/public/assets/logo/dark.svg +9 -0
  229. package/renderer/public/assets/logo/light.svg +9 -0
  230. package/renderer/public/assets/logo.svg +9 -0
  231. package/renderer/public/file.svg +1 -0
  232. package/renderer/public/globe.svg +1 -0
  233. package/renderer/public/icon.png +0 -0
  234. package/renderer/public/logo.svg +9 -0
  235. package/renderer/public/window.svg +1 -0
  236. package/renderer/tsconfig.json +28 -0
  237. package/templates/basic/README.md +139 -0
  238. package/templates/basic/assets/favicon.svg +4 -0
  239. package/templates/basic/assets/logo.svg +9 -0
  240. package/templates/basic/docs.json +47 -0
  241. package/templates/basic/guides/configuration.mdx +149 -0
  242. package/templates/basic/guides/overview.mdx +96 -0
  243. package/templates/basic/index.mdx +39 -0
  244. package/templates/basic/package.json +14 -0
  245. package/templates/basic/quickstart.mdx +92 -0
  246. package/templates/basic/vercel.json +6 -0
  247. package/templates/graphql/README.md +139 -0
  248. package/templates/graphql/api-reference/schema.graphql +305 -0
  249. package/templates/graphql/assets/favicon.svg +4 -0
  250. package/templates/graphql/assets/logo.svg +9 -0
  251. package/templates/graphql/docs.json +54 -0
  252. package/templates/graphql/guides/configuration.mdx +149 -0
  253. package/templates/graphql/guides/overview.mdx +96 -0
  254. package/templates/graphql/index.mdx +39 -0
  255. package/templates/graphql/package.json +14 -0
  256. package/templates/graphql/quickstart.mdx +92 -0
  257. package/templates/graphql/vercel.json +6 -0
  258. package/templates/openapi/README.md +139 -0
  259. package/templates/openapi/api-reference/openapi.json +419 -0
  260. package/templates/openapi/assets/favicon.svg +4 -0
  261. package/templates/openapi/assets/logo.svg +9 -0
  262. package/templates/openapi/docs.json +61 -0
  263. package/templates/openapi/guides/configuration.mdx +149 -0
  264. package/templates/openapi/guides/overview.mdx +96 -0
  265. package/templates/openapi/index.mdx +39 -0
  266. package/templates/openapi/package.json +14 -0
  267. package/templates/openapi/quickstart.mdx +92 -0
  268. package/templates/openapi/vercel.json +6 -0
@@ -0,0 +1,721 @@
1
+ 'use client'
2
+
3
+ import {
4
+ MagnifyingGlass,
5
+ CheckCircle,
6
+ XCircle,
7
+ FilePlus,
8
+ FolderPlus,
9
+ Note,
10
+ File,
11
+ PaperPlaneTilt,
12
+ CheckSquare,
13
+ Warning,
14
+ Trash,
15
+ TestTube,
16
+ ArrowSquareOut,
17
+ Clock,
18
+ } from '@phosphor-icons/react'
19
+ import { MethodBadge } from '../../shared/method-badge'
20
+ import type { HTTPMethod } from '@/lib/api-docs/types'
21
+ import type { ToolCallDisplayProps, EndpointIndex, PrefillData } from './types'
22
+ import type { RequestValidationResult } from '@/lib/api-docs/agent/types'
23
+ import { getFileType } from '@/lib/api-docs/code-editor'
24
+
25
+ export function ToolCallDisplay({ tool, onNavigate, onNavigateToDocPage, onOpenFile }: ToolCallDisplayProps) {
26
+ const { toolName, args, result, state } = tool
27
+
28
+ // Helper to render clickable file name
29
+ const ClickableFileName = ({ path, className = '' }: { path?: string; className?: string }) => {
30
+ if (!path) return <span className="font-mono text-muted-foreground">unknown</span>
31
+ const fileName = path.split('/').pop() || path
32
+ return (
33
+ <button
34
+ onClick={() => onOpenFile?.(path)}
35
+ className={`font-mono font-medium text-foreground hover:text-primary hover:underline cursor-pointer transition-colors ${className}`}
36
+ >
37
+ {fileName}
38
+ </button>
39
+ )
40
+ }
41
+
42
+ // Helper to handle search result click
43
+ const handleResultClick = (item: { type: string; id: string }) => {
44
+ if (item.type === 'endpoint') {
45
+ onNavigate?.(item.id)
46
+ } else if (item.type === 'doc') {
47
+ onNavigateToDocPage?.(item.id)
48
+ } else if (item.type === 'changelog') {
49
+ onNavigateToDocPage?.(`changelog/${item.id}`)
50
+ }
51
+ }
52
+
53
+ // Loading state - with subtle background
54
+ if (state === 'pending') {
55
+ return (
56
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
57
+ <div className="size-3.5 rounded-full border-2 border-primary/30 border-t-primary animate-spin" />
58
+ <span className="text-muted-foreground">
59
+ {toolName === 'search' && 'Searching...'}
60
+ {toolName === 'navigate' && 'Navigating...'}
61
+ {toolName === 'search_endpoints' && 'Searching...'}
62
+ {toolName === 'navigate_to_endpoint' && 'Navigating...'}
63
+ {toolName === 'prefill_parameters' && 'Setting params...'}
64
+ {toolName === 'get_endpoint_details' && 'Loading details...'}
65
+ {toolName === 'check_request_validity' && 'Validating request...'}
66
+ {toolName === 'send_request' && (
67
+ <span className="flex items-center gap-1.5">
68
+ <PaperPlaneTilt className="size-3.5 text-blue-500 animate-pulse" weight="fill" />
69
+ Sending request...
70
+ </span>
71
+ )}
72
+ {toolName === 'switch_to_notes' && 'Opening Notes...'}
73
+ {toolName === 'switch_to_docs' && 'Opening Docs...'}
74
+ {toolName === 'switch_to_api_client' && 'Opening API Client...'}
75
+ {toolName === 'navigate_to_doc_section' && 'Navigating to section...'}
76
+ {toolName === 'navigate_to_doc_page' && 'Navigating to page...'}
77
+ {toolName === 'search_docs' && 'Searching documentation...'}
78
+ {toolName === 'scroll_to_section' && 'Scrolling to section...'}
79
+ {toolName === 'list_notes' && 'Checking existing files...'}
80
+ {toolName === 'open_file' && `Opening ${(args.path as string)?.split('/').pop() || 'file'}...`}
81
+ {toolName === 'create_folder' && `Creating folder ${args.path as string || ''}...`}
82
+ {toolName === 'create_file' && `Creating ${(args.path as string)?.split('/').pop() || 'file'}...`}
83
+ {toolName === 'write_file' && `Writing ${(args.path as string)?.split('/').pop() || 'file'}...`}
84
+ {toolName === 'delete_file' && `Deleting ${(args.path as string)?.split('/').pop() || 'file'}...`}
85
+ {toolName === 'delete_all_notes' && 'Deleting all notes...'}
86
+ </span>
87
+ </div>
88
+ )
89
+ }
90
+
91
+ // Error state
92
+ if (state === 'error') {
93
+ return (
94
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-destructive/10 border border-destructive/20">
95
+ <XCircle className="size-3.5 text-destructive" weight="fill" />
96
+ <span className="text-destructive">Failed</span>
97
+ </div>
98
+ )
99
+ }
100
+
101
+ // === UNIFIED TOOLS ===
102
+
103
+ // Unified search
104
+ if (toolName === 'search') {
105
+ const resultData = result as { results?: Array<{ type: string; id: string; title: string; method?: string; group?: string }>; message?: string } | undefined
106
+ const results = resultData?.results || []
107
+
108
+ if (results.length === 0) {
109
+ return (
110
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
111
+ <MagnifyingGlass className="size-3.5 text-muted-foreground" weight="bold" />
112
+ <span className="text-muted-foreground">No results found</span>
113
+ </div>
114
+ )
115
+ }
116
+
117
+ if (results.length === 1) {
118
+ const item = results[0]
119
+ return (
120
+ <button
121
+ onClick={() => handleResultClick(item)}
122
+ className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50 hover:bg-muted/80 hover:border-border transition-colors cursor-pointer w-full text-left"
123
+ >
124
+ <CheckCircle className="size-3.5 text-green-600 dark:text-green-500" weight="fill" />
125
+ <span className="text-muted-foreground">Found</span>
126
+ {item.method && <MethodBadge method={item.method as HTTPMethod} size="sm" />}
127
+ <span className="font-medium truncate">{item.title}</span>
128
+ </button>
129
+ )
130
+ }
131
+
132
+ return (
133
+ <div className="py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
134
+ <div className="flex items-center gap-2 text-xs text-muted-foreground mb-1.5">
135
+ <MagnifyingGlass className="size-3.5" weight="bold" />
136
+ <span>Found {results.length} results</span>
137
+ </div>
138
+ <div className="space-y-0.5 ml-5">
139
+ {results.slice(0, 4).map((item, i) => (
140
+ <button
141
+ key={i}
142
+ onClick={() => handleResultClick(item)}
143
+ className="flex items-center gap-1.5 text-xs py-1 px-1.5 -ml-1.5 rounded text-muted-foreground hover:bg-muted/80 transition-colors cursor-pointer w-full text-left"
144
+ >
145
+ {item.type === 'endpoint' && item.method && (
146
+ <MethodBadge method={item.method as HTTPMethod} size="sm" />
147
+ )}
148
+ {item.type === 'doc' && (
149
+ <File className="size-3 text-purple-500" weight="fill" />
150
+ )}
151
+ {item.type === 'changelog' && (
152
+ <Clock className="size-3 text-orange-500" weight="fill" />
153
+ )}
154
+ <span className="truncate font-medium text-foreground">{item.title}</span>
155
+ {item.group && <span className="text-muted-foreground/60 text-[10px]">({item.group})</span>}
156
+ </button>
157
+ ))}
158
+ {results.length > 4 && (
159
+ <span className="text-[11px] text-muted-foreground">+{results.length - 4} more</span>
160
+ )}
161
+ </div>
162
+ </div>
163
+ )
164
+ }
165
+
166
+ // Unified navigate
167
+ if (toolName === 'navigate') {
168
+ const resultData = result as { success?: boolean; type?: string; title?: string; method?: string; message?: string } | undefined
169
+
170
+ if (!resultData?.success) {
171
+ return (
172
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
173
+ <XCircle className="size-3.5 text-muted-foreground" />
174
+ <span className="text-muted-foreground">{resultData?.message || 'Navigation failed'}</span>
175
+ </div>
176
+ )
177
+ }
178
+
179
+ return (
180
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
181
+ <CheckCircle className="size-3.5 text-green-600 dark:text-green-500" weight="fill" />
182
+ <span className="text-muted-foreground">Opened</span>
183
+ {resultData.method && <MethodBadge method={resultData.method as HTTPMethod} size="sm" />}
184
+ {resultData.type === 'changelog' && <Clock className="size-3 text-orange-500" weight="fill" />}
185
+ {resultData.type === 'doc' && <File className="size-3 text-purple-500" weight="fill" />}
186
+ <span className="font-medium truncate">{resultData.title}</span>
187
+ </div>
188
+ )
189
+ }
190
+
191
+ // Navigate to endpoint
192
+ if (toolName === 'navigate_to_endpoint') {
193
+ const resultData = result as { success?: boolean; endpoint?: EndpointIndex; reason?: string } | undefined
194
+ const endpoint = resultData?.endpoint
195
+
196
+ if (!endpoint) return null
197
+
198
+ return (
199
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
200
+ <CheckCircle className="size-3.5 text-green-600 dark:text-green-500" weight="fill" />
201
+ <span className="text-muted-foreground">Opened</span>
202
+ <MethodBadge method={endpoint.method as HTTPMethod} size="sm" />
203
+ <span className="font-medium truncate">{endpoint.name}</span>
204
+ </div>
205
+ )
206
+ }
207
+
208
+ // Prefill parameters
209
+ if (toolName === 'prefill_parameters') {
210
+ const prefillData = (result as { params?: Array<{key: string; value: string}>; headers?: Array<{key: string; value: string}>; body?: string; pathVariables?: Array<{key: string; value: string}> }) || args as PrefillData
211
+ const items: string[] = []
212
+ if (prefillData.pathVariables?.length) items.push(`${prefillData.pathVariables.length} path vars`)
213
+ if (prefillData.params?.length) items.push(`${prefillData.params.length} params`)
214
+ if (prefillData.headers?.length) items.push(`${prefillData.headers.length} headers`)
215
+ if (prefillData.body) items.push('body')
216
+
217
+ return (
218
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
219
+ <CheckCircle className="size-3.5 text-green-600 dark:text-green-500" weight="fill" />
220
+ <span className="text-muted-foreground">Set {items.join(', ')}</span>
221
+ </div>
222
+ )
223
+ }
224
+
225
+ // Search endpoints
226
+ if (toolName === 'search_endpoints') {
227
+ const results = result as EndpointIndex[] | undefined
228
+
229
+ if (!results || results.length === 0) return null
230
+ if (results.length === 1) {
231
+ return (
232
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
233
+ <CheckCircle className="size-3.5 text-green-600 dark:text-green-500" weight="fill" />
234
+ <span className="text-muted-foreground">Found</span>
235
+ <MethodBadge method={results[0].method as HTTPMethod} size="sm" />
236
+ <button
237
+ onClick={() => onNavigate?.(results[0].id)}
238
+ className="font-medium truncate hover:underline"
239
+ >
240
+ {results[0].name}
241
+ </button>
242
+ </div>
243
+ )
244
+ }
245
+
246
+ return (
247
+ <div className="py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
248
+ <div className="flex items-center gap-2 text-xs text-muted-foreground mb-1.5">
249
+ <MagnifyingGlass className="size-3.5" weight="bold" />
250
+ <span>Found {results.length} endpoints</span>
251
+ </div>
252
+ <div className="space-y-0.5 ml-5">
253
+ {results.slice(0, 4).map((ep) => (
254
+ <button
255
+ key={ep.id}
256
+ onClick={() => onNavigate?.(ep.id)}
257
+ className="flex items-center gap-1.5 text-xs py-0.5 hover:text-foreground transition-colors text-muted-foreground"
258
+ >
259
+ <MethodBadge method={ep.method as HTTPMethod} size="sm" />
260
+ <span className="truncate">{ep.name}</span>
261
+ </button>
262
+ ))}
263
+ {results.length > 4 && (
264
+ <span className="text-[11px] text-muted-foreground">+{results.length - 4} more</span>
265
+ )}
266
+ </div>
267
+ </div>
268
+ )
269
+ }
270
+
271
+ // Get endpoint details
272
+ if (toolName === 'get_endpoint_details') {
273
+ const endpoint = result as {
274
+ id?: string
275
+ name?: string
276
+ method?: string
277
+ path?: string
278
+ description?: string | null
279
+ } | undefined
280
+
281
+ if (!endpoint) return null
282
+
283
+ return (
284
+ <div className="py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
285
+ <div className="flex items-center gap-2 text-xs">
286
+ <MethodBadge method={(endpoint.method || 'GET') as HTTPMethod} size="sm" />
287
+ <span className="font-medium">{endpoint.name}</span>
288
+ </div>
289
+ <code className="text-[11px] text-muted-foreground block truncate mt-0.5">{endpoint.path}</code>
290
+ </div>
291
+ )
292
+ }
293
+
294
+ // Switch to notes
295
+ if (toolName === 'switch_to_notes') {
296
+ return (
297
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
298
+ <Note className="size-3.5 text-green-600 dark:text-green-500" weight="fill" />
299
+ <span className="text-muted-foreground">Opened Notes workspace</span>
300
+ </div>
301
+ )
302
+ }
303
+
304
+ // Switch to docs
305
+ if (toolName === 'switch_to_docs') {
306
+ return (
307
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
308
+ <File className="size-3.5 text-purple-600 dark:text-purple-500" weight="fill" />
309
+ <span className="text-muted-foreground">Opened Docs</span>
310
+ </div>
311
+ )
312
+ }
313
+
314
+ // Switch to API client
315
+ if (toolName === 'switch_to_api_client') {
316
+ return (
317
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
318
+ <TestTube className="size-3.5 text-blue-600 dark:text-blue-500" weight="fill" />
319
+ <span className="text-muted-foreground">Opened API Client</span>
320
+ </div>
321
+ )
322
+ }
323
+
324
+ // Navigate to documentation section
325
+ if (toolName === 'navigate_to_doc_section') {
326
+ const sectionName = args.sectionName as string || 'section'
327
+ return (
328
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
329
+ <ArrowSquareOut className="size-3.5 text-purple-600 dark:text-purple-500" weight="fill" />
330
+ <span className="text-muted-foreground">Navigated to <span className="font-medium text-foreground">{sectionName}</span></span>
331
+ </div>
332
+ )
333
+ }
334
+
335
+ // Navigate to documentation page
336
+ if (toolName === 'navigate_to_doc_page') {
337
+ const resultData = result as { success?: boolean; title?: string; slug?: string } | undefined
338
+ const title = args.title as string || resultData?.title || 'page'
339
+ return (
340
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
341
+ <File className="size-3.5 text-purple-600 dark:text-purple-500" weight="fill" />
342
+ <span className="text-muted-foreground">
343
+ Opened <span className="font-medium text-foreground">{title}</span>
344
+ </span>
345
+ </div>
346
+ )
347
+ }
348
+
349
+ // Search documentation pages
350
+ if (toolName === 'search_docs') {
351
+ const resultData = result as { results?: Array<{ title: string; slug: string; group: string }>; message?: string } | undefined
352
+ const results = resultData?.results || []
353
+
354
+ if (results.length === 0) {
355
+ return (
356
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
357
+ <MagnifyingGlass className="size-3.5 text-muted-foreground" weight="bold" />
358
+ <span className="text-muted-foreground">No documentation pages found</span>
359
+ </div>
360
+ )
361
+ }
362
+
363
+ if (results.length === 1) {
364
+ return (
365
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
366
+ <CheckCircle className="size-3.5 text-green-600 dark:text-green-500" weight="fill" />
367
+ <span className="text-muted-foreground">Found</span>
368
+ <span className="font-medium truncate">{results[0].title}</span>
369
+ <span className="text-muted-foreground/60">in {results[0].group}</span>
370
+ </div>
371
+ )
372
+ }
373
+
374
+ return (
375
+ <div className="py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
376
+ <div className="flex items-center gap-2 text-xs text-muted-foreground mb-1.5">
377
+ <MagnifyingGlass className="size-3.5" weight="bold" />
378
+ <span>Found {results.length} documentation pages</span>
379
+ </div>
380
+ <div className="space-y-0.5 ml-5">
381
+ {results.slice(0, 4).map((doc, i) => (
382
+ <div
383
+ key={i}
384
+ className="flex items-center gap-1.5 text-xs py-0.5 text-muted-foreground"
385
+ >
386
+ <File className="size-3 text-purple-500" weight="fill" />
387
+ <span className="truncate font-medium text-foreground">{doc.title}</span>
388
+ <span className="text-muted-foreground/60 text-[10px]">({doc.group})</span>
389
+ </div>
390
+ ))}
391
+ {results.length > 4 && (
392
+ <span className="text-[11px] text-muted-foreground">+{results.length - 4} more</span>
393
+ )}
394
+ </div>
395
+ </div>
396
+ )
397
+ }
398
+
399
+ // Scroll to section within page
400
+ if (toolName === 'scroll_to_section') {
401
+ const resultData = result as { success?: boolean; heading?: string; id?: string; message?: string } | undefined
402
+
403
+ if (!resultData?.success) {
404
+ return (
405
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
406
+ <MagnifyingGlass className="size-3.5 text-muted-foreground" weight="bold" />
407
+ <span className="text-muted-foreground">{resultData?.message || 'Section not found'}</span>
408
+ </div>
409
+ )
410
+ }
411
+
412
+ return (
413
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
414
+ <ArrowSquareOut className="size-3.5 text-yellow-600 dark:text-yellow-500" weight="fill" />
415
+ <span className="text-muted-foreground">
416
+ Scrolled to <span className="font-medium text-foreground">{resultData.heading}</span>
417
+ </span>
418
+ </div>
419
+ )
420
+ }
421
+
422
+ // List notes (check for existing files)
423
+ if (toolName === 'list_notes') {
424
+ const resultData = result as { success?: boolean; count?: number; files?: Array<{ path: string }>; message?: string } | undefined
425
+ const count = resultData?.count ?? 0
426
+ const files = resultData?.files ?? []
427
+
428
+ // File type color mapping - matches notes-mode.tsx colors exactly
429
+ const getFileColor = (path: string) => {
430
+ const type = getFileType(path)
431
+ const colorMap: Record<string, string> = {
432
+ 'javascript': 'text-yellow-400',
433
+ 'typescript': 'text-blue-400',
434
+ 'python': 'text-green-500',
435
+ 'go': 'text-cyan-400',
436
+ 'ruby': 'text-red-400',
437
+ 'php': 'text-purple-400',
438
+ 'shell': 'text-green-400',
439
+ 'markdown': 'text-blue-300',
440
+ 'mermaid': 'text-pink-400',
441
+ 'json': 'text-yellow-500',
442
+ 'yaml': 'text-red-300',
443
+ 'text': 'text-zinc-400',
444
+ }
445
+ return colorMap[type] || 'text-zinc-400'
446
+ }
447
+
448
+ return (
449
+ <div className="text-xs py-2 px-2.5 rounded-md bg-muted/50 border border-border/50">
450
+ <div className="flex items-center gap-2 mb-1.5">
451
+ <MagnifyingGlass className="size-3.5 text-violet-600 dark:text-violet-500 flex-shrink-0" weight="fill" />
452
+ <span className="text-muted-foreground">
453
+ {count > 0 ? `Found ${count} existing file${count !== 1 ? 's' : ''}` : 'No existing files found'}
454
+ </span>
455
+ </div>
456
+ {count > 0 && files.length > 0 && (
457
+ <div className="ml-5 space-y-1">
458
+ {files.slice(0, 5).map((f, i) => {
459
+ const fileName = f.path.split('/').pop() || f.path
460
+ const folder = f.path.includes('/') ? f.path.split('/').slice(0, -1).join('/') + '/' : ''
461
+ return (
462
+ <button
463
+ key={i}
464
+ onClick={() => onOpenFile?.(f.path)}
465
+ className="flex items-center gap-1.5 hover:bg-muted/50 rounded px-1 -mx-1 py-0.5 transition-colors text-left w-full"
466
+ >
467
+ <File className={`size-3.5 ${getFileColor(f.path)} flex-shrink-0`} weight="fill" />
468
+ {folder && <span className="text-muted-foreground/60">{folder}</span>}
469
+ <span className="font-mono text-foreground/80 hover:text-primary transition-colors">{fileName}</span>
470
+ </button>
471
+ )
472
+ })}
473
+ {files.length > 5 && (
474
+ <div className="text-muted-foreground/60 italic pl-5">...and {files.length - 5} more</div>
475
+ )}
476
+ </div>
477
+ )}
478
+ </div>
479
+ )
480
+ }
481
+
482
+ // Open existing file
483
+ if (toolName === 'open_file') {
484
+ const path = args.path as string
485
+
486
+ return (
487
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
488
+ <File className="size-3.5 text-blue-600 dark:text-blue-500" weight="fill" />
489
+ <span className="text-muted-foreground">Opened </span>
490
+ <ClickableFileName path={path} />
491
+ </div>
492
+ )
493
+ }
494
+
495
+ // Create folder
496
+ if (toolName === 'create_folder') {
497
+ const resultData = result as { success?: boolean; path?: string; description?: string } | undefined
498
+ const path = args.path as string || resultData?.path
499
+ const description = args.description as string || resultData?.description
500
+
501
+ return (
502
+ <div className="flex items-start gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
503
+ <FolderPlus className="size-3.5 text-amber-600 dark:text-amber-500 mt-0.5 flex-shrink-0" weight="fill" />
504
+ <div>
505
+ <span className="text-muted-foreground">Created folder </span>
506
+ <span className="font-mono font-medium text-foreground">{path}</span>
507
+ {description && (
508
+ <p className="text-muted-foreground mt-0.5">{description}</p>
509
+ )}
510
+ </div>
511
+ </div>
512
+ )
513
+ }
514
+
515
+ // Create file (empty)
516
+ if (toolName === 'create_file') {
517
+ const resultData = result as { success?: boolean; path?: string; description?: string } | undefined
518
+ const path = args.path as string || resultData?.path
519
+ const description = args.description as string || resultData?.description
520
+
521
+ return (
522
+ <div className="flex items-start gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
523
+ <FilePlus className="size-3.5 text-blue-600 dark:text-blue-500 mt-0.5 flex-shrink-0" weight="fill" />
524
+ <div>
525
+ <span className="text-muted-foreground">Created </span>
526
+ <ClickableFileName path={path} />
527
+ {description && (
528
+ <p className="text-muted-foreground mt-0.5">{description}</p>
529
+ )}
530
+ </div>
531
+ </div>
532
+ )
533
+ }
534
+
535
+ // Write file (content)
536
+ if (toolName === 'write_file') {
537
+ const resultData = result as { success?: boolean; path?: string } | undefined
538
+ const path = args.path as string || resultData?.path
539
+
540
+ return (
541
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
542
+ <CheckCircle className="size-3.5 text-green-600 dark:text-green-500" weight="fill" />
543
+ <span className="text-muted-foreground">Wrote </span>
544
+ <ClickableFileName path={path} />
545
+ </div>
546
+ )
547
+ }
548
+
549
+ // Delete file
550
+ if (toolName === 'delete_file') {
551
+ const resultData = result as { success?: boolean; path?: string; reason?: string; error?: string } | undefined
552
+ const path = args.path as string || resultData?.path
553
+ const reason = args.reason as string || resultData?.reason
554
+ const fileName = path?.split('/').pop() || path
555
+
556
+ if (resultData?.error) {
557
+ return (
558
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-destructive/10 border border-destructive/20">
559
+ <XCircle className="size-3.5 text-destructive" weight="fill" />
560
+ <span className="text-destructive">Failed to delete: {resultData.error}</span>
561
+ </div>
562
+ )
563
+ }
564
+
565
+ return (
566
+ <div className="flex items-start gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
567
+ <Trash className="size-3.5 text-red-600 dark:text-red-500 mt-0.5 flex-shrink-0" weight="fill" />
568
+ <div>
569
+ <span className="text-muted-foreground">Deleted </span>
570
+ <span className="font-mono font-medium text-foreground">{fileName}</span>
571
+ {reason && (
572
+ <p className="text-muted-foreground mt-0.5">{reason}</p>
573
+ )}
574
+ </div>
575
+ </div>
576
+ )
577
+ }
578
+
579
+ // Delete all notes
580
+ if (toolName === 'delete_all_notes') {
581
+ const resultData = result as { success?: boolean; deletedCount?: number; files?: string[]; message?: string; error?: string } | undefined
582
+
583
+ if (resultData?.error) {
584
+ return (
585
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-destructive/10 border border-destructive/20">
586
+ <XCircle className="size-3.5 text-destructive" weight="fill" />
587
+ <span className="text-destructive">{resultData.error}</span>
588
+ </div>
589
+ )
590
+ }
591
+
592
+ const count = resultData?.deletedCount ?? 0
593
+ const files = resultData?.files ?? []
594
+
595
+ // File type color mapping - matches notes-mode.tsx colors exactly
596
+ const getFileColor = (path: string) => {
597
+ const type = getFileType(path)
598
+ const colorMap: Record<string, string> = {
599
+ 'javascript': 'text-yellow-400',
600
+ 'typescript': 'text-blue-400',
601
+ 'python': 'text-green-500',
602
+ 'go': 'text-cyan-400',
603
+ 'ruby': 'text-red-400',
604
+ 'php': 'text-purple-400',
605
+ 'shell': 'text-green-400',
606
+ 'markdown': 'text-blue-300',
607
+ 'mermaid': 'text-pink-400',
608
+ 'json': 'text-yellow-500',
609
+ 'yaml': 'text-red-300',
610
+ 'text': 'text-zinc-400',
611
+ }
612
+ return colorMap[type] || 'text-zinc-400'
613
+ }
614
+
615
+ if (count === 0) {
616
+ return (
617
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-muted/50 border border-border/50">
618
+ <Trash className="size-3.5 text-muted-foreground" />
619
+ <span className="text-muted-foreground">No notes to delete</span>
620
+ </div>
621
+ )
622
+ }
623
+
624
+ const displayFiles = files.slice(0, 5)
625
+ const remainingCount = files.length - displayFiles.length
626
+
627
+ return (
628
+ <div className="text-xs py-2 px-2.5 rounded-md bg-red-500/5 border border-red-500/20">
629
+ <div className="flex items-center gap-2 mb-2">
630
+ <Trash className="size-3.5 text-red-600 dark:text-red-500" weight="fill" />
631
+ <span className="font-medium text-red-700 dark:text-red-400">
632
+ Deleted {count} file{count !== 1 ? 's' : ''}
633
+ </span>
634
+ </div>
635
+ <div className="space-y-1 ml-5">
636
+ {displayFiles.map((filePath, i) => {
637
+ const fileName = filePath.split('/').pop() || filePath
638
+ const folder = filePath.includes('/') ? filePath.split('/').slice(0, -1).join('/') + '/' : ''
639
+ return (
640
+ <div key={i} className="flex items-center gap-1.5">
641
+ <File className={`size-3.5 ${getFileColor(filePath)}`} weight="fill" />
642
+ <span className="text-muted-foreground/70">{folder}</span>
643
+ <span className="font-mono text-foreground/80">{fileName}</span>
644
+ </div>
645
+ )
646
+ })}
647
+ {remainingCount > 0 && (
648
+ <div className="text-muted-foreground/60 italic">
649
+ ...and {remainingCount} more
650
+ </div>
651
+ )}
652
+ </div>
653
+ </div>
654
+ )
655
+ }
656
+
657
+ // Check request validity
658
+ if (toolName === 'check_request_validity') {
659
+ const validation = result as RequestValidationResult | undefined
660
+
661
+ if (!validation) return null
662
+
663
+ const { isValid, validation: v } = validation
664
+ const issues: string[] = []
665
+
666
+ if (v?.pathParams?.missing?.length) issues.push(`path: ${v.pathParams.missing.join(', ')}`)
667
+ if (v?.queryParams?.missing?.length) issues.push(`params: ${v.queryParams.missing.join(', ')}`)
668
+ if (v?.body?.required && !v?.body?.hasContent) issues.push('body empty')
669
+ if (v?.auth?.required && !v?.auth?.configured) issues.push('auth missing')
670
+
671
+ if (isValid) {
672
+ return (
673
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-green-500/10 border border-green-500/20">
674
+ <CheckSquare className="size-3.5 text-green-600 dark:text-green-500" weight="fill" />
675
+ <span className="text-green-700 dark:text-green-400">Ready to send</span>
676
+ </div>
677
+ )
678
+ }
679
+
680
+ return (
681
+ <div className="py-1.5 px-2.5 rounded-md bg-amber-500/10 border border-amber-500/20">
682
+ <div className="flex items-center gap-2 text-xs">
683
+ <Warning className="size-3.5 text-amber-600 dark:text-amber-500" weight="fill" />
684
+ <span className="text-amber-700 dark:text-amber-400">Missing: {issues.join(', ')}</span>
685
+ </div>
686
+ </div>
687
+ )
688
+ }
689
+
690
+ // Send request
691
+ if (toolName === 'send_request') {
692
+ const response = result as {
693
+ success?: boolean
694
+ message?: string
695
+ error?: string
696
+ } | undefined
697
+
698
+ if (!response) return null
699
+
700
+ const { success, error } = response
701
+
702
+ if (error || !success) {
703
+ return (
704
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-destructive/10 border border-destructive/20">
705
+ <XCircle className="size-3.5 text-destructive" weight="fill" />
706
+ <span className="text-destructive">{error || 'Request failed'}</span>
707
+ </div>
708
+ )
709
+ }
710
+
711
+ return (
712
+ <div className="flex items-center gap-2 text-xs py-1.5 px-2.5 rounded-md bg-blue-500/10 border border-blue-500/20">
713
+ <PaperPlaneTilt className="size-3.5 text-blue-600 dark:text-blue-500" weight="fill" />
714
+ <span className="text-blue-700 dark:text-blue-400">Request sent → check response viewer</span>
715
+ </div>
716
+ )
717
+ }
718
+
719
+ // Hide unknown tools
720
+ return null
721
+ }