@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,261 @@
1
+ 'use client'
2
+
3
+ import { useCallback, useMemo } from 'react'
4
+ import { CheckCircle, Circle, Trash, Plus, WarningCircle } from '@phosphor-icons/react'
5
+ import { Input } from '@/components/ui/input'
6
+ import { Button } from '@/components/ui/button'
7
+ import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from '@/components/ui/tooltip'
8
+ import { cn } from '@/lib/utils'
9
+
10
+ export interface KeyValueItem {
11
+ id: string
12
+ key: string
13
+ value: string
14
+ active: boolean
15
+ description?: string
16
+ }
17
+
18
+ interface KeyValueEditorProps {
19
+ items: KeyValueItem[]
20
+ onChange: (items: KeyValueItem[]) => void
21
+ title?: string
22
+ keyLabel?: string
23
+ valueLabel?: string
24
+ showDescription?: boolean
25
+ readOnly?: boolean
26
+ /** Show error styling to indicate this section needs attention */
27
+ showError?: boolean
28
+ }
29
+
30
+ let idCounter = 0
31
+ const generateId = () => `kv_${Date.now()}_${idCounter++}`
32
+
33
+ export function KeyValueEditor({
34
+ items,
35
+ onChange,
36
+ title = 'Parameters',
37
+ keyLabel = 'Name',
38
+ valueLabel = 'Value',
39
+ showDescription = false,
40
+ readOnly = false,
41
+ showError = false,
42
+ }: KeyValueEditorProps) {
43
+ // Ensure there's always an empty row at the end for new entries
44
+ const workingItems = useMemo(() => {
45
+ if (items.length === 0 || items[items.length - 1].key !== '') {
46
+ return [...items, { id: generateId(), key: '', value: '', active: true, description: '' }]
47
+ }
48
+ return items
49
+ }, [items])
50
+
51
+ const updateItem = useCallback((index: number, updates: Partial<KeyValueItem>) => {
52
+ const newItems = workingItems.map((item, i) =>
53
+ i === index ? { ...item, ...updates } : item
54
+ )
55
+ const filtered = newItems.filter((item, i) =>
56
+ i === newItems.length - 1 || item.key !== '' || item.value !== ''
57
+ )
58
+ onChange(filtered.filter(item => item.key !== '' || item.value !== ''))
59
+ }, [workingItems, onChange])
60
+
61
+ const deleteItem = useCallback((index: number) => {
62
+ const newItems = workingItems.filter((_, i) => i !== index)
63
+ onChange(newItems.filter(item => item.key !== ''))
64
+ }, [workingItems, onChange])
65
+
66
+ const addItem = useCallback(() => {
67
+ onChange([...items, { id: generateId(), key: '', value: '', active: true, description: '' }])
68
+ }, [items, onChange])
69
+
70
+ const clearAll = useCallback(() => {
71
+ onChange([])
72
+ }, [onChange])
73
+
74
+ const toggleActive = useCallback((index: number) => {
75
+ const item = workingItems[index]
76
+ if (item) {
77
+ updateItem(index, { active: !item.active })
78
+ }
79
+ }, [workingItems, updateItem])
80
+
81
+ const activeCount = useMemo(() => {
82
+ return items.filter(item => item.active && item.key).length
83
+ }, [items])
84
+
85
+ const hasItems = items.some(item => item.key !== '')
86
+
87
+ return (
88
+ <TooltipProvider>
89
+ <div className="flex flex-1 flex-col h-full">
90
+ {/* Header */}
91
+ <div className={cn(
92
+ "sticky top-0 z-10 flex items-center justify-between border-b bg-muted/30 px-4 py-2",
93
+ showError ? "border-red-500 bg-red-500/5" : "border-border"
94
+ )}>
95
+ <div className="flex items-center gap-2">
96
+ {showError && (
97
+ <WarningCircle className="h-4 w-4 text-red-500 animate-pulse" weight="fill" />
98
+ )}
99
+ <label className={cn(
100
+ "truncate font-semibold text-sm",
101
+ showError ? "text-red-500" : "text-foreground"
102
+ )}>
103
+ {title}
104
+ </label>
105
+ {activeCount > 0 && (
106
+ <span className="text-xs text-muted-foreground bg-muted px-2 py-0.5 rounded-full">
107
+ {activeCount}
108
+ </span>
109
+ )}
110
+ </div>
111
+ <div className="flex items-center gap-1">
112
+ {hasItems && (
113
+ <Tooltip>
114
+ <TooltipTrigger asChild>
115
+ <Button
116
+ variant="ghost"
117
+ size="icon"
118
+ className="h-7 w-7 text-muted-foreground hover:text-destructive"
119
+ onClick={clearAll}
120
+ disabled={readOnly}
121
+ >
122
+ <Trash className="h-3.5 w-3.5" />
123
+ </Button>
124
+ </TooltipTrigger>
125
+ <TooltipContent>Clear all</TooltipContent>
126
+ </Tooltip>
127
+ )}
128
+ <Tooltip>
129
+ <TooltipTrigger asChild>
130
+ <Button
131
+ variant="ghost"
132
+ size="icon"
133
+ className="h-7 w-7 text-muted-foreground hover:text-foreground"
134
+ onClick={addItem}
135
+ disabled={readOnly}
136
+ >
137
+ <Plus className="h-3.5 w-3.5" />
138
+ </Button>
139
+ </TooltipTrigger>
140
+ <TooltipContent>Add new</TooltipContent>
141
+ </Tooltip>
142
+ </div>
143
+ </div>
144
+
145
+ {/* Items */}
146
+ <div className="flex-1 overflow-auto">
147
+ {workingItems.length > 0 && (
148
+ <table className="w-full text-sm">
149
+ <thead className="text-xs text-muted-foreground sticky top-0 z-10">
150
+ <tr className="bg-background border-b border-border/40 shadow-sm">
151
+ <th className="text-left font-medium px-3 py-2.5 w-[45%] bg-background">{keyLabel}</th>
152
+ <th className="text-left font-medium px-3 py-2.5 w-[45%] bg-background">{valueLabel}</th>
153
+ {showDescription && (
154
+ <th className="text-left font-medium px-3 py-2.5 bg-background">Description</th>
155
+ )}
156
+ <th className="w-16 bg-background"></th>
157
+ </tr>
158
+ </thead>
159
+ <tbody>
160
+ {workingItems.map((item, index) => {
161
+ const isLastEmpty = index === workingItems.length - 1 && !item.key
162
+ return (
163
+ <tr
164
+ key={item.id}
165
+ className={`border-b border-border/50 hover:bg-muted/20 group ${!item.active ? 'opacity-50' : ''}`}
166
+ >
167
+ <td className="px-2 py-1.5">
168
+ <Input
169
+ type="text"
170
+ value={item.key}
171
+ onChange={(e) => updateItem(index, { key: e.target.value })}
172
+ placeholder={isLastEmpty ? `Add ${keyLabel.toLowerCase()}...` : ''}
173
+ className="h-9 bg-background border border-border/60 focus-visible:ring-1 focus-visible:ring-primary/30 focus-visible:border-primary/50 placeholder:text-muted-foreground/50"
174
+ disabled={readOnly}
175
+ />
176
+ </td>
177
+ <td className="px-2 py-1.5">
178
+ <Input
179
+ type="text"
180
+ value={item.value}
181
+ onChange={(e) => updateItem(index, { value: e.target.value })}
182
+ placeholder={isLastEmpty ? `Add ${valueLabel.toLowerCase()}...` : ''}
183
+ className="h-9 bg-background border border-border/60 focus-visible:ring-1 focus-visible:ring-primary/30 focus-visible:border-primary/50 placeholder:text-muted-foreground/50"
184
+ disabled={readOnly}
185
+ />
186
+ </td>
187
+ {showDescription && (
188
+ <td className="px-2 py-1.5">
189
+ <Input
190
+ type="text"
191
+ value={item.description || ''}
192
+ onChange={(e) => updateItem(index, { description: e.target.value })}
193
+ placeholder=""
194
+ className="h-9 bg-background border border-border/60 focus-visible:ring-1 focus-visible:ring-primary/30 focus-visible:border-primary/50 text-muted-foreground"
195
+ disabled={readOnly}
196
+ />
197
+ </td>
198
+ )}
199
+ <td className="px-2 py-1.5">
200
+ <div className="flex items-center justify-end gap-0.5">
201
+ {!isLastEmpty && (
202
+ <>
203
+ <Tooltip>
204
+ <TooltipTrigger asChild>
205
+ <Button
206
+ variant="ghost"
207
+ size="icon"
208
+ className="h-7 w-7"
209
+ onClick={() => toggleActive(index)}
210
+ disabled={readOnly}
211
+ >
212
+ {item.active ? (
213
+ <CheckCircle className="h-4 w-4 text-green-500" weight="fill" />
214
+ ) : (
215
+ <Circle className="h-4 w-4 text-muted-foreground" />
216
+ )}
217
+ </Button>
218
+ </TooltipTrigger>
219
+ <TooltipContent>{item.active ? 'Disable' : 'Enable'}</TooltipContent>
220
+ </Tooltip>
221
+ <Tooltip>
222
+ <TooltipTrigger asChild>
223
+ <Button
224
+ variant="ghost"
225
+ size="icon"
226
+ className="h-7 w-7 text-muted-foreground hover:text-destructive"
227
+ onClick={() => deleteItem(index)}
228
+ disabled={readOnly}
229
+ >
230
+ <Trash className="h-3.5 w-3.5" />
231
+ </Button>
232
+ </TooltipTrigger>
233
+ <TooltipContent>Remove</TooltipContent>
234
+ </Tooltip>
235
+ </>
236
+ )}
237
+ </div>
238
+ </td>
239
+ </tr>
240
+ )
241
+ })}
242
+ </tbody>
243
+ </table>
244
+ )}
245
+
246
+ {/* Empty state */}
247
+ {!hasItems && workingItems.length <= 1 && (
248
+ <div className="flex flex-col items-center justify-center py-8 text-center">
249
+ <p className="text-xs text-muted-foreground">
250
+ No {title.toLowerCase()} added yet
251
+ </p>
252
+ <p className="text-xs text-muted-foreground mt-1">
253
+ Click <Plus className="h-3 w-3 inline" /> to add one
254
+ </p>
255
+ </div>
256
+ )}
257
+ </div>
258
+ </div>
259
+ </TooltipProvider>
260
+ )
261
+ }
@@ -0,0 +1,60 @@
1
+ 'use client'
2
+
3
+ import { CaretDown, Check } from '@phosphor-icons/react'
4
+ import {
5
+ DropdownMenu,
6
+ DropdownMenuContent,
7
+ DropdownMenuItem,
8
+ DropdownMenuTrigger,
9
+ } from '@/components/ui/dropdown-menu'
10
+ import { Button } from '@/components/ui/button'
11
+ import type { HTTPMethod } from '@/lib/api-docs/types'
12
+
13
+ const METHODS: HTTPMethod[] = ['GET', 'POST', 'PUT', 'PATCH', 'DELETE', 'HEAD', 'OPTIONS']
14
+
15
+ const METHOD_COLORS: Record<HTTPMethod, string> = {
16
+ GET: 'text-green-500',
17
+ POST: 'text-blue-500',
18
+ PUT: 'text-orange-500',
19
+ PATCH: 'text-purple-500',
20
+ DELETE: 'text-red-500',
21
+ HEAD: 'text-gray-500',
22
+ OPTIONS: 'text-gray-500',
23
+ }
24
+
25
+ interface MethodSelectorProps {
26
+ value: HTTPMethod
27
+ onChange: (method: HTTPMethod) => void
28
+ disabled?: boolean
29
+ }
30
+
31
+ export function MethodSelector({ value, onChange, disabled = false }: MethodSelectorProps) {
32
+ return (
33
+ <DropdownMenu>
34
+ <DropdownMenuTrigger asChild>
35
+ <Button
36
+ variant="ghost"
37
+ className={`flex items-center gap-1 px-4 py-2 h-auto bg-muted/50 hover:bg-muted rounded-l-md rounded-r-none border-r border-border font-semibold text-sm ${
38
+ METHOD_COLORS[value]
39
+ }`}
40
+ disabled={disabled}
41
+ >
42
+ <span className="w-16 text-left">{value}</span>
43
+ <CaretDown className="h-4 w-4" weight="bold" />
44
+ </Button>
45
+ </DropdownMenuTrigger>
46
+ <DropdownMenuContent align="start" className="min-w-[120px]">
47
+ {METHODS.map((method) => (
48
+ <DropdownMenuItem
49
+ key={method}
50
+ onClick={() => onChange(method)}
51
+ className={`font-semibold ${METHOD_COLORS[method]}`}
52
+ >
53
+ <span className="flex-1">{method}</span>
54
+ {value === method && <Check className="h-4 w-4 ml-2" weight="bold" />}
55
+ </DropdownMenuItem>
56
+ ))}
57
+ </DropdownMenuContent>
58
+ </DropdownMenu>
59
+ )
60
+ }
@@ -0,0 +1,179 @@
1
+ 'use client'
2
+
3
+ import type { BrainfishRESTRequest } from '@/lib/api-docs/types'
4
+
5
+ interface RequestBuilderProps {
6
+ request: BrainfishRESTRequest
7
+ params: Array<{ key: string; value: string; active: boolean }>
8
+ headers: Array<{ key: string; value: string; active: boolean }>
9
+ body: string | null
10
+ variables: Array<{ key: string; value: string; active: boolean }>
11
+ onParamsChange: (params: Array<{ key: string; value: string; active: boolean }>) => void
12
+ onHeadersChange: (headers: Array<{ key: string; value: string; active: boolean }>) => void
13
+ onBodyChange: (body: string | null) => void
14
+ onVariablesChange: (variables: Array<{ key: string; value: string; active: boolean }>) => void
15
+ }
16
+
17
+ export function RequestBuilder({
18
+ request,
19
+ params,
20
+ headers,
21
+ body,
22
+ variables,
23
+ onParamsChange,
24
+ onHeadersChange,
25
+ onBodyChange,
26
+ onVariablesChange,
27
+ }: RequestBuilderProps) {
28
+ return (
29
+ <div className="p-6 space-y-6">
30
+ {/* URL */}
31
+ <div>
32
+ <label className="block text-sm font-medium mb-2">Endpoint</label>
33
+ <input
34
+ type="text"
35
+ value={request.endpoint}
36
+ disabled
37
+ className="w-full px-3 py-2 border rounded bg-muted text-sm font-mono"
38
+ />
39
+ </div>
40
+
41
+ {/* Path Variables */}
42
+ {variables.length > 0 && (
43
+ <div>
44
+ <h3 className="text-sm font-semibold mb-2">Path Variables</h3>
45
+ <div className="space-y-2">
46
+ {variables.map((variable, index) => (
47
+ <div key={index} className="flex gap-2">
48
+ <input
49
+ type="text"
50
+ value={variable.key}
51
+ disabled
52
+ className="flex-1 px-3 py-2 border rounded bg-muted text-sm font-mono"
53
+ />
54
+ <input
55
+ type="text"
56
+ value={variable.value}
57
+ onChange={(e) => {
58
+ const newVars = [...variables]
59
+ newVars[index].value = e.target.value
60
+ onVariablesChange(newVars)
61
+ }}
62
+ placeholder="value"
63
+ className="flex-1 px-3 py-2 border rounded text-sm"
64
+ />
65
+ </div>
66
+ ))}
67
+ </div>
68
+ </div>
69
+ )}
70
+
71
+ {/* Query Parameters */}
72
+ {params.length > 0 && (
73
+ <div>
74
+ <h3 className="text-sm font-semibold mb-2">Query Parameters</h3>
75
+ <div className="space-y-2">
76
+ {params.map((param, index) => (
77
+ <div key={index} className="flex gap-2 items-center">
78
+ <input
79
+ type="checkbox"
80
+ checked={param.active}
81
+ onChange={(e) => {
82
+ const newParams = [...params]
83
+ newParams[index].active = e.target.checked
84
+ onParamsChange(newParams)
85
+ }}
86
+ className="w-4 h-4"
87
+ />
88
+ <input
89
+ type="text"
90
+ value={param.key}
91
+ disabled
92
+ className="flex-1 px-3 py-2 border rounded bg-muted text-sm font-mono"
93
+ />
94
+ <input
95
+ type="text"
96
+ value={param.value}
97
+ onChange={(e) => {
98
+ const newParams = [...params]
99
+ newParams[index].value = e.target.value
100
+ onParamsChange(newParams)
101
+ }}
102
+ placeholder="value"
103
+ className="flex-1 px-3 py-2 border rounded text-sm"
104
+ />
105
+ </div>
106
+ ))}
107
+ </div>
108
+ </div>
109
+ )}
110
+
111
+ {/* Headers */}
112
+ {headers.length > 0 && (
113
+ <div>
114
+ <h3 className="text-sm font-semibold mb-2">Headers</h3>
115
+ <div className="space-y-2">
116
+ {headers.map((header, index) => (
117
+ <div key={index} className="flex gap-2 items-center">
118
+ <input
119
+ type="checkbox"
120
+ checked={header.active}
121
+ onChange={(e) => {
122
+ const newHeaders = [...headers]
123
+ newHeaders[index].active = e.target.checked
124
+ onHeadersChange(newHeaders)
125
+ }}
126
+ className="w-4 h-4"
127
+ />
128
+ <input
129
+ type="text"
130
+ value={header.key}
131
+ disabled
132
+ className="flex-1 px-3 py-2 border rounded bg-muted text-sm font-mono"
133
+ />
134
+ <input
135
+ type="text"
136
+ value={header.value}
137
+ onChange={(e) => {
138
+ const newHeaders = [...headers]
139
+ newHeaders[index].value = e.target.value
140
+ onHeadersChange(newHeaders)
141
+ }}
142
+ placeholder="value"
143
+ className="flex-1 px-3 py-2 border rounded text-sm"
144
+ />
145
+ </div>
146
+ ))}
147
+ </div>
148
+ </div>
149
+ )}
150
+
151
+ {/* Body */}
152
+ {request.body.body !== null && request.body.contentType && (
153
+ <div>
154
+ <h3 className="text-sm font-semibold mb-2">Request Body</h3>
155
+ <div className="mb-2">
156
+ <span className="text-xs text-muted-foreground">
157
+ Content-Type: {request.body.contentType}
158
+ </span>
159
+ </div>
160
+ {request.body.contentType === 'application/x-www-form-urlencoded' ? (
161
+ <textarea
162
+ value={body || ''}
163
+ onChange={(e) => onBodyChange(e.target.value)}
164
+ className="w-full h-64 px-3 py-2 border rounded font-mono text-sm"
165
+ placeholder="key1: value1&#10;key2: value2"
166
+ />
167
+ ) : (
168
+ <textarea
169
+ value={body || ''}
170
+ onChange={(e) => onBodyChange(e.target.value)}
171
+ className="w-full h-64 px-3 py-2 border rounded font-mono text-sm"
172
+ placeholder="Enter request body..."
173
+ />
174
+ )}
175
+ </div>
176
+ )}
177
+ </div>
178
+ )
179
+ }