@brainfish-ai/devdoc 0.1.21

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (268) hide show
  1. package/LICENSE +33 -0
  2. package/README.md +415 -0
  3. package/bin/devdoc.js +13 -0
  4. package/dist/cli/commands/build.d.ts +5 -0
  5. package/dist/cli/commands/build.js +87 -0
  6. package/dist/cli/commands/check.d.ts +1 -0
  7. package/dist/cli/commands/check.js +143 -0
  8. package/dist/cli/commands/create.d.ts +24 -0
  9. package/dist/cli/commands/create.js +387 -0
  10. package/dist/cli/commands/deploy.d.ts +9 -0
  11. package/dist/cli/commands/deploy.js +433 -0
  12. package/dist/cli/commands/dev.d.ts +6 -0
  13. package/dist/cli/commands/dev.js +139 -0
  14. package/dist/cli/commands/init.d.ts +11 -0
  15. package/dist/cli/commands/init.js +238 -0
  16. package/dist/cli/commands/keys.d.ts +12 -0
  17. package/dist/cli/commands/keys.js +165 -0
  18. package/dist/cli/commands/start.d.ts +5 -0
  19. package/dist/cli/commands/start.js +56 -0
  20. package/dist/cli/commands/upload.d.ts +13 -0
  21. package/dist/cli/commands/upload.js +238 -0
  22. package/dist/cli/commands/whoami.d.ts +8 -0
  23. package/dist/cli/commands/whoami.js +91 -0
  24. package/dist/cli/index.d.ts +1 -0
  25. package/dist/cli/index.js +106 -0
  26. package/dist/config/index.d.ts +80 -0
  27. package/dist/config/index.js +133 -0
  28. package/dist/constants.d.ts +9 -0
  29. package/dist/constants.js +13 -0
  30. package/dist/index.d.ts +7 -0
  31. package/dist/index.js +12 -0
  32. package/dist/utils/logger.d.ts +16 -0
  33. package/dist/utils/logger.js +61 -0
  34. package/dist/utils/paths.d.ts +16 -0
  35. package/dist/utils/paths.js +50 -0
  36. package/package.json +51 -0
  37. package/renderer/app/api/assets/[...path]/route.ts +123 -0
  38. package/renderer/app/api/assets/route.ts +124 -0
  39. package/renderer/app/api/assets/upload/route.ts +177 -0
  40. package/renderer/app/api/auth-schemes/route.ts +77 -0
  41. package/renderer/app/api/chat/route.ts +858 -0
  42. package/renderer/app/api/codegen/route.ts +72 -0
  43. package/renderer/app/api/collections/route.ts +1016 -0
  44. package/renderer/app/api/debug/route.ts +53 -0
  45. package/renderer/app/api/deploy/route.ts +234 -0
  46. package/renderer/app/api/device/route.ts +42 -0
  47. package/renderer/app/api/docs/route.ts +187 -0
  48. package/renderer/app/api/keys/regenerate/route.ts +80 -0
  49. package/renderer/app/api/openapi-spec/route.ts +151 -0
  50. package/renderer/app/api/projects/[slug]/route.ts +153 -0
  51. package/renderer/app/api/projects/[slug]/stats/route.ts +96 -0
  52. package/renderer/app/api/projects/register/route.ts +152 -0
  53. package/renderer/app/api/proxy/route.ts +149 -0
  54. package/renderer/app/api/proxy-stream/route.ts +168 -0
  55. package/renderer/app/api/redirects/route.ts +47 -0
  56. package/renderer/app/api/schema/route.ts +65 -0
  57. package/renderer/app/api/subdomains/check/route.ts +172 -0
  58. package/renderer/app/api/suggestions/route.ts +144 -0
  59. package/renderer/app/favicon.ico +0 -0
  60. package/renderer/app/globals.css +1103 -0
  61. package/renderer/app/layout.tsx +47 -0
  62. package/renderer/app/llms-full.txt/route.ts +346 -0
  63. package/renderer/app/llms.txt/route.ts +279 -0
  64. package/renderer/app/page.tsx +14 -0
  65. package/renderer/app/robots.txt/route.ts +84 -0
  66. package/renderer/app/sitemap.xml/route.ts +199 -0
  67. package/renderer/components/docs/index.ts +12 -0
  68. package/renderer/components/docs/mdx/accordion.tsx +169 -0
  69. package/renderer/components/docs/mdx/badge.tsx +132 -0
  70. package/renderer/components/docs/mdx/callouts.tsx +154 -0
  71. package/renderer/components/docs/mdx/cards.tsx +213 -0
  72. package/renderer/components/docs/mdx/changelog.tsx +120 -0
  73. package/renderer/components/docs/mdx/code-block.tsx +186 -0
  74. package/renderer/components/docs/mdx/code-group.tsx +421 -0
  75. package/renderer/components/docs/mdx/file-embeds.tsx +105 -0
  76. package/renderer/components/docs/mdx/frame.tsx +112 -0
  77. package/renderer/components/docs/mdx/highlight.tsx +151 -0
  78. package/renderer/components/docs/mdx/iframe.tsx +134 -0
  79. package/renderer/components/docs/mdx/image.tsx +235 -0
  80. package/renderer/components/docs/mdx/index.ts +204 -0
  81. package/renderer/components/docs/mdx/mermaid.tsx +240 -0
  82. package/renderer/components/docs/mdx/param-field.tsx +200 -0
  83. package/renderer/components/docs/mdx/steps.tsx +113 -0
  84. package/renderer/components/docs/mdx/tabs.tsx +86 -0
  85. package/renderer/components/docs/mdx-renderer.tsx +100 -0
  86. package/renderer/components/docs/navigation/breadcrumbs.tsx +76 -0
  87. package/renderer/components/docs/navigation/index.ts +8 -0
  88. package/renderer/components/docs/navigation/page-nav.tsx +64 -0
  89. package/renderer/components/docs/navigation/sidebar.tsx +515 -0
  90. package/renderer/components/docs/navigation/toc.tsx +113 -0
  91. package/renderer/components/docs/notice.tsx +105 -0
  92. package/renderer/components/docs-header.tsx +274 -0
  93. package/renderer/components/docs-viewer/agent/agent-chat.tsx +2076 -0
  94. package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +90 -0
  95. package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +49 -0
  96. package/renderer/components/docs-viewer/agent/cards/index.tsx +50 -0
  97. package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +212 -0
  98. package/renderer/components/docs-viewer/agent/cards/types.ts +84 -0
  99. package/renderer/components/docs-viewer/agent/chat-message.tsx +17 -0
  100. package/renderer/components/docs-viewer/agent/index.tsx +6 -0
  101. package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +119 -0
  102. package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +46 -0
  103. package/renderer/components/docs-viewer/agent/messages/index.ts +17 -0
  104. package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +721 -0
  105. package/renderer/components/docs-viewer/agent/messages/types.ts +61 -0
  106. package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +24 -0
  107. package/renderer/components/docs-viewer/agent/messages/user-message.tsx +51 -0
  108. package/renderer/components/docs-viewer/code-editor/index.tsx +2 -0
  109. package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +1283 -0
  110. package/renderer/components/docs-viewer/content/changelog-page.tsx +331 -0
  111. package/renderer/components/docs-viewer/content/doc-page.tsx +285 -0
  112. package/renderer/components/docs-viewer/content/documentation-viewer.tsx +17 -0
  113. package/renderer/components/docs-viewer/content/index.tsx +29 -0
  114. package/renderer/components/docs-viewer/content/introduction.tsx +21 -0
  115. package/renderer/components/docs-viewer/content/request-details.tsx +330 -0
  116. package/renderer/components/docs-viewer/content/sections/auth.tsx +69 -0
  117. package/renderer/components/docs-viewer/content/sections/body.tsx +66 -0
  118. package/renderer/components/docs-viewer/content/sections/headers.tsx +43 -0
  119. package/renderer/components/docs-viewer/content/sections/overview.tsx +40 -0
  120. package/renderer/components/docs-viewer/content/sections/parameters.tsx +43 -0
  121. package/renderer/components/docs-viewer/content/sections/responses.tsx +87 -0
  122. package/renderer/components/docs-viewer/global-auth-modal.tsx +352 -0
  123. package/renderer/components/docs-viewer/index.tsx +1466 -0
  124. package/renderer/components/docs-viewer/playground/auth-editor.tsx +280 -0
  125. package/renderer/components/docs-viewer/playground/body-editor.tsx +221 -0
  126. package/renderer/components/docs-viewer/playground/code-editor.tsx +224 -0
  127. package/renderer/components/docs-viewer/playground/code-snippet.tsx +387 -0
  128. package/renderer/components/docs-viewer/playground/graphql-playground.tsx +745 -0
  129. package/renderer/components/docs-viewer/playground/index.tsx +671 -0
  130. package/renderer/components/docs-viewer/playground/key-value-editor.tsx +261 -0
  131. package/renderer/components/docs-viewer/playground/method-selector.tsx +60 -0
  132. package/renderer/components/docs-viewer/playground/request-builder.tsx +179 -0
  133. package/renderer/components/docs-viewer/playground/request-tabs.tsx +237 -0
  134. package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +21 -0
  135. package/renderer/components/docs-viewer/playground/response-cards/index.tsx +93 -0
  136. package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +16 -0
  137. package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +23 -0
  138. package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +268 -0
  139. package/renderer/components/docs-viewer/playground/response-cards/types.ts +82 -0
  140. package/renderer/components/docs-viewer/playground/response-viewer.tsx +43 -0
  141. package/renderer/components/docs-viewer/search/index.ts +2 -0
  142. package/renderer/components/docs-viewer/search/search-dialog.tsx +331 -0
  143. package/renderer/components/docs-viewer/search/use-search.ts +117 -0
  144. package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +431 -0
  145. package/renderer/components/docs-viewer/shared/method-badge.tsx +41 -0
  146. package/renderer/components/docs-viewer/shared/schema-viewer.tsx +349 -0
  147. package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +239 -0
  148. package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +316 -0
  149. package/renderer/components/docs-viewer/sidebar/index.tsx +343 -0
  150. package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +202 -0
  151. package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +118 -0
  152. package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +226 -0
  153. package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +52 -0
  154. package/renderer/components/theme-provider.tsx +11 -0
  155. package/renderer/components/theme-toggle.tsx +76 -0
  156. package/renderer/components/ui/badge.tsx +46 -0
  157. package/renderer/components/ui/button.tsx +59 -0
  158. package/renderer/components/ui/dialog.tsx +118 -0
  159. package/renderer/components/ui/dropdown-menu.tsx +257 -0
  160. package/renderer/components/ui/input.tsx +21 -0
  161. package/renderer/components/ui/label.tsx +24 -0
  162. package/renderer/components/ui/navigation-menu.tsx +168 -0
  163. package/renderer/components/ui/select.tsx +190 -0
  164. package/renderer/components/ui/spinner.tsx +114 -0
  165. package/renderer/components/ui/tabs.tsx +66 -0
  166. package/renderer/components/ui/tooltip.tsx +61 -0
  167. package/renderer/hooks/use-code-copy.ts +88 -0
  168. package/renderer/hooks/use-openapi-title.ts +44 -0
  169. package/renderer/lib/api-docs/agent/index.ts +6 -0
  170. package/renderer/lib/api-docs/agent/indexer.ts +323 -0
  171. package/renderer/lib/api-docs/agent/spec-summary.ts +335 -0
  172. package/renderer/lib/api-docs/agent/types.ts +116 -0
  173. package/renderer/lib/api-docs/auth/auth-context.tsx +225 -0
  174. package/renderer/lib/api-docs/auth/auth-storage.ts +87 -0
  175. package/renderer/lib/api-docs/auth/crypto.ts +89 -0
  176. package/renderer/lib/api-docs/auth/index.ts +4 -0
  177. package/renderer/lib/api-docs/code-editor/db.ts +164 -0
  178. package/renderer/lib/api-docs/code-editor/hooks.ts +266 -0
  179. package/renderer/lib/api-docs/code-editor/index.ts +6 -0
  180. package/renderer/lib/api-docs/code-editor/mode-context.tsx +207 -0
  181. package/renderer/lib/api-docs/code-editor/types.ts +105 -0
  182. package/renderer/lib/api-docs/codegen/definitions.ts +297 -0
  183. package/renderer/lib/api-docs/codegen/har.ts +251 -0
  184. package/renderer/lib/api-docs/codegen/index.ts +159 -0
  185. package/renderer/lib/api-docs/factories.ts +151 -0
  186. package/renderer/lib/api-docs/index.ts +17 -0
  187. package/renderer/lib/api-docs/mobile-context.tsx +112 -0
  188. package/renderer/lib/api-docs/navigation-context.tsx +88 -0
  189. package/renderer/lib/api-docs/parsers/graphql/README.md +129 -0
  190. package/renderer/lib/api-docs/parsers/graphql/index.ts +91 -0
  191. package/renderer/lib/api-docs/parsers/graphql/parser.ts +491 -0
  192. package/renderer/lib/api-docs/parsers/graphql/transformer.ts +246 -0
  193. package/renderer/lib/api-docs/parsers/graphql/types.ts +283 -0
  194. package/renderer/lib/api-docs/parsers/openapi/README.md +32 -0
  195. package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +60 -0
  196. package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +574 -0
  197. package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +403 -0
  198. package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +232 -0
  199. package/renderer/lib/api-docs/parsers/openapi/index.ts +171 -0
  200. package/renderer/lib/api-docs/parsers/openapi/transformer.ts +277 -0
  201. package/renderer/lib/api-docs/parsers/openapi/validator.ts +31 -0
  202. package/renderer/lib/api-docs/playground/context.tsx +107 -0
  203. package/renderer/lib/api-docs/playground/navigation-context.tsx +124 -0
  204. package/renderer/lib/api-docs/playground/request-builder.ts +223 -0
  205. package/renderer/lib/api-docs/playground/request-runner.ts +282 -0
  206. package/renderer/lib/api-docs/playground/types.ts +35 -0
  207. package/renderer/lib/api-docs/types.ts +269 -0
  208. package/renderer/lib/api-docs/utils.ts +311 -0
  209. package/renderer/lib/cache.ts +193 -0
  210. package/renderer/lib/docs/config/index.ts +29 -0
  211. package/renderer/lib/docs/config/loader.ts +142 -0
  212. package/renderer/lib/docs/config/schema.ts +298 -0
  213. package/renderer/lib/docs/index.ts +12 -0
  214. package/renderer/lib/docs/mdx/compiler.ts +176 -0
  215. package/renderer/lib/docs/mdx/frontmatter.ts +80 -0
  216. package/renderer/lib/docs/mdx/index.ts +26 -0
  217. package/renderer/lib/docs/navigation/generator.ts +348 -0
  218. package/renderer/lib/docs/navigation/index.ts +12 -0
  219. package/renderer/lib/docs/navigation/types.ts +123 -0
  220. package/renderer/lib/docs-navigation-context.tsx +80 -0
  221. package/renderer/lib/multi-tenant/context.ts +105 -0
  222. package/renderer/lib/storage/blob.ts +845 -0
  223. package/renderer/lib/utils.ts +6 -0
  224. package/renderer/next.config.ts +76 -0
  225. package/renderer/package.json +66 -0
  226. package/renderer/postcss.config.mjs +5 -0
  227. package/renderer/public/assets/images/screenshot.png +0 -0
  228. package/renderer/public/assets/logo/dark.svg +9 -0
  229. package/renderer/public/assets/logo/light.svg +9 -0
  230. package/renderer/public/assets/logo.svg +9 -0
  231. package/renderer/public/file.svg +1 -0
  232. package/renderer/public/globe.svg +1 -0
  233. package/renderer/public/icon.png +0 -0
  234. package/renderer/public/logo.svg +9 -0
  235. package/renderer/public/window.svg +1 -0
  236. package/renderer/tsconfig.json +28 -0
  237. package/templates/basic/README.md +139 -0
  238. package/templates/basic/assets/favicon.svg +4 -0
  239. package/templates/basic/assets/logo.svg +9 -0
  240. package/templates/basic/docs.json +47 -0
  241. package/templates/basic/guides/configuration.mdx +149 -0
  242. package/templates/basic/guides/overview.mdx +96 -0
  243. package/templates/basic/index.mdx +39 -0
  244. package/templates/basic/package.json +14 -0
  245. package/templates/basic/quickstart.mdx +92 -0
  246. package/templates/basic/vercel.json +6 -0
  247. package/templates/graphql/README.md +139 -0
  248. package/templates/graphql/api-reference/schema.graphql +305 -0
  249. package/templates/graphql/assets/favicon.svg +4 -0
  250. package/templates/graphql/assets/logo.svg +9 -0
  251. package/templates/graphql/docs.json +54 -0
  252. package/templates/graphql/guides/configuration.mdx +149 -0
  253. package/templates/graphql/guides/overview.mdx +96 -0
  254. package/templates/graphql/index.mdx +39 -0
  255. package/templates/graphql/package.json +14 -0
  256. package/templates/graphql/quickstart.mdx +92 -0
  257. package/templates/graphql/vercel.json +6 -0
  258. package/templates/openapi/README.md +139 -0
  259. package/templates/openapi/api-reference/openapi.json +419 -0
  260. package/templates/openapi/assets/favicon.svg +4 -0
  261. package/templates/openapi/assets/logo.svg +9 -0
  262. package/templates/openapi/docs.json +61 -0
  263. package/templates/openapi/guides/configuration.mdx +149 -0
  264. package/templates/openapi/guides/overview.mdx +96 -0
  265. package/templates/openapi/index.mdx +39 -0
  266. package/templates/openapi/package.json +14 -0
  267. package/templates/openapi/quickstart.mdx +92 -0
  268. package/templates/openapi/vercel.json +6 -0
@@ -0,0 +1,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
+ }