@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,431 @@
1
+ 'use client'
2
+
3
+ import React, { useEffect, useState, useRef } from 'react'
4
+ import ReactMarkdown from 'react-markdown'
5
+ import remarkGfm from 'remark-gfm'
6
+ import type { Components } from 'react-markdown'
7
+ import { CodeViewer, SupportedLanguage } from '../playground/code-editor'
8
+ import { useNavigation } from '@/lib/api-docs/navigation-context'
9
+
10
+ // Error Boundary for Mermaid Preview
11
+ class MermaidErrorBoundary extends React.Component<
12
+ { children: React.ReactNode; code: string },
13
+ { hasError: boolean; error: string | null }
14
+ > {
15
+ constructor(props: { children: React.ReactNode; code: string }) {
16
+ super(props)
17
+ this.state = { hasError: false, error: null }
18
+ }
19
+
20
+ static getDerivedStateFromError(error: Error) {
21
+ return { hasError: true, error: error.message }
22
+ }
23
+
24
+ componentDidCatch(error: Error, errorInfo: React.ErrorInfo) {
25
+ console.error('[MermaidErrorBoundary] Error:', error, errorInfo)
26
+ }
27
+
28
+ render() {
29
+ if (this.state.hasError) {
30
+ return (
31
+ <div className="my-4 p-4 border border-red-500/30 bg-red-500/10 rounded-lg">
32
+ <div className="text-red-400 text-sm font-medium mb-2">Mermaid Diagram Error</div>
33
+ <div className="text-red-300 text-xs mb-2">{this.state.error}</div>
34
+ <pre className="text-xs text-muted-foreground overflow-x-auto p-2 bg-muted rounded">
35
+ {this.props.code}
36
+ </pre>
37
+ </div>
38
+ )
39
+ }
40
+ return this.props.children
41
+ }
42
+ }
43
+
44
+ // Mermaid preview component
45
+ function MermaidPreviewInner({ code }: { code: string }) {
46
+ const [svg, setSvg] = useState<string>('')
47
+ const [error, setError] = useState<string | null>(null)
48
+ const [loading, setLoading] = useState(true)
49
+ const renderIdRef = useRef(0)
50
+
51
+ useEffect(() => {
52
+ if (!code.trim()) {
53
+ setLoading(false)
54
+ setSvg('')
55
+ setError(null)
56
+ return
57
+ }
58
+
59
+ const currentRenderId = ++renderIdRef.current
60
+ setLoading(true)
61
+ setError(null)
62
+
63
+ const renderDiagram = async () => {
64
+ try {
65
+ const mermaid = (await import('mermaid')).default
66
+
67
+ mermaid.initialize({
68
+ startOnLoad: false,
69
+ theme: 'dark',
70
+ themeVariables: {
71
+ background: '#1e1e1e',
72
+ primaryColor: '#3b82f6',
73
+ primaryTextColor: '#ffffff',
74
+ primaryBorderColor: '#60a5fa',
75
+ secondaryColor: '#3b82f6',
76
+ secondaryTextColor: '#ffffff',
77
+ secondaryBorderColor: '#60a5fa',
78
+ tertiaryColor: '#3b82f6',
79
+ tertiaryTextColor: '#ffffff',
80
+ tertiaryBorderColor: '#60a5fa',
81
+ mainBkg: '#3b82f6',
82
+ nodeBkg: '#3b82f6',
83
+ nodeBorder: '#60a5fa',
84
+ nodeTextColor: '#ffffff',
85
+ clusterBkg: '#27272a',
86
+ clusterBorder: '#52525b',
87
+ titleColor: '#ffffff',
88
+ lineColor: '#94a3b8',
89
+ fontFamily: 'system-ui, sans-serif',
90
+ fontSize: '14px',
91
+ },
92
+ flowchart: {
93
+ htmlLabels: true,
94
+ useMaxWidth: true,
95
+ }
96
+ })
97
+
98
+ const { svg: renderedSvg } = await mermaid.render(
99
+ `mermaid-md-${currentRenderId}-${Date.now()}`,
100
+ code
101
+ )
102
+
103
+ if (currentRenderId === renderIdRef.current) {
104
+ setSvg(renderedSvg)
105
+ setError(null)
106
+ }
107
+ } catch (err: unknown) {
108
+ if (currentRenderId === renderIdRef.current) {
109
+ let errorMessage = 'Failed to render diagram'
110
+ if (err instanceof Error) {
111
+ errorMessage = err.message
112
+ if (errorMessage.includes('Parse error')) {
113
+ const match = errorMessage.match(/Parse error on line (\d+)/)
114
+ if (match) {
115
+ errorMessage = `Syntax error on line ${match[1]}`
116
+ }
117
+ }
118
+ } else if (typeof err === 'string') {
119
+ errorMessage = err
120
+ }
121
+ console.error('[MermaidPreview] Render error:', errorMessage)
122
+ setError(errorMessage)
123
+ }
124
+ } finally {
125
+ if (currentRenderId === renderIdRef.current) {
126
+ setLoading(false)
127
+ }
128
+ }
129
+ }
130
+
131
+ const debounceTimer = setTimeout(renderDiagram, 300)
132
+ return () => clearTimeout(debounceTimer)
133
+ }, [code])
134
+
135
+ if (loading) {
136
+ return (
137
+ <div className="my-4 p-4 bg-muted/30 rounded-lg flex items-center justify-center min-h-[100px]">
138
+ <div className="text-muted-foreground text-sm">Rendering diagram...</div>
139
+ </div>
140
+ )
141
+ }
142
+
143
+ if (error) {
144
+ return (
145
+ <div className="my-4 p-4 border border-yellow-500/30 bg-yellow-500/10 rounded-lg">
146
+ <div className="text-yellow-400 text-sm font-medium mb-2">Diagram Syntax Error</div>
147
+ <div className="text-yellow-300 text-xs mb-2">{error}</div>
148
+ <pre className="text-xs text-muted-foreground overflow-x-auto p-2 bg-muted rounded">
149
+ {code}
150
+ </pre>
151
+ </div>
152
+ )
153
+ }
154
+
155
+ if (!svg) {
156
+ return null
157
+ }
158
+
159
+ return (
160
+ <div
161
+ className="my-4 p-4 bg-zinc-900 rounded-lg overflow-x-auto mermaid"
162
+ dangerouslySetInnerHTML={{ __html: svg }}
163
+ />
164
+ )
165
+ }
166
+
167
+ // Mermaid preview with error boundary wrapper
168
+ function MermaidPreview({ code }: { code: string }) {
169
+ return (
170
+ <MermaidErrorBoundary code={code}>
171
+ <MermaidPreviewInner code={code} />
172
+ </MermaidErrorBoundary>
173
+ )
174
+ }
175
+
176
+ interface MarkdownRendererProps {
177
+ content: string
178
+ className?: string
179
+ }
180
+
181
+
182
+
183
+ // Map common language identifiers to Monaco supported languages
184
+ function getMonacoLanguage(lang: string | undefined): SupportedLanguage {
185
+ if (!lang) return 'plaintext'
186
+
187
+ const langMap: Record<string, SupportedLanguage> = {
188
+ 'js': 'javascript',
189
+ 'javascript': 'javascript',
190
+ 'ts': 'typescript',
191
+ 'typescript': 'typescript',
192
+ 'json': 'json',
193
+ 'xml': 'xml',
194
+ 'html': 'html',
195
+ 'htm': 'html',
196
+ 'bash': 'shell',
197
+ 'shell': 'shell',
198
+ 'sh': 'shell',
199
+ 'zsh': 'shell',
200
+ 'curl': 'shell',
201
+ 'text': 'plaintext',
202
+ 'plaintext': 'plaintext',
203
+ }
204
+
205
+ return langMap[lang.toLowerCase()] || 'plaintext'
206
+ }
207
+
208
+ /**
209
+ * Generate a URL-friendly slug from heading text
210
+ */
211
+ function generateSlug(text: string): string {
212
+ return text
213
+ .toLowerCase()
214
+ .replace(/[^a-z0-9]+/g, '-')
215
+ .replace(/^-+|-+$/g, '')
216
+ }
217
+
218
+ /**
219
+ * Extract text content from React children
220
+ */
221
+ function getTextContent(children: React.ReactNode): string {
222
+ if (typeof children === 'string') return children
223
+ if (typeof children === 'number') return String(children)
224
+ if (Array.isArray(children)) return children.map(getTextContent).join('')
225
+ if (children && typeof children === 'object' && 'props' in children) {
226
+ const element = children as React.ReactElement<{ children?: React.ReactNode }>
227
+ return getTextContent(element.props.children)
228
+ }
229
+ return ''
230
+ }
231
+
232
+ /**
233
+ * Custom link component that handles internal endpoint links
234
+ */
235
+ function MarkdownLink({ href, children }: { href?: string; children: React.ReactNode }) {
236
+ const { navigateToEndpoint } = useNavigation()
237
+
238
+ // Check if this is an internal endpoint link
239
+ // Formats: #operation/operationId, #/paths/~1endpoint/method
240
+ const isEndpointLink = href?.startsWith('#operation/') || href?.startsWith('#/paths/')
241
+
242
+ if (isEndpointLink) {
243
+ let operationId = ''
244
+
245
+ if (href?.startsWith('#operation/')) {
246
+ // Format: #operation/post_carts
247
+ operationId = href.replace('#operation/', '')
248
+ } else if (href?.startsWith('#/paths/')) {
249
+ // Format: #/paths/~1carts/post → extract operation (simplified)
250
+ const parts = href.replace('#/paths/', '').split('/')
251
+ const method = parts[parts.length - 1]
252
+ const path = parts.slice(0, -1).join('/').replace(/~1/g, '/')
253
+ operationId = `${method}_${path.replace(/\//g, '_').replace(/^_/, '')}`
254
+ }
255
+
256
+ return (
257
+ <button
258
+ onClick={() => navigateToEndpoint(operationId)}
259
+ className="text-primary hover:underline font-medium cursor-pointer bg-transparent border-none p-0 inline"
260
+ >
261
+ {children}
262
+ </button>
263
+ )
264
+ }
265
+
266
+ // External link - open in new tab
267
+ return (
268
+ <a
269
+ href={href}
270
+ target="_blank"
271
+ rel="noopener noreferrer"
272
+ className="text-primary hover:underline font-medium"
273
+ >
274
+ {children}
275
+ </a>
276
+ )
277
+ }
278
+
279
+ const markdownComponents: Partial<Components> = {
280
+ h1: ({ children }) => {
281
+ const text = getTextContent(children)
282
+ const id = generateSlug(text)
283
+ return (
284
+ <h1 id={id} className="text-2xl font-bold mt-8 mb-4 pb-2 border-b border-border scroll-mt-20">{children}</h1>
285
+ )
286
+ },
287
+ h2: ({ children }) => {
288
+ const text = getTextContent(children)
289
+ const id = generateSlug(text)
290
+ return (
291
+ <h2 id={id} className="text-xl font-semibold mt-6 mb-3 scroll-mt-20">{children}</h2>
292
+ )
293
+ },
294
+ h3: ({ children }) => {
295
+ const text = getTextContent(children)
296
+ const id = generateSlug(text)
297
+ return (
298
+ <h3 id={id} className="text-lg font-semibold mt-5 mb-2 scroll-mt-20">{children}</h3>
299
+ )
300
+ },
301
+ h4: ({ children }) => {
302
+ const text = getTextContent(children)
303
+ const id = generateSlug(text)
304
+ return (
305
+ <h4 id={id} className="text-base font-semibold mt-4 mb-2 scroll-mt-20">{children}</h4>
306
+ )
307
+ },
308
+ h5: ({ children }) => {
309
+ const text = getTextContent(children)
310
+ const id = generateSlug(text)
311
+ return (
312
+ <h5 id={id} className="text-sm font-semibold mt-3 mb-1 scroll-mt-20">{children}</h5>
313
+ )
314
+ },
315
+ h6: ({ children }) => {
316
+ const text = getTextContent(children)
317
+ const id = generateSlug(text)
318
+ return (
319
+ <h6 id={id} className="text-sm font-medium mt-2 mb-1 text-muted-foreground scroll-mt-20">{children}</h6>
320
+ )
321
+ },
322
+ p: ({ children }) => (
323
+ <p className="mb-4 leading-7 text-foreground/90">{children}</p>
324
+ ),
325
+ ul: ({ children }) => (
326
+ <ul className="list-disc pl-6 mb-4 space-y-2">{children}</ul>
327
+ ),
328
+ ol: ({ children }) => (
329
+ <ol className="list-decimal pl-6 mb-4 space-y-2">{children}</ol>
330
+ ),
331
+ li: ({ children }) => (
332
+ <li className="leading-7">{children}</li>
333
+ ),
334
+ code: ({ className, children, ...props }) => {
335
+ const match = /language-(\w+)/.exec(className || '')
336
+ const isCodeBlock = !!match
337
+ const detectedLanguage = match?.[1]?.toLowerCase()
338
+ const codeContent = String(children).replace(/\n$/, '')
339
+
340
+ // Inline code
341
+ if (!isCodeBlock) {
342
+ return (
343
+ <code
344
+ className="px-1.5 py-0.5 bg-muted rounded text-sm font-mono text-foreground"
345
+ {...props}
346
+ >
347
+ {children}
348
+ </code>
349
+ )
350
+ }
351
+
352
+ // Mermaid code blocks - render as diagrams
353
+ if (detectedLanguage === 'mermaid') {
354
+ return <MermaidPreview code={codeContent} />
355
+ }
356
+
357
+ // Code block - use Monaco with dark theme and no line numbers
358
+ const language = getMonacoLanguage(detectedLanguage)
359
+
360
+ return (
361
+ <div className="my-4 dark-editor">
362
+ <CodeViewer
363
+ value={codeContent}
364
+ language={language}
365
+ height="auto"
366
+ minHeight={Math.min(Math.max(codeContent.split('\n').length * 20 + 24, 60), 400)}
367
+ theme="dark"
368
+ showLineNumbers={false}
369
+ />
370
+ </div>
371
+ )
372
+ },
373
+ pre: ({ children }) => {
374
+ // Pre is handled by code block, just pass through
375
+ return <>{children}</>
376
+ },
377
+ blockquote: ({ children }) => (
378
+ <blockquote className="border-l-4 border-primary/50 pl-4 py-1 my-4 bg-muted/30 rounded-r">
379
+ <div className="text-muted-foreground italic">{children}</div>
380
+ </blockquote>
381
+ ),
382
+ a: ({ href, children }) => (
383
+ <MarkdownLink href={href}>{children}</MarkdownLink>
384
+ ),
385
+ strong: ({ children }) => (
386
+ <strong className="font-bold text-foreground">{children}</strong>
387
+ ),
388
+ em: ({ children }) => (
389
+ <em className="italic">{children}</em>
390
+ ),
391
+ hr: () => (
392
+ <hr className="my-8 border-border" />
393
+ ),
394
+ table: ({ children }) => (
395
+ <div className="my-4 overflow-x-auto">
396
+ <table className="min-w-full border border-border rounded-sm overflow-hidden">
397
+ {children}
398
+ </table>
399
+ </div>
400
+ ),
401
+ thead: ({ children }) => (
402
+ <thead className="bg-muted">{children}</thead>
403
+ ),
404
+ tbody: ({ children }) => (
405
+ <tbody className="divide-y divide-border">{children}</tbody>
406
+ ),
407
+ tr: ({ children }) => (
408
+ <tr className="hover:bg-muted/50">{children}</tr>
409
+ ),
410
+ th: ({ children }) => (
411
+ <th className="px-4 py-2 text-left text-sm font-semibold">{children}</th>
412
+ ),
413
+ td: ({ children }) => (
414
+ <td className="px-4 py-2 text-sm">{children}</td>
415
+ ),
416
+ }
417
+
418
+ export function MarkdownRenderer({ content, className = '' }: MarkdownRendererProps) {
419
+ if (!content) return null
420
+
421
+ return (
422
+ <div className={`max-w-none ${className}`}>
423
+ <ReactMarkdown
424
+ components={markdownComponents}
425
+ remarkPlugins={[remarkGfm]}
426
+ >
427
+ {content}
428
+ </ReactMarkdown>
429
+ </div>
430
+ )
431
+ }
@@ -0,0 +1,41 @@
1
+ 'use client'
2
+
3
+ import { cn } from '@/lib/utils'
4
+ import type { HTTPMethod } from '@/lib/api-docs/types'
5
+
6
+ interface MethodBadgeProps {
7
+ method: HTTPMethod
8
+ size?: 'sm' | 'md' | 'lg'
9
+ className?: string
10
+ }
11
+
12
+ const methodColors: Record<HTTPMethod, string> = {
13
+ GET: 'bg-emerald-500/90 text-white',
14
+ POST: 'bg-blue-500/90 text-white',
15
+ PUT: 'bg-amber-500/90 text-white',
16
+ DELETE: 'bg-red-500/90 text-white',
17
+ PATCH: 'bg-violet-500/90 text-white',
18
+ HEAD: 'bg-slate-500/90 text-white',
19
+ OPTIONS: 'bg-slate-500/90 text-white',
20
+ }
21
+
22
+ const sizeClasses = {
23
+ sm: 'px-1.5 py-0.5 text-[10px]',
24
+ md: 'px-2 py-0.5 text-xs',
25
+ lg: 'px-2.5 py-1 text-sm',
26
+ }
27
+
28
+ export function MethodBadge({ method, size = 'md', className = '' }: MethodBadgeProps) {
29
+ return (
30
+ <span
31
+ className={cn(
32
+ 'font-semibold rounded uppercase tracking-wide',
33
+ methodColors[method],
34
+ sizeClasses[size],
35
+ className
36
+ )}
37
+ >
38
+ {method}
39
+ </span>
40
+ )
41
+ }