@polymorphism-tech/morph-spec 3.0.0 → 3.0.1
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/CLAUDE.md +75 -371
- package/LICENSE +72 -72
- package/bin/detect-agents.js +225 -225
- package/bin/render-template.js +302 -302
- package/bin/semantic-detect-agents.js +246 -246
- package/bin/validate-agents-skills.js +251 -251
- package/bin/validate-agents.js +69 -69
- package/bin/validate-phase.js +263 -263
- package/content/.azure/README.md +293 -293
- package/content/.azure/docs/azure-devops-setup.md +454 -454
- package/content/.azure/docs/branch-strategy.md +398 -398
- package/content/.azure/docs/local-development.md +515 -515
- package/content/.azure/pipelines/pipeline-variables.yml +34 -34
- package/content/.azure/pipelines/prod-pipeline.yml +319 -319
- package/content/.azure/pipelines/staging-pipeline.yml +234 -234
- package/content/.azure/pipelines/templates/build-dotnet.yml +75 -75
- package/content/.azure/pipelines/templates/deploy-app-service.yml +94 -94
- package/content/.azure/pipelines/templates/deploy-container-app.yml +120 -120
- package/content/.azure/pipelines/templates/infra-deploy.yml +90 -90
- package/content/.claude/commands/morph-archive.md +79 -79
- package/content/.claude/commands/morph-deploy.md +529 -529
- package/content/.claude/commands/morph-infra.md +209 -209
- package/content/.claude/commands/morph-preflight.md +227 -227
- package/content/.claude/commands/morph-troubleshoot.md +122 -122
- package/content/.claude/settings.local.json +15 -15
- package/content/.claude/skills/{specialists → level-2-domains/architecture}/prompt-engineer.md +189 -189
- package/content/.claude/skills/{specialists → level-2-domains/architecture}/seo-growth-hacker.md +320 -320
- package/content/.claude/skills/{infra → level-2-domains/infrastructure}/azure-deploy-specialist.md +699 -699
- package/content/.morph/.morphversion +5 -5
- package/content/.morph/archive/.gitkeep +25 -25
- package/content/.morph/config/agents.json +7 -5
- package/content/.morph/docs/STORY-DRIVEN-DEVELOPMENT.md +392 -392
- package/content/.morph/examples/api-nextjs/README.md +241 -241
- package/content/.morph/examples/api-nextjs/contracts.ts +307 -307
- package/content/.morph/examples/api-nextjs/spec.md +399 -399
- package/content/.morph/examples/api-nextjs/tasks.md +168 -168
- package/content/.morph/examples/micro-saas/README.md +125 -125
- package/content/.morph/examples/micro-saas/contracts.cs +358 -358
- package/content/.morph/examples/micro-saas/decisions.md +246 -246
- package/content/.morph/examples/micro-saas/spec.md +236 -236
- package/content/.morph/examples/micro-saas/tasks.md +150 -150
- package/content/.morph/examples/multi-agent/README.md +309 -309
- package/content/.morph/examples/multi-agent/contracts.cs +433 -433
- package/content/.morph/examples/multi-agent/spec.md +479 -479
- package/content/.morph/examples/multi-agent/tasks.md +185 -185
- package/content/.morph/examples/state-v3.json +188 -188
- package/content/.morph/features/.gitkeep +25 -25
- package/content/.morph/hooks/pre-commit-all.sh +48 -48
- package/content/.morph/hooks/pre-commit-specs.sh +49 -49
- package/content/.morph/hooks/pre-commit-tests.sh +60 -60
- package/content/.morph/project.md +160 -160
- package/content/.morph/schemas/agent.schema.json +296 -296
- package/content/.morph/specs/.gitkeep +20 -20
- package/content/.morph/standards/coding.md +377 -377
- package/content/.morph/standards/fluent-ui-setup.md +590 -590
- package/content/.morph/standards/migration-guide.md +514 -514
- package/content/.morph/standards/passkeys-auth.md +423 -423
- package/content/.morph/standards/vector-search-rag.md +536 -536
- package/content/.morph/state.json +17 -17
- package/content/.morph/templates/FluentDesignTheme.cs +149 -149
- package/content/.morph/templates/MudTheme.cs +281 -281
- package/content/.morph/templates/component.razor +239 -239
- package/content/.morph/templates/contracts.cs +217 -217
- package/content/.morph/templates/design-system.css +226 -226
- package/content/.morph/templates/infra/.dockerignore.example +89 -89
- package/content/.morph/templates/infra/Dockerfile.example +82 -82
- package/content/.morph/templates/infra/README.md +286 -286
- package/content/.morph/templates/infra/app-insights.bicep +63 -63
- package/content/.morph/templates/infra/app-service.bicep +164 -164
- package/content/.morph/templates/infra/azure-pipelines-deploy.yml +480 -480
- package/content/.morph/templates/infra/container-app-env.bicep +49 -49
- package/content/.morph/templates/infra/container-app.bicep +156 -156
- package/content/.morph/templates/infra/deploy-checklist.md +426 -426
- package/content/.morph/templates/infra/deploy.ps1 +229 -229
- package/content/.morph/templates/infra/deploy.sh +208 -208
- package/content/.morph/templates/infra/key-vault.bicep +91 -91
- package/content/.morph/templates/infra/main.bicep +189 -189
- package/content/.morph/templates/infra/parameters.dev.json +29 -29
- package/content/.morph/templates/infra/parameters.prod.json +29 -29
- package/content/.morph/templates/infra/parameters.staging.json +29 -29
- package/content/.morph/templates/infra/sql-database.bicep +103 -103
- package/content/.morph/templates/infra/storage.bicep +106 -106
- package/content/.morph/templates/integrations/asaas-client.cs +387 -387
- package/content/.morph/templates/integrations/asaas-webhook.cs +351 -351
- package/content/.morph/templates/integrations/azure-identity-config.cs +288 -288
- package/content/.morph/templates/integrations/clerk-config.cs +258 -258
- package/content/.morph/templates/job.cs +171 -171
- package/content/.morph/templates/migration.cs +83 -83
- package/content/.morph/templates/repository.cs +141 -141
- package/content/.morph/templates/saas/subscription.cs +347 -347
- package/content/.morph/templates/saas/tenant.cs +338 -338
- package/content/.morph/templates/service.cs +139 -139
- package/content/.morph/templates/sprint-status.yaml +68 -68
- package/content/.morph/templates/story.md +143 -143
- package/content/.morph/templates/test.cs +239 -239
- package/content/.morph/templates/ui-design-system.md +286 -286
- package/content/.morph/templates/ui-flows.md +336 -336
- package/content/.morph/templates/ui-mockups.md +133 -133
- package/content/.morph/test-infra/example.bicep +59 -59
- package/content/README.md +79 -79
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-light-webfont.svg +977 -977
- package/docs/api/fonts/Source-Sans-Pro/sourcesanspro-regular-webfont.svg +1048 -1048
- package/docs/api/scripts/collapse.js +38 -38
- package/docs/api/scripts/commonNav.js +28 -28
- package/docs/api/scripts/linenumber.js +25 -25
- package/docs/api/scripts/nav.js +12 -12
- package/docs/api/scripts/polyfill.js +3 -3
- package/docs/api/scripts/prettify/Apache-License-2.0.txt +202 -202
- package/docs/api/scripts/prettify/lang-css.js +2 -2
- package/docs/api/scripts/prettify/prettify.js +28 -28
- package/docs/api/scripts/search.js +98 -98
- package/docs/api/styles/jsdoc.css +776 -776
- package/docs/api/styles/prettify.css +80 -80
- package/docs/examples.md +328 -328
- package/docs/templates.md +418 -418
- package/package.json +1 -2
- package/scripts/postinstall.js +132 -132
- package/scripts/reorganize-skills.cjs +175 -0
- package/scripts/validate-agents-structure.cjs +52 -0
- package/scripts/validate-skills.cjs +180 -0
- package/src/commands/analyze-blazor-concurrency.js +193 -193
- package/src/commands/create-story.js +351 -351
- package/src/commands/deploy.js +780 -780
- package/src/commands/detect-agents.js +9 -0
- package/src/commands/detect.js +104 -104
- package/src/commands/generate.js +149 -149
- package/src/commands/lint-fluent.js +352 -352
- package/src/commands/rollback-phase.js +185 -185
- package/src/commands/session-summary.js +291 -291
- package/src/commands/shard-spec.js +224 -224
- package/src/commands/sprint-status.js +250 -250
- package/src/commands/state.js +334 -333
- package/src/commands/sync.js +167 -167
- package/src/commands/troubleshoot.js +222 -222
- package/src/commands/update.js +13 -1
- package/src/commands/validate-blazor-state.js +210 -210
- package/src/commands/validate-blazor.js +156 -156
- package/src/commands/validate-css.js +84 -84
- package/src/commands/validate-phase.js +221 -221
- package/src/lib/blazor-concurrency-analyzer.js +288 -288
- package/src/lib/blazor-state-validator.js +291 -291
- package/src/lib/blazor-validator.js +374 -374
- package/src/lib/css-validator.js +352 -352
- package/src/lib/design-system-generator.js +298 -298
- package/{detectors → src/lib/detectors}/config-detector.js +223 -223
- package/{detectors → src/lib/detectors}/conversation-analyzer.js +163 -163
- package/{detectors → src/lib/detectors}/index.js +84 -84
- package/{detectors → src/lib/detectors}/standards-generator.js +275 -275
- package/src/lib/learning-system.js +520 -520
- package/src/lib/mockup-generator.js +366 -366
- package/src/lib/state-manager.js +21 -4
- package/src/lib/troubleshoot-grep.js +194 -194
- package/src/lib/troubleshoot-index.js +144 -144
- package/src/lib/ui-detector.js +350 -350
- package/src/lib/validators/architecture-validator.js +387 -387
- package/src/lib/validators/package-validator.js +360 -360
- package/src/lib/validators/ui-contrast-validator.js +422 -422
- package/src/utils/logger.js +32 -32
- package/src/utils/version-checker.js +175 -175
- /package/{detectors → src/lib/detectors}/structure-detector.js +0 -0
|
@@ -38,6 +38,13 @@ function detectAgents(userInput, config) {
|
|
|
38
38
|
for (const agent of allAgents) {
|
|
39
39
|
if (agent.always_active) continue; // Skip core agents
|
|
40
40
|
|
|
41
|
+
// MORPH 3.0.1: Check negative keywords first — if any match, skip agent entirely
|
|
42
|
+
const negativeKeywords = agent.negativeKeywords || [];
|
|
43
|
+
const hasNegativeMatch = negativeKeywords.some(neg =>
|
|
44
|
+
input.includes(neg.toLowerCase())
|
|
45
|
+
);
|
|
46
|
+
if (hasNegativeMatch) continue;
|
|
47
|
+
|
|
41
48
|
// MORPH 3.0: keywords is directly on agent, not nested in autoActivation
|
|
42
49
|
const keywords = agent.keywords || [];
|
|
43
50
|
const matchedKeywords = [];
|
|
@@ -100,6 +107,8 @@ function formatVerbose(result) {
|
|
|
100
107
|
logger.blank();
|
|
101
108
|
}
|
|
102
109
|
|
|
110
|
+
export { detectAgents, loadAgentsConfig };
|
|
111
|
+
|
|
103
112
|
export function detectAgentsCommand(input, options) {
|
|
104
113
|
const userInput = Array.isArray(input) ? input.join(' ') : input;
|
|
105
114
|
|
package/src/commands/detect.js
CHANGED
|
@@ -1,104 +1,104 @@
|
|
|
1
|
-
import { join } from 'path';
|
|
2
|
-
import ora from 'ora';
|
|
3
|
-
import chalk from 'chalk';
|
|
4
|
-
import { logger } from '../utils/logger.js';
|
|
5
|
-
import { detectProject, getDetectionSummary } from '
|
|
6
|
-
import { ensureDir, writeFile } from '../utils/file-copier.js';
|
|
7
|
-
|
|
8
|
-
export async function detectCommand(options) {
|
|
9
|
-
const targetPath = options.path || process.cwd();
|
|
10
|
-
|
|
11
|
-
logger.header('MORPH-SPEC Project Detection');
|
|
12
|
-
logger.dim(`Analyzing: ${targetPath}`);
|
|
13
|
-
logger.blank();
|
|
14
|
-
|
|
15
|
-
const spinner = ora('Detecting project structure...').start();
|
|
16
|
-
|
|
17
|
-
try {
|
|
18
|
-
// Run detection
|
|
19
|
-
const results = await detectProject(targetPath, {
|
|
20
|
-
structure: true,
|
|
21
|
-
config: true,
|
|
22
|
-
conversation: true,
|
|
23
|
-
generateStandards: true
|
|
24
|
-
});
|
|
25
|
-
|
|
26
|
-
spinner.succeed('Detection complete!');
|
|
27
|
-
logger.blank();
|
|
28
|
-
|
|
29
|
-
// Display summary
|
|
30
|
-
logger.header('Detection Results');
|
|
31
|
-
logger.blank();
|
|
32
|
-
|
|
33
|
-
// Stack
|
|
34
|
-
logger.info(`Stack: ${chalk.cyan(results.structure.stack)}`);
|
|
35
|
-
logger.info(`Architecture: ${chalk.cyan(results.structure.architecture)}`);
|
|
36
|
-
if (results.structure.uiLibrary) {
|
|
37
|
-
logger.info(`UI Library: ${chalk.cyan(results.structure.uiLibrary)}`);
|
|
38
|
-
}
|
|
39
|
-
logger.blank();
|
|
40
|
-
|
|
41
|
-
// Technologies
|
|
42
|
-
if (results.config.technologies.length > 0) {
|
|
43
|
-
logger.header('Technologies:');
|
|
44
|
-
results.config.technologies.forEach(tech => {
|
|
45
|
-
logger.dim(` - ${tech}`);
|
|
46
|
-
});
|
|
47
|
-
logger.blank();
|
|
48
|
-
}
|
|
49
|
-
|
|
50
|
-
// Patterns
|
|
51
|
-
if (results.structure.patterns.length > 0) {
|
|
52
|
-
logger.header('Patterns:');
|
|
53
|
-
results.structure.patterns.forEach(pattern => {
|
|
54
|
-
logger.dim(` - ${pattern}`);
|
|
55
|
-
});
|
|
56
|
-
logger.blank();
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
// Recommendations
|
|
60
|
-
if (results.inferred.recommendations.length > 0) {
|
|
61
|
-
logger.header('Recommendations:');
|
|
62
|
-
results.inferred.recommendations.forEach(rec => {
|
|
63
|
-
logger.warn(` ⚠ ${rec}`);
|
|
64
|
-
});
|
|
65
|
-
logger.blank();
|
|
66
|
-
}
|
|
67
|
-
|
|
68
|
-
// Save results if requested
|
|
69
|
-
if (options.save !== false) {
|
|
70
|
-
spinner.start('Saving detection results...');
|
|
71
|
-
|
|
72
|
-
const outputDir = join(targetPath, '.morph', 'project', 'context');
|
|
73
|
-
await ensureDir(outputDir);
|
|
74
|
-
|
|
75
|
-
// Save detection log
|
|
76
|
-
const logPath = join(outputDir, 'detection-log.md');
|
|
77
|
-
const summary = getDetectionSummary(results);
|
|
78
|
-
await writeFile(logPath, summary);
|
|
79
|
-
|
|
80
|
-
// Save inferred standards
|
|
81
|
-
const standardsDir = join(targetPath, '.morph', 'project', 'standards');
|
|
82
|
-
await ensureDir(standardsDir);
|
|
83
|
-
|
|
84
|
-
const standardsPath = join(standardsDir, 'inferred.md');
|
|
85
|
-
await writeFile(standardsPath, results.inferred.markdown);
|
|
86
|
-
|
|
87
|
-
spinner.succeed('Results saved!');
|
|
88
|
-
logger.dim(` - ${logPath}`);
|
|
89
|
-
logger.dim(` - ${standardsPath}`);
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
// Verbose output
|
|
93
|
-
if (options.verbose) {
|
|
94
|
-
logger.blank();
|
|
95
|
-
logger.header('Detailed Results (JSON):');
|
|
96
|
-
console.log(JSON.stringify(results, null, 2));
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
} catch (error) {
|
|
100
|
-
spinner.fail('Detection failed');
|
|
101
|
-
logger.error(error.message);
|
|
102
|
-
process.exit(1);
|
|
103
|
-
}
|
|
104
|
-
}
|
|
1
|
+
import { join } from 'path';
|
|
2
|
+
import ora from 'ora';
|
|
3
|
+
import chalk from 'chalk';
|
|
4
|
+
import { logger } from '../utils/logger.js';
|
|
5
|
+
import { detectProject, getDetectionSummary } from '../lib/detectors/index.js';
|
|
6
|
+
import { ensureDir, writeFile } from '../utils/file-copier.js';
|
|
7
|
+
|
|
8
|
+
export async function detectCommand(options) {
|
|
9
|
+
const targetPath = options.path || process.cwd();
|
|
10
|
+
|
|
11
|
+
logger.header('MORPH-SPEC Project Detection');
|
|
12
|
+
logger.dim(`Analyzing: ${targetPath}`);
|
|
13
|
+
logger.blank();
|
|
14
|
+
|
|
15
|
+
const spinner = ora('Detecting project structure...').start();
|
|
16
|
+
|
|
17
|
+
try {
|
|
18
|
+
// Run detection
|
|
19
|
+
const results = await detectProject(targetPath, {
|
|
20
|
+
structure: true,
|
|
21
|
+
config: true,
|
|
22
|
+
conversation: true,
|
|
23
|
+
generateStandards: true
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
spinner.succeed('Detection complete!');
|
|
27
|
+
logger.blank();
|
|
28
|
+
|
|
29
|
+
// Display summary
|
|
30
|
+
logger.header('Detection Results');
|
|
31
|
+
logger.blank();
|
|
32
|
+
|
|
33
|
+
// Stack
|
|
34
|
+
logger.info(`Stack: ${chalk.cyan(results.structure.stack)}`);
|
|
35
|
+
logger.info(`Architecture: ${chalk.cyan(results.structure.architecture)}`);
|
|
36
|
+
if (results.structure.uiLibrary) {
|
|
37
|
+
logger.info(`UI Library: ${chalk.cyan(results.structure.uiLibrary)}`);
|
|
38
|
+
}
|
|
39
|
+
logger.blank();
|
|
40
|
+
|
|
41
|
+
// Technologies
|
|
42
|
+
if (results.config.technologies.length > 0) {
|
|
43
|
+
logger.header('Technologies:');
|
|
44
|
+
results.config.technologies.forEach(tech => {
|
|
45
|
+
logger.dim(` - ${tech}`);
|
|
46
|
+
});
|
|
47
|
+
logger.blank();
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
// Patterns
|
|
51
|
+
if (results.structure.patterns.length > 0) {
|
|
52
|
+
logger.header('Patterns:');
|
|
53
|
+
results.structure.patterns.forEach(pattern => {
|
|
54
|
+
logger.dim(` - ${pattern}`);
|
|
55
|
+
});
|
|
56
|
+
logger.blank();
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
// Recommendations
|
|
60
|
+
if (results.inferred.recommendations.length > 0) {
|
|
61
|
+
logger.header('Recommendations:');
|
|
62
|
+
results.inferred.recommendations.forEach(rec => {
|
|
63
|
+
logger.warn(` ⚠ ${rec}`);
|
|
64
|
+
});
|
|
65
|
+
logger.blank();
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
// Save results if requested
|
|
69
|
+
if (options.save !== false) {
|
|
70
|
+
spinner.start('Saving detection results...');
|
|
71
|
+
|
|
72
|
+
const outputDir = join(targetPath, '.morph', 'project', 'context');
|
|
73
|
+
await ensureDir(outputDir);
|
|
74
|
+
|
|
75
|
+
// Save detection log
|
|
76
|
+
const logPath = join(outputDir, 'detection-log.md');
|
|
77
|
+
const summary = getDetectionSummary(results);
|
|
78
|
+
await writeFile(logPath, summary);
|
|
79
|
+
|
|
80
|
+
// Save inferred standards
|
|
81
|
+
const standardsDir = join(targetPath, '.morph', 'project', 'standards');
|
|
82
|
+
await ensureDir(standardsDir);
|
|
83
|
+
|
|
84
|
+
const standardsPath = join(standardsDir, 'inferred.md');
|
|
85
|
+
await writeFile(standardsPath, results.inferred.markdown);
|
|
86
|
+
|
|
87
|
+
spinner.succeed('Results saved!');
|
|
88
|
+
logger.dim(` - ${logPath}`);
|
|
89
|
+
logger.dim(` - ${standardsPath}`);
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
// Verbose output
|
|
93
|
+
if (options.verbose) {
|
|
94
|
+
logger.blank();
|
|
95
|
+
logger.header('Detailed Results (JSON):');
|
|
96
|
+
console.log(JSON.stringify(results, null, 2));
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
} catch (error) {
|
|
100
|
+
spinner.fail('Detection failed');
|
|
101
|
+
logger.error(error.message);
|
|
102
|
+
process.exit(1);
|
|
103
|
+
}
|
|
104
|
+
}
|
package/src/commands/generate.js
CHANGED
|
@@ -1,149 +1,149 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* MORPH-SPEC Generate Command
|
|
3
|
-
* CLI wrapper for code generation operations
|
|
4
|
-
*/
|
|
5
|
-
|
|
6
|
-
import { mkdirSync, writeFileSync } from 'fs';
|
|
7
|
-
import { dirname } from 'path';
|
|
8
|
-
import ora from 'ora';
|
|
9
|
-
import chalk from 'chalk';
|
|
10
|
-
import { logger } from '../utils/logger.js';
|
|
11
|
-
import * as DesignSystemGenerator from '../lib/design-system-generator.js';
|
|
12
|
-
|
|
13
|
-
// ============================================================================
|
|
14
|
-
// Design System Subcommand
|
|
15
|
-
// ============================================================================
|
|
16
|
-
|
|
17
|
-
/**
|
|
18
|
-
* Generate design system files (CSS + themes) from ui-design-system.md
|
|
19
|
-
* @param {string} designSystemPath - Path to ui-design-system.md
|
|
20
|
-
* @param {Object} options - CLI options
|
|
21
|
-
*/
|
|
22
|
-
export async function generateDesignSystemCommand(designSystemPath, options) {
|
|
23
|
-
if (!designSystemPath) {
|
|
24
|
-
logger.error('Design system file path required');
|
|
25
|
-
logger.dim(' Usage: morph-spec generate design-system <ui-design-system.md>');
|
|
26
|
-
logger.blank();
|
|
27
|
-
logger.dim(' Example:');
|
|
28
|
-
logger.dim(' morph-spec generate design-system .morph/project/outputs/my-feature/ui-design-system.md');
|
|
29
|
-
process.exit(1);
|
|
30
|
-
}
|
|
31
|
-
|
|
32
|
-
logger.header('MORPH-SPEC Design System Generator');
|
|
33
|
-
logger.blank();
|
|
34
|
-
|
|
35
|
-
const spinner = ora('Parsing design system...').start();
|
|
36
|
-
|
|
37
|
-
try {
|
|
38
|
-
// Determine mode
|
|
39
|
-
let mode = 'both';
|
|
40
|
-
if (options.fluent && !options.mud) mode = 'fluent';
|
|
41
|
-
if (options.mud && !options.fluent) mode = 'mud';
|
|
42
|
-
|
|
43
|
-
// Generate design system
|
|
44
|
-
const generated = DesignSystemGenerator.generateDesignSystem(designSystemPath, {
|
|
45
|
-
mode,
|
|
46
|
-
namespace: options.namespace || 'YourProject.Themes'
|
|
47
|
-
});
|
|
48
|
-
|
|
49
|
-
spinner.succeed('Design system parsed!');
|
|
50
|
-
logger.blank();
|
|
51
|
-
|
|
52
|
-
// Display stats
|
|
53
|
-
logger.header('Parsed Design System:');
|
|
54
|
-
logger.info(`Primary Colors: ${chalk.cyan(generated.stats.primaryColors)}`);
|
|
55
|
-
logger.info(`Neutral Colors: ${chalk.cyan(generated.stats.neutralColors)}`);
|
|
56
|
-
logger.info(`Semantic Colors: ${chalk.cyan(generated.stats.semanticColors)}`);
|
|
57
|
-
logger.info(`Font Sizes: ${chalk.cyan(generated.stats.fontSizes)}`);
|
|
58
|
-
logger.info(`Spacing Values: ${chalk.cyan(generated.stats.spacingValues)}`);
|
|
59
|
-
logger.blank();
|
|
60
|
-
|
|
61
|
-
if (options.dryRun) {
|
|
62
|
-
logger.warn('Dry run - files not written');
|
|
63
|
-
logger.blank();
|
|
64
|
-
logger.header('Would generate:');
|
|
65
|
-
logger.dim(' - wwwroot/css/design-system.css');
|
|
66
|
-
if (generated.fluentTheme) logger.dim(' - Themes/FluentDesignTheme.cs');
|
|
67
|
-
if (generated.mudTheme) logger.dim(' - Themes/MudDesignTheme.cs');
|
|
68
|
-
logger.blank();
|
|
69
|
-
return;
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
// Write files
|
|
73
|
-
const writeSpinner = ora('Writing files...').start();
|
|
74
|
-
|
|
75
|
-
// Write CSS
|
|
76
|
-
const cssPath = options.cssOutput || 'wwwroot/css/design-system.css';
|
|
77
|
-
mkdirSync(dirname(cssPath), { recursive: true });
|
|
78
|
-
writeFileSync(cssPath, generated.css);
|
|
79
|
-
writeSpinner.text = `Created ${cssPath}`;
|
|
80
|
-
|
|
81
|
-
// Write Fluent theme
|
|
82
|
-
if (generated.fluentTheme) {
|
|
83
|
-
const fluentPath = options.fluentOutput || 'Themes/FluentDesignTheme.cs';
|
|
84
|
-
mkdirSync(dirname(fluentPath), { recursive: true });
|
|
85
|
-
writeFileSync(fluentPath, generated.fluentTheme);
|
|
86
|
-
writeSpinner.text = `Created ${fluentPath}`;
|
|
87
|
-
}
|
|
88
|
-
|
|
89
|
-
// Write MudBlazor theme
|
|
90
|
-
if (generated.mudTheme) {
|
|
91
|
-
const mudPath = options.mudOutput || 'Themes/MudDesignTheme.cs';
|
|
92
|
-
mkdirSync(dirname(mudPath), { recursive: true });
|
|
93
|
-
writeFileSync(mudPath, generated.mudTheme);
|
|
94
|
-
writeSpinner.text = `Created ${mudPath}`;
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
writeSpinner.succeed('Files generated!');
|
|
98
|
-
logger.blank();
|
|
99
|
-
|
|
100
|
-
// List generated files
|
|
101
|
-
logger.header('Generated Files:');
|
|
102
|
-
logger.success(` ✓ ${cssPath}`);
|
|
103
|
-
if (generated.fluentTheme) logger.success(` ✓ ${options.fluentOutput || 'Themes/FluentDesignTheme.cs'}`);
|
|
104
|
-
if (generated.mudTheme) logger.success(` ✓ ${options.mudOutput || 'Themes/MudDesignTheme.cs'}`);
|
|
105
|
-
logger.blank();
|
|
106
|
-
|
|
107
|
-
// Next steps
|
|
108
|
-
logger.header('Next Steps:');
|
|
109
|
-
logger.dim(' 1. Reference design-system.css in your layout');
|
|
110
|
-
logger.dim(' 2. Register theme in Program.cs');
|
|
111
|
-
logger.dim(' 3. Apply theme to your components');
|
|
112
|
-
logger.blank();
|
|
113
|
-
|
|
114
|
-
} catch (error) {
|
|
115
|
-
spinner.fail('Generation failed');
|
|
116
|
-
logger.error(error.message);
|
|
117
|
-
process.exit(1);
|
|
118
|
-
}
|
|
119
|
-
}
|
|
120
|
-
|
|
121
|
-
// ============================================================================
|
|
122
|
-
// Main Generate Command (router for future subcommands)
|
|
123
|
-
// ============================================================================
|
|
124
|
-
|
|
125
|
-
/**
|
|
126
|
-
* Main generate command router
|
|
127
|
-
* Future: Could add more generation commands here
|
|
128
|
-
*/
|
|
129
|
-
export async function generateCommand(subcommand, args, options) {
|
|
130
|
-
switch (subcommand) {
|
|
131
|
-
case 'design-system':
|
|
132
|
-
await generateDesignSystemCommand(args[0], options);
|
|
133
|
-
break;
|
|
134
|
-
|
|
135
|
-
// Future: Add more generation commands
|
|
136
|
-
// case 'component':
|
|
137
|
-
// await generateComponentCommand(args[0], options);
|
|
138
|
-
// break;
|
|
139
|
-
|
|
140
|
-
default:
|
|
141
|
-
logger.error(`Unknown subcommand: ${subcommand}`);
|
|
142
|
-
logger.blank();
|
|
143
|
-
logger.info('Available subcommands:');
|
|
144
|
-
logger.dim(' design-system Generate CSS + theme files from ui-design-system.md');
|
|
145
|
-
logger.blank();
|
|
146
|
-
logger.dim('Run "morph-spec generate --help" for more information');
|
|
147
|
-
process.exit(1);
|
|
148
|
-
}
|
|
149
|
-
}
|
|
1
|
+
/**
|
|
2
|
+
* MORPH-SPEC Generate Command
|
|
3
|
+
* CLI wrapper for code generation operations
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { mkdirSync, writeFileSync } from 'fs';
|
|
7
|
+
import { dirname } from 'path';
|
|
8
|
+
import ora from 'ora';
|
|
9
|
+
import chalk from 'chalk';
|
|
10
|
+
import { logger } from '../utils/logger.js';
|
|
11
|
+
import * as DesignSystemGenerator from '../lib/design-system-generator.js';
|
|
12
|
+
|
|
13
|
+
// ============================================================================
|
|
14
|
+
// Design System Subcommand
|
|
15
|
+
// ============================================================================
|
|
16
|
+
|
|
17
|
+
/**
|
|
18
|
+
* Generate design system files (CSS + themes) from ui-design-system.md
|
|
19
|
+
* @param {string} designSystemPath - Path to ui-design-system.md
|
|
20
|
+
* @param {Object} options - CLI options
|
|
21
|
+
*/
|
|
22
|
+
export async function generateDesignSystemCommand(designSystemPath, options) {
|
|
23
|
+
if (!designSystemPath) {
|
|
24
|
+
logger.error('Design system file path required');
|
|
25
|
+
logger.dim(' Usage: morph-spec generate design-system <ui-design-system.md>');
|
|
26
|
+
logger.blank();
|
|
27
|
+
logger.dim(' Example:');
|
|
28
|
+
logger.dim(' morph-spec generate design-system .morph/project/outputs/my-feature/ui-design-system.md');
|
|
29
|
+
process.exit(1);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
logger.header('MORPH-SPEC Design System Generator');
|
|
33
|
+
logger.blank();
|
|
34
|
+
|
|
35
|
+
const spinner = ora('Parsing design system...').start();
|
|
36
|
+
|
|
37
|
+
try {
|
|
38
|
+
// Determine mode
|
|
39
|
+
let mode = 'both';
|
|
40
|
+
if (options.fluent && !options.mud) mode = 'fluent';
|
|
41
|
+
if (options.mud && !options.fluent) mode = 'mud';
|
|
42
|
+
|
|
43
|
+
// Generate design system
|
|
44
|
+
const generated = DesignSystemGenerator.generateDesignSystem(designSystemPath, {
|
|
45
|
+
mode,
|
|
46
|
+
namespace: options.namespace || 'YourProject.Themes'
|
|
47
|
+
});
|
|
48
|
+
|
|
49
|
+
spinner.succeed('Design system parsed!');
|
|
50
|
+
logger.blank();
|
|
51
|
+
|
|
52
|
+
// Display stats
|
|
53
|
+
logger.header('Parsed Design System:');
|
|
54
|
+
logger.info(`Primary Colors: ${chalk.cyan(generated.stats.primaryColors)}`);
|
|
55
|
+
logger.info(`Neutral Colors: ${chalk.cyan(generated.stats.neutralColors)}`);
|
|
56
|
+
logger.info(`Semantic Colors: ${chalk.cyan(generated.stats.semanticColors)}`);
|
|
57
|
+
logger.info(`Font Sizes: ${chalk.cyan(generated.stats.fontSizes)}`);
|
|
58
|
+
logger.info(`Spacing Values: ${chalk.cyan(generated.stats.spacingValues)}`);
|
|
59
|
+
logger.blank();
|
|
60
|
+
|
|
61
|
+
if (options.dryRun) {
|
|
62
|
+
logger.warn('Dry run - files not written');
|
|
63
|
+
logger.blank();
|
|
64
|
+
logger.header('Would generate:');
|
|
65
|
+
logger.dim(' - wwwroot/css/design-system.css');
|
|
66
|
+
if (generated.fluentTheme) logger.dim(' - Themes/FluentDesignTheme.cs');
|
|
67
|
+
if (generated.mudTheme) logger.dim(' - Themes/MudDesignTheme.cs');
|
|
68
|
+
logger.blank();
|
|
69
|
+
return;
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
// Write files
|
|
73
|
+
const writeSpinner = ora('Writing files...').start();
|
|
74
|
+
|
|
75
|
+
// Write CSS
|
|
76
|
+
const cssPath = options.cssOutput || 'wwwroot/css/design-system.css';
|
|
77
|
+
mkdirSync(dirname(cssPath), { recursive: true });
|
|
78
|
+
writeFileSync(cssPath, generated.css);
|
|
79
|
+
writeSpinner.text = `Created ${cssPath}`;
|
|
80
|
+
|
|
81
|
+
// Write Fluent theme
|
|
82
|
+
if (generated.fluentTheme) {
|
|
83
|
+
const fluentPath = options.fluentOutput || 'Themes/FluentDesignTheme.cs';
|
|
84
|
+
mkdirSync(dirname(fluentPath), { recursive: true });
|
|
85
|
+
writeFileSync(fluentPath, generated.fluentTheme);
|
|
86
|
+
writeSpinner.text = `Created ${fluentPath}`;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
// Write MudBlazor theme
|
|
90
|
+
if (generated.mudTheme) {
|
|
91
|
+
const mudPath = options.mudOutput || 'Themes/MudDesignTheme.cs';
|
|
92
|
+
mkdirSync(dirname(mudPath), { recursive: true });
|
|
93
|
+
writeFileSync(mudPath, generated.mudTheme);
|
|
94
|
+
writeSpinner.text = `Created ${mudPath}`;
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
writeSpinner.succeed('Files generated!');
|
|
98
|
+
logger.blank();
|
|
99
|
+
|
|
100
|
+
// List generated files
|
|
101
|
+
logger.header('Generated Files:');
|
|
102
|
+
logger.success(` ✓ ${cssPath}`);
|
|
103
|
+
if (generated.fluentTheme) logger.success(` ✓ ${options.fluentOutput || 'Themes/FluentDesignTheme.cs'}`);
|
|
104
|
+
if (generated.mudTheme) logger.success(` ✓ ${options.mudOutput || 'Themes/MudDesignTheme.cs'}`);
|
|
105
|
+
logger.blank();
|
|
106
|
+
|
|
107
|
+
// Next steps
|
|
108
|
+
logger.header('Next Steps:');
|
|
109
|
+
logger.dim(' 1. Reference design-system.css in your layout');
|
|
110
|
+
logger.dim(' 2. Register theme in Program.cs');
|
|
111
|
+
logger.dim(' 3. Apply theme to your components');
|
|
112
|
+
logger.blank();
|
|
113
|
+
|
|
114
|
+
} catch (error) {
|
|
115
|
+
spinner.fail('Generation failed');
|
|
116
|
+
logger.error(error.message);
|
|
117
|
+
process.exit(1);
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// ============================================================================
|
|
122
|
+
// Main Generate Command (router for future subcommands)
|
|
123
|
+
// ============================================================================
|
|
124
|
+
|
|
125
|
+
/**
|
|
126
|
+
* Main generate command router
|
|
127
|
+
* Future: Could add more generation commands here
|
|
128
|
+
*/
|
|
129
|
+
export async function generateCommand(subcommand, args, options) {
|
|
130
|
+
switch (subcommand) {
|
|
131
|
+
case 'design-system':
|
|
132
|
+
await generateDesignSystemCommand(args[0], options);
|
|
133
|
+
break;
|
|
134
|
+
|
|
135
|
+
// Future: Add more generation commands
|
|
136
|
+
// case 'component':
|
|
137
|
+
// await generateComponentCommand(args[0], options);
|
|
138
|
+
// break;
|
|
139
|
+
|
|
140
|
+
default:
|
|
141
|
+
logger.error(`Unknown subcommand: ${subcommand}`);
|
|
142
|
+
logger.blank();
|
|
143
|
+
logger.info('Available subcommands:');
|
|
144
|
+
logger.dim(' design-system Generate CSS + theme files from ui-design-system.md');
|
|
145
|
+
logger.blank();
|
|
146
|
+
logger.dim('Run "morph-spec generate --help" for more information');
|
|
147
|
+
process.exit(1);
|
|
148
|
+
}
|
|
149
|
+
}
|