@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,331 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { useEffect, useState } from 'react'
|
|
4
|
+
import { Spinner } from '@phosphor-icons/react'
|
|
5
|
+
import { MDXRemote, MDXRemoteSerializeResult } from 'next-mdx-remote'
|
|
6
|
+
import { cn } from '@/lib/utils'
|
|
7
|
+
|
|
8
|
+
// Import base MDX components (using relative path for Vercel compatibility)
|
|
9
|
+
import {
|
|
10
|
+
Note,
|
|
11
|
+
Warning as WarningCallout,
|
|
12
|
+
Info,
|
|
13
|
+
Tip,
|
|
14
|
+
Check,
|
|
15
|
+
Error as ErrorCallout,
|
|
16
|
+
Callout,
|
|
17
|
+
} from '../../docs/mdx/index'
|
|
18
|
+
|
|
19
|
+
interface ChangelogRelease {
|
|
20
|
+
version: string
|
|
21
|
+
date: string
|
|
22
|
+
title: string
|
|
23
|
+
slug: string
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
interface ChangelogPageProps {
|
|
27
|
+
releases: ChangelogRelease[]
|
|
28
|
+
selectedVersion?: string | null
|
|
29
|
+
onSelectVersion?: (version: string) => void
|
|
30
|
+
tabName?: string // Tab name from config (e.g., "Changelog", "Releases")
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
interface ReleaseContent {
|
|
34
|
+
frontmatter: {
|
|
35
|
+
title: string
|
|
36
|
+
version: string
|
|
37
|
+
date: string
|
|
38
|
+
}
|
|
39
|
+
mdxSource: MDXRemoteSerializeResult
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
// Custom styled components for changelog content
|
|
43
|
+
const changelogMdxComponents = {
|
|
44
|
+
h2: ({ children, ...props }: { children: React.ReactNode }) => (
|
|
45
|
+
<h2 className="text-lg font-semibold text-foreground mt-6 mb-3" {...props}>{children}</h2>
|
|
46
|
+
),
|
|
47
|
+
h3: ({ children, ...props }: { children: React.ReactNode }) => (
|
|
48
|
+
<h3 className="text-base font-medium text-foreground mt-4 mb-2" {...props}>{children}</h3>
|
|
49
|
+
),
|
|
50
|
+
p: ({ children, ...props }: { children: React.ReactNode }) => (
|
|
51
|
+
<p className="text-muted-foreground mb-4" {...props}>{children}</p>
|
|
52
|
+
),
|
|
53
|
+
ul: ({ children, ...props }: { children: React.ReactNode }) => (
|
|
54
|
+
<ul className="space-y-2 text-muted-foreground mb-4" {...props}>{children}</ul>
|
|
55
|
+
),
|
|
56
|
+
li: ({ children, ...props }: { children: React.ReactNode }) => (
|
|
57
|
+
<li className="flex items-start gap-2" {...props}>
|
|
58
|
+
<span className="mt-2 h-1.5 w-1.5 rounded-full bg-foreground shrink-0" />
|
|
59
|
+
<span>{children}</span>
|
|
60
|
+
</li>
|
|
61
|
+
),
|
|
62
|
+
strong: ({ children, ...props }: { children: React.ReactNode }) => (
|
|
63
|
+
<strong className="font-semibold text-foreground" {...props}>{children}</strong>
|
|
64
|
+
),
|
|
65
|
+
// Keep callout components
|
|
66
|
+
Note,
|
|
67
|
+
Warning: WarningCallout,
|
|
68
|
+
Info,
|
|
69
|
+
Tip,
|
|
70
|
+
Check,
|
|
71
|
+
Error: ErrorCallout,
|
|
72
|
+
Callout,
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
// Format date to be more readable
|
|
76
|
+
function formatDate(dateString: string): string {
|
|
77
|
+
try {
|
|
78
|
+
const date = new Date(dateString)
|
|
79
|
+
return date.toLocaleDateString('en-US', {
|
|
80
|
+
year: 'numeric',
|
|
81
|
+
month: 'long',
|
|
82
|
+
day: 'numeric',
|
|
83
|
+
})
|
|
84
|
+
} catch {
|
|
85
|
+
return dateString
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// Extract version from slug (e.g., "changelog/v1.2.0" -> "v1.2.0")
|
|
90
|
+
function getVersionFromSlug(slug: string): string {
|
|
91
|
+
return slug.replace(/^changelog\//, '')
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
// Single release entry component
|
|
95
|
+
function ReleaseEntry({
|
|
96
|
+
release,
|
|
97
|
+
content,
|
|
98
|
+
isLatest,
|
|
99
|
+
isLast,
|
|
100
|
+
}: {
|
|
101
|
+
release: ChangelogRelease
|
|
102
|
+
content: ReleaseContent | null
|
|
103
|
+
isLatest: boolean
|
|
104
|
+
isLast: boolean
|
|
105
|
+
}) {
|
|
106
|
+
// Use version extracted from slug for the ID (without changelog/ prefix)
|
|
107
|
+
const versionId = getVersionFromSlug(release.slug)
|
|
108
|
+
|
|
109
|
+
return (
|
|
110
|
+
<article
|
|
111
|
+
id={`release-${versionId}`}
|
|
112
|
+
className="docs-changelog-entry relative scroll-mt-24 pl-8"
|
|
113
|
+
>
|
|
114
|
+
{/* Timeline line - extends full height except for last item */}
|
|
115
|
+
{!isLast && (
|
|
116
|
+
<div className="absolute left-[3px] top-3 bottom-0 w-[2px] bg-border" />
|
|
117
|
+
)}
|
|
118
|
+
|
|
119
|
+
{/* Timeline dot */}
|
|
120
|
+
<div className={cn(
|
|
121
|
+
"absolute left-0 top-[6px] w-[8px] h-[8px] rounded-full border-2 z-10",
|
|
122
|
+
isLatest
|
|
123
|
+
? "bg-primary border-primary"
|
|
124
|
+
: "bg-background border-muted-foreground/40"
|
|
125
|
+
)} />
|
|
126
|
+
|
|
127
|
+
<div className="pb-10">
|
|
128
|
+
{/* Version badge and date header */}
|
|
129
|
+
<div className="flex flex-wrap items-center gap-3 mb-4">
|
|
130
|
+
<span className={cn(
|
|
131
|
+
"docs-changelog-version inline-flex items-center px-3 py-1 text-sm font-semibold rounded-full",
|
|
132
|
+
isLatest
|
|
133
|
+
? "bg-primary/10 text-primary border border-primary/20"
|
|
134
|
+
: "bg-muted text-foreground border border-border"
|
|
135
|
+
)}>
|
|
136
|
+
{release.version}
|
|
137
|
+
{isLatest && (
|
|
138
|
+
<span className="ml-2 text-xs font-normal opacity-75">Latest</span>
|
|
139
|
+
)}
|
|
140
|
+
</span>
|
|
141
|
+
<time className="docs-changelog-date text-sm text-muted-foreground">
|
|
142
|
+
{formatDate(release.date)}
|
|
143
|
+
</time>
|
|
144
|
+
</div>
|
|
145
|
+
|
|
146
|
+
{/* Release title */}
|
|
147
|
+
<h2 className="docs-changelog-title text-xl font-semibold text-foreground mb-4">
|
|
148
|
+
{release.title}
|
|
149
|
+
</h2>
|
|
150
|
+
|
|
151
|
+
{/* Release content */}
|
|
152
|
+
{content ? (
|
|
153
|
+
<div className="docs-changelog-content prose prose-sm dark:prose-invert max-w-none">
|
|
154
|
+
<MDXRemote {...content.mdxSource} components={changelogMdxComponents} />
|
|
155
|
+
</div>
|
|
156
|
+
) : (
|
|
157
|
+
<div className="flex items-center gap-2 text-muted-foreground py-4">
|
|
158
|
+
<Spinner className="h-4 w-4 animate-spin" />
|
|
159
|
+
<span className="text-sm">Loading...</span>
|
|
160
|
+
</div>
|
|
161
|
+
)}
|
|
162
|
+
</div>
|
|
163
|
+
</article>
|
|
164
|
+
)
|
|
165
|
+
}
|
|
166
|
+
|
|
167
|
+
export function ChangelogPage({ releases, tabName = 'Changelog' }: ChangelogPageProps) {
|
|
168
|
+
const [releaseContents, setReleaseContents] = useState<Map<string, ReleaseContent>>(new Map())
|
|
169
|
+
const [loadingAll, setLoadingAll] = useState(true)
|
|
170
|
+
const [error, setError] = useState<string | null>(null)
|
|
171
|
+
const [activeVersion, setActiveVersion] = useState<string | null>(releases[0]?.version || null)
|
|
172
|
+
|
|
173
|
+
// Load all releases on mount
|
|
174
|
+
useEffect(() => {
|
|
175
|
+
async function fetchAllReleases() {
|
|
176
|
+
if (releases.length === 0) {
|
|
177
|
+
setLoadingAll(false)
|
|
178
|
+
return
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
setLoadingAll(true)
|
|
182
|
+
setError(null)
|
|
183
|
+
|
|
184
|
+
const contents = new Map<string, ReleaseContent>()
|
|
185
|
+
|
|
186
|
+
// Fetch all releases in parallel
|
|
187
|
+
const fetchPromises = releases.map(async (release) => {
|
|
188
|
+
try {
|
|
189
|
+
const response = await fetch(`/api/docs?slug=${encodeURIComponent(release.slug)}`)
|
|
190
|
+
if (response.ok) {
|
|
191
|
+
const data = await response.json()
|
|
192
|
+
contents.set(release.version, data)
|
|
193
|
+
}
|
|
194
|
+
} catch (err) {
|
|
195
|
+
console.error(`Error loading release ${release.version}:`, err)
|
|
196
|
+
}
|
|
197
|
+
})
|
|
198
|
+
|
|
199
|
+
await Promise.all(fetchPromises)
|
|
200
|
+
setReleaseContents(contents)
|
|
201
|
+
setLoadingAll(false)
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
fetchAllReleases()
|
|
205
|
+
}, [releases])
|
|
206
|
+
|
|
207
|
+
// Track active version based on scroll position
|
|
208
|
+
useEffect(() => {
|
|
209
|
+
const observer = new IntersectionObserver(
|
|
210
|
+
(entries) => {
|
|
211
|
+
for (const entry of entries) {
|
|
212
|
+
if (entry.isIntersecting) {
|
|
213
|
+
const version = entry.target.getAttribute('data-version')
|
|
214
|
+
if (version) {
|
|
215
|
+
setActiveVersion(version)
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
},
|
|
220
|
+
{ rootMargin: '-100px 0px -70% 0px', threshold: 0 }
|
|
221
|
+
)
|
|
222
|
+
|
|
223
|
+
// Observe all release entries
|
|
224
|
+
releases.forEach((release) => {
|
|
225
|
+
const versionId = getVersionFromSlug(release.slug)
|
|
226
|
+
const element = document.getElementById(`release-${versionId}`)
|
|
227
|
+
if (element) {
|
|
228
|
+
element.setAttribute('data-version', release.version)
|
|
229
|
+
observer.observe(element)
|
|
230
|
+
}
|
|
231
|
+
})
|
|
232
|
+
|
|
233
|
+
return () => observer.disconnect()
|
|
234
|
+
}, [releases, releaseContents])
|
|
235
|
+
|
|
236
|
+
const scrollToRelease = (slug: string) => {
|
|
237
|
+
const versionId = getVersionFromSlug(slug)
|
|
238
|
+
const element = document.getElementById(`release-${versionId}`)
|
|
239
|
+
if (element) {
|
|
240
|
+
element.scrollIntoView({ behavior: 'smooth', block: 'start' })
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
|
|
244
|
+
if (releases.length === 0) {
|
|
245
|
+
return (
|
|
246
|
+
<div className="flex-1 flex items-center justify-center py-12">
|
|
247
|
+
<p className="text-muted-foreground">No changelog releases found.</p>
|
|
248
|
+
</div>
|
|
249
|
+
)
|
|
250
|
+
}
|
|
251
|
+
|
|
252
|
+
return (
|
|
253
|
+
<div className="docs-changelog flex flex-1 h-full overflow-hidden">
|
|
254
|
+
{/* Main content area */}
|
|
255
|
+
<div className="docs-changelog-main flex-1 overflow-y-auto">
|
|
256
|
+
<div className="max-w-3xl mx-auto px-6 py-8 lg:px-8">
|
|
257
|
+
{/* Header */}
|
|
258
|
+
<header className="docs-changelog-header mb-10">
|
|
259
|
+
<h1 className="docs-content-title text-3xl font-bold text-foreground mb-2">
|
|
260
|
+
{tabName}
|
|
261
|
+
</h1>
|
|
262
|
+
<p className="text-muted-foreground">
|
|
263
|
+
All notable changes and updates to this project.
|
|
264
|
+
</p>
|
|
265
|
+
</header>
|
|
266
|
+
|
|
267
|
+
{/* Loading state */}
|
|
268
|
+
{loadingAll && releaseContents.size === 0 && (
|
|
269
|
+
<div className="docs-loading flex items-center justify-center py-12">
|
|
270
|
+
<Spinner className="h-6 w-6 animate-spin text-muted-foreground" />
|
|
271
|
+
</div>
|
|
272
|
+
)}
|
|
273
|
+
|
|
274
|
+
{/* Error state */}
|
|
275
|
+
{error && (
|
|
276
|
+
<div className="docs-error text-center py-12">
|
|
277
|
+
<p className="text-destructive">{error}</p>
|
|
278
|
+
</div>
|
|
279
|
+
)}
|
|
280
|
+
|
|
281
|
+
{/* Releases list with connected timeline */}
|
|
282
|
+
<div className="docs-changelog-releases relative">
|
|
283
|
+
{releases.map((release, index) => (
|
|
284
|
+
<ReleaseEntry
|
|
285
|
+
key={release.version}
|
|
286
|
+
release={release}
|
|
287
|
+
content={releaseContents.get(release.version) || null}
|
|
288
|
+
isLatest={index === 0}
|
|
289
|
+
isLast={index === releases.length - 1}
|
|
290
|
+
/>
|
|
291
|
+
))}
|
|
292
|
+
</div>
|
|
293
|
+
</div>
|
|
294
|
+
</div>
|
|
295
|
+
|
|
296
|
+
{/* Right sidebar - Version navigation */}
|
|
297
|
+
{releases.length > 1 && (
|
|
298
|
+
<aside className="docs-changelog-sidebar hidden lg:flex lg:flex-col w-48 h-full border-l border-border bg-muted/20 overflow-y-auto shrink-0">
|
|
299
|
+
<div className="p-4 sticky top-0">
|
|
300
|
+
<h3 className="docs-changelog-sidebar-title text-xs font-semibold text-muted-foreground uppercase tracking-wider mb-3">
|
|
301
|
+
Releases
|
|
302
|
+
</h3>
|
|
303
|
+
<nav className="docs-changelog-nav space-y-1">
|
|
304
|
+
{releases.map((release, index) => (
|
|
305
|
+
<button
|
|
306
|
+
key={release.version}
|
|
307
|
+
onClick={() => scrollToRelease(release.slug)}
|
|
308
|
+
className={cn(
|
|
309
|
+
'docs-changelog-nav-item block w-full text-left px-3 py-2 text-sm rounded-md transition-all duration-200',
|
|
310
|
+
activeVersion === release.version
|
|
311
|
+
? 'docs-changelog-nav-item-active bg-primary/10 text-primary font-medium'
|
|
312
|
+
: 'text-muted-foreground hover:text-foreground hover:bg-muted'
|
|
313
|
+
)}
|
|
314
|
+
>
|
|
315
|
+
<span className="flex items-center gap-2">
|
|
316
|
+
{release.version}
|
|
317
|
+
{index === 0 && (
|
|
318
|
+
<span className="text-[10px] px-1.5 py-0.5 rounded bg-primary/10 text-primary">
|
|
319
|
+
Latest
|
|
320
|
+
</span>
|
|
321
|
+
)}
|
|
322
|
+
</span>
|
|
323
|
+
</button>
|
|
324
|
+
))}
|
|
325
|
+
</nav>
|
|
326
|
+
</div>
|
|
327
|
+
</aside>
|
|
328
|
+
)}
|
|
329
|
+
</div>
|
|
330
|
+
)
|
|
331
|
+
}
|
|
@@ -0,0 +1,285 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { useEffect, useState, useCallback, useRef } from 'react'
|
|
4
|
+
import { Spinner } from '@phosphor-icons/react'
|
|
5
|
+
import { MDXRemote, MDXRemoteSerializeResult } from 'next-mdx-remote'
|
|
6
|
+
import { useDocsNavigation } from '@/lib/docs-navigation-context'
|
|
7
|
+
import { useCodeCopy } from '@/hooks/use-code-copy'
|
|
8
|
+
|
|
9
|
+
// Custom Link component for MDX - uses docs navigation context
|
|
10
|
+
function MdxLink({ href, children, ...props }: React.AnchorHTMLAttributes<HTMLAnchorElement>) {
|
|
11
|
+
const docsNav = useDocsNavigation()
|
|
12
|
+
|
|
13
|
+
const handleClick = useCallback((e: React.MouseEvent<HTMLAnchorElement>) => {
|
|
14
|
+
// Handle anchor links (same page)
|
|
15
|
+
if (href?.startsWith('#')) {
|
|
16
|
+
return // Let default behavior handle anchor scrolling
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
// Handle internal doc links
|
|
20
|
+
if (href?.startsWith('/') && docsNav?.isApiDocsView) {
|
|
21
|
+
e.preventDefault()
|
|
22
|
+
|
|
23
|
+
// Extract the slug from the path
|
|
24
|
+
let slug = href
|
|
25
|
+
|
|
26
|
+
// Remove leading slash
|
|
27
|
+
if (slug.startsWith('/')) {
|
|
28
|
+
slug = slug.slice(1)
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
// Handle special paths
|
|
32
|
+
if (slug.startsWith('api-reference')) {
|
|
33
|
+
docsNav.switchToTab('api-reference')
|
|
34
|
+
return
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
if (slug.startsWith('changelog')) {
|
|
38
|
+
docsNav.switchToTab('changelog')
|
|
39
|
+
return
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
// Remove docs/ prefix if present
|
|
43
|
+
if (slug.startsWith('docs/')) {
|
|
44
|
+
slug = slug.slice(5)
|
|
45
|
+
}
|
|
46
|
+
|
|
47
|
+
// Navigate to the page
|
|
48
|
+
docsNav.navigateToPage(slug)
|
|
49
|
+
}
|
|
50
|
+
}, [href, docsNav])
|
|
51
|
+
|
|
52
|
+
// Determine link styling - better visibility and accessibility
|
|
53
|
+
// Using visited: to override browser's purple visited link color
|
|
54
|
+
const linkClassName = "text-primary visited:text-primary font-medium underline decoration-primary/30 underline-offset-2 hover:decoration-primary transition-colors"
|
|
55
|
+
|
|
56
|
+
// External links
|
|
57
|
+
if (href && !href.startsWith('/') && !href.startsWith('#')) {
|
|
58
|
+
return (
|
|
59
|
+
<a
|
|
60
|
+
href={href}
|
|
61
|
+
target="_blank"
|
|
62
|
+
rel="noopener noreferrer"
|
|
63
|
+
className={linkClassName}
|
|
64
|
+
{...props}
|
|
65
|
+
>
|
|
66
|
+
{children}
|
|
67
|
+
</a>
|
|
68
|
+
)
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
return (
|
|
72
|
+
<a
|
|
73
|
+
href={href}
|
|
74
|
+
onClick={handleClick}
|
|
75
|
+
className={linkClassName}
|
|
76
|
+
{...props}
|
|
77
|
+
>
|
|
78
|
+
{children}
|
|
79
|
+
</a>
|
|
80
|
+
)
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
// Import MDX components
|
|
84
|
+
import {
|
|
85
|
+
Note,
|
|
86
|
+
Warning,
|
|
87
|
+
Info,
|
|
88
|
+
Tip,
|
|
89
|
+
Check,
|
|
90
|
+
Error as ErrorCallout,
|
|
91
|
+
Callout,
|
|
92
|
+
Card,
|
|
93
|
+
CardGroup,
|
|
94
|
+
Accordion,
|
|
95
|
+
AccordionGroup,
|
|
96
|
+
Steps,
|
|
97
|
+
Step,
|
|
98
|
+
Tabs,
|
|
99
|
+
Tab,
|
|
100
|
+
CodeGroup,
|
|
101
|
+
Frame,
|
|
102
|
+
Columns,
|
|
103
|
+
Snippet,
|
|
104
|
+
Latex,
|
|
105
|
+
ParamField,
|
|
106
|
+
ResponseField,
|
|
107
|
+
Expandable,
|
|
108
|
+
Iframe,
|
|
109
|
+
Video,
|
|
110
|
+
Loom,
|
|
111
|
+
Image,
|
|
112
|
+
Screenshot,
|
|
113
|
+
Logo,
|
|
114
|
+
Icon,
|
|
115
|
+
Highlight,
|
|
116
|
+
Marker,
|
|
117
|
+
Underline,
|
|
118
|
+
Badge,
|
|
119
|
+
Mermaid,
|
|
120
|
+
PDF,
|
|
121
|
+
Audio,
|
|
122
|
+
Download,
|
|
123
|
+
} from '../../docs/mdx/index'
|
|
124
|
+
|
|
125
|
+
// MDX components mapping
|
|
126
|
+
const mdxComponents = {
|
|
127
|
+
// Callouts
|
|
128
|
+
Note,
|
|
129
|
+
Warning,
|
|
130
|
+
Info,
|
|
131
|
+
Tip,
|
|
132
|
+
Check,
|
|
133
|
+
Error: ErrorCallout,
|
|
134
|
+
Callout,
|
|
135
|
+
|
|
136
|
+
// Cards
|
|
137
|
+
Card,
|
|
138
|
+
CardGroup,
|
|
139
|
+
|
|
140
|
+
// Accordion
|
|
141
|
+
Accordion,
|
|
142
|
+
AccordionGroup,
|
|
143
|
+
|
|
144
|
+
// Steps
|
|
145
|
+
Steps,
|
|
146
|
+
Step,
|
|
147
|
+
|
|
148
|
+
// Tabs
|
|
149
|
+
Tabs,
|
|
150
|
+
Tab,
|
|
151
|
+
|
|
152
|
+
// Code
|
|
153
|
+
CodeGroup,
|
|
154
|
+
|
|
155
|
+
// Layout
|
|
156
|
+
Frame,
|
|
157
|
+
Columns,
|
|
158
|
+
Snippet,
|
|
159
|
+
Latex,
|
|
160
|
+
|
|
161
|
+
// API docs
|
|
162
|
+
ParamField,
|
|
163
|
+
ResponseField,
|
|
164
|
+
Expandable,
|
|
165
|
+
|
|
166
|
+
// Embeds
|
|
167
|
+
Iframe,
|
|
168
|
+
Video,
|
|
169
|
+
Loom,
|
|
170
|
+
|
|
171
|
+
// Images
|
|
172
|
+
Image,
|
|
173
|
+
Screenshot,
|
|
174
|
+
Logo,
|
|
175
|
+
Icon,
|
|
176
|
+
|
|
177
|
+
// Highlight
|
|
178
|
+
Highlight,
|
|
179
|
+
Marker,
|
|
180
|
+
Underline,
|
|
181
|
+
|
|
182
|
+
// Badge
|
|
183
|
+
Badge,
|
|
184
|
+
|
|
185
|
+
// Mermaid
|
|
186
|
+
Mermaid,
|
|
187
|
+
|
|
188
|
+
// File Embeds
|
|
189
|
+
PDF,
|
|
190
|
+
Audio,
|
|
191
|
+
Download,
|
|
192
|
+
|
|
193
|
+
// Links
|
|
194
|
+
a: MdxLink,
|
|
195
|
+
}
|
|
196
|
+
|
|
197
|
+
interface DocPageProps {
|
|
198
|
+
slug: string
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
interface DocPageData {
|
|
202
|
+
slug: string
|
|
203
|
+
frontmatter: {
|
|
204
|
+
title: string
|
|
205
|
+
description?: string
|
|
206
|
+
icon?: string
|
|
207
|
+
}
|
|
208
|
+
mdxSource: MDXRemoteSerializeResult
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
export function DocPage({ slug }: DocPageProps) {
|
|
212
|
+
const [pageData, setPageData] = useState<DocPageData | null>(null)
|
|
213
|
+
const [loading, setLoading] = useState(true)
|
|
214
|
+
const [error, setError] = useState<string | null>(null)
|
|
215
|
+
const contentRef = useRef<HTMLDivElement>(null)
|
|
216
|
+
|
|
217
|
+
// Add copy buttons to code blocks after content loads
|
|
218
|
+
useCodeCopy(contentRef, [pageData])
|
|
219
|
+
|
|
220
|
+
useEffect(() => {
|
|
221
|
+
async function fetchPage() {
|
|
222
|
+
try {
|
|
223
|
+
setLoading(true)
|
|
224
|
+
setError(null)
|
|
225
|
+
|
|
226
|
+
const response = await fetch(`/api/docs?slug=${encodeURIComponent(slug)}`)
|
|
227
|
+
|
|
228
|
+
if (!response.ok) {
|
|
229
|
+
throw new Error('Failed to load page')
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
const data = await response.json()
|
|
233
|
+
setPageData(data)
|
|
234
|
+
} catch (err) {
|
|
235
|
+
console.error('Error loading doc page:', err)
|
|
236
|
+
setError(err instanceof Error ? err.message : 'Failed to load page')
|
|
237
|
+
} finally {
|
|
238
|
+
setLoading(false)
|
|
239
|
+
}
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
fetchPage()
|
|
243
|
+
}, [slug])
|
|
244
|
+
|
|
245
|
+
if (loading) {
|
|
246
|
+
return (
|
|
247
|
+
<div className="docs-page docs-page-loading max-w-4xl mx-auto px-4 py-6 sm:px-8 sm:py-8">
|
|
248
|
+
<div className="docs-loading flex items-center justify-center py-12">
|
|
249
|
+
<Spinner className="h-6 w-6 animate-spin text-muted-foreground" />
|
|
250
|
+
</div>
|
|
251
|
+
</div>
|
|
252
|
+
)
|
|
253
|
+
}
|
|
254
|
+
|
|
255
|
+
if (error || !pageData) {
|
|
256
|
+
return (
|
|
257
|
+
<div className="docs-page docs-page-error max-w-4xl mx-auto px-4 py-6 sm:px-8 sm:py-8">
|
|
258
|
+
<div className="docs-error text-center py-12">
|
|
259
|
+
<p className="text-destructive">{error || 'Page not found'}</p>
|
|
260
|
+
</div>
|
|
261
|
+
</div>
|
|
262
|
+
)
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
return (
|
|
266
|
+
<div ref={contentRef} className="docs-page docs-content max-w-4xl mx-auto px-4 py-6 sm:px-8 sm:py-8">
|
|
267
|
+
{/* Page header */}
|
|
268
|
+
<div className="docs-page-header mb-6">
|
|
269
|
+
<h1 className="docs-content-title text-2xl sm:text-3xl font-bold mb-2 text-foreground">
|
|
270
|
+
{pageData.frontmatter.title}
|
|
271
|
+
</h1>
|
|
272
|
+
{pageData.frontmatter.description && (
|
|
273
|
+
<p className="docs-content-description text-lg text-muted-foreground">
|
|
274
|
+
{pageData.frontmatter.description}
|
|
275
|
+
</p>
|
|
276
|
+
)}
|
|
277
|
+
</div>
|
|
278
|
+
|
|
279
|
+
{/* Page content - MDX rendered with custom components */}
|
|
280
|
+
<div className="docs-prose prose prose-sm max-w-none prose-headings:text-foreground prose-p:text-muted-foreground prose-strong:text-foreground prose-code:text-foreground prose-pre:bg-muted prose-code:bg-muted prose-code:px-1 prose-code:py-0.5 prose-code:rounded prose-pre:overflow-x-auto prose-table:w-full prose-th:text-left prose-th:p-3 prose-th:bg-muted prose-td:p-3 prose-td:border-b prose-td:border-border">
|
|
281
|
+
<MDXRemote {...pageData.mdxSource} components={mdxComponents} />
|
|
282
|
+
</div>
|
|
283
|
+
</div>
|
|
284
|
+
)
|
|
285
|
+
}
|
|
@@ -0,0 +1,17 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { MarkdownRenderer } from '../shared/markdown-renderer'
|
|
4
|
+
|
|
5
|
+
interface DocumentationViewerProps {
|
|
6
|
+
content: string
|
|
7
|
+
}
|
|
8
|
+
|
|
9
|
+
export function DocumentationViewer({ content }: DocumentationViewerProps) {
|
|
10
|
+
return (
|
|
11
|
+
<div className="max-w-4xl mx-auto p-8">
|
|
12
|
+
<div className="prose prose-sm max-w-none">
|
|
13
|
+
<MarkdownRenderer content={content} />
|
|
14
|
+
</div>
|
|
15
|
+
</div>
|
|
16
|
+
)
|
|
17
|
+
}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import type { BrainfishRESTRequest } from '@/lib/api-docs/types'
|
|
4
|
+
import { RequestDetails } from './request-details'
|
|
5
|
+
|
|
6
|
+
interface ApiDocsContentProps {
|
|
7
|
+
request: BrainfishRESTRequest | null
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
export function ApiDocsContent({ request }: ApiDocsContentProps) {
|
|
11
|
+
if (!request) {
|
|
12
|
+
return (
|
|
13
|
+
<div className="flex-1 flex items-center justify-center bg-background">
|
|
14
|
+
<div className="text-center">
|
|
15
|
+
<p className="text-muted-foreground text-lg mb-2">Select an endpoint</p>
|
|
16
|
+
<p className="text-sm text-muted-foreground">
|
|
17
|
+
Choose an endpoint from the sidebar to view its documentation
|
|
18
|
+
</p>
|
|
19
|
+
</div>
|
|
20
|
+
</div>
|
|
21
|
+
)
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
return (
|
|
25
|
+
<div className="flex-1 overflow-y-auto bg-background">
|
|
26
|
+
<RequestDetails request={request} />
|
|
27
|
+
</div>
|
|
28
|
+
)
|
|
29
|
+
}
|
|
@@ -0,0 +1,21 @@
|
|
|
1
|
+
'use client'
|
|
2
|
+
|
|
3
|
+
import { MarkdownRenderer } from '../shared/markdown-renderer'
|
|
4
|
+
import type { BrainfishCollection } from '@/lib/api-docs/types'
|
|
5
|
+
|
|
6
|
+
interface IntroductionProps {
|
|
7
|
+
collection: BrainfishCollection
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
export function Introduction({ collection }: IntroductionProps) {
|
|
11
|
+
return (
|
|
12
|
+
<div className="docs-introduction docs-content max-w-4xl mx-auto px-4 py-6 sm:px-8 sm:py-8">
|
|
13
|
+
<h1 className="docs-content-title text-2xl sm:text-3xl font-bold mb-4 sm:mb-6 text-foreground">{collection.name}</h1>
|
|
14
|
+
{collection.description && (
|
|
15
|
+
<div className="docs-prose prose prose-sm max-w-none prose-headings:text-foreground prose-p:text-muted-foreground prose-strong:text-foreground prose-code:text-foreground prose-pre:bg-muted prose-code:bg-muted prose-code:px-1 prose-code:py-0.5 prose-code:rounded prose-pre:overflow-x-auto prose-table:w-full prose-th:text-left prose-th:p-3 prose-th:bg-muted prose-td:p-3 prose-td:border-b prose-td:border-border">
|
|
16
|
+
<MarkdownRenderer content={collection.description} />
|
|
17
|
+
</div>
|
|
18
|
+
)}
|
|
19
|
+
</div>
|
|
20
|
+
)
|
|
21
|
+
}
|