@polymorphism-tech/morph-spec 2.1.2 → 2.3.0

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.
@@ -0,0 +1,366 @@
1
+ /**
2
+ * Mockup Generator
3
+ *
4
+ * Generates high-quality ASCII wireframes for UI components.
5
+ * Used in FASE 1.5: UI/UX Design (conception phase).
6
+ *
7
+ * MORPH-SPEC 3.0
8
+ */
9
+
10
+ /**
11
+ * Generate wireframe based on component spec
12
+ * @param {Object} componentSpec - Component specification
13
+ * @returns {string} ASCII wireframe
14
+ */
15
+ export function generateWireframe(componentSpec) {
16
+ switch (componentSpec.type) {
17
+ case 'form':
18
+ return generateFormWireframe(componentSpec);
19
+ case 'list':
20
+ case 'data-grid':
21
+ return generateListWireframe(componentSpec);
22
+ case 'dashboard':
23
+ return generateDashboardWireframe(componentSpec);
24
+ case 'wizard':
25
+ return generateWizardWireframe(componentSpec);
26
+ case 'file-upload':
27
+ return generateFileUploadWireframe(componentSpec);
28
+ case 'confirm-dialog':
29
+ return generateConfirmDialogWireframe(componentSpec);
30
+ case 'detail-view':
31
+ return generateDetailViewWireframe(componentSpec);
32
+ default:
33
+ return generateGenericWireframe(componentSpec);
34
+ }
35
+ }
36
+
37
+ /**
38
+ * Generate form wireframe
39
+ */
40
+ function generateFormWireframe(spec) {
41
+ const { entity, operation = 'create', fields = [] } = spec;
42
+ const title = operation === 'create' ? `Create ${entity}` : `Edit ${entity}`;
43
+
44
+ const maxLabelLength = Math.max(...fields.map(f => (f.label || f.name).length), 15);
45
+ const fieldWidth = 40;
46
+ const totalWidth = maxLabelLength + fieldWidth + 10;
47
+
48
+ const fieldLines = fields.map(field => {
49
+ const label = (field.label || field.name).padEnd(maxLabelLength);
50
+ const inputType = getInputType(field.type);
51
+ const required = field.isRequired ? '*' : ' ';
52
+
53
+ return `│ ${required} ${label} ${inputType.padEnd(fieldWidth)} │`;
54
+ }).join('\n');
55
+
56
+ return `
57
+ ┌${'─'.repeat(totalWidth)}┐
58
+ │ ${title.padEnd(totalWidth - 4)} │
59
+ ├${'─'.repeat(totalWidth)}┤
60
+ │${' '.repeat(totalWidth)}│
61
+ ${fieldLines}
62
+ │${' '.repeat(totalWidth)}│
63
+ │ ┌──────────┐ ┌────────┐${' '.repeat(totalWidth - 34)}│
64
+ │ │ ${operation === 'create' ? 'Create' : 'Save'} │ │ Cancel │${' '.repeat(totalWidth - 34)}│
65
+ │ └──────────┘ └────────┘${' '.repeat(totalWidth - 34)}│
66
+ └${'─'.repeat(totalWidth)}┘
67
+
68
+ Legend: * = Required field
69
+ `.trim();
70
+ }
71
+
72
+ /**
73
+ * Get input type representation
74
+ */
75
+ function getInputType(type) {
76
+ if (!type) return '[_________________]';
77
+
78
+ const lower = type.toLowerCase();
79
+
80
+ if (lower.includes('string') || lower.includes('text')) {
81
+ return '[_________________]';
82
+ } else if (lower.includes('int') || lower.includes('decimal') || lower.includes('number')) {
83
+ return '[______] (number)';
84
+ } else if (lower.includes('date') || lower.includes('datetime')) {
85
+ return '[____/____/____] 📅';
86
+ } else if (lower.includes('bool')) {
87
+ return '☐ Yes ☐ No';
88
+ } else if (lower.includes('enum') || lower.includes('status') || lower.includes('type')) {
89
+ return '[Select option ▼]';
90
+ } else if (lower.includes('email')) {
91
+ return '[___@___.___]';
92
+ } else if (lower.includes('url')) {
93
+ return '[https://____]';
94
+ } else {
95
+ return '[_________________]';
96
+ }
97
+ }
98
+
99
+ /**
100
+ * Generate list/grid wireframe
101
+ */
102
+ function generateListWireframe(spec) {
103
+ const { entity, fields = [], features = {} } = spec;
104
+ const title = `${entity} List`;
105
+
106
+ const headers = fields.slice(0, 5).map(f => (f.label || f.name).substring(0, 15).padEnd(15));
107
+ const headerLine = headers.join(' │ ');
108
+
109
+ const sampleRow = fields.slice(0, 5).map(f => {
110
+ const sample = getSampleValue(f.type);
111
+ return sample.substring(0, 15).padEnd(15);
112
+ }).join(' │ ');
113
+
114
+ const totalWidth = headers.length * 16 + headers.length - 1 + 12; // +12 for actions
115
+
116
+ return `
117
+ ┌${'─'.repeat(totalWidth + 4)}┐
118
+ │ ${title.padEnd(totalWidth - 20)} [+ New] [⚙ Filters] │
119
+ ├${'─'.repeat(totalWidth + 4)}┤
120
+ │ ${headerLine} │ Actions │
121
+ ├${'─'.repeat(totalWidth + 4)}┤
122
+ │ ${sampleRow} │ ✏️ 🗑️ │
123
+ │ ${sampleRow} │ ✏️ 🗑️ │
124
+ │ ${sampleRow} │ ✏️ 🗑️ │
125
+ ${features.pagination !== false ? `├${'─'.repeat(totalWidth + 4)}┤
126
+ │ ◀ 1 2 3 ... 10 ▶${' '.repeat(totalWidth - 40)}Showing 1-10 of 47 │` : ''}
127
+ └${'─'.repeat(totalWidth + 4)}┘
128
+
129
+ ${features.filtering !== false ? '\nFeatures: Search, Filter, Sort' : ''}
130
+ ${features.selection !== false ? 'Selection: Multi-select with checkboxes' : ''}
131
+ `.trim();
132
+ }
133
+
134
+ /**
135
+ * Get sample value for field type
136
+ */
137
+ function getSampleValue(type) {
138
+ if (!type) return 'Sample Value';
139
+
140
+ const lower = type.toLowerCase();
141
+
142
+ if (lower.includes('name')) return 'John Doe';
143
+ if (lower.includes('email')) return 'user@example.com';
144
+ if (lower.includes('phone')) return '(555) 123-4567';
145
+ if (lower.includes('date')) return '2025-01-17';
146
+ if (lower.includes('status')) return 'Active';
147
+ if (lower.includes('amount') || lower.includes('price')) return '$99.99';
148
+ if (lower.includes('quantity') || lower.includes('count')) return '42';
149
+ if (lower.includes('bool')) return 'Yes';
150
+
151
+ return 'Sample';
152
+ }
153
+
154
+ /**
155
+ * Generate dashboard wireframe
156
+ */
157
+ function generateDashboardWireframe(spec) {
158
+ const { widgets = [] } = spec;
159
+
160
+ const hasStats = widgets.some(w => w.type === 'stat-card');
161
+ const hasChart = widgets.some(w => w.type === 'chart');
162
+ const hasTable = widgets.some(w => w.type === 'data-table');
163
+
164
+ return `
165
+ ┌────────────────────────────────────────────────────────────────────────────┐
166
+ │ Dashboard Overview [User Icon] ▼ │
167
+ ├────────────────────────────────────────────────────────────────────────────┤
168
+ │ │
169
+ ${hasStats ? `│ ┌──────────────┐ ┌──────────────┐ ┌──────────────┐ │
170
+ │ │ Total Sales │ │ New Orders │ │ Revenue │ │
171
+ │ │ $125,430 │ │ 47 │ │ +12.5% │ │
172
+ │ │ +8.2% ↑ │ │ +5 today │ │ This month │ │
173
+ │ └──────────────┘ └──────────────┘ └──────────────┘ │
174
+ │ │` : ''}
175
+ ${hasChart ? `│ ┌────────────────────────────────────────────────────────────────┐ │
176
+ │ │ Sales Trend (Last 30 Days) │ │
177
+ │ │ │ │
178
+ │ │ $ ╭─╮ │ │
179
+ │ │ │ │ ╭─╮ │ │
180
+ │ │ │ │ ╭─╯ ╰─╮ ╭─╮ │ │
181
+ │ │ ────┴─┴───┴─────┴───┴─┴───────────────────────────────────── │ │
182
+ │ │ Week 1 Week 2 Week 3 Week 4 │ │
183
+ │ └────────────────────────────────────────────────────────────────┘ │
184
+ │ │` : ''}
185
+ ${hasTable ? `│ ┌────────────────────────────────────────────────────────────────┐ │
186
+ │ │ Recent Orders [View All →] │ │
187
+ │ ├────────────────────────────────────────────────────────────────┤ │
188
+ │ │ Order # Customer Amount Status Actions │ │
189
+ │ ├────────────────────────────────────────────────────────────────┤ │
190
+ │ │ #1234 John Doe $99.99 Pending 👁️ ✏️ │ │
191
+ │ │ #1233 Jane Smith $149.99 Completed 👁️ ✏️ │ │
192
+ │ │ #1232 Bob Johnson $79.99 Shipped 👁️ ✏️ │ │
193
+ │ └────────────────────────────────────────────────────────────────┘ │` : ''}
194
+ │ │
195
+ └────────────────────────────────────────────────────────────────────────────┘
196
+ `.trim();
197
+ }
198
+
199
+ /**
200
+ * Generate wizard wireframe
201
+ */
202
+ function generateWizardWireframe(spec) {
203
+ const { steps = [] } = spec;
204
+ const currentStep = steps[0] || { number: 1, title: 'Step 1' };
205
+
206
+ const stepIndicators = steps.map((step, i) => {
207
+ const isCurrent = i === 0;
208
+ return isCurrent ? `● ${step.number}` : `○ ${step.number}`;
209
+ }).join(' ── ');
210
+
211
+ return `
212
+ ┌────────────────────────────────────────────────────────────────────────────┐
213
+ │ ${currentStep.title.padEnd(70)} │
214
+ ├────────────────────────────────────────────────────────────────────────────┤
215
+ │ │
216
+ │ Progress: ${stepIndicators.padEnd(60)} │
217
+ │ │
218
+ │ ┌──────────────────────────────────────────────────────────────────────┐ │
219
+ │ │ │ │
220
+ │ │ {Step ${currentStep.number} Content} │ │
221
+ │ │ - Input fields │ │
222
+ │ │ - Instructions │ │
223
+ │ │ - Validation messages │ │
224
+ │ │ │ │
225
+ │ └──────────────────────────────────────────────────────────────────────┘ │
226
+ │ │
227
+ │ ┌────────┐ ┌────────────┐ │
228
+ │ │ Back │ │ Next → │ │
229
+ │ └────────┘ └────────────┘ │
230
+ └────────────────────────────────────────────────────────────────────────────┘
231
+ `.trim();
232
+ }
233
+
234
+ /**
235
+ * Generate file upload wireframe
236
+ */
237
+ function generateFileUploadWireframe(spec) {
238
+ const { maxSize = '10MB', allowedTypes = ['.jpg', '.png'] } = spec;
239
+
240
+ return `
241
+ ┌────────────────────────────────────────────────────────────────────────────┐
242
+ │ Upload File │
243
+ ├────────────────────────────────────────────────────────────────────────────┤
244
+ │ │
245
+ │ ┌──────────────────────────────────────────────────────────────────────┐ │
246
+ │ │ │ │
247
+ │ │ 📷 Drag & Drop │ │
248
+ │ │ │ │
249
+ │ │ Drag your file here or click to browse │ │
250
+ │ │ │ │
251
+ │ │ Accepted: ${allowedTypes.join(', ').padEnd(30)} │ │
252
+ │ │ Max size: ${maxSize.padEnd(30)} │ │
253
+ │ │ │ │
254
+ │ └──────────────────────────────────────────────────────────────────────┘ │
255
+ │ │
256
+ │ [Preview will appear here after file selection] │
257
+ │ │
258
+ │ ┌────────────┐ ┌────────┐ │
259
+ │ │ Upload │ │ Cancel │ │
260
+ │ └────────────┘ └────────┘ │
261
+ └────────────────────────────────────────────────────────────────────────────┘
262
+ `.trim();
263
+ }
264
+
265
+ /**
266
+ * Generate confirm dialog wireframe
267
+ */
268
+ function generateConfirmDialogWireframe(spec) {
269
+ const { entity, operation = 'delete' } = spec;
270
+
271
+ return `
272
+ ┌────────────────────────────────────────┐
273
+ │ Confirm ${operation} │
274
+ ├────────────────────────────────────────┤
275
+ │ │
276
+ │ Are you sure you want to ${operation} │
277
+ │ this ${entity}? │
278
+ │ │
279
+ │ This action cannot be undone. │
280
+ │ │
281
+ │ ┌────────────┐ ┌────────┐ │
282
+ │ │ ${operation} │ │ Cancel │ │
283
+ │ └────────────┘ └────────┘ │
284
+ └────────────────────────────────────────┘
285
+ `.trim();
286
+ }
287
+
288
+ /**
289
+ * Generate detail view wireframe
290
+ */
291
+ function generateDetailViewWireframe(spec) {
292
+ const { entity, fields = [] } = spec;
293
+
294
+ const fieldLines = fields.slice(0, 8).map(field => {
295
+ const label = (field.label || field.name).padEnd(20);
296
+ const value = getSampleValue(field.type).padEnd(30);
297
+ return `│ ${label}: ${value} │`;
298
+ }).join('\n');
299
+
300
+ return `
301
+ ┌────────────────────────────────────────────────────────────────────────────┐
302
+ │ ${entity} Details [✏️ Edit] [🗑️ Delete]│
303
+ ├────────────────────────────────────────────────────────────────────────────┤
304
+ │ │
305
+ ${fieldLines}
306
+ │ │
307
+ │ ┌────────┐ │
308
+ │ │ Back │ │
309
+ │ └────────┘ │
310
+ └────────────────────────────────────────────────────────────────────────────┘
311
+ `.trim();
312
+ }
313
+
314
+ /**
315
+ * Generate generic wireframe
316
+ */
317
+ function generateGenericWireframe(spec) {
318
+ const { entity = 'Component', type = 'generic' } = spec;
319
+
320
+ return `
321
+ ┌────────────────────────────────────────────────────────────────────────────┐
322
+ │ ${entity} (${type}) │
323
+ ├────────────────────────────────────────────────────────────────────────────┤
324
+ │ │
325
+ │ [Component content will be defined based on requirements] │
326
+ │ │
327
+ │ - Feature 1 │
328
+ │ - Feature 2 │
329
+ │ - Feature 3 │
330
+ │ │
331
+ └────────────────────────────────────────────────────────────────────────────┘
332
+ `.trim();
333
+ }
334
+
335
+ /**
336
+ * Generate complete UI mockup set
337
+ */
338
+ export function generateCompleteMockupSet(uiNeeds) {
339
+ const mockups = {};
340
+
341
+ for (const need of uiNeeds) {
342
+ const key = `${need.type}-${need.entity || need.operation || 'generic'}`;
343
+ mockups[key] = generateWireframe(need);
344
+ }
345
+
346
+ return mockups;
347
+ }
348
+
349
+ /**
350
+ * Format mockups for markdown output
351
+ */
352
+ export function formatMockupsForMarkdown(mockups) {
353
+ const sections = [];
354
+
355
+ for (const [key, wireframe] of Object.entries(mockups)) {
356
+ const [type, entity] = key.split('-');
357
+ sections.push(`### ${entity} - ${type.toUpperCase()}`);
358
+ sections.push('');
359
+ sections.push('```');
360
+ sections.push(wireframe);
361
+ sections.push('```');
362
+ sections.push('');
363
+ }
364
+
365
+ return sections.join('\n');
366
+ }