@su-record/vibe 2.6.28 → 2.6.31

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 (243) hide show
  1. package/CLAUDE.md +235 -202
  2. package/LICENSE +21 -21
  3. package/README.md +276 -267
  4. package/agents/architect-low.md +41 -41
  5. package/agents/architect-medium.md +59 -59
  6. package/agents/architect.md +80 -80
  7. package/agents/build-error-resolver.md +115 -115
  8. package/agents/compounder.md +261 -261
  9. package/agents/diagrammer.md +178 -178
  10. package/agents/docs/api-documenter.md +99 -99
  11. package/agents/docs/changelog-writer.md +93 -93
  12. package/agents/e2e-tester.md +266 -266
  13. package/agents/explorer-low.md +42 -42
  14. package/agents/explorer-medium.md +59 -59
  15. package/agents/explorer.md +48 -48
  16. package/agents/implementer-low.md +43 -43
  17. package/agents/implementer-medium.md +52 -52
  18. package/agents/implementer.md +54 -54
  19. package/agents/planning/requirements-analyst.md +84 -84
  20. package/agents/planning/ux-advisor.md +83 -83
  21. package/agents/qa/acceptance-tester.md +86 -86
  22. package/agents/qa/edge-case-finder.md +93 -93
  23. package/agents/refactor-cleaner.md +143 -143
  24. package/agents/research/best-practices-agent.md +199 -199
  25. package/agents/research/codebase-patterns-agent.md +157 -157
  26. package/agents/research/framework-docs-agent.md +188 -188
  27. package/agents/research/security-advisory-agent.md +213 -213
  28. package/agents/review/architecture-reviewer.md +107 -107
  29. package/agents/review/complexity-reviewer.md +116 -116
  30. package/agents/review/data-integrity-reviewer.md +88 -88
  31. package/agents/review/git-history-reviewer.md +103 -103
  32. package/agents/review/performance-reviewer.md +86 -86
  33. package/agents/review/python-reviewer.md +150 -150
  34. package/agents/review/rails-reviewer.md +139 -139
  35. package/agents/review/react-reviewer.md +144 -144
  36. package/agents/review/security-reviewer.md +80 -80
  37. package/agents/review/simplicity-reviewer.md +140 -140
  38. package/agents/review/test-coverage-reviewer.md +116 -116
  39. package/agents/review/typescript-reviewer.md +127 -127
  40. package/agents/searcher.md +54 -54
  41. package/agents/simplifier.md +120 -120
  42. package/agents/tester.md +49 -49
  43. package/agents/ui-previewer.md +268 -268
  44. package/commands/vibe.analyze.md +356 -356
  45. package/commands/vibe.reason.md +329 -329
  46. package/commands/vibe.review.md +423 -423
  47. package/commands/vibe.run.md +1423 -1313
  48. package/commands/vibe.spec.md +1054 -1054
  49. package/commands/vibe.spec.review.md +412 -412
  50. package/commands/vibe.trace.md +161 -161
  51. package/commands/vibe.utils.md +376 -376
  52. package/commands/vibe.verify.md +375 -375
  53. package/dist/cli/collaborator.js +52 -52
  54. package/dist/cli/detect.js +32 -32
  55. package/dist/cli/hud.js +20 -20
  56. package/dist/cli/index.js +112 -112
  57. package/dist/cli/llm.js +144 -144
  58. package/dist/cli/postinstall.js +858 -858
  59. package/dist/lib/DeepInit.js +24 -24
  60. package/dist/lib/IterationTracker.js +11 -11
  61. package/dist/lib/PythonParser.js +108 -108
  62. package/dist/lib/ReviewRace.js +96 -96
  63. package/dist/lib/SkillFrontmatter.js +28 -28
  64. package/dist/lib/SkillQualityGate.js +9 -9
  65. package/dist/lib/SkillRepository.js +159 -159
  66. package/dist/lib/UltraQA.js +77 -77
  67. package/dist/lib/gemini-api.d.ts +13 -0
  68. package/dist/lib/gemini-api.d.ts.map +1 -1
  69. package/dist/lib/gemini-api.js +63 -5
  70. package/dist/lib/gemini-api.js.map +1 -1
  71. package/dist/lib/gpt-api.js +4 -4
  72. package/dist/lib/memory/KnowledgeGraph.js +4 -4
  73. package/dist/lib/memory/MemorySearch.js +43 -43
  74. package/dist/lib/memory/MemoryStorage.js +130 -130
  75. package/dist/lib/memory/ObservationStore.js +28 -28
  76. package/dist/lib/memory/SessionRAGRetriever.js +7 -7
  77. package/dist/lib/memory/SessionRAGStore.js +216 -216
  78. package/dist/lib/memory/SessionSummarizer.js +9 -9
  79. package/dist/orchestrator/AgentManager.js +12 -12
  80. package/dist/orchestrator/MultiLlmResearch.js +8 -8
  81. package/dist/orchestrator/SmartRouter.js +11 -11
  82. package/dist/orchestrator/SwarmOrchestrator.test.js +16 -16
  83. package/dist/orchestrator/parallelResearch.js +24 -24
  84. package/dist/tools/convention/analyzeComplexity.test.js +115 -115
  85. package/dist/tools/convention/validateCodeQuality.test.js +104 -104
  86. package/dist/tools/spec/prdParser.test.js +171 -171
  87. package/dist/tools/spec/specGenerator.js +169 -169
  88. package/dist/tools/spec/traceabilityMatrix.js +64 -64
  89. package/dist/tools/spec/traceabilityMatrix.test.js +28 -28
  90. package/hooks/hooks.json +115 -115
  91. package/hooks/scripts/code-check.js +70 -70
  92. package/hooks/scripts/code-review.js +22 -22
  93. package/hooks/scripts/complexity.js +22 -22
  94. package/hooks/scripts/compound.js +23 -23
  95. package/hooks/scripts/context-save.js +53 -53
  96. package/hooks/scripts/gemini-ui-gen.js +281 -281
  97. package/hooks/scripts/generate-brand-assets.js +474 -474
  98. package/hooks/scripts/hud-multiline.js +262 -262
  99. package/hooks/scripts/hud-status.js +291 -291
  100. package/hooks/scripts/keyword-detector.js +214 -214
  101. package/hooks/scripts/llm-orchestrate.js +245 -171
  102. package/hooks/scripts/post-edit.js +97 -97
  103. package/hooks/scripts/post-tool-verify.js +210 -210
  104. package/hooks/scripts/pre-tool-guard.js +125 -125
  105. package/hooks/scripts/prompt-dispatcher.js +161 -161
  106. package/hooks/scripts/recall.js +22 -22
  107. package/hooks/scripts/session-start.js +30 -30
  108. package/hooks/scripts/skill-injector.js +191 -191
  109. package/hooks/scripts/utils.js +97 -97
  110. package/languages/csharp-unity.md +515 -515
  111. package/languages/gdscript-godot.md +470 -470
  112. package/languages/ruby-rails.md +489 -489
  113. package/languages/typescript-angular.md +433 -433
  114. package/languages/typescript-astro.md +416 -416
  115. package/languages/typescript-electron.md +406 -406
  116. package/languages/typescript-nestjs.md +524 -524
  117. package/languages/typescript-svelte.md +407 -407
  118. package/languages/typescript-tauri.md +365 -365
  119. package/package.json +84 -84
  120. package/skills/brand-assets.md +141 -141
  121. package/skills/commerce-patterns.md +361 -361
  122. package/skills/context7-usage.md +102 -102
  123. package/skills/e2e-commerce.md +304 -304
  124. package/skills/frontend-design.md +92 -92
  125. package/skills/git-worktree.md +181 -181
  126. package/skills/parallel-research.md +77 -77
  127. package/skills/priority-todos.md +239 -239
  128. package/skills/seo-checklist.md +244 -244
  129. package/skills/tool-fallback.md +190 -190
  130. package/skills/vibe-capabilities.md +161 -161
  131. package/vibe/constitution.md +227 -227
  132. package/vibe/rules/core/communication-guide.md +98 -98
  133. package/vibe/rules/core/development-philosophy.md +52 -52
  134. package/vibe/rules/core/quick-start.md +102 -102
  135. package/vibe/rules/quality/bdd-contract-testing.md +393 -393
  136. package/vibe/rules/quality/checklist.md +276 -276
  137. package/vibe/rules/quality/testing-strategy.md +440 -440
  138. package/vibe/rules/standards/anti-patterns.md +541 -541
  139. package/vibe/rules/standards/code-structure.md +291 -291
  140. package/vibe/rules/standards/complexity-metrics.md +313 -313
  141. package/vibe/rules/standards/naming-conventions.md +198 -198
  142. package/vibe/setup.sh +31 -31
  143. package/vibe/templates/constitution-template.md +252 -252
  144. package/vibe/templates/contract-backend-template.md +526 -526
  145. package/vibe/templates/contract-frontend-template.md +599 -599
  146. package/vibe/templates/feature-template.md +96 -96
  147. package/vibe/templates/spec-template.md +221 -221
  148. package/dist/cli/mcp.d.ts +0 -49
  149. package/dist/cli/mcp.d.ts.map +0 -1
  150. package/dist/cli/mcp.js +0 -169
  151. package/dist/cli/mcp.js.map +0 -1
  152. package/dist/lib/gemini-mcp.d.ts +0 -10
  153. package/dist/lib/gemini-mcp.d.ts.map +0 -1
  154. package/dist/lib/gemini-mcp.js +0 -353
  155. package/dist/lib/gemini-mcp.js.map +0 -1
  156. package/dist/lib/gpt-mcp.d.ts +0 -10
  157. package/dist/lib/gpt-mcp.d.ts.map +0 -1
  158. package/dist/lib/gpt-mcp.js +0 -352
  159. package/dist/lib/gpt-mcp.js.map +0 -1
  160. package/dist/tools/analytics/getUsageAnalytics.d.ts +0 -10
  161. package/dist/tools/analytics/getUsageAnalytics.d.ts.map +0 -1
  162. package/dist/tools/analytics/getUsageAnalytics.js +0 -246
  163. package/dist/tools/analytics/getUsageAnalytics.js.map +0 -1
  164. package/dist/tools/analytics/index.d.ts +0 -5
  165. package/dist/tools/analytics/index.d.ts.map +0 -1
  166. package/dist/tools/analytics/index.js +0 -5
  167. package/dist/tools/analytics/index.js.map +0 -1
  168. package/dist/tools/convention/getCodingGuide.d.ts +0 -7
  169. package/dist/tools/convention/getCodingGuide.d.ts.map +0 -1
  170. package/dist/tools/convention/getCodingGuide.js +0 -69
  171. package/dist/tools/convention/getCodingGuide.js.map +0 -1
  172. package/dist/tools/planning/analyzeRequirements.d.ts +0 -9
  173. package/dist/tools/planning/analyzeRequirements.d.ts.map +0 -1
  174. package/dist/tools/planning/analyzeRequirements.js +0 -171
  175. package/dist/tools/planning/analyzeRequirements.js.map +0 -1
  176. package/dist/tools/planning/createUserStories.d.ts +0 -9
  177. package/dist/tools/planning/createUserStories.d.ts.map +0 -1
  178. package/dist/tools/planning/createUserStories.js +0 -124
  179. package/dist/tools/planning/createUserStories.js.map +0 -1
  180. package/dist/tools/planning/featureRoadmap.d.ts +0 -10
  181. package/dist/tools/planning/featureRoadmap.d.ts.map +0 -1
  182. package/dist/tools/planning/featureRoadmap.js +0 -207
  183. package/dist/tools/planning/featureRoadmap.js.map +0 -1
  184. package/dist/tools/planning/generatePrd.d.ts +0 -11
  185. package/dist/tools/planning/generatePrd.d.ts.map +0 -1
  186. package/dist/tools/planning/generatePrd.js +0 -161
  187. package/dist/tools/planning/generatePrd.js.map +0 -1
  188. package/dist/tools/planning/index.d.ts +0 -8
  189. package/dist/tools/planning/index.d.ts.map +0 -1
  190. package/dist/tools/planning/index.js +0 -8
  191. package/dist/tools/planning/index.js.map +0 -1
  192. package/dist/tools/prompt/analyzePrompt.d.ts +0 -7
  193. package/dist/tools/prompt/analyzePrompt.d.ts.map +0 -1
  194. package/dist/tools/prompt/analyzePrompt.js +0 -150
  195. package/dist/tools/prompt/analyzePrompt.js.map +0 -1
  196. package/dist/tools/prompt/enhancePrompt.d.ts +0 -8
  197. package/dist/tools/prompt/enhancePrompt.d.ts.map +0 -1
  198. package/dist/tools/prompt/enhancePrompt.js +0 -110
  199. package/dist/tools/prompt/enhancePrompt.js.map +0 -1
  200. package/dist/tools/prompt/enhancePromptGemini.d.ts +0 -8
  201. package/dist/tools/prompt/enhancePromptGemini.d.ts.map +0 -1
  202. package/dist/tools/prompt/enhancePromptGemini.js +0 -332
  203. package/dist/tools/prompt/enhancePromptGemini.js.map +0 -1
  204. package/dist/tools/prompt/index.d.ts +0 -7
  205. package/dist/tools/prompt/index.d.ts.map +0 -1
  206. package/dist/tools/prompt/index.js +0 -7
  207. package/dist/tools/prompt/index.js.map +0 -1
  208. package/dist/tools/reasoning/applyReasoningFramework.d.ts +0 -8
  209. package/dist/tools/reasoning/applyReasoningFramework.d.ts.map +0 -1
  210. package/dist/tools/reasoning/applyReasoningFramework.js +0 -266
  211. package/dist/tools/reasoning/applyReasoningFramework.js.map +0 -1
  212. package/dist/tools/reasoning/index.d.ts +0 -5
  213. package/dist/tools/reasoning/index.d.ts.map +0 -1
  214. package/dist/tools/reasoning/index.js +0 -5
  215. package/dist/tools/reasoning/index.js.map +0 -1
  216. package/dist/tools/thinking/analyzeProblem.d.ts +0 -7
  217. package/dist/tools/thinking/analyzeProblem.d.ts.map +0 -1
  218. package/dist/tools/thinking/analyzeProblem.js +0 -55
  219. package/dist/tools/thinking/analyzeProblem.js.map +0 -1
  220. package/dist/tools/thinking/breakDownProblem.d.ts +0 -8
  221. package/dist/tools/thinking/breakDownProblem.d.ts.map +0 -1
  222. package/dist/tools/thinking/breakDownProblem.js +0 -145
  223. package/dist/tools/thinking/breakDownProblem.js.map +0 -1
  224. package/dist/tools/thinking/createThinkingChain.d.ts +0 -7
  225. package/dist/tools/thinking/createThinkingChain.d.ts.map +0 -1
  226. package/dist/tools/thinking/createThinkingChain.js +0 -44
  227. package/dist/tools/thinking/createThinkingChain.js.map +0 -1
  228. package/dist/tools/thinking/formatAsPlan.d.ts +0 -9
  229. package/dist/tools/thinking/formatAsPlan.d.ts.map +0 -1
  230. package/dist/tools/thinking/formatAsPlan.js +0 -78
  231. package/dist/tools/thinking/formatAsPlan.js.map +0 -1
  232. package/dist/tools/thinking/index.d.ts +0 -10
  233. package/dist/tools/thinking/index.d.ts.map +0 -1
  234. package/dist/tools/thinking/index.js +0 -10
  235. package/dist/tools/thinking/index.js.map +0 -1
  236. package/dist/tools/thinking/stepByStepAnalysis.d.ts +0 -8
  237. package/dist/tools/thinking/stepByStepAnalysis.d.ts.map +0 -1
  238. package/dist/tools/thinking/stepByStepAnalysis.js +0 -63
  239. package/dist/tools/thinking/stepByStepAnalysis.js.map +0 -1
  240. package/dist/tools/thinking/thinkAloudProcess.d.ts +0 -8
  241. package/dist/tools/thinking/thinkAloudProcess.d.ts.map +0 -1
  242. package/dist/tools/thinking/thinkAloudProcess.js +0 -80
  243. package/dist/tools/thinking/thinkAloudProcess.js.map +0 -1
@@ -1,281 +1,281 @@
1
- #!/usr/bin/env node
2
-
3
- /**
4
- * Gemini UI Code Generator
5
- *
6
- * ė””ģžģø ķŒŒģ¼(ģ“ėÆøģ§€, HTML 등)ģ„ ė¶„ģ„ķ•“ģ„œ UI ģ½”ė“œė„¼ ģƒģ„±ķ•©ė‹ˆė‹¤.
7
- * 기씓 gemini-api ģøķ”„ė¼ ģ‚¬ģš©.
8
- *
9
- * Usage:
10
- * node gemini-ui-gen.js --image ./design.png --framework react --output ./src/components
11
- * node gemini-ui-gen.js --html ./mockup.html --framework vue --output ./src/components
12
- * node gemini-ui-gen.js --design-folder ./design/ --framework react --output ./src
13
- */
14
-
15
- import fs from 'fs';
16
- import path from 'path';
17
- import { getLibBaseUrl } from './utils.js';
18
-
19
- const LIB_URL = getLibBaseUrl();
20
-
21
- // ============================================
22
- // Gemini API (기씓 ģøķ”„ė¼ ģ‚¬ģš©)
23
- // ============================================
24
-
25
- let geminiApi = null;
26
-
27
- async function getGeminiApi() {
28
- if (!geminiApi) {
29
- geminiApi = await import(`${LIB_URL}gemini-api.js`);
30
- }
31
- return geminiApi;
32
- }
33
-
34
- async function askGemini(prompt) {
35
- const api = await getGeminiApi();
36
- return api.ask(prompt, { model: 'gemini-3-flash', maxTokens: 8192, temperature: 0.3 });
37
- }
38
-
39
- // ============================================
40
- // UI Code Generation
41
- // ============================================
42
-
43
- function getFrameworkPrompt(framework) {
44
- const prompts = {
45
- react: `Generate React TypeScript components using:
46
- - Functional components with hooks
47
- - Tailwind CSS for styling
48
- - Proper TypeScript types/interfaces
49
- - Export as default`,
50
-
51
- vue: `Generate Vue 3 components using:
52
- - Composition API with <script setup>
53
- - Tailwind CSS for styling
54
- - TypeScript support
55
- - Single File Component format`,
56
-
57
- svelte: `Generate Svelte components using:
58
- - Svelte 5 runes syntax
59
- - Tailwind CSS for styling
60
- - TypeScript support`,
61
-
62
- html: `Generate semantic HTML5 with:
63
- - Tailwind CSS classes
64
- - Accessible markup
65
- - Responsive design`,
66
- };
67
-
68
- return prompts[framework] || prompts.react;
69
- }
70
-
71
- async function generateUIFromImage(imagePath, framework) {
72
- const imageBuffer = fs.readFileSync(imagePath);
73
- const imageBase64 = imageBuffer.toString('base64');
74
-
75
- const ext = path.extname(imagePath).toLowerCase();
76
- const mimeTypes = {
77
- '.png': 'image/png',
78
- '.jpg': 'image/jpeg',
79
- '.jpeg': 'image/jpeg',
80
- '.webp': 'image/webp',
81
- '.gif': 'image/gif',
82
- };
83
- const mimeType = mimeTypes[ext] || 'image/png';
84
-
85
- const prompt = `[Image attached as base64: ${mimeType}]
86
- data:${mimeType};base64,${imageBase64}
87
-
88
- Analyze this UI design image and generate production-ready code.
89
-
90
- ${getFrameworkPrompt(framework)}
91
-
92
- Requirements:
93
- 1. Match the visual design exactly (colors, spacing, typography, layout)
94
- 2. Extract exact colors as hex values
95
- 3. Use proper semantic HTML structure
96
- 4. Make it responsive (mobile-first)
97
- 5. Include hover/focus states where appropriate
98
- 6. Add appropriate accessibility attributes
99
-
100
- Output format:
101
- \`\`\`${framework === 'html' ? 'html' : 'tsx'}
102
- // Component code here
103
- \`\`\`
104
-
105
- Also provide a summary of:
106
- - Colors extracted
107
- - Components identified
108
- - Layout structure`;
109
-
110
- return askGemini(prompt);
111
- }
112
-
113
- async function generateUIFromHTML(htmlPath, framework) {
114
- const htmlContent = fs.readFileSync(htmlPath, 'utf-8');
115
-
116
- const prompt = `Convert this HTML mockup to production-ready ${framework} code.
117
-
118
- HTML Mockup:
119
- \`\`\`html
120
- ${htmlContent}
121
- \`\`\`
122
-
123
- ${getFrameworkPrompt(framework)}
124
-
125
- Requirements:
126
- 1. Preserve the exact visual appearance
127
- 2. Extract inline styles to Tailwind classes
128
- 3. Create reusable components where appropriate
129
- 4. Add proper TypeScript types
130
- 5. Make it responsive
131
-
132
- Output the converted code in proper format.`;
133
-
134
- return askGemini(prompt);
135
- }
136
-
137
- async function analyzeDesignFolder(folderPath, framework) {
138
- const files = fs.readdirSync(folderPath);
139
- let combinedPrompt = `Analyze the following design files and generate production-ready ${framework} code.\n\n`;
140
-
141
- for (const file of files) {
142
- const filePath = path.join(folderPath, file);
143
- const ext = path.extname(file).toLowerCase();
144
-
145
- if (['.png', '.jpg', '.jpeg', '.webp', '.gif'].includes(ext)) {
146
- const imageBuffer = fs.readFileSync(filePath);
147
- const imageBase64 = imageBuffer.toString('base64');
148
- const mimeType = ext === '.png' ? 'image/png' : ext === '.webp' ? 'image/webp' : 'image/jpeg';
149
- combinedPrompt += `\n--- Image: ${file} ---\ndata:${mimeType};base64,${imageBase64}\n`;
150
- console.log(`šŸ“· ${file}`);
151
- } else if (ext === '.html') {
152
- const content = fs.readFileSync(filePath, 'utf-8');
153
- combinedPrompt += `\n--- HTML: ${file} ---\n${content}\n`;
154
- console.log(`šŸ“„ ${file}`);
155
- } else if (ext === '.json') {
156
- const content = fs.readFileSync(filePath, 'utf-8');
157
- combinedPrompt += `\n--- Design Tokens: ${file} ---\n${content}\n`;
158
- console.log(`šŸ“‹ ${file}`);
159
- } else if (['.css', '.scss'].includes(ext)) {
160
- const content = fs.readFileSync(filePath, 'utf-8');
161
- combinedPrompt += `\n--- Styles: ${file} ---\n${content}\n`;
162
- console.log(`šŸŽØ ${file}`);
163
- } else if (ext === '.md') {
164
- const content = fs.readFileSync(filePath, 'utf-8');
165
- combinedPrompt += `\n--- Guide: ${file} ---\n${content}\n`;
166
- console.log(`šŸ“ ${file}`);
167
- }
168
- }
169
-
170
- combinedPrompt += `\n${getFrameworkPrompt(framework)}
171
-
172
- Requirements:
173
- 1. Match the visual design exactly
174
- 2. Extract design tokens from JSON if provided
175
- 3. Use CSS variables from stylesheets if provided
176
- 4. Create separate component files for each major UI section
177
- 5. Make it responsive (mobile-first)
178
- 6. Include accessibility attributes
179
-
180
- Output complete component code.`;
181
-
182
- return askGemini(combinedPrompt);
183
- }
184
-
185
- // ============================================
186
- // CLI
187
- // ============================================
188
-
189
- async function main() {
190
- const args = process.argv.slice(2);
191
-
192
- const options = {
193
- image: null,
194
- html: null,
195
- designFolder: null,
196
- framework: 'react',
197
- output: './generated',
198
- };
199
-
200
- for (let i = 0; i < args.length; i++) {
201
- switch (args[i]) {
202
- case '--image':
203
- options.image = args[++i];
204
- break;
205
- case '--html':
206
- options.html = args[++i];
207
- break;
208
- case '--design-folder':
209
- case '--folder':
210
- options.designFolder = args[++i];
211
- break;
212
- case '--framework':
213
- case '-f':
214
- options.framework = args[++i];
215
- break;
216
- case '--output':
217
- case '-o':
218
- options.output = args[++i];
219
- break;
220
- case '--help':
221
- case '-h':
222
- console.log(`
223
- Gemini UI Code Generator
224
-
225
- Usage:
226
- node gemini-ui-gen.js --image ./design.png --framework react
227
- node gemini-ui-gen.js --html ./mockup.html --framework vue
228
- node gemini-ui-gen.js --design-folder ./design/ --framework react
229
-
230
- Options:
231
- --image <path> Image file to analyze
232
- --html <path> HTML mockup to convert
233
- --design-folder <path> Folder with design files
234
- --framework <name> Target framework (react, vue, svelte, html)
235
- --output <path> Output directory
236
- --help Show this help
237
- `);
238
- process.exit(0);
239
- }
240
- }
241
-
242
- console.log(`šŸ¤– Gemini UI Generator`);
243
- console.log(`šŸ“¦ Framework: ${options.framework}`);
244
-
245
- try {
246
- let result;
247
-
248
- if (options.image) {
249
- console.log(`\nšŸ“· Analyzing: ${options.image}\n`);
250
- result = await generateUIFromImage(options.image, options.framework);
251
- } else if (options.html) {
252
- console.log(`\nšŸ“„ Converting: ${options.html}\n`);
253
- result = await generateUIFromHTML(options.html, options.framework);
254
- } else if (options.designFolder) {
255
- console.log(`\nšŸ“‚ Analyzing folder: ${options.designFolder}\n`);
256
- result = await analyzeDesignFolder(options.designFolder, options.framework);
257
- } else {
258
- console.error('āŒ No input specified. Use --image, --html, or --design-folder');
259
- process.exit(1);
260
- }
261
-
262
- console.log('\n' + '='.repeat(60) + '\n');
263
- console.log(result);
264
- console.log('\n' + '='.repeat(60));
265
-
266
- if (options.output && result) {
267
- if (!fs.existsSync(options.output)) {
268
- fs.mkdirSync(options.output, { recursive: true });
269
- }
270
- const outputFile = path.join(options.output, `generated-${Date.now()}.txt`);
271
- fs.writeFileSync(outputFile, result);
272
- console.log(`\nāœ… Output saved to: ${outputFile}`);
273
- }
274
-
275
- } catch (error) {
276
- console.error(`\nāŒ Error: ${error.message}`);
277
- process.exit(1);
278
- }
279
- }
280
-
281
- main();
1
+ #!/usr/bin/env node
2
+
3
+ /**
4
+ * Gemini UI Code Generator
5
+ *
6
+ * ė””ģžģø ķŒŒģ¼(ģ“ėÆøģ§€, HTML 등)ģ„ ė¶„ģ„ķ•“ģ„œ UI ģ½”ė“œė„¼ ģƒģ„±ķ•©ė‹ˆė‹¤.
7
+ * 기씓 gemini-api ģøķ”„ė¼ ģ‚¬ģš©.
8
+ *
9
+ * Usage:
10
+ * node gemini-ui-gen.js --image ./design.png --framework react --output ./src/components
11
+ * node gemini-ui-gen.js --html ./mockup.html --framework vue --output ./src/components
12
+ * node gemini-ui-gen.js --design-folder ./design/ --framework react --output ./src
13
+ */
14
+
15
+ import fs from 'fs';
16
+ import path from 'path';
17
+ import { getLibBaseUrl } from './utils.js';
18
+
19
+ const LIB_URL = getLibBaseUrl();
20
+
21
+ // ============================================
22
+ // Gemini API (기씓 ģøķ”„ė¼ ģ‚¬ģš©)
23
+ // ============================================
24
+
25
+ let geminiApi = null;
26
+
27
+ async function getGeminiApi() {
28
+ if (!geminiApi) {
29
+ geminiApi = await import(`${LIB_URL}gemini-api.js`);
30
+ }
31
+ return geminiApi;
32
+ }
33
+
34
+ async function askGemini(prompt) {
35
+ const api = await getGeminiApi();
36
+ return api.ask(prompt, { model: 'gemini-3-flash', maxTokens: 8192, temperature: 0.3 });
37
+ }
38
+
39
+ // ============================================
40
+ // UI Code Generation
41
+ // ============================================
42
+
43
+ function getFrameworkPrompt(framework) {
44
+ const prompts = {
45
+ react: `Generate React TypeScript components using:
46
+ - Functional components with hooks
47
+ - Tailwind CSS for styling
48
+ - Proper TypeScript types/interfaces
49
+ - Export as default`,
50
+
51
+ vue: `Generate Vue 3 components using:
52
+ - Composition API with <script setup>
53
+ - Tailwind CSS for styling
54
+ - TypeScript support
55
+ - Single File Component format`,
56
+
57
+ svelte: `Generate Svelte components using:
58
+ - Svelte 5 runes syntax
59
+ - Tailwind CSS for styling
60
+ - TypeScript support`,
61
+
62
+ html: `Generate semantic HTML5 with:
63
+ - Tailwind CSS classes
64
+ - Accessible markup
65
+ - Responsive design`,
66
+ };
67
+
68
+ return prompts[framework] || prompts.react;
69
+ }
70
+
71
+ async function generateUIFromImage(imagePath, framework) {
72
+ const imageBuffer = fs.readFileSync(imagePath);
73
+ const imageBase64 = imageBuffer.toString('base64');
74
+
75
+ const ext = path.extname(imagePath).toLowerCase();
76
+ const mimeTypes = {
77
+ '.png': 'image/png',
78
+ '.jpg': 'image/jpeg',
79
+ '.jpeg': 'image/jpeg',
80
+ '.webp': 'image/webp',
81
+ '.gif': 'image/gif',
82
+ };
83
+ const mimeType = mimeTypes[ext] || 'image/png';
84
+
85
+ const prompt = `[Image attached as base64: ${mimeType}]
86
+ data:${mimeType};base64,${imageBase64}
87
+
88
+ Analyze this UI design image and generate production-ready code.
89
+
90
+ ${getFrameworkPrompt(framework)}
91
+
92
+ Requirements:
93
+ 1. Match the visual design exactly (colors, spacing, typography, layout)
94
+ 2. Extract exact colors as hex values
95
+ 3. Use proper semantic HTML structure
96
+ 4. Make it responsive (mobile-first)
97
+ 5. Include hover/focus states where appropriate
98
+ 6. Add appropriate accessibility attributes
99
+
100
+ Output format:
101
+ \`\`\`${framework === 'html' ? 'html' : 'tsx'}
102
+ // Component code here
103
+ \`\`\`
104
+
105
+ Also provide a summary of:
106
+ - Colors extracted
107
+ - Components identified
108
+ - Layout structure`;
109
+
110
+ return askGemini(prompt);
111
+ }
112
+
113
+ async function generateUIFromHTML(htmlPath, framework) {
114
+ const htmlContent = fs.readFileSync(htmlPath, 'utf-8');
115
+
116
+ const prompt = `Convert this HTML mockup to production-ready ${framework} code.
117
+
118
+ HTML Mockup:
119
+ \`\`\`html
120
+ ${htmlContent}
121
+ \`\`\`
122
+
123
+ ${getFrameworkPrompt(framework)}
124
+
125
+ Requirements:
126
+ 1. Preserve the exact visual appearance
127
+ 2. Extract inline styles to Tailwind classes
128
+ 3. Create reusable components where appropriate
129
+ 4. Add proper TypeScript types
130
+ 5. Make it responsive
131
+
132
+ Output the converted code in proper format.`;
133
+
134
+ return askGemini(prompt);
135
+ }
136
+
137
+ async function analyzeDesignFolder(folderPath, framework) {
138
+ const files = fs.readdirSync(folderPath);
139
+ let combinedPrompt = `Analyze the following design files and generate production-ready ${framework} code.\n\n`;
140
+
141
+ for (const file of files) {
142
+ const filePath = path.join(folderPath, file);
143
+ const ext = path.extname(file).toLowerCase();
144
+
145
+ if (['.png', '.jpg', '.jpeg', '.webp', '.gif'].includes(ext)) {
146
+ const imageBuffer = fs.readFileSync(filePath);
147
+ const imageBase64 = imageBuffer.toString('base64');
148
+ const mimeType = ext === '.png' ? 'image/png' : ext === '.webp' ? 'image/webp' : 'image/jpeg';
149
+ combinedPrompt += `\n--- Image: ${file} ---\ndata:${mimeType};base64,${imageBase64}\n`;
150
+ console.log(`šŸ“· ${file}`);
151
+ } else if (ext === '.html') {
152
+ const content = fs.readFileSync(filePath, 'utf-8');
153
+ combinedPrompt += `\n--- HTML: ${file} ---\n${content}\n`;
154
+ console.log(`šŸ“„ ${file}`);
155
+ } else if (ext === '.json') {
156
+ const content = fs.readFileSync(filePath, 'utf-8');
157
+ combinedPrompt += `\n--- Design Tokens: ${file} ---\n${content}\n`;
158
+ console.log(`šŸ“‹ ${file}`);
159
+ } else if (['.css', '.scss'].includes(ext)) {
160
+ const content = fs.readFileSync(filePath, 'utf-8');
161
+ combinedPrompt += `\n--- Styles: ${file} ---\n${content}\n`;
162
+ console.log(`šŸŽØ ${file}`);
163
+ } else if (ext === '.md') {
164
+ const content = fs.readFileSync(filePath, 'utf-8');
165
+ combinedPrompt += `\n--- Guide: ${file} ---\n${content}\n`;
166
+ console.log(`šŸ“ ${file}`);
167
+ }
168
+ }
169
+
170
+ combinedPrompt += `\n${getFrameworkPrompt(framework)}
171
+
172
+ Requirements:
173
+ 1. Match the visual design exactly
174
+ 2. Extract design tokens from JSON if provided
175
+ 3. Use CSS variables from stylesheets if provided
176
+ 4. Create separate component files for each major UI section
177
+ 5. Make it responsive (mobile-first)
178
+ 6. Include accessibility attributes
179
+
180
+ Output complete component code.`;
181
+
182
+ return askGemini(combinedPrompt);
183
+ }
184
+
185
+ // ============================================
186
+ // CLI
187
+ // ============================================
188
+
189
+ async function main() {
190
+ const args = process.argv.slice(2);
191
+
192
+ const options = {
193
+ image: null,
194
+ html: null,
195
+ designFolder: null,
196
+ framework: 'react',
197
+ output: './generated',
198
+ };
199
+
200
+ for (let i = 0; i < args.length; i++) {
201
+ switch (args[i]) {
202
+ case '--image':
203
+ options.image = args[++i];
204
+ break;
205
+ case '--html':
206
+ options.html = args[++i];
207
+ break;
208
+ case '--design-folder':
209
+ case '--folder':
210
+ options.designFolder = args[++i];
211
+ break;
212
+ case '--framework':
213
+ case '-f':
214
+ options.framework = args[++i];
215
+ break;
216
+ case '--output':
217
+ case '-o':
218
+ options.output = args[++i];
219
+ break;
220
+ case '--help':
221
+ case '-h':
222
+ console.log(`
223
+ Gemini UI Code Generator
224
+
225
+ Usage:
226
+ node gemini-ui-gen.js --image ./design.png --framework react
227
+ node gemini-ui-gen.js --html ./mockup.html --framework vue
228
+ node gemini-ui-gen.js --design-folder ./design/ --framework react
229
+
230
+ Options:
231
+ --image <path> Image file to analyze
232
+ --html <path> HTML mockup to convert
233
+ --design-folder <path> Folder with design files
234
+ --framework <name> Target framework (react, vue, svelte, html)
235
+ --output <path> Output directory
236
+ --help Show this help
237
+ `);
238
+ process.exit(0);
239
+ }
240
+ }
241
+
242
+ console.log(`šŸ¤– Gemini UI Generator`);
243
+ console.log(`šŸ“¦ Framework: ${options.framework}`);
244
+
245
+ try {
246
+ let result;
247
+
248
+ if (options.image) {
249
+ console.log(`\nšŸ“· Analyzing: ${options.image}\n`);
250
+ result = await generateUIFromImage(options.image, options.framework);
251
+ } else if (options.html) {
252
+ console.log(`\nšŸ“„ Converting: ${options.html}\n`);
253
+ result = await generateUIFromHTML(options.html, options.framework);
254
+ } else if (options.designFolder) {
255
+ console.log(`\nšŸ“‚ Analyzing folder: ${options.designFolder}\n`);
256
+ result = await analyzeDesignFolder(options.designFolder, options.framework);
257
+ } else {
258
+ console.error('āŒ No input specified. Use --image, --html, or --design-folder');
259
+ process.exit(1);
260
+ }
261
+
262
+ console.log('\n' + '='.repeat(60) + '\n');
263
+ console.log(result);
264
+ console.log('\n' + '='.repeat(60));
265
+
266
+ if (options.output && result) {
267
+ if (!fs.existsSync(options.output)) {
268
+ fs.mkdirSync(options.output, { recursive: true });
269
+ }
270
+ const outputFile = path.join(options.output, `generated-${Date.now()}.txt`);
271
+ fs.writeFileSync(outputFile, result);
272
+ console.log(`\nāœ… Output saved to: ${outputFile}`);
273
+ }
274
+
275
+ } catch (error) {
276
+ console.error(`\nāŒ Error: ${error.message}`);
277
+ process.exit(1);
278
+ }
279
+ }
280
+
281
+ main();