@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,421 @@
1
+ 'use client'
2
+
3
+ import React, { useState, Children, isValidElement, ReactNode } from 'react'
4
+ import { cn } from '@/lib/utils'
5
+ import { Copy, Check } from '@phosphor-icons/react'
6
+
7
+ /**
8
+ * CodeGroup Component for MDX Documentation
9
+ *
10
+ * Groups multiple code blocks with tabs for language/file selection.
11
+ * Similar to Mintlify's CodeGroup component.
12
+ *
13
+ * Usage:
14
+ * <CodeGroup>
15
+ * ```js title="helloWorld.js"
16
+ * console.log("Hello World");
17
+ * ```
18
+ *
19
+ * ```python title="hello_world.py"
20
+ * print("Hello World")
21
+ * ```
22
+ * </CodeGroup>
23
+ */
24
+
25
+ interface CodeBlockInfo {
26
+ title: string
27
+ language: string
28
+ content: string
29
+ element: React.ReactNode
30
+ }
31
+
32
+ interface CodeGroupProps {
33
+ children: React.ReactNode
34
+ className?: string
35
+ /** Display variant: 'default' (dark with underline), 'tabs' (light tab style), 'pills' (pill buttons) */
36
+ variant?: 'default' | 'tabs' | 'pills'
37
+ }
38
+
39
+ // Extract text content from nested React elements
40
+ function extractTextContent(node: ReactNode): string {
41
+ if (typeof node === 'string') return node
42
+ if (typeof node === 'number') return String(node)
43
+ if (Array.isArray(node)) return node.map(extractTextContent).join('')
44
+ if (isValidElement(node)) {
45
+ const props = node.props as { children?: ReactNode }
46
+ if (props.children) {
47
+ return extractTextContent(props.children)
48
+ }
49
+ }
50
+ return ''
51
+ }
52
+
53
+ // Extract title from code comment on first line (e.g., "// npm" or "# yarn")
54
+ function extractTitleFromComment(code: string): string | null {
55
+ const firstLine = code.trim().split('\n')[0]
56
+
57
+ // Match various comment styles
58
+ const patterns = [
59
+ /^\/\/\s*(.+)$/, // // comment
60
+ /^#\s*(.+)$/, // # comment
61
+ /^\/\*\s*(.+?)\s*\*\/$/, // /* comment */
62
+ /^<!--\s*(.+?)\s*-->$/, // <!-- comment -->
63
+ /^--\s*(.+)$/, // -- comment (SQL)
64
+ ]
65
+
66
+ for (const pattern of patterns) {
67
+ const match = firstLine.match(pattern)
68
+ if (match && match[1] && match[1].length < 30) {
69
+ return match[1].trim()
70
+ }
71
+ }
72
+
73
+ return null
74
+ }
75
+
76
+ // Check if element type matches a tag name (handles both string and component types)
77
+ function isElementType(element: React.ReactElement, tagName: string): boolean {
78
+ const type = element.type
79
+ if (typeof type === 'string') {
80
+ return type === tagName
81
+ }
82
+ // Check displayName or name for components
83
+ if (typeof type === 'function') {
84
+ const funcType = type as { displayName?: string; name?: string }
85
+ return funcType.displayName === tagName || funcType.name === tagName
86
+ }
87
+ return false
88
+ }
89
+
90
+ // Recursively find all pre elements
91
+ function findPreElements(children: ReactNode, results: CodeBlockInfo[] = []): CodeBlockInfo[] {
92
+ Children.forEach(children, (child) => {
93
+ if (!child) return
94
+
95
+ if (isValidElement(child)) {
96
+ const props = child.props as Record<string, unknown>
97
+
98
+ // Check if it's a pre element (either native or component)
99
+ if (isElementType(child, 'pre')) {
100
+ const codeChild = props.children
101
+ let language = 'text'
102
+ let content = ''
103
+
104
+ if (isValidElement(codeChild)) {
105
+ const codeProps = codeChild.props as Record<string, unknown>
106
+ const className = (codeProps.className as string) || ''
107
+ const match = className.match(/language-(\w+)/)
108
+ language = match ? match[1] : 'text'
109
+ content = extractTextContent(codeProps.children as ReactNode)
110
+ } else if (typeof codeChild === 'string') {
111
+ content = codeChild
112
+ } else {
113
+ // Try to get content from nested structure
114
+ content = extractTextContent(codeChild as ReactNode)
115
+ }
116
+
117
+ // Also check pre element's className for language
118
+ const preClassName = (props.className as string) || ''
119
+ const preMatch = preClassName.match(/language-(\w+)/)
120
+ if (preMatch) {
121
+ language = preMatch[1]
122
+ }
123
+
124
+ // Check data-language attribute
125
+ if (props['data-language']) {
126
+ language = props['data-language'] as string
127
+ }
128
+
129
+ // Extract title from various possible locations
130
+ // Priority: data-title > title prop > comment in code > language name
131
+ const commentTitle = extractTitleFromComment(content)
132
+ const title = (props['data-title'] as string) ||
133
+ (props.title as string) ||
134
+ commentTitle ||
135
+ getDisplayName(language)
136
+
137
+ results.push({ title, language, content, element: child })
138
+ }
139
+ // Check for figure elements (used by some syntax highlighters like rehype-pretty-code)
140
+ else if (isElementType(child, 'figure') && props['data-rehype-pretty-code-figure'] !== undefined) {
141
+ // Find pre inside figure
142
+ findPreElements(props.children as ReactNode, results)
143
+ }
144
+ // Check if it has children and recurse
145
+ else if (props.children) {
146
+ findPreElements(props.children as ReactNode, results)
147
+ }
148
+ }
149
+ })
150
+
151
+ return results
152
+ }
153
+
154
+ // Get display name for language
155
+ function getDisplayName(language: string): string {
156
+ const displayNames: Record<string, string> = {
157
+ javascript: 'JavaScript',
158
+ typescript: 'TypeScript',
159
+ python: 'Python',
160
+ java: 'Java',
161
+ go: 'Go',
162
+ rust: 'Rust',
163
+ ruby: 'Ruby',
164
+ php: 'PHP',
165
+ csharp: 'C#',
166
+ cpp: 'C++',
167
+ c: 'C',
168
+ swift: 'Swift',
169
+ kotlin: 'Kotlin',
170
+ bash: 'Bash',
171
+ shell: 'Shell',
172
+ json: 'JSON',
173
+ yaml: 'YAML',
174
+ html: 'HTML',
175
+ css: 'CSS',
176
+ sql: 'SQL',
177
+ markdown: 'Markdown',
178
+ text: 'Text',
179
+ plaintext: 'Text',
180
+ }
181
+ return displayNames[language.toLowerCase()] || language.charAt(0).toUpperCase() + language.slice(1)
182
+ }
183
+
184
+ export function CodeGroup({ children, className, variant = 'default' }: CodeGroupProps) {
185
+ const blocks = findPreElements(children)
186
+ const [activeIndex, setActiveIndex] = useState(0)
187
+ const [copied, setCopied] = useState(false)
188
+
189
+ // If no code blocks found, just render children as-is
190
+ if (blocks.length === 0) {
191
+ return <div className={className}>{children}</div>
192
+ }
193
+
194
+ const activeBlock = blocks[activeIndex]
195
+
196
+ const handleCopy = async () => {
197
+ if (activeBlock?.content) {
198
+ try {
199
+ await navigator.clipboard.writeText(activeBlock.content.trim())
200
+ setCopied(true)
201
+ setTimeout(() => setCopied(false), 2000)
202
+ } catch (err) {
203
+ console.error('Failed to copy:', err)
204
+ }
205
+ }
206
+ }
207
+
208
+ // Tabs variant - dark theme with border tabs
209
+ if (variant === 'tabs') {
210
+ return (
211
+ <div className={cn(
212
+ 'docs-code-group docs-code-group-tabs my-4 rounded-xl border border-zinc-800 overflow-hidden bg-[#1a1a1a]',
213
+ className
214
+ )}>
215
+ {/* Tab headers */}
216
+ <div className="docs-code-group-header flex items-center justify-between border-b border-zinc-800 bg-[#262626]">
217
+ <div className="flex overflow-x-auto">
218
+ {blocks.map((block, index) => (
219
+ <button
220
+ key={index}
221
+ type="button"
222
+ onClick={() => setActiveIndex(index)}
223
+ className={cn(
224
+ 'docs-code-group-tab px-4 py-2.5 text-sm font-medium whitespace-nowrap transition-colors border-b-2 -mb-px',
225
+ 'hover:text-white',
226
+ 'focus:outline-none focus-visible:ring-2 focus-visible:ring-primary/50',
227
+ activeIndex === index
228
+ ? 'text-white border-emerald-500'
229
+ : 'text-zinc-400 border-transparent hover:border-zinc-600'
230
+ )}
231
+ >
232
+ {block.title}
233
+ </button>
234
+ ))}
235
+ </div>
236
+
237
+ {/* Copy button */}
238
+ <button
239
+ type="button"
240
+ onClick={handleCopy}
241
+ className={cn(
242
+ 'mr-3 p-1.5 rounded text-zinc-500 hover:text-zinc-300',
243
+ 'hover:bg-zinc-700/50 transition-colors',
244
+ 'focus:outline-none focus-visible:ring-2 focus-visible:ring-primary/50'
245
+ )}
246
+ title="Copy code"
247
+ >
248
+ {copied ? (
249
+ <Check className="h-4 w-4 text-emerald-500" weight="bold" />
250
+ ) : (
251
+ <Copy className="h-4 w-4" />
252
+ )}
253
+ </button>
254
+ </div>
255
+
256
+ {/* Code content */}
257
+ <div className={cn(
258
+ 'docs-code-group-content',
259
+ '[&>pre]:my-0 [&>pre]:rounded-none [&>pre]:border-0 [&>pre]:bg-[#1a1a1a]',
260
+ '[&_pre]:my-0 [&_pre]:rounded-none [&_pre]:border-0 [&_pre]:bg-[#1a1a1a]',
261
+ '[&_code]:bg-transparent [&_pre]:p-4',
262
+ '[&_pre]:text-sm [&_code]:text-sm'
263
+ )}>
264
+ {activeBlock?.element}
265
+ </div>
266
+ </div>
267
+ )
268
+ }
269
+
270
+ // Pills variant - pill-shaped buttons with dark theme
271
+ if (variant === 'pills') {
272
+ return (
273
+ <div className={cn(
274
+ 'docs-code-group docs-code-group-pills my-4 rounded-xl border border-zinc-800 overflow-hidden bg-[#1a1a1a]',
275
+ className
276
+ )}>
277
+ {/* Pill headers */}
278
+ <div className="docs-code-group-header flex items-center justify-between p-2 bg-[#262626] border-b border-zinc-800">
279
+ <div className="flex gap-1 overflow-x-auto">
280
+ {blocks.map((block, index) => (
281
+ <button
282
+ key={index}
283
+ type="button"
284
+ onClick={() => setActiveIndex(index)}
285
+ className={cn(
286
+ 'docs-code-group-tab px-3 py-1.5 text-xs font-medium whitespace-nowrap rounded-full transition-colors',
287
+ 'focus:outline-none focus-visible:ring-2 focus-visible:ring-primary/50',
288
+ activeIndex === index
289
+ ? 'bg-emerald-500 text-white'
290
+ : 'text-zinc-400 hover:text-white hover:bg-zinc-700'
291
+ )}
292
+ >
293
+ {block.title}
294
+ </button>
295
+ ))}
296
+ </div>
297
+
298
+ {/* Copy button */}
299
+ <button
300
+ type="button"
301
+ onClick={handleCopy}
302
+ className={cn(
303
+ 'p-1.5 rounded text-zinc-500 hover:text-zinc-300',
304
+ 'hover:bg-zinc-700/50 transition-colors',
305
+ 'focus:outline-none focus-visible:ring-2 focus-visible:ring-primary/50'
306
+ )}
307
+ title="Copy code"
308
+ >
309
+ {copied ? (
310
+ <Check className="h-4 w-4 text-emerald-500" weight="bold" />
311
+ ) : (
312
+ <Copy className="h-4 w-4" />
313
+ )}
314
+ </button>
315
+ </div>
316
+
317
+ {/* Code content */}
318
+ <div className={cn(
319
+ 'docs-code-group-content',
320
+ '[&>pre]:my-0 [&>pre]:rounded-none [&>pre]:border-0 [&>pre]:bg-[#1a1a1a]',
321
+ '[&_pre]:my-0 [&_pre]:rounded-none [&_pre]:border-0 [&_pre]:bg-[#1a1a1a]',
322
+ '[&_code]:bg-transparent [&_pre]:p-4',
323
+ '[&_pre]:text-sm [&_code]:text-sm'
324
+ )}>
325
+ {activeBlock?.element}
326
+ </div>
327
+ </div>
328
+ )
329
+ }
330
+
331
+ // Default variant - dark theme with underline indicator
332
+ return (
333
+ <div className={cn(
334
+ 'docs-code-group docs-code-group-default my-4 rounded-xl border border-zinc-800 overflow-hidden bg-[#1a1a1a]',
335
+ className
336
+ )}>
337
+ {/* Tab headers */}
338
+ <div className="docs-code-group-header flex items-center justify-between bg-[#262626] px-2">
339
+ <div className="flex overflow-x-auto">
340
+ {blocks.map((block, index) => (
341
+ <button
342
+ key={index}
343
+ type="button"
344
+ onClick={() => setActiveIndex(index)}
345
+ className={cn(
346
+ 'docs-code-group-tab px-4 py-2.5 text-sm font-medium whitespace-nowrap transition-colors relative',
347
+ 'hover:text-white',
348
+ 'focus:outline-none focus-visible:ring-2 focus-visible:ring-primary/50',
349
+ activeIndex === index
350
+ ? 'text-white'
351
+ : 'text-zinc-400'
352
+ )}
353
+ >
354
+ {block.title}
355
+ {/* Active indicator line */}
356
+ {activeIndex === index && (
357
+ <span className="absolute bottom-0 left-0 right-0 h-0.5 bg-emerald-500 rounded-full" />
358
+ )}
359
+ </button>
360
+ ))}
361
+ </div>
362
+
363
+ {/* Copy button */}
364
+ <div className="docs-code-group-actions flex items-center gap-1 pr-2">
365
+ <button
366
+ type="button"
367
+ onClick={handleCopy}
368
+ className={cn(
369
+ 'p-1.5 rounded text-zinc-500 hover:text-zinc-300',
370
+ 'hover:bg-zinc-700/50 transition-colors',
371
+ 'focus:outline-none focus-visible:ring-2 focus-visible:ring-primary/50'
372
+ )}
373
+ title="Copy code"
374
+ >
375
+ {copied ? (
376
+ <Check className="h-4 w-4 text-emerald-500" weight="bold" />
377
+ ) : (
378
+ <Copy className="h-4 w-4" />
379
+ )}
380
+ </button>
381
+ </div>
382
+ </div>
383
+
384
+ {/* Code content */}
385
+ <div className={cn(
386
+ 'docs-code-group-content',
387
+ '[&>pre]:my-0 [&>pre]:rounded-none [&>pre]:border-0 [&>pre]:bg-[#1a1a1a]',
388
+ '[&_pre]:my-0 [&_pre]:rounded-none [&_pre]:border-0 [&_pre]:bg-[#1a1a1a]',
389
+ '[&_code]:bg-transparent [&_pre]:p-4',
390
+ '[&_pre]:text-sm [&_code]:text-sm'
391
+ )}>
392
+ {activeBlock?.element}
393
+ </div>
394
+ </div>
395
+ )
396
+ }
397
+
398
+ /**
399
+ * CodeBlock component for individual code blocks within CodeGroup
400
+ * or standalone use with filename headers
401
+ */
402
+ interface CodeBlockProps {
403
+ children: React.ReactNode
404
+ title?: string
405
+ language?: string
406
+ className?: string
407
+ }
408
+
409
+ export function CodeBlockItem({ children, title, language, className }: CodeBlockProps) {
410
+ return (
411
+ <pre
412
+ className={cn('p-4 overflow-x-auto', className)}
413
+ data-title={title}
414
+ data-language={language}
415
+ >
416
+ <code className={language ? `language-${language}` : ''}>
417
+ {children}
418
+ </code>
419
+ </pre>
420
+ )
421
+ }
@@ -0,0 +1,105 @@
1
+ 'use client'
2
+
3
+ import { cn } from '@/lib/utils'
4
+ import { DownloadSimple, FileAudio, FilePdf } from '@phosphor-icons/react'
5
+
6
+ /**
7
+ * PDF Component - Embed PDF files with preview
8
+ */
9
+ interface PDFProps {
10
+ src: string
11
+ title?: string
12
+ height?: number
13
+ className?: string
14
+ }
15
+
16
+ export function PDF({ src, title, height = 500, className }: PDFProps) {
17
+ return (
18
+ <div className={cn('my-4', className)}>
19
+ {title && (
20
+ <div className="flex items-center gap-2 mb-2">
21
+ <FilePdf className="h-5 w-5 text-red-500" weight="fill" />
22
+ <span className="text-sm font-medium">{title}</span>
23
+ </div>
24
+ )}
25
+ <div
26
+ className="rounded-lg border border-border overflow-hidden bg-muted"
27
+ style={{ height }}
28
+ >
29
+ <iframe
30
+ src={`${src}#view=FitH`}
31
+ className="w-full h-full"
32
+ title={title || 'PDF Document'}
33
+ />
34
+ </div>
35
+ </div>
36
+ )
37
+ }
38
+
39
+ /**
40
+ * Audio Component - Embed audio files with player
41
+ */
42
+ interface AudioProps {
43
+ src: string
44
+ title?: string
45
+ autoplay?: boolean
46
+ loop?: boolean
47
+ className?: string
48
+ }
49
+
50
+ export function Audio({ src, title, autoplay = false, loop = false, className }: AudioProps) {
51
+ return (
52
+ <div className={cn('my-4', className)}>
53
+ {title && (
54
+ <div className="flex items-center gap-2 mb-2">
55
+ <FileAudio className="h-5 w-5 text-primary" weight="fill" />
56
+ <span className="text-sm font-medium">{title}</span>
57
+ </div>
58
+ )}
59
+ <div className="rounded-lg border border-border bg-muted p-4">
60
+ <audio
61
+ src={src}
62
+ controls
63
+ autoPlay={autoplay}
64
+ loop={loop}
65
+ className="w-full"
66
+ >
67
+ Your browser does not support the audio element.
68
+ </audio>
69
+ </div>
70
+ </div>
71
+ )
72
+ }
73
+
74
+ /**
75
+ * Download Component - Download button for files
76
+ */
77
+ interface DownloadProps {
78
+ src: string
79
+ label?: string
80
+ icon?: string
81
+ filename?: string
82
+ className?: string
83
+ }
84
+
85
+ export function Download({ src, label, filename, className }: DownloadProps) {
86
+ const displayLabel = label || 'Download File'
87
+ const downloadName = filename || src.split('/').pop() || 'download'
88
+
89
+ return (
90
+ <a
91
+ href={src}
92
+ download={downloadName}
93
+ className={cn(
94
+ 'inline-flex items-center gap-2 px-4 py-2 rounded-lg',
95
+ 'bg-primary text-primary-foreground hover:bg-primary/90',
96
+ 'text-sm font-medium transition-colors',
97
+ 'no-underline',
98
+ className
99
+ )}
100
+ >
101
+ <DownloadSimple className="h-4 w-4" weight="bold" />
102
+ {displayLabel}
103
+ </a>
104
+ )
105
+ }
@@ -0,0 +1,112 @@
1
+ 'use client'
2
+
3
+ import React from 'react'
4
+ import { cn } from '@/lib/utils'
5
+
6
+ /**
7
+ * Frame Component for MDX Documentation
8
+ *
9
+ * Wraps images and content with a styled border and optional caption.
10
+ * Similar to Mintlify's Frame component.
11
+ */
12
+
13
+ interface FrameProps {
14
+ children: React.ReactNode
15
+ caption?: string
16
+ className?: string
17
+ }
18
+
19
+ export function Frame({ children, caption, className }: FrameProps) {
20
+ return (
21
+ <figure className={cn('my-6', className)}>
22
+ <div className="overflow-hidden rounded-lg border border-border bg-muted/30">
23
+ <div className="[&>img]:m-0 [&>img]:w-full [&>img]:h-auto">
24
+ {children}
25
+ </div>
26
+ </div>
27
+ {caption && (
28
+ <figcaption className="mt-2 text-center text-sm text-muted-foreground">
29
+ {caption}
30
+ </figcaption>
31
+ )}
32
+ </figure>
33
+ )
34
+ }
35
+
36
+ /**
37
+ * Columns Component for MDX Documentation
38
+ *
39
+ * Creates a multi-column layout for content.
40
+ */
41
+
42
+ interface ColumnsProps {
43
+ children: React.ReactNode
44
+ cols?: 1 | 2 | 3 | 4
45
+ className?: string
46
+ }
47
+
48
+ export function Columns({ children, cols = 2, className }: ColumnsProps) {
49
+ const gridCols = {
50
+ 1: 'grid-cols-1',
51
+ 2: 'grid-cols-1 md:grid-cols-2',
52
+ 3: 'grid-cols-1 md:grid-cols-2 lg:grid-cols-3',
53
+ 4: 'grid-cols-1 md:grid-cols-2 lg:grid-cols-4',
54
+ }
55
+
56
+ return (
57
+ <div className={cn('grid gap-4 my-4', gridCols[cols], className)}>
58
+ {children}
59
+ </div>
60
+ )
61
+ }
62
+
63
+ /**
64
+ * Snippet Component for MDX Documentation
65
+ *
66
+ * Placeholder for reusable content snippets.
67
+ * In a full implementation, this would load content from a snippets directory.
68
+ */
69
+
70
+ interface SnippetProps {
71
+ file: string
72
+ className?: string
73
+ }
74
+
75
+ export function Snippet({ file, className }: SnippetProps) {
76
+ // In production, this would dynamically load the snippet content
77
+ // For now, we show a placeholder
78
+ return (
79
+ <div className={cn('my-4 p-4 rounded-lg border border-dashed border-border bg-muted/30', className)}>
80
+ <div className="text-sm text-muted-foreground">
81
+ Snippet: <code className="text-xs bg-muted px-1 py-0.5 rounded">{file}</code>
82
+ </div>
83
+ </div>
84
+ )
85
+ }
86
+
87
+ /**
88
+ * Latex Component for MDX Documentation
89
+ *
90
+ * Renders LaTeX mathematical expressions.
91
+ * In a full implementation, this would use KaTeX or MathJax.
92
+ */
93
+
94
+ interface LatexProps {
95
+ children: string
96
+ className?: string
97
+ }
98
+
99
+ export function Latex({ children, className }: LatexProps) {
100
+ // In production, this would render LaTeX using KaTeX
101
+ // For now, we show the raw expression in a styled container
102
+ return (
103
+ <span
104
+ className={cn(
105
+ 'inline-block px-2 py-1 font-mono text-sm bg-muted rounded',
106
+ className
107
+ )}
108
+ >
109
+ {children}
110
+ </span>
111
+ )
112
+ }