@brainfish-ai/devdoc 0.1.21
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +33 -0
- package/README.md +415 -0
- package/bin/devdoc.js +13 -0
- package/dist/cli/commands/build.d.ts +5 -0
- package/dist/cli/commands/build.js +87 -0
- package/dist/cli/commands/check.d.ts +1 -0
- package/dist/cli/commands/check.js +143 -0
- package/dist/cli/commands/create.d.ts +24 -0
- package/dist/cli/commands/create.js +387 -0
- package/dist/cli/commands/deploy.d.ts +9 -0
- package/dist/cli/commands/deploy.js +433 -0
- package/dist/cli/commands/dev.d.ts +6 -0
- package/dist/cli/commands/dev.js +139 -0
- package/dist/cli/commands/init.d.ts +11 -0
- package/dist/cli/commands/init.js +238 -0
- package/dist/cli/commands/keys.d.ts +12 -0
- package/dist/cli/commands/keys.js +165 -0
- package/dist/cli/commands/start.d.ts +5 -0
- package/dist/cli/commands/start.js +56 -0
- package/dist/cli/commands/upload.d.ts +13 -0
- package/dist/cli/commands/upload.js +238 -0
- package/dist/cli/commands/whoami.d.ts +8 -0
- package/dist/cli/commands/whoami.js +91 -0
- package/dist/cli/index.d.ts +1 -0
- package/dist/cli/index.js +106 -0
- package/dist/config/index.d.ts +80 -0
- package/dist/config/index.js +133 -0
- package/dist/constants.d.ts +9 -0
- package/dist/constants.js +13 -0
- package/dist/index.d.ts +7 -0
- package/dist/index.js +12 -0
- package/dist/utils/logger.d.ts +16 -0
- package/dist/utils/logger.js +61 -0
- package/dist/utils/paths.d.ts +16 -0
- package/dist/utils/paths.js +50 -0
- package/package.json +51 -0
- package/renderer/app/api/assets/[...path]/route.ts +123 -0
- package/renderer/app/api/assets/route.ts +124 -0
- package/renderer/app/api/assets/upload/route.ts +177 -0
- package/renderer/app/api/auth-schemes/route.ts +77 -0
- package/renderer/app/api/chat/route.ts +858 -0
- package/renderer/app/api/codegen/route.ts +72 -0
- package/renderer/app/api/collections/route.ts +1016 -0
- package/renderer/app/api/debug/route.ts +53 -0
- package/renderer/app/api/deploy/route.ts +234 -0
- package/renderer/app/api/device/route.ts +42 -0
- package/renderer/app/api/docs/route.ts +187 -0
- package/renderer/app/api/keys/regenerate/route.ts +80 -0
- package/renderer/app/api/openapi-spec/route.ts +151 -0
- package/renderer/app/api/projects/[slug]/route.ts +153 -0
- package/renderer/app/api/projects/[slug]/stats/route.ts +96 -0
- package/renderer/app/api/projects/register/route.ts +152 -0
- package/renderer/app/api/proxy/route.ts +149 -0
- package/renderer/app/api/proxy-stream/route.ts +168 -0
- package/renderer/app/api/redirects/route.ts +47 -0
- package/renderer/app/api/schema/route.ts +65 -0
- package/renderer/app/api/subdomains/check/route.ts +172 -0
- package/renderer/app/api/suggestions/route.ts +144 -0
- package/renderer/app/favicon.ico +0 -0
- package/renderer/app/globals.css +1103 -0
- package/renderer/app/layout.tsx +47 -0
- package/renderer/app/llms-full.txt/route.ts +346 -0
- package/renderer/app/llms.txt/route.ts +279 -0
- package/renderer/app/page.tsx +14 -0
- package/renderer/app/robots.txt/route.ts +84 -0
- package/renderer/app/sitemap.xml/route.ts +199 -0
- package/renderer/components/docs/index.ts +12 -0
- package/renderer/components/docs/mdx/accordion.tsx +169 -0
- package/renderer/components/docs/mdx/badge.tsx +132 -0
- package/renderer/components/docs/mdx/callouts.tsx +154 -0
- package/renderer/components/docs/mdx/cards.tsx +213 -0
- package/renderer/components/docs/mdx/changelog.tsx +120 -0
- package/renderer/components/docs/mdx/code-block.tsx +186 -0
- package/renderer/components/docs/mdx/code-group.tsx +421 -0
- package/renderer/components/docs/mdx/file-embeds.tsx +105 -0
- package/renderer/components/docs/mdx/frame.tsx +112 -0
- package/renderer/components/docs/mdx/highlight.tsx +151 -0
- package/renderer/components/docs/mdx/iframe.tsx +134 -0
- package/renderer/components/docs/mdx/image.tsx +235 -0
- package/renderer/components/docs/mdx/index.ts +204 -0
- package/renderer/components/docs/mdx/mermaid.tsx +240 -0
- package/renderer/components/docs/mdx/param-field.tsx +200 -0
- package/renderer/components/docs/mdx/steps.tsx +113 -0
- package/renderer/components/docs/mdx/tabs.tsx +86 -0
- package/renderer/components/docs/mdx-renderer.tsx +100 -0
- package/renderer/components/docs/navigation/breadcrumbs.tsx +76 -0
- package/renderer/components/docs/navigation/index.ts +8 -0
- package/renderer/components/docs/navigation/page-nav.tsx +64 -0
- package/renderer/components/docs/navigation/sidebar.tsx +515 -0
- package/renderer/components/docs/navigation/toc.tsx +113 -0
- package/renderer/components/docs/notice.tsx +105 -0
- package/renderer/components/docs-header.tsx +274 -0
- package/renderer/components/docs-viewer/agent/agent-chat.tsx +2076 -0
- package/renderer/components/docs-viewer/agent/cards/debug-context-card.tsx +90 -0
- package/renderer/components/docs-viewer/agent/cards/endpoint-context-card.tsx +49 -0
- package/renderer/components/docs-viewer/agent/cards/index.tsx +50 -0
- package/renderer/components/docs-viewer/agent/cards/response-options-card.tsx +212 -0
- package/renderer/components/docs-viewer/agent/cards/types.ts +84 -0
- package/renderer/components/docs-viewer/agent/chat-message.tsx +17 -0
- package/renderer/components/docs-viewer/agent/index.tsx +6 -0
- package/renderer/components/docs-viewer/agent/messages/assistant-message.tsx +119 -0
- package/renderer/components/docs-viewer/agent/messages/chat-message.tsx +46 -0
- package/renderer/components/docs-viewer/agent/messages/index.ts +17 -0
- package/renderer/components/docs-viewer/agent/messages/tool-call-display.tsx +721 -0
- package/renderer/components/docs-viewer/agent/messages/types.ts +61 -0
- package/renderer/components/docs-viewer/agent/messages/typing-indicator.tsx +24 -0
- package/renderer/components/docs-viewer/agent/messages/user-message.tsx +51 -0
- package/renderer/components/docs-viewer/code-editor/index.tsx +2 -0
- package/renderer/components/docs-viewer/code-editor/notes-mode.tsx +1283 -0
- package/renderer/components/docs-viewer/content/changelog-page.tsx +331 -0
- package/renderer/components/docs-viewer/content/doc-page.tsx +285 -0
- package/renderer/components/docs-viewer/content/documentation-viewer.tsx +17 -0
- package/renderer/components/docs-viewer/content/index.tsx +29 -0
- package/renderer/components/docs-viewer/content/introduction.tsx +21 -0
- package/renderer/components/docs-viewer/content/request-details.tsx +330 -0
- package/renderer/components/docs-viewer/content/sections/auth.tsx +69 -0
- package/renderer/components/docs-viewer/content/sections/body.tsx +66 -0
- package/renderer/components/docs-viewer/content/sections/headers.tsx +43 -0
- package/renderer/components/docs-viewer/content/sections/overview.tsx +40 -0
- package/renderer/components/docs-viewer/content/sections/parameters.tsx +43 -0
- package/renderer/components/docs-viewer/content/sections/responses.tsx +87 -0
- package/renderer/components/docs-viewer/global-auth-modal.tsx +352 -0
- package/renderer/components/docs-viewer/index.tsx +1466 -0
- package/renderer/components/docs-viewer/playground/auth-editor.tsx +280 -0
- package/renderer/components/docs-viewer/playground/body-editor.tsx +221 -0
- package/renderer/components/docs-viewer/playground/code-editor.tsx +224 -0
- package/renderer/components/docs-viewer/playground/code-snippet.tsx +387 -0
- package/renderer/components/docs-viewer/playground/graphql-playground.tsx +745 -0
- package/renderer/components/docs-viewer/playground/index.tsx +671 -0
- package/renderer/components/docs-viewer/playground/key-value-editor.tsx +261 -0
- package/renderer/components/docs-viewer/playground/method-selector.tsx +60 -0
- package/renderer/components/docs-viewer/playground/request-builder.tsx +179 -0
- package/renderer/components/docs-viewer/playground/request-tabs.tsx +237 -0
- package/renderer/components/docs-viewer/playground/response-cards/idle-card.tsx +21 -0
- package/renderer/components/docs-viewer/playground/response-cards/index.tsx +93 -0
- package/renderer/components/docs-viewer/playground/response-cards/loading-card.tsx +16 -0
- package/renderer/components/docs-viewer/playground/response-cards/network-error-card.tsx +23 -0
- package/renderer/components/docs-viewer/playground/response-cards/response-body-card.tsx +268 -0
- package/renderer/components/docs-viewer/playground/response-cards/types.ts +82 -0
- package/renderer/components/docs-viewer/playground/response-viewer.tsx +43 -0
- package/renderer/components/docs-viewer/search/index.ts +2 -0
- package/renderer/components/docs-viewer/search/search-dialog.tsx +331 -0
- package/renderer/components/docs-viewer/search/use-search.ts +117 -0
- package/renderer/components/docs-viewer/shared/markdown-renderer.tsx +431 -0
- package/renderer/components/docs-viewer/shared/method-badge.tsx +41 -0
- package/renderer/components/docs-viewer/shared/schema-viewer.tsx +349 -0
- package/renderer/components/docs-viewer/sidebar/collection-tree.tsx +239 -0
- package/renderer/components/docs-viewer/sidebar/endpoint-options.tsx +316 -0
- package/renderer/components/docs-viewer/sidebar/index.tsx +343 -0
- package/renderer/components/docs-viewer/sidebar/right-sidebar.tsx +202 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-group.tsx +118 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-item.tsx +226 -0
- package/renderer/components/docs-viewer/sidebar/sidebar-section.tsx +52 -0
- package/renderer/components/theme-provider.tsx +11 -0
- package/renderer/components/theme-toggle.tsx +76 -0
- package/renderer/components/ui/badge.tsx +46 -0
- package/renderer/components/ui/button.tsx +59 -0
- package/renderer/components/ui/dialog.tsx +118 -0
- package/renderer/components/ui/dropdown-menu.tsx +257 -0
- package/renderer/components/ui/input.tsx +21 -0
- package/renderer/components/ui/label.tsx +24 -0
- package/renderer/components/ui/navigation-menu.tsx +168 -0
- package/renderer/components/ui/select.tsx +190 -0
- package/renderer/components/ui/spinner.tsx +114 -0
- package/renderer/components/ui/tabs.tsx +66 -0
- package/renderer/components/ui/tooltip.tsx +61 -0
- package/renderer/hooks/use-code-copy.ts +88 -0
- package/renderer/hooks/use-openapi-title.ts +44 -0
- package/renderer/lib/api-docs/agent/index.ts +6 -0
- package/renderer/lib/api-docs/agent/indexer.ts +323 -0
- package/renderer/lib/api-docs/agent/spec-summary.ts +335 -0
- package/renderer/lib/api-docs/agent/types.ts +116 -0
- package/renderer/lib/api-docs/auth/auth-context.tsx +225 -0
- package/renderer/lib/api-docs/auth/auth-storage.ts +87 -0
- package/renderer/lib/api-docs/auth/crypto.ts +89 -0
- package/renderer/lib/api-docs/auth/index.ts +4 -0
- package/renderer/lib/api-docs/code-editor/db.ts +164 -0
- package/renderer/lib/api-docs/code-editor/hooks.ts +266 -0
- package/renderer/lib/api-docs/code-editor/index.ts +6 -0
- package/renderer/lib/api-docs/code-editor/mode-context.tsx +207 -0
- package/renderer/lib/api-docs/code-editor/types.ts +105 -0
- package/renderer/lib/api-docs/codegen/definitions.ts +297 -0
- package/renderer/lib/api-docs/codegen/har.ts +251 -0
- package/renderer/lib/api-docs/codegen/index.ts +159 -0
- package/renderer/lib/api-docs/factories.ts +151 -0
- package/renderer/lib/api-docs/index.ts +17 -0
- package/renderer/lib/api-docs/mobile-context.tsx +112 -0
- package/renderer/lib/api-docs/navigation-context.tsx +88 -0
- package/renderer/lib/api-docs/parsers/graphql/README.md +129 -0
- package/renderer/lib/api-docs/parsers/graphql/index.ts +91 -0
- package/renderer/lib/api-docs/parsers/graphql/parser.ts +491 -0
- package/renderer/lib/api-docs/parsers/graphql/transformer.ts +246 -0
- package/renderer/lib/api-docs/parsers/graphql/types.ts +283 -0
- package/renderer/lib/api-docs/parsers/openapi/README.md +32 -0
- package/renderer/lib/api-docs/parsers/openapi/dereferencer.ts +60 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/auth.ts +574 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/body.ts +403 -0
- package/renderer/lib/api-docs/parsers/openapi/extractors/index.ts +232 -0
- package/renderer/lib/api-docs/parsers/openapi/index.ts +171 -0
- package/renderer/lib/api-docs/parsers/openapi/transformer.ts +277 -0
- package/renderer/lib/api-docs/parsers/openapi/validator.ts +31 -0
- package/renderer/lib/api-docs/playground/context.tsx +107 -0
- package/renderer/lib/api-docs/playground/navigation-context.tsx +124 -0
- package/renderer/lib/api-docs/playground/request-builder.ts +223 -0
- package/renderer/lib/api-docs/playground/request-runner.ts +282 -0
- package/renderer/lib/api-docs/playground/types.ts +35 -0
- package/renderer/lib/api-docs/types.ts +269 -0
- package/renderer/lib/api-docs/utils.ts +311 -0
- package/renderer/lib/cache.ts +193 -0
- package/renderer/lib/docs/config/index.ts +29 -0
- package/renderer/lib/docs/config/loader.ts +142 -0
- package/renderer/lib/docs/config/schema.ts +298 -0
- package/renderer/lib/docs/index.ts +12 -0
- package/renderer/lib/docs/mdx/compiler.ts +176 -0
- package/renderer/lib/docs/mdx/frontmatter.ts +80 -0
- package/renderer/lib/docs/mdx/index.ts +26 -0
- package/renderer/lib/docs/navigation/generator.ts +348 -0
- package/renderer/lib/docs/navigation/index.ts +12 -0
- package/renderer/lib/docs/navigation/types.ts +123 -0
- package/renderer/lib/docs-navigation-context.tsx +80 -0
- package/renderer/lib/multi-tenant/context.ts +105 -0
- package/renderer/lib/storage/blob.ts +845 -0
- package/renderer/lib/utils.ts +6 -0
- package/renderer/next.config.ts +76 -0
- package/renderer/package.json +66 -0
- package/renderer/postcss.config.mjs +5 -0
- package/renderer/public/assets/images/screenshot.png +0 -0
- package/renderer/public/assets/logo/dark.svg +9 -0
- package/renderer/public/assets/logo/light.svg +9 -0
- package/renderer/public/assets/logo.svg +9 -0
- package/renderer/public/file.svg +1 -0
- package/renderer/public/globe.svg +1 -0
- package/renderer/public/icon.png +0 -0
- package/renderer/public/logo.svg +9 -0
- package/renderer/public/window.svg +1 -0
- package/renderer/tsconfig.json +28 -0
- package/templates/basic/README.md +139 -0
- package/templates/basic/assets/favicon.svg +4 -0
- package/templates/basic/assets/logo.svg +9 -0
- package/templates/basic/docs.json +47 -0
- package/templates/basic/guides/configuration.mdx +149 -0
- package/templates/basic/guides/overview.mdx +96 -0
- package/templates/basic/index.mdx +39 -0
- package/templates/basic/package.json +14 -0
- package/templates/basic/quickstart.mdx +92 -0
- package/templates/basic/vercel.json +6 -0
- package/templates/graphql/README.md +139 -0
- package/templates/graphql/api-reference/schema.graphql +305 -0
- package/templates/graphql/assets/favicon.svg +4 -0
- package/templates/graphql/assets/logo.svg +9 -0
- package/templates/graphql/docs.json +54 -0
- package/templates/graphql/guides/configuration.mdx +149 -0
- package/templates/graphql/guides/overview.mdx +96 -0
- package/templates/graphql/index.mdx +39 -0
- package/templates/graphql/package.json +14 -0
- package/templates/graphql/quickstart.mdx +92 -0
- package/templates/graphql/vercel.json +6 -0
- package/templates/openapi/README.md +139 -0
- package/templates/openapi/api-reference/openapi.json +419 -0
- package/templates/openapi/assets/favicon.svg +4 -0
- package/templates/openapi/assets/logo.svg +9 -0
- package/templates/openapi/docs.json +61 -0
- package/templates/openapi/guides/configuration.mdx +149 -0
- package/templates/openapi/guides/overview.mdx +96 -0
- package/templates/openapi/index.mdx +39 -0
- package/templates/openapi/package.json +14 -0
- package/templates/openapi/quickstart.mdx +92 -0
- package/templates/openapi/vercel.json +6 -0
|
@@ -0,0 +1,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
|
+
}
|