@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.
- package/LICENSE +33 -0
- package/README.md +415 -0
- package/bin/devdoc.js +13 -0
- package/dist/cli/commands/build.d.ts +5 -0
- package/dist/cli/commands/build.js +87 -0
- package/dist/cli/commands/check.d.ts +1 -0
- package/dist/cli/commands/check.js +143 -0
- package/dist/cli/commands/create.d.ts +24 -0
- package/dist/cli/commands/create.js +387 -0
- package/dist/cli/commands/deploy.d.ts +9 -0
- package/dist/cli/commands/deploy.js +433 -0
- package/dist/cli/commands/dev.d.ts +6 -0
- package/dist/cli/commands/dev.js +139 -0
- package/dist/cli/commands/init.d.ts +11 -0
- package/dist/cli/commands/init.js +238 -0
- package/dist/cli/commands/keys.d.ts +12 -0
- package/dist/cli/commands/keys.js +165 -0
- package/dist/cli/commands/start.d.ts +5 -0
- package/dist/cli/commands/start.js +56 -0
- package/dist/cli/commands/upload.d.ts +13 -0
- package/dist/cli/commands/upload.js +238 -0
- package/dist/cli/commands/whoami.d.ts +8 -0
- package/dist/cli/commands/whoami.js +91 -0
- package/dist/cli/index.d.ts +1 -0
- package/dist/cli/index.js +106 -0
- package/dist/config/index.d.ts +80 -0
- package/dist/config/index.js +133 -0
- package/dist/constants.d.ts +9 -0
- package/dist/constants.js +13 -0
- package/dist/index.d.ts +7 -0
- package/dist/index.js +12 -0
- package/dist/utils/logger.d.ts +16 -0
- package/dist/utils/logger.js +61 -0
- package/dist/utils/paths.d.ts +16 -0
- package/dist/utils/paths.js +50 -0
- package/package.json +51 -0
- package/renderer/app/api/assets/[...path]/route.ts +123 -0
- package/renderer/app/api/assets/route.ts +124 -0
- package/renderer/app/api/assets/upload/route.ts +177 -0
- package/renderer/app/api/auth-schemes/route.ts +77 -0
- package/renderer/app/api/chat/route.ts +858 -0
- package/renderer/app/api/codegen/route.ts +72 -0
- package/renderer/app/api/collections/route.ts +1016 -0
- package/renderer/app/api/debug/route.ts +53 -0
- package/renderer/app/api/deploy/route.ts +234 -0
- package/renderer/app/api/device/route.ts +42 -0
- package/renderer/app/api/docs/route.ts +187 -0
- package/renderer/app/api/keys/regenerate/route.ts +80 -0
- package/renderer/app/api/openapi-spec/route.ts +151 -0
- package/renderer/app/api/projects/[slug]/route.ts +153 -0
- package/renderer/app/api/projects/[slug]/stats/route.ts +96 -0
- package/renderer/app/api/projects/register/route.ts +152 -0
- package/renderer/app/api/proxy/route.ts +149 -0
- package/renderer/app/api/proxy-stream/route.ts +168 -0
- package/renderer/app/api/redirects/route.ts +47 -0
- package/renderer/app/api/schema/route.ts +65 -0
- package/renderer/app/api/subdomains/check/route.ts +172 -0
- package/renderer/app/api/suggestions/route.ts +144 -0
- package/renderer/app/favicon.ico +0 -0
- package/renderer/app/globals.css +1103 -0
- package/renderer/app/layout.tsx +47 -0
- package/renderer/app/llms-full.txt/route.ts +346 -0
- package/renderer/app/llms.txt/route.ts +279 -0
- package/renderer/app/page.tsx +14 -0
- package/renderer/app/robots.txt/route.ts +84 -0
- package/renderer/app/sitemap.xml/route.ts +199 -0
- package/renderer/components/docs/index.ts +12 -0
- package/renderer/components/docs/mdx/accordion.tsx +169 -0
- package/renderer/components/docs/mdx/badge.tsx +132 -0
- package/renderer/components/docs/mdx/callouts.tsx +154 -0
- package/renderer/components/docs/mdx/cards.tsx +213 -0
- package/renderer/components/docs/mdx/changelog.tsx +120 -0
- package/renderer/components/docs/mdx/code-block.tsx +186 -0
- package/renderer/components/docs/mdx/code-group.tsx +421 -0
- package/renderer/components/docs/mdx/file-embeds.tsx +105 -0
- package/renderer/components/docs/mdx/frame.tsx +112 -0
- package/renderer/components/docs/mdx/highlight.tsx +151 -0
- package/renderer/components/docs/mdx/iframe.tsx +134 -0
- package/renderer/components/docs/mdx/image.tsx +235 -0
- package/renderer/components/docs/mdx/index.ts +204 -0
- package/renderer/components/docs/mdx/mermaid.tsx +240 -0
- package/renderer/components/docs/mdx/param-field.tsx +200 -0
- package/renderer/components/docs/mdx/steps.tsx +113 -0
- package/renderer/components/docs/mdx/tabs.tsx +86 -0
- package/renderer/components/docs/mdx-renderer.tsx +100 -0
- package/renderer/components/docs/navigation/breadcrumbs.tsx +76 -0
- package/renderer/components/docs/navigation/index.ts +8 -0
- package/renderer/components/docs/navigation/page-nav.tsx +64 -0
- package/renderer/components/docs/navigation/sidebar.tsx +515 -0
- package/renderer/components/docs/navigation/toc.tsx +113 -0
- package/renderer/components/docs/notice.tsx +105 -0
- package/renderer/components/docs-header.tsx +274 -0
- package/renderer/components/docs-viewer/agent/agent-chat.tsx +2076 -0
- package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +90 -0
- package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +49 -0
- package/renderer/components/docs-viewer/agent/cards/index.tsx +50 -0
- package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +212 -0
- package/renderer/components/docs-viewer/agent/cards/types.ts +84 -0
- package/renderer/components/docs-viewer/agent/chat-message.tsx +17 -0
- package/renderer/components/docs-viewer/agent/index.tsx +6 -0
- package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +119 -0
- package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +46 -0
- package/renderer/components/docs-viewer/agent/messages/index.ts +17 -0
- package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +721 -0
- package/renderer/components/docs-viewer/agent/messages/types.ts +61 -0
- package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +24 -0
- package/renderer/components/docs-viewer/agent/messages/user-message.tsx +51 -0
- package/renderer/components/docs-viewer/code-editor/index.tsx +2 -0
- package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +1283 -0
- package/renderer/components/docs-viewer/content/changelog-page.tsx +331 -0
- package/renderer/components/docs-viewer/content/doc-page.tsx +285 -0
- package/renderer/components/docs-viewer/content/documentation-viewer.tsx +17 -0
- package/renderer/components/docs-viewer/content/index.tsx +29 -0
- package/renderer/components/docs-viewer/content/introduction.tsx +21 -0
- package/renderer/components/docs-viewer/content/request-details.tsx +330 -0
- package/renderer/components/docs-viewer/content/sections/auth.tsx +69 -0
- package/renderer/components/docs-viewer/content/sections/body.tsx +66 -0
- package/renderer/components/docs-viewer/content/sections/headers.tsx +43 -0
- package/renderer/components/docs-viewer/content/sections/overview.tsx +40 -0
- package/renderer/components/docs-viewer/content/sections/parameters.tsx +43 -0
- package/renderer/components/docs-viewer/content/sections/responses.tsx +87 -0
- package/renderer/components/docs-viewer/global-auth-modal.tsx +352 -0
- package/renderer/components/docs-viewer/index.tsx +1466 -0
- package/renderer/components/docs-viewer/playground/auth-editor.tsx +280 -0
- package/renderer/components/docs-viewer/playground/body-editor.tsx +221 -0
- package/renderer/components/docs-viewer/playground/code-editor.tsx +224 -0
- package/renderer/components/docs-viewer/playground/code-snippet.tsx +387 -0
- package/renderer/components/docs-viewer/playground/graphql-playground.tsx +745 -0
- package/renderer/components/docs-viewer/playground/index.tsx +671 -0
- package/renderer/components/docs-viewer/playground/key-value-editor.tsx +261 -0
- package/renderer/components/docs-viewer/playground/method-selector.tsx +60 -0
- package/renderer/components/docs-viewer/playground/request-builder.tsx +179 -0
- package/renderer/components/docs-viewer/playground/request-tabs.tsx +237 -0
- package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +21 -0
- package/renderer/components/docs-viewer/playground/response-cards/index.tsx +93 -0
- package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +16 -0
- package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +23 -0
- package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +268 -0
- package/renderer/components/docs-viewer/playground/response-cards/types.ts +82 -0
- package/renderer/components/docs-viewer/playground/response-viewer.tsx +43 -0
- package/renderer/components/docs-viewer/search/index.ts +2 -0
- package/renderer/components/docs-viewer/search/search-dialog.tsx +331 -0
- package/renderer/components/docs-viewer/search/use-search.ts +117 -0
- package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +431 -0
- package/renderer/components/docs-viewer/shared/method-badge.tsx +41 -0
- package/renderer/components/docs-viewer/shared/schema-viewer.tsx +349 -0
- package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +239 -0
- package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +316 -0
- package/renderer/components/docs-viewer/sidebar/index.tsx +343 -0
- package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +202 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +118 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +226 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +52 -0
- package/renderer/components/theme-provider.tsx +11 -0
- package/renderer/components/theme-toggle.tsx +76 -0
- package/renderer/components/ui/badge.tsx +46 -0
- package/renderer/components/ui/button.tsx +59 -0
- package/renderer/components/ui/dialog.tsx +118 -0
- package/renderer/components/ui/dropdown-menu.tsx +257 -0
- package/renderer/components/ui/input.tsx +21 -0
- package/renderer/components/ui/label.tsx +24 -0
- package/renderer/components/ui/navigation-menu.tsx +168 -0
- package/renderer/components/ui/select.tsx +190 -0
- package/renderer/components/ui/spinner.tsx +114 -0
- package/renderer/components/ui/tabs.tsx +66 -0
- package/renderer/components/ui/tooltip.tsx +61 -0
- package/renderer/hooks/use-code-copy.ts +88 -0
- package/renderer/hooks/use-openapi-title.ts +44 -0
- package/renderer/lib/api-docs/agent/index.ts +6 -0
- package/renderer/lib/api-docs/agent/indexer.ts +323 -0
- package/renderer/lib/api-docs/agent/spec-summary.ts +335 -0
- package/renderer/lib/api-docs/agent/types.ts +116 -0
- package/renderer/lib/api-docs/auth/auth-context.tsx +225 -0
- package/renderer/lib/api-docs/auth/auth-storage.ts +87 -0
- package/renderer/lib/api-docs/auth/crypto.ts +89 -0
- package/renderer/lib/api-docs/auth/index.ts +4 -0
- package/renderer/lib/api-docs/code-editor/db.ts +164 -0
- package/renderer/lib/api-docs/code-editor/hooks.ts +266 -0
- package/renderer/lib/api-docs/code-editor/index.ts +6 -0
- package/renderer/lib/api-docs/code-editor/mode-context.tsx +207 -0
- package/renderer/lib/api-docs/code-editor/types.ts +105 -0
- package/renderer/lib/api-docs/codegen/definitions.ts +297 -0
- package/renderer/lib/api-docs/codegen/har.ts +251 -0
- package/renderer/lib/api-docs/codegen/index.ts +159 -0
- package/renderer/lib/api-docs/factories.ts +151 -0
- package/renderer/lib/api-docs/index.ts +17 -0
- package/renderer/lib/api-docs/mobile-context.tsx +112 -0
- package/renderer/lib/api-docs/navigation-context.tsx +88 -0
- package/renderer/lib/api-docs/parsers/graphql/README.md +129 -0
- package/renderer/lib/api-docs/parsers/graphql/index.ts +91 -0
- package/renderer/lib/api-docs/parsers/graphql/parser.ts +491 -0
- package/renderer/lib/api-docs/parsers/graphql/transformer.ts +246 -0
- package/renderer/lib/api-docs/parsers/graphql/types.ts +283 -0
- package/renderer/lib/api-docs/parsers/openapi/README.md +32 -0
- package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +60 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +574 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +403 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +232 -0
- package/renderer/lib/api-docs/parsers/openapi/index.ts +171 -0
- package/renderer/lib/api-docs/parsers/openapi/transformer.ts +277 -0
- package/renderer/lib/api-docs/parsers/openapi/validator.ts +31 -0
- package/renderer/lib/api-docs/playground/context.tsx +107 -0
- package/renderer/lib/api-docs/playground/navigation-context.tsx +124 -0
- package/renderer/lib/api-docs/playground/request-builder.ts +223 -0
- package/renderer/lib/api-docs/playground/request-runner.ts +282 -0
- package/renderer/lib/api-docs/playground/types.ts +35 -0
- package/renderer/lib/api-docs/types.ts +269 -0
- package/renderer/lib/api-docs/utils.ts +311 -0
- package/renderer/lib/cache.ts +193 -0
- package/renderer/lib/docs/config/index.ts +29 -0
- package/renderer/lib/docs/config/loader.ts +142 -0
- package/renderer/lib/docs/config/schema.ts +298 -0
- package/renderer/lib/docs/index.ts +12 -0
- package/renderer/lib/docs/mdx/compiler.ts +176 -0
- package/renderer/lib/docs/mdx/frontmatter.ts +80 -0
- package/renderer/lib/docs/mdx/index.ts +26 -0
- package/renderer/lib/docs/navigation/generator.ts +348 -0
- package/renderer/lib/docs/navigation/index.ts +12 -0
- package/renderer/lib/docs/navigation/types.ts +123 -0
- package/renderer/lib/docs-navigation-context.tsx +80 -0
- package/renderer/lib/multi-tenant/context.ts +105 -0
- package/renderer/lib/storage/blob.ts +845 -0
- package/renderer/lib/utils.ts +6 -0
- package/renderer/next.config.ts +76 -0
- package/renderer/package.json +66 -0
- package/renderer/postcss.config.mjs +5 -0
- package/renderer/public/assets/images/screenshot.png +0 -0
- package/renderer/public/assets/logo/dark.svg +9 -0
- package/renderer/public/assets/logo/light.svg +9 -0
- package/renderer/public/assets/logo.svg +9 -0
- package/renderer/public/file.svg +1 -0
- package/renderer/public/globe.svg +1 -0
- package/renderer/public/icon.png +0 -0
- package/renderer/public/logo.svg +9 -0
- package/renderer/public/window.svg +1 -0
- package/renderer/tsconfig.json +28 -0
- package/templates/basic/README.md +139 -0
- package/templates/basic/assets/favicon.svg +4 -0
- package/templates/basic/assets/logo.svg +9 -0
- package/templates/basic/docs.json +47 -0
- package/templates/basic/guides/configuration.mdx +149 -0
- package/templates/basic/guides/overview.mdx +96 -0
- package/templates/basic/index.mdx +39 -0
- package/templates/basic/package.json +14 -0
- package/templates/basic/quickstart.mdx +92 -0
- package/templates/basic/vercel.json +6 -0
- package/templates/graphql/README.md +139 -0
- package/templates/graphql/api-reference/schema.graphql +305 -0
- package/templates/graphql/assets/favicon.svg +4 -0
- package/templates/graphql/assets/logo.svg +9 -0
- package/templates/graphql/docs.json +54 -0
- package/templates/graphql/guides/configuration.mdx +149 -0
- package/templates/graphql/guides/overview.mdx +96 -0
- package/templates/graphql/index.mdx +39 -0
- package/templates/graphql/package.json +14 -0
- package/templates/graphql/quickstart.mdx +92 -0
- package/templates/graphql/vercel.json +6 -0
- package/templates/openapi/README.md +139 -0
- package/templates/openapi/api-reference/openapi.json +419 -0
- package/templates/openapi/assets/favicon.svg +4 -0
- package/templates/openapi/assets/logo.svg +9 -0
- package/templates/openapi/docs.json +61 -0
- package/templates/openapi/guides/configuration.mdx +149 -0
- package/templates/openapi/guides/overview.mdx +96 -0
- package/templates/openapi/index.mdx +39 -0
- package/templates/openapi/package.json +14 -0
- package/templates/openapi/quickstart.mdx +92 -0
- 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
|
+
}
|