@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,90 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { Bug, ArrowsClockwise } from '@phosphor-icons/react'
|
|
4
|
+
import { Badge } from '@/components/ui/badge'
|
|
5
|
+
import { MethodBadge } from '../../shared/method-badge'
|
|
6
|
+
import type { DebugContext } from '../../playground/response-viewer'
|
|
7
|
+
import type { HTTPMethod } from '@/lib/api-docs/types'
|
|
8
|
+
import { cn } from '@/lib/utils'
|
|
9
|
+
|
|
10
|
+
interface DebugContextCardProps {
|
|
11
|
+
context: DebugContext
|
|
12
|
+
}
|
|
13
|
+
|
|
14
|
+
export function DebugContextCard({ context }: DebugContextCardProps) {
|
|
15
|
+
const isError = context.status >= 400
|
|
16
|
+
|
|
17
|
+
// Try to format the response body
|
|
18
|
+
let formattedBody = context.responseBody
|
|
19
|
+
try {
|
|
20
|
+
const parsed = JSON.parse(context.responseBody)
|
|
21
|
+
formattedBody = JSON.stringify(parsed, null, 2)
|
|
22
|
+
} catch {
|
|
23
|
+
// Keep original
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
// Truncate if too long
|
|
27
|
+
const truncatedBody = formattedBody.length > 300
|
|
28
|
+
? formattedBody.slice(0, 300) + '...'
|
|
29
|
+
: formattedBody
|
|
30
|
+
|
|
31
|
+
return (
|
|
32
|
+
<div
|
|
33
|
+
className={cn(
|
|
34
|
+
"fade-in slide-in-from-bottom-1 animate-in duration-150",
|
|
35
|
+
"mx-auto w-full max-w-[var(--thread-max-width)] px-2 py-3"
|
|
36
|
+
)}
|
|
37
|
+
>
|
|
38
|
+
<div className="rounded-lg border border-border bg-muted/30 overflow-hidden">
|
|
39
|
+
{/* Header */}
|
|
40
|
+
<div className="flex items-center gap-2 px-3 py-2 bg-muted/50 border-b border-border">
|
|
41
|
+
<Bug className="h-4 w-4 text-amber-500" weight="bold" />
|
|
42
|
+
<span className="text-xs font-medium text-muted-foreground">Debug Context</span>
|
|
43
|
+
</div>
|
|
44
|
+
|
|
45
|
+
{/* Endpoint info */}
|
|
46
|
+
{context.endpointName && (
|
|
47
|
+
<div className="px-3 py-2 border-b border-border">
|
|
48
|
+
<div className="flex items-center gap-2">
|
|
49
|
+
{context.endpointMethod && (
|
|
50
|
+
<MethodBadge method={context.endpointMethod as HTTPMethod} size="sm" />
|
|
51
|
+
)}
|
|
52
|
+
<span className="text-sm font-medium text-foreground">{context.endpointName}</span>
|
|
53
|
+
</div>
|
|
54
|
+
{context.endpointPath && (
|
|
55
|
+
<p className="text-xs text-muted-foreground mt-1 font-mono">{context.endpointPath}</p>
|
|
56
|
+
)}
|
|
57
|
+
</div>
|
|
58
|
+
)}
|
|
59
|
+
|
|
60
|
+
{/* Response status */}
|
|
61
|
+
<div className="px-3 py-2 border-b border-border">
|
|
62
|
+
<div className="flex items-center gap-2">
|
|
63
|
+
<ArrowsClockwise className="h-3.5 w-3.5 text-muted-foreground" weight="bold" />
|
|
64
|
+
<span className="text-xs text-muted-foreground">Response</span>
|
|
65
|
+
<Badge variant={isError ? 'destructive' : 'default'} className="text-xs h-5">
|
|
66
|
+
{context.status} {context.statusText}
|
|
67
|
+
</Badge>
|
|
68
|
+
</div>
|
|
69
|
+
</div>
|
|
70
|
+
|
|
71
|
+
{/* Response body preview */}
|
|
72
|
+
<div className="px-3 py-2">
|
|
73
|
+
<pre className="text-xs font-mono text-muted-foreground bg-background/50 rounded p-2 overflow-x-auto max-h-32 overflow-y-auto">
|
|
74
|
+
{truncatedBody}
|
|
75
|
+
</pre>
|
|
76
|
+
</div>
|
|
77
|
+
|
|
78
|
+
{/* Request body if present */}
|
|
79
|
+
{context.requestBody && (
|
|
80
|
+
<div className="px-3 py-2 border-t border-border">
|
|
81
|
+
<p className="text-xs text-muted-foreground mb-1">Request Body</p>
|
|
82
|
+
<pre className="text-xs font-mono text-muted-foreground bg-background/50 rounded p-2 overflow-x-auto max-h-20 overflow-y-auto">
|
|
83
|
+
{context.requestBody.length > 200 ? context.requestBody.slice(0, 200) + '...' : context.requestBody}
|
|
84
|
+
</pre>
|
|
85
|
+
</div>
|
|
86
|
+
)}
|
|
87
|
+
</div>
|
|
88
|
+
</div>
|
|
89
|
+
)
|
|
90
|
+
}
|
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { Crosshair } from '@phosphor-icons/react'
|
|
4
|
+
import { MethodBadge } from '../../shared/method-badge'
|
|
5
|
+
import type { EndpointContextCardData } from './types'
|
|
6
|
+
import { cn } from '@/lib/utils'
|
|
7
|
+
|
|
8
|
+
interface EndpointContextCardProps {
|
|
9
|
+
endpoint: EndpointContextCardData['endpoint']
|
|
10
|
+
}
|
|
11
|
+
|
|
12
|
+
export function EndpointContextCard({ endpoint }: EndpointContextCardProps) {
|
|
13
|
+
return (
|
|
14
|
+
<div
|
|
15
|
+
className={cn(
|
|
16
|
+
"fade-in slide-in-from-bottom-1 animate-in duration-150",
|
|
17
|
+
"mx-auto w-full max-w-[var(--thread-max-width)] px-2 py-2"
|
|
18
|
+
)}
|
|
19
|
+
>
|
|
20
|
+
<div className="rounded-lg border border-border/60 bg-muted/20 overflow-hidden">
|
|
21
|
+
{/* Header */}
|
|
22
|
+
<div className="flex items-center gap-2 px-3 py-2 bg-muted/30 border-b border-border/40">
|
|
23
|
+
<Crosshair className="h-3.5 w-3.5 text-muted-foreground" weight="bold" />
|
|
24
|
+
<span className="text-xs font-medium text-muted-foreground">Asking about</span>
|
|
25
|
+
</div>
|
|
26
|
+
|
|
27
|
+
{/* Content */}
|
|
28
|
+
<div className="p-3">
|
|
29
|
+
<div className="flex items-start gap-3">
|
|
30
|
+
<MethodBadge method={endpoint.method} size="sm" />
|
|
31
|
+
<div className="flex-1 min-w-0">
|
|
32
|
+
<p className="text-sm font-medium text-foreground truncate">
|
|
33
|
+
{endpoint.name}
|
|
34
|
+
</p>
|
|
35
|
+
<code className="text-xs text-muted-foreground font-mono block truncate mt-0.5">
|
|
36
|
+
{endpoint.path}
|
|
37
|
+
</code>
|
|
38
|
+
{endpoint.description && (
|
|
39
|
+
<p className="text-xs text-muted-foreground mt-1.5 line-clamp-2">
|
|
40
|
+
{endpoint.description}
|
|
41
|
+
</p>
|
|
42
|
+
)}
|
|
43
|
+
</div>
|
|
44
|
+
</div>
|
|
45
|
+
</div>
|
|
46
|
+
</div>
|
|
47
|
+
</div>
|
|
48
|
+
)
|
|
49
|
+
}
|
|
@@ -0,0 +1,50 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { DebugContextCard } from './debug-context-card'
|
|
4
|
+
import { ResponseOptionsCard } from './response-options-card'
|
|
5
|
+
import { EndpointContextCard } from './endpoint-context-card'
|
|
6
|
+
import { decodeCardMessage, type CardData, type CardActions } from './types'
|
|
7
|
+
|
|
8
|
+
// Export types and utilities
|
|
9
|
+
export * from './types'
|
|
10
|
+
export { DebugContextCard } from './debug-context-card'
|
|
11
|
+
export { ResponseOptionsCard, detectErrorType, type ErrorType } from './response-options-card'
|
|
12
|
+
export { EndpointContextCard } from './endpoint-context-card'
|
|
13
|
+
|
|
14
|
+
interface CardRendererProps {
|
|
15
|
+
content: string
|
|
16
|
+
actions?: CardActions
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
/**
|
|
20
|
+
* Renders a card based on the encoded message content.
|
|
21
|
+
* Returns null if the content is not a valid card message.
|
|
22
|
+
*/
|
|
23
|
+
export function CardRenderer({ content, actions }: CardRendererProps) {
|
|
24
|
+
const cardData = decodeCardMessage(content)
|
|
25
|
+
|
|
26
|
+
if (!cardData) {
|
|
27
|
+
return null
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
return <CardByType data={cardData} actions={actions} />
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
/**
|
|
34
|
+
* Renders the appropriate card component based on the card type.
|
|
35
|
+
*/
|
|
36
|
+
function CardByType({ data, actions }: { data: CardData; actions?: CardActions }) {
|
|
37
|
+
switch (data.type) {
|
|
38
|
+
case 'debug_context':
|
|
39
|
+
return <DebugContextCard context={data.context} />
|
|
40
|
+
|
|
41
|
+
case 'response_options':
|
|
42
|
+
return <ResponseOptionsCard errorType={data.errorType} actions={actions} />
|
|
43
|
+
|
|
44
|
+
case 'endpoint_context':
|
|
45
|
+
return <EndpointContextCard endpoint={data.endpoint} />
|
|
46
|
+
|
|
47
|
+
default:
|
|
48
|
+
return null
|
|
49
|
+
}
|
|
50
|
+
}
|
|
@@ -0,0 +1,212 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { ShieldCheck, CaretRight, Sliders, Code, ListBullets, type Icon } from '@phosphor-icons/react'
|
|
4
|
+
import type { CardActions } from './types'
|
|
5
|
+
import { cn } from '@/lib/utils'
|
|
6
|
+
|
|
7
|
+
// Extended error type with more specific categories
|
|
8
|
+
export type ErrorType =
|
|
9
|
+
| 'auth'
|
|
10
|
+
| 'missing_params'
|
|
11
|
+
| 'invalid_body'
|
|
12
|
+
| 'missing_headers'
|
|
13
|
+
| 'validation'
|
|
14
|
+
| 'not_found'
|
|
15
|
+
| 'server'
|
|
16
|
+
| 'generic'
|
|
17
|
+
|
|
18
|
+
interface ResponseOptionsCardProps {
|
|
19
|
+
errorType: ErrorType
|
|
20
|
+
actions?: CardActions
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
interface ActionButton {
|
|
24
|
+
id: string
|
|
25
|
+
icon: Icon
|
|
26
|
+
iconColor: string
|
|
27
|
+
bgColor: string
|
|
28
|
+
hoverBorder: string
|
|
29
|
+
title: string
|
|
30
|
+
description: string
|
|
31
|
+
onClick?: () => void
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
export function ResponseOptionsCard({ errorType, actions }: ResponseOptionsCardProps) {
|
|
35
|
+
// Build list of actions based on error type
|
|
36
|
+
const actionButtons: ActionButton[] = []
|
|
37
|
+
|
|
38
|
+
// Auth errors
|
|
39
|
+
if (errorType === 'auth') {
|
|
40
|
+
if (actions?.onOpenGlobalAuth) {
|
|
41
|
+
actionButtons.push({
|
|
42
|
+
id: 'global-auth',
|
|
43
|
+
icon: ShieldCheck,
|
|
44
|
+
iconColor: 'text-green-600',
|
|
45
|
+
bgColor: 'bg-green-500/10',
|
|
46
|
+
hoverBorder: 'hover:border-green-500/30',
|
|
47
|
+
title: 'Set Global Auth',
|
|
48
|
+
description: 'Configure API credentials for all requests',
|
|
49
|
+
onClick: actions.onOpenGlobalAuth,
|
|
50
|
+
})
|
|
51
|
+
}
|
|
52
|
+
if (actions?.onNavigateToAuthTab) {
|
|
53
|
+
actionButtons.push({
|
|
54
|
+
id: 'auth-tab',
|
|
55
|
+
icon: ShieldCheck,
|
|
56
|
+
iconColor: 'text-blue-600',
|
|
57
|
+
bgColor: 'bg-blue-500/10',
|
|
58
|
+
hoverBorder: 'hover:border-blue-500/30',
|
|
59
|
+
title: 'Go to Authorization Tab',
|
|
60
|
+
description: 'Set auth for this specific request',
|
|
61
|
+
onClick: actions.onNavigateToAuthTab,
|
|
62
|
+
})
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
// Missing params errors
|
|
67
|
+
if (errorType === 'missing_params' || errorType === 'validation') {
|
|
68
|
+
if (actions?.onNavigateToParamsTab) {
|
|
69
|
+
actionButtons.push({
|
|
70
|
+
id: 'params-tab',
|
|
71
|
+
icon: Sliders,
|
|
72
|
+
iconColor: 'text-purple-600',
|
|
73
|
+
bgColor: 'bg-purple-500/10',
|
|
74
|
+
hoverBorder: 'hover:border-purple-500/30',
|
|
75
|
+
title: 'Go to Parameters Tab',
|
|
76
|
+
description: 'Check and fill in required parameters',
|
|
77
|
+
onClick: actions.onNavigateToParamsTab,
|
|
78
|
+
})
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
// Invalid body errors
|
|
83
|
+
if (errorType === 'invalid_body' || errorType === 'validation') {
|
|
84
|
+
if (actions?.onNavigateToBodyTab) {
|
|
85
|
+
actionButtons.push({
|
|
86
|
+
id: 'body-tab',
|
|
87
|
+
icon: Code,
|
|
88
|
+
iconColor: 'text-amber-600',
|
|
89
|
+
bgColor: 'bg-amber-500/10',
|
|
90
|
+
hoverBorder: 'hover:border-amber-500/30',
|
|
91
|
+
title: 'Go to Body Tab',
|
|
92
|
+
description: 'Review and fix the request body',
|
|
93
|
+
onClick: actions.onNavigateToBodyTab,
|
|
94
|
+
})
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
// Missing headers errors
|
|
99
|
+
if (errorType === 'missing_headers') {
|
|
100
|
+
if (actions?.onNavigateToHeadersTab) {
|
|
101
|
+
actionButtons.push({
|
|
102
|
+
id: 'headers-tab',
|
|
103
|
+
icon: ListBullets,
|
|
104
|
+
iconColor: 'text-cyan-600',
|
|
105
|
+
bgColor: 'bg-cyan-500/10',
|
|
106
|
+
hoverBorder: 'hover:border-cyan-500/30',
|
|
107
|
+
title: 'Go to Headers Tab',
|
|
108
|
+
description: 'Add required request headers',
|
|
109
|
+
onClick: actions.onNavigateToHeadersTab,
|
|
110
|
+
})
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
// Don't render if no actions
|
|
115
|
+
if (actionButtons.length === 0) {
|
|
116
|
+
return null
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
return (
|
|
120
|
+
<div
|
|
121
|
+
className={cn(
|
|
122
|
+
"fade-in slide-in-from-bottom-1 animate-in duration-150",
|
|
123
|
+
"mx-auto w-full max-w-[var(--thread-max-width)] px-2 py-2"
|
|
124
|
+
)}
|
|
125
|
+
>
|
|
126
|
+
<div className="space-y-2">
|
|
127
|
+
<p className="text-xs font-medium text-muted-foreground px-1 mb-3">Suggested Actions</p>
|
|
128
|
+
{actionButtons.map((action) => (
|
|
129
|
+
<button
|
|
130
|
+
key={action.id}
|
|
131
|
+
onClick={action.onClick}
|
|
132
|
+
className={cn(
|
|
133
|
+
"w-full flex items-center gap-3 p-3 rounded-lg border border-border",
|
|
134
|
+
"bg-muted/30 hover:bg-muted/60 transition-all group text-left",
|
|
135
|
+
action.hoverBorder
|
|
136
|
+
)}
|
|
137
|
+
>
|
|
138
|
+
<div className={cn("shrink-0 w-8 h-8 rounded-md flex items-center justify-center", action.bgColor)}>
|
|
139
|
+
<action.icon className={cn("h-4 w-4", action.iconColor)} weight="bold" />
|
|
140
|
+
</div>
|
|
141
|
+
<div className="flex-1 min-w-0">
|
|
142
|
+
<p className="text-sm font-medium text-foreground">{action.title}</p>
|
|
143
|
+
<p className="text-xs text-muted-foreground truncate">{action.description}</p>
|
|
144
|
+
</div>
|
|
145
|
+
<CaretRight className="shrink-0 h-4 w-4 text-muted-foreground group-hover:text-foreground transition-colors" weight="bold" />
|
|
146
|
+
</button>
|
|
147
|
+
))}
|
|
148
|
+
</div>
|
|
149
|
+
</div>
|
|
150
|
+
)
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
// Helper to determine error type from status and response body
|
|
154
|
+
export function detectErrorType(status: number, responseBody: string): ErrorType {
|
|
155
|
+
const lowerBody = responseBody.toLowerCase()
|
|
156
|
+
|
|
157
|
+
// Auth errors (401, 403)
|
|
158
|
+
if (status === 401 || status === 403) return 'auth'
|
|
159
|
+
if (
|
|
160
|
+
lowerBody.includes('unauthorized') ||
|
|
161
|
+
lowerBody.includes('authentication') ||
|
|
162
|
+
lowerBody.includes('api key') ||
|
|
163
|
+
lowerBody.includes('api_key') ||
|
|
164
|
+
lowerBody.includes('token') ||
|
|
165
|
+
lowerBody.includes('credentials') ||
|
|
166
|
+
lowerBody.includes('forbidden')
|
|
167
|
+
) {
|
|
168
|
+
return 'auth'
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
// Missing parameters
|
|
172
|
+
if (
|
|
173
|
+
lowerBody.includes('missing') && (lowerBody.includes('param') || lowerBody.includes('query')) ||
|
|
174
|
+
lowerBody.includes('required parameter') ||
|
|
175
|
+
lowerBody.includes('required field') ||
|
|
176
|
+
lowerBody.includes('parameter is required')
|
|
177
|
+
) {
|
|
178
|
+
return 'missing_params'
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
// Invalid body
|
|
182
|
+
if (
|
|
183
|
+
lowerBody.includes('invalid json') ||
|
|
184
|
+
lowerBody.includes('parse error') ||
|
|
185
|
+
lowerBody.includes('body') && (lowerBody.includes('invalid') || lowerBody.includes('malformed')) ||
|
|
186
|
+
lowerBody.includes('request body') ||
|
|
187
|
+
lowerBody.includes('payload')
|
|
188
|
+
) {
|
|
189
|
+
return 'invalid_body'
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
// Missing headers
|
|
193
|
+
if (
|
|
194
|
+
lowerBody.includes('missing header') ||
|
|
195
|
+
lowerBody.includes('required header') ||
|
|
196
|
+
lowerBody.includes('header') && lowerBody.includes('required') ||
|
|
197
|
+
lowerBody.includes('content-type')
|
|
198
|
+
) {
|
|
199
|
+
return 'missing_headers'
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
// Generic validation errors (400, 422)
|
|
203
|
+
if (status === 400 || status === 422) return 'validation'
|
|
204
|
+
|
|
205
|
+
// Not found
|
|
206
|
+
if (status === 404) return 'not_found'
|
|
207
|
+
|
|
208
|
+
// Server errors
|
|
209
|
+
if (status >= 500) return 'server'
|
|
210
|
+
|
|
211
|
+
return 'generic'
|
|
212
|
+
}
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
import type { DebugContext } from '../../playground/response-viewer'
|
|
2
|
+
import type { HTTPMethod } from '@/lib/api-docs/types'
|
|
3
|
+
|
|
4
|
+
// Card type identifiers
|
|
5
|
+
export type CardType = 'debug_context' | 'response_options' | 'endpoint_context'
|
|
6
|
+
|
|
7
|
+
// Error types for response options
|
|
8
|
+
export type ErrorType =
|
|
9
|
+
| 'auth'
|
|
10
|
+
| 'missing_params'
|
|
11
|
+
| 'invalid_body'
|
|
12
|
+
| 'missing_headers'
|
|
13
|
+
| 'validation'
|
|
14
|
+
| 'not_found'
|
|
15
|
+
| 'server'
|
|
16
|
+
| 'generic'
|
|
17
|
+
|
|
18
|
+
// Base interface for all cards
|
|
19
|
+
export interface BaseCardData {
|
|
20
|
+
type: CardType
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
// Debug context card data
|
|
24
|
+
export interface DebugContextCardData extends BaseCardData {
|
|
25
|
+
type: 'debug_context'
|
|
26
|
+
context: DebugContext
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
// Response options card data
|
|
30
|
+
export interface ResponseOptionsCardData extends BaseCardData {
|
|
31
|
+
type: 'response_options'
|
|
32
|
+
errorType: ErrorType
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
// Endpoint context card data
|
|
36
|
+
export interface EndpointContextCardData extends BaseCardData {
|
|
37
|
+
type: 'endpoint_context'
|
|
38
|
+
endpoint: {
|
|
39
|
+
id: string
|
|
40
|
+
name: string
|
|
41
|
+
method: HTTPMethod
|
|
42
|
+
path: string
|
|
43
|
+
description?: string | null
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
// Union type for all card data
|
|
48
|
+
export type CardData = DebugContextCardData | ResponseOptionsCardData | EndpointContextCardData
|
|
49
|
+
|
|
50
|
+
// Card action callbacks
|
|
51
|
+
export interface CardActions {
|
|
52
|
+
onOpenGlobalAuth?: () => void
|
|
53
|
+
onNavigateToAuthTab?: () => void
|
|
54
|
+
onNavigateToParamsTab?: () => void
|
|
55
|
+
onNavigateToBodyTab?: () => void
|
|
56
|
+
onNavigateToHeadersTab?: () => void
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
// Prefix used to identify card messages
|
|
60
|
+
export const CARD_MESSAGE_PREFIX = '__CARD__'
|
|
61
|
+
|
|
62
|
+
// Helper to encode card data into a message
|
|
63
|
+
export function encodeCardMessage(data: CardData): string {
|
|
64
|
+
return `${CARD_MESSAGE_PREFIX}${JSON.stringify(data)}`
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
// Helper to decode card data from a message
|
|
68
|
+
export function decodeCardMessage(content: string): CardData | null {
|
|
69
|
+
if (!content?.startsWith(CARD_MESSAGE_PREFIX)) {
|
|
70
|
+
return null
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
try {
|
|
74
|
+
const json = content.slice(CARD_MESSAGE_PREFIX.length)
|
|
75
|
+
return JSON.parse(json) as CardData
|
|
76
|
+
} catch {
|
|
77
|
+
return null
|
|
78
|
+
}
|
|
79
|
+
}
|
|
80
|
+
|
|
81
|
+
// Check if a message is a card message
|
|
82
|
+
export function isCardMessage(content: string): boolean {
|
|
83
|
+
return content?.startsWith(CARD_MESSAGE_PREFIX) ?? false
|
|
84
|
+
}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
// Re-export from messages folder for backwards compatibility
|
|
2
|
+
export {
|
|
3
|
+
ChatMessage,
|
|
4
|
+
TypingIndicator,
|
|
5
|
+
UserMessage,
|
|
6
|
+
AssistantMessage,
|
|
7
|
+
ToolCallDisplay,
|
|
8
|
+
} from './messages'
|
|
9
|
+
|
|
10
|
+
export type {
|
|
11
|
+
ToolInvocation,
|
|
12
|
+
MessageImage,
|
|
13
|
+
ChatMessageProps,
|
|
14
|
+
UserMessageProps,
|
|
15
|
+
AssistantMessageProps,
|
|
16
|
+
ToolCallDisplayProps,
|
|
17
|
+
} from './messages'
|
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { useState, useCallback } from 'react'
|
|
4
|
+
import { Copy, Check, Trash } from '@phosphor-icons/react'
|
|
5
|
+
import { cn } from '@/lib/utils'
|
|
6
|
+
import { MarkdownRenderer } from '../../shared/markdown-renderer'
|
|
7
|
+
import {
|
|
8
|
+
Tooltip,
|
|
9
|
+
TooltipContent,
|
|
10
|
+
TooltipTrigger,
|
|
11
|
+
} from '@/components/ui/tooltip'
|
|
12
|
+
import { ToolCallDisplay } from './tool-call-display'
|
|
13
|
+
import type { AssistantMessageProps, AssistantActionBarProps } from './types'
|
|
14
|
+
|
|
15
|
+
export function AssistantMessage({
|
|
16
|
+
content,
|
|
17
|
+
messageId,
|
|
18
|
+
toolInvocations,
|
|
19
|
+
isStreaming,
|
|
20
|
+
onDeleteMessage,
|
|
21
|
+
onNavigate,
|
|
22
|
+
onNavigateToDocPage,
|
|
23
|
+
onOpenFile
|
|
24
|
+
}: AssistantMessageProps) {
|
|
25
|
+
return (
|
|
26
|
+
<div
|
|
27
|
+
className={cn(
|
|
28
|
+
"aui-assistant-message-root",
|
|
29
|
+
"fade-in slide-in-from-bottom-1 animate-in duration-150",
|
|
30
|
+
"relative mx-auto w-full max-w-[var(--thread-max-width)] py-3"
|
|
31
|
+
)}
|
|
32
|
+
data-role="assistant"
|
|
33
|
+
>
|
|
34
|
+
{/* Text content */}
|
|
35
|
+
{content && (
|
|
36
|
+
<div className="aui-assistant-message-content wrap-break-word px-2 text-foreground leading-relaxed">
|
|
37
|
+
<div className="prose prose-sm dark:prose-invert max-w-none">
|
|
38
|
+
<MarkdownRenderer content={content} />
|
|
39
|
+
</div>
|
|
40
|
+
</div>
|
|
41
|
+
)}
|
|
42
|
+
|
|
43
|
+
{/* Tool invocations */}
|
|
44
|
+
{toolInvocations && toolInvocations.length > 0 && (
|
|
45
|
+
<div className="mt-2 px-2 space-y-1">
|
|
46
|
+
{toolInvocations.map((tool) => (
|
|
47
|
+
<ToolCallDisplay key={tool.toolCallId} tool={tool} onNavigate={onNavigate} onNavigateToDocPage={onNavigateToDocPage} onOpenFile={onOpenFile} />
|
|
48
|
+
))}
|
|
49
|
+
</div>
|
|
50
|
+
)}
|
|
51
|
+
|
|
52
|
+
{/* Action bar - only show when streaming is finished */}
|
|
53
|
+
{!isStreaming && (
|
|
54
|
+
<div className="aui-assistant-message-footer mt-1 px-2 flex justify-end">
|
|
55
|
+
<AssistantActionBar content={content} messageId={messageId} onDeleteMessage={onDeleteMessage} />
|
|
56
|
+
</div>
|
|
57
|
+
)}
|
|
58
|
+
</div>
|
|
59
|
+
)
|
|
60
|
+
}
|
|
61
|
+
|
|
62
|
+
function AssistantActionBar({
|
|
63
|
+
content,
|
|
64
|
+
messageId,
|
|
65
|
+
onDeleteMessage
|
|
66
|
+
}: AssistantActionBarProps) {
|
|
67
|
+
const [copied, setCopied] = useState(false)
|
|
68
|
+
|
|
69
|
+
const handleCopy = useCallback(async () => {
|
|
70
|
+
try {
|
|
71
|
+
await navigator.clipboard.writeText(content)
|
|
72
|
+
setCopied(true)
|
|
73
|
+
setTimeout(() => setCopied(false), 2000)
|
|
74
|
+
} catch (err) {
|
|
75
|
+
console.error('Failed to copy:', err)
|
|
76
|
+
}
|
|
77
|
+
}, [content])
|
|
78
|
+
|
|
79
|
+
const handleDelete = useCallback(() => {
|
|
80
|
+
if (messageId && onDeleteMessage) {
|
|
81
|
+
onDeleteMessage(messageId)
|
|
82
|
+
}
|
|
83
|
+
}, [messageId, onDeleteMessage])
|
|
84
|
+
|
|
85
|
+
return (
|
|
86
|
+
<div className={cn(
|
|
87
|
+
"aui-assistant-action-bar-root",
|
|
88
|
+
"flex gap-1 text-muted-foreground"
|
|
89
|
+
)}>
|
|
90
|
+
<Tooltip>
|
|
91
|
+
<TooltipTrigger asChild>
|
|
92
|
+
<button
|
|
93
|
+
onClick={handleCopy}
|
|
94
|
+
className="aui-button-icon size-6 p-1 rounded hover:bg-accent hover:text-foreground transition-colors"
|
|
95
|
+
aria-label={copied ? "Copied" : "Copy"}
|
|
96
|
+
>
|
|
97
|
+
{copied ? <Check className="size-4" weight="bold" /> : <Copy className="size-4" />}
|
|
98
|
+
</button>
|
|
99
|
+
</TooltipTrigger>
|
|
100
|
+
<TooltipContent>{copied ? "Copied!" : "Copy"}</TooltipContent>
|
|
101
|
+
</Tooltip>
|
|
102
|
+
|
|
103
|
+
{messageId && onDeleteMessage && (
|
|
104
|
+
<Tooltip>
|
|
105
|
+
<TooltipTrigger asChild>
|
|
106
|
+
<button
|
|
107
|
+
onClick={handleDelete}
|
|
108
|
+
className="aui-button-icon size-6 p-1 rounded hover:bg-accent hover:text-foreground transition-colors"
|
|
109
|
+
aria-label="Delete message"
|
|
110
|
+
>
|
|
111
|
+
<Trash className="size-4" />
|
|
112
|
+
</button>
|
|
113
|
+
</TooltipTrigger>
|
|
114
|
+
<TooltipContent>Delete message</TooltipContent>
|
|
115
|
+
</Tooltip>
|
|
116
|
+
)}
|
|
117
|
+
</div>
|
|
118
|
+
)
|
|
119
|
+
}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { CardRenderer, isCardMessage } from '../cards'
|
|
4
|
+
import { UserMessage } from './user-message'
|
|
5
|
+
import { AssistantMessage } from './assistant-message'
|
|
6
|
+
import type { ChatMessageProps } from './types'
|
|
7
|
+
|
|
8
|
+
export function ChatMessage({
|
|
9
|
+
role,
|
|
10
|
+
content,
|
|
11
|
+
messageId,
|
|
12
|
+
images,
|
|
13
|
+
toolInvocations,
|
|
14
|
+
isStreaming,
|
|
15
|
+
onDeleteMessage,
|
|
16
|
+
onNavigate,
|
|
17
|
+
onNavigateToDocPage,
|
|
18
|
+
onOpenFile,
|
|
19
|
+
cardActions
|
|
20
|
+
}: ChatMessageProps) {
|
|
21
|
+
if (role === 'user') {
|
|
22
|
+
return <UserMessage content={content} images={images} cardActions={cardActions} />
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
if (role === 'assistant') {
|
|
26
|
+
// Check if this is a card message (like response options)
|
|
27
|
+
if (isCardMessage(content)) {
|
|
28
|
+
return <CardRenderer content={content} actions={cardActions} />
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
return (
|
|
32
|
+
<AssistantMessage
|
|
33
|
+
content={content}
|
|
34
|
+
messageId={messageId}
|
|
35
|
+
toolInvocations={toolInvocations}
|
|
36
|
+
isStreaming={isStreaming}
|
|
37
|
+
onDeleteMessage={onDeleteMessage}
|
|
38
|
+
onNavigate={onNavigate}
|
|
39
|
+
onNavigateToDocPage={onNavigateToDocPage}
|
|
40
|
+
onOpenFile={onOpenFile}
|
|
41
|
+
/>
|
|
42
|
+
)
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
return null
|
|
46
|
+
}
|