create-hq 5.3.2 → 6.0.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.
- package/dist/deps.d.ts +2 -2
- package/dist/deps.d.ts.map +1 -1
- package/dist/deps.js +29 -138
- package/dist/deps.js.map +1 -1
- package/dist/git.d.ts.map +1 -1
- package/dist/git.js +5 -0
- package/dist/git.js.map +1 -1
- package/dist/index.js +6 -3
- package/dist/index.js.map +1 -1
- package/dist/scaffold.d.ts +2 -1
- package/dist/scaffold.d.ts.map +1 -1
- package/dist/scaffold.js +238 -54
- package/dist/scaffold.js.map +1 -1
- package/dist/ui.d.ts +2 -0
- package/dist/ui.d.ts.map +1 -1
- package/dist/ui.js +31 -1
- package/dist/ui.js.map +1 -1
- package/package.json +6 -3
- package/template/.claude/CLAUDE.md +202 -0
- package/template/.claude/commands/checkpoint.md +127 -0
- package/template/.claude/commands/cleanup.md +307 -0
- package/template/.claude/commands/execute-task.md +440 -0
- package/template/.claude/commands/exit-plan.md +41 -0
- package/template/.claude/commands/handoff.md +97 -0
- package/template/.claude/commands/learn.md +218 -0
- package/template/.claude/commands/metrics.md +118 -0
- package/template/.claude/commands/newworker.md +162 -0
- package/template/.claude/commands/nexttask.md +67 -0
- package/template/.claude/commands/prd.md +238 -0
- package/template/.claude/commands/reanchor.md +51 -0
- package/template/.claude/commands/remember.md +126 -0
- package/template/.claude/commands/run-project.md +348 -0
- package/template/.claude/commands/run.md +110 -0
- package/template/.claude/commands/search-reindex.md +62 -0
- package/template/.claude/commands/search.md +100 -0
- package/template/.claude/commands/setup.md +381 -0
- package/template/.claude/scripts/pure-ralph-loop.ps1 +312 -0
- package/template/.claude/scripts/pure-ralph-loop.sh +859 -0
- package/template/CHANGELOG.md +220 -0
- package/template/LICENSE +21 -0
- package/template/MIGRATION.md +259 -0
- package/template/README.md +368 -0
- package/template/data/journal/.gitkeep +0 -0
- package/template/docs/images/ascii-banner-options.md +122 -0
- package/template/docs/images/hq-banner.svg +105 -0
- package/template/knowledge/Ralph/01-overview.md +71 -0
- package/template/knowledge/Ralph/02-core-concepts.md +114 -0
- package/template/knowledge/Ralph/03-how-ralph-works.md +184 -0
- package/template/knowledge/Ralph/04-back-pressure.md +222 -0
- package/template/knowledge/Ralph/05-specifications.md +210 -0
- package/template/knowledge/Ralph/06-agents-md.md +222 -0
- package/template/knowledge/Ralph/07-implementation.md +316 -0
- package/template/knowledge/Ralph/08-economics.md +182 -0
- package/template/knowledge/Ralph/09-resources.md +145 -0
- package/template/knowledge/Ralph/10-claude-code-workflow.md +212 -0
- package/template/knowledge/Ralph/11-team-training-guide.md +383 -0
- package/template/knowledge/Ralph/README.md +40 -0
- package/template/knowledge/ai-security-framework/CONTRIBUTING.md +139 -0
- package/template/knowledge/ai-security-framework/GLOSSARY.md +176 -0
- package/template/knowledge/ai-security-framework/LICENSE +21 -0
- package/template/knowledge/ai-security-framework/QUICK-START.md +172 -0
- package/template/knowledge/ai-security-framework/README.md +232 -0
- package/template/knowledge/ai-security-framework/checklists/browser-security.md +301 -0
- package/template/knowledge/ai-security-framework/checklists/credential-isolation.md +322 -0
- package/template/knowledge/ai-security-framework/checklists/incident-response.md +288 -0
- package/template/knowledge/ai-security-framework/checklists/pre-flight.md +249 -0
- package/template/knowledge/ai-security-framework/checklists/weekly-audit.md +159 -0
- package/template/knowledge/ai-security-framework/configs/audit-logging.md +372 -0
- package/template/knowledge/ai-security-framework/configs/kill-switches.md +354 -0
- package/template/knowledge/ai-security-framework/docs/01-core-principles.md +256 -0
- package/template/knowledge/ai-security-framework/docs/02-threat-landscape.md +326 -0
- package/template/knowledge/ai-security-framework/docs/03-security-posture.md +250 -0
- package/template/knowledge/ai-security-framework/templates/agents-security.md +233 -0
- package/template/knowledge/design-styles/README.md +42 -0
- package/template/knowledge/design-styles/american-industrial.md +136 -0
- package/template/knowledge/design-styles/ethereal-abstract.md +133 -0
- package/template/knowledge/design-styles/liminal-portal.md +111 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G-3m4YPW0AADdu2.jpeg +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G-JJlt5WwAABK3K.png +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G-JJmj5W0AEbJ-7.png +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G59fgNuXkAAKLJQ (1).jpeg +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G59fgNuXkAAKLJQ.jpeg +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G7fVkn3WEAAM-ST.jpeg +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G8ECO5JWEAIksyn.png +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G9-3GQSWoAA8eqZ.png +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G9xEOqrXkAEZRcs.png +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G_MVeJrXQAA8sx4.jpeg +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/G_RSkmGXkAAgAVZ.png +0 -0
- package/template/knowledge/design-styles/swipes/american-industrial/README.md +31 -0
- package/template/knowledge/design-styles/swipes/american-industrial/qyqtg7Dq.png +0 -0
- package/template/knowledge/dev-team/README.md +35 -0
- package/template/knowledge/dev-team/patterns/README.md +34 -0
- package/template/knowledge/dev-team/patterns/frontend/react-best-practices.md +178 -0
- package/template/knowledge/dev-team/troubleshooting/README.md +31 -0
- package/template/knowledge/dev-team/workflows/README.md +49 -0
- package/template/knowledge/hq/checkpoint-schema.json +51 -0
- package/template/knowledge/hq/index-md-spec.md +74 -0
- package/template/knowledge/hq/thread-schema.md +153 -0
- package/template/knowledge/hq-core/checkpoint-schema.json +51 -0
- package/template/knowledge/hq-core/index-md-spec.md +74 -0
- package/template/knowledge/hq-core/thread-schema.md +153 -0
- package/template/knowledge/loom/README.md +51 -0
- package/template/knowledge/loom/architecture.md +125 -0
- package/template/knowledge/loom/code-style.md +169 -0
- package/template/knowledge/loom/llm-proxy.md +132 -0
- package/template/knowledge/loom/state-machine.md +131 -0
- package/template/knowledge/loom/thread-system.md +117 -0
- package/template/knowledge/loom/tools.md +94 -0
- package/template/knowledge/loom/weaver.md +96 -0
- package/template/knowledge/loom/web-frontend.md +131 -0
- package/template/knowledge/projects/README.md +72 -0
- package/template/knowledge/projects/templates/README.template.md +28 -0
- package/template/knowledge/workers/README.md +195 -0
- package/template/knowledge/workers/ralph-loop-pattern.md +157 -0
- package/template/knowledge/workers/skill-schema.md +182 -0
- package/template/knowledge/workers/state-machine.md +102 -0
- package/template/knowledge/workers/templates/base-worker.yaml +73 -0
- package/template/knowledge/workers/templates/code-worker.yaml +85 -0
- package/template/knowledge/workers/templates/skill.yaml +49 -0
- package/template/knowledge/workers/templates/social-worker.yaml +70 -0
- package/template/modules/examples/full-manifest.yaml +92 -0
- package/template/modules/examples/minimal.yaml +14 -0
- package/template/modules/modules.yaml +59 -0
- package/template/projects/.gitkeep +0 -0
- package/template/projects/incorporate-workers-into-pure-ralph/prd.json +88 -0
- package/template/projects/pure-ralph-branch-isolation/README.md +114 -0
- package/template/projects/pure-ralph-branch-isolation/prd.json +123 -0
- package/template/projects/purist-ralph-loop/README.md +148 -0
- package/template/projects/purist-ralph-loop/prd.json +135 -0
- package/template/projects/ralph-test/prd.json +50 -0
- package/template/prompts/pure-ralph-base.md +551 -0
- package/template/settings/.gitkeep +0 -0
- package/template/settings/pure-ralph.json +42 -0
- package/template/social-content/drafts/INDEX.md +21 -0
- package/template/social-content/drafts/linkedin/.gitkeep +1 -0
- package/template/social-content/drafts/x/.gitkeep +1 -0
- package/template/social-content/images/.gitkeep +1 -0
- package/template/starter-projects/code-worker/README.md +97 -0
- package/template/starter-projects/code-worker/prd.json +45 -0
- package/template/starter-projects/personal-assistant/README.md +42 -0
- package/template/starter-projects/personal-assistant/prd.json +43 -0
- package/template/starter-projects/social-media/README.md +60 -0
- package/template/starter-projects/social-media/prd.json +43 -0
- package/template/workers/content-brand/README.md +59 -0
- package/template/workers/content-brand/skills/messaging-alignment.md +91 -0
- package/template/workers/content-brand/skills/tone-check.md +76 -0
- package/template/workers/content-brand/skills/voice-analysis.md +68 -0
- package/template/workers/content-brand/worker.yaml +81 -0
- package/template/workers/content-legal/README.md +80 -0
- package/template/workers/content-legal/skills/claim-substantiation.md +150 -0
- package/template/workers/content-legal/skills/compliance-scan.md +123 -0
- package/template/workers/content-legal/skills/disclaimer-check.md +146 -0
- package/template/workers/content-legal/worker.yaml +118 -0
- package/template/workers/content-product/README.md +77 -0
- package/template/workers/content-product/skills/claim-verification.md +96 -0
- package/template/workers/content-product/skills/feature-accuracy.md +117 -0
- package/template/workers/content-product/skills/stats-check.md +128 -0
- package/template/workers/content-product/worker.yaml +97 -0
- package/template/workers/content-sales/README.md +70 -0
- package/template/workers/content-sales/skills/conversion-analysis.md +96 -0
- package/template/workers/content-sales/skills/cta-audit.md +107 -0
- package/template/workers/content-sales/skills/value-prop-check.md +114 -0
- package/template/workers/content-sales/worker.yaml +93 -0
- package/template/workers/content-shared/cli.ts +242 -0
- package/template/workers/content-shared/index.ts +234 -0
- package/template/workers/content-shared/lib/accuracy-analyzer.ts +661 -0
- package/template/workers/content-shared/lib/analyze.ts +370 -0
- package/template/workers/content-shared/lib/brand-analyzer.ts +526 -0
- package/template/workers/content-shared/lib/cms-integration.ts +446 -0
- package/template/workers/content-shared/lib/compliance-analyzer.ts +655 -0
- package/template/workers/content-shared/lib/conversion-analyzer.ts +555 -0
- package/template/workers/content-shared/lib/github-integration.ts +582 -0
- package/template/workers/content-shared/lib/output.ts +373 -0
- package/template/workers/content-shared/lib/parser.ts +771 -0
- package/template/workers/content-shared/lib/priority.ts +439 -0
- package/template/workers/content-shared/lib/recommendations.ts +512 -0
- package/template/workers/content-shared/lib/reporter.ts +749 -0
- package/template/workers/content-shared/lib/restructure.ts +664 -0
- package/template/workers/content-shared/lib/scorer.ts +140 -0
- package/template/workers/content-shared/lib/types.ts +227 -0
- package/template/workers/content-shared/lib/variants.ts +595 -0
- package/template/workers/content-shared/package.json +51 -0
- package/template/workers/content-shared/pnpm-lock.yaml +39 -0
- package/template/workers/content-shared/test/sample-page.json +115 -0
- package/template/workers/content-shared/tsconfig.json +20 -0
- package/template/workers/dev-team/README.md +166 -0
- package/template/workers/dev-team/_template.yaml +70 -0
- package/template/workers/dev-team/architect/package.json +27 -0
- package/template/workers/dev-team/architect/skills/api-design.md +89 -0
- package/template/workers/dev-team/architect/skills/refactor-plan.md +96 -0
- package/template/workers/dev-team/architect/skills/system-design.md +100 -0
- package/template/workers/dev-team/architect/src/index.ts +49 -0
- package/template/workers/dev-team/architect/src/mcp-server.ts +122 -0
- package/template/workers/dev-team/architect/src/skills/api-design.ts +316 -0
- package/template/workers/dev-team/architect/src/skills/refactor-plan.ts +264 -0
- package/template/workers/dev-team/architect/src/skills/system-design.ts +212 -0
- package/template/workers/dev-team/architect/tsconfig.json +19 -0
- package/template/workers/dev-team/architect/worker.yaml +128 -0
- package/template/workers/dev-team/backend-dev/package-lock.json +1252 -0
- package/template/workers/dev-team/backend-dev/package.json +27 -0
- package/template/workers/dev-team/backend-dev/skills/implement-endpoint.md +70 -0
- package/template/workers/dev-team/backend-dev/skills/implement-service.md +62 -0
- package/template/workers/dev-team/backend-dev/src/index.ts +51 -0
- package/template/workers/dev-team/backend-dev/src/mcp-server.ts +109 -0
- package/template/workers/dev-team/backend-dev/src/skills/implement-endpoint.ts +122 -0
- package/template/workers/dev-team/backend-dev/src/skills/implement-service.ts +126 -0
- package/template/workers/dev-team/backend-dev/tsconfig.json +19 -0
- package/template/workers/dev-team/backend-dev/worker.yaml +128 -0
- package/template/workers/dev-team/code-reviewer/package-lock.json +1080 -0
- package/template/workers/dev-team/code-reviewer/package.json +24 -0
- package/template/workers/dev-team/code-reviewer/skills/merge-to-production.md +61 -0
- package/template/workers/dev-team/code-reviewer/skills/merge-to-staging.md +54 -0
- package/template/workers/dev-team/code-reviewer/skills/request-changes.md +63 -0
- package/template/workers/dev-team/code-reviewer/skills/review-pr.md +77 -0
- package/template/workers/dev-team/code-reviewer/src/index.ts +56 -0
- package/template/workers/dev-team/code-reviewer/src/mcp-server.ts +101 -0
- package/template/workers/dev-team/code-reviewer/tsconfig.json +19 -0
- package/template/workers/dev-team/code-reviewer/worker.yaml +90 -0
- package/template/workers/dev-team/database-dev/package.json +22 -0
- package/template/workers/dev-team/database-dev/skills/create-schema.md +48 -0
- package/template/workers/dev-team/database-dev/src/index.ts +50 -0
- package/template/workers/dev-team/database-dev/src/mcp-server.ts +76 -0
- package/template/workers/dev-team/database-dev/tsconfig.json +18 -0
- package/template/workers/dev-team/database-dev/worker.yaml +90 -0
- package/template/workers/dev-team/frontend-dev/package.json +22 -0
- package/template/workers/dev-team/frontend-dev/skills/create-component.md +26 -0
- package/template/workers/dev-team/frontend-dev/src/index.ts +50 -0
- package/template/workers/dev-team/frontend-dev/src/mcp-server.ts +77 -0
- package/template/workers/dev-team/frontend-dev/tsconfig.json +18 -0
- package/template/workers/dev-team/frontend-dev/worker.yaml +132 -0
- package/template/workers/dev-team/infra-dev/package.json +24 -0
- package/template/workers/dev-team/infra-dev/skills/add-monitoring.md +73 -0
- package/template/workers/dev-team/infra-dev/skills/configure-deployment.md +80 -0
- package/template/workers/dev-team/infra-dev/skills/create-dockerfile.md +62 -0
- package/template/workers/dev-team/infra-dev/skills/setup-cicd.md +63 -0
- package/template/workers/dev-team/infra-dev/src/index.ts +55 -0
- package/template/workers/dev-team/infra-dev/src/mcp-server.ts +82 -0
- package/template/workers/dev-team/infra-dev/tsconfig.json +19 -0
- package/template/workers/dev-team/infra-dev/worker.yaml +92 -0
- package/template/workers/dev-team/knowledge-curator/package.json +24 -0
- package/template/workers/dev-team/knowledge-curator/skills/curate-troubleshooting.md +63 -0
- package/template/workers/dev-team/knowledge-curator/skills/process-learnings.md +61 -0
- package/template/workers/dev-team/knowledge-curator/skills/sync-documentation.md +76 -0
- package/template/workers/dev-team/knowledge-curator/skills/update-patterns.md +63 -0
- package/template/workers/dev-team/knowledge-curator/src/index.ts +53 -0
- package/template/workers/dev-team/knowledge-curator/src/mcp-server.ts +92 -0
- package/template/workers/dev-team/knowledge-curator/tsconfig.json +19 -0
- package/template/workers/dev-team/knowledge-curator/worker.yaml +80 -0
- package/template/workers/dev-team/motion-designer/package.json +22 -0
- package/template/workers/dev-team/motion-designer/skills/add-animation.md +25 -0
- package/template/workers/dev-team/motion-designer/skills/generate-image.md +36 -0
- package/template/workers/dev-team/motion-designer/src/index.ts +63 -0
- package/template/workers/dev-team/motion-designer/src/mcp-server.ts +79 -0
- package/template/workers/dev-team/motion-designer/tsconfig.json +18 -0
- package/template/workers/dev-team/motion-designer/worker.yaml +84 -0
- package/template/workers/dev-team/product-planner/queue.json +4 -0
- package/template/workers/dev-team/product-planner/worker.yaml +220 -0
- package/template/workers/dev-team/project-manager/package-lock.json +1252 -0
- package/template/workers/dev-team/project-manager/package.json +27 -0
- package/template/workers/dev-team/project-manager/skills/create-prd.md +66 -0
- package/template/workers/dev-team/project-manager/skills/next-issue.md +51 -0
- package/template/workers/dev-team/project-manager/skills/project-status.md +59 -0
- package/template/workers/dev-team/project-manager/skills/update-learnings.md +65 -0
- package/template/workers/dev-team/project-manager/src/index.ts +54 -0
- package/template/workers/dev-team/project-manager/src/mcp-server.ts +207 -0
- package/template/workers/dev-team/project-manager/src/skills/create-prd.ts +86 -0
- package/template/workers/dev-team/project-manager/src/skills/next-issue.ts +137 -0
- package/template/workers/dev-team/project-manager/src/skills/project-status.ts +131 -0
- package/template/workers/dev-team/project-manager/src/skills/update-learnings.ts +94 -0
- package/template/workers/dev-team/project-manager/tsconfig.json +19 -0
- package/template/workers/dev-team/project-manager/worker.yaml +96 -0
- package/template/workers/dev-team/qa-tester/package.json +24 -0
- package/template/workers/dev-team/qa-tester/skills/create-demo-account.md +36 -0
- package/template/workers/dev-team/qa-tester/skills/run-tests.md +36 -0
- package/template/workers/dev-team/qa-tester/skills/write-test.md +27 -0
- package/template/workers/dev-team/qa-tester/src/index.ts +61 -0
- package/template/workers/dev-team/qa-tester/src/mcp-server.ts +88 -0
- package/template/workers/dev-team/qa-tester/tsconfig.json +18 -0
- package/template/workers/dev-team/qa-tester/worker.yaml +116 -0
- package/template/workers/dev-team/task-executor/package-lock.json +1252 -0
- package/template/workers/dev-team/task-executor/package.json +27 -0
- package/template/workers/dev-team/task-executor/skills/analyze-issue.md +101 -0
- package/template/workers/dev-team/task-executor/skills/execute.md +133 -0
- package/template/workers/dev-team/task-executor/skills/report-learnings.md +106 -0
- package/template/workers/dev-team/task-executor/skills/validate-completion.md +121 -0
- package/template/workers/dev-team/task-executor/src/index.ts +54 -0
- package/template/workers/dev-team/task-executor/src/mcp-server.ts +139 -0
- package/template/workers/dev-team/task-executor/src/skills/analyze-issue.ts +219 -0
- package/template/workers/dev-team/task-executor/src/skills/execute.ts +132 -0
- package/template/workers/dev-team/task-executor/src/skills/report-learnings.ts +119 -0
- package/template/workers/dev-team/task-executor/src/skills/validate-completion.ts +142 -0
- package/template/workers/dev-team/task-executor/tsconfig.json +19 -0
- package/template/workers/dev-team/task-executor/worker.yaml +110 -0
- package/template/workers/registry.yaml +171 -0
- package/template/workers/security-scanner/README.md +73 -0
- package/template/workers/security-scanner/skills/pre-deploy-check.md +205 -0
- package/template/workers/security-scanner/worker.yaml +26 -0
- package/template/workspace/checkpoints/.gitkeep +0 -0
- package/template/workspace/content-ideas/inbox.jsonl +0 -0
- package/template/workspace/drafts/.gitkeep +0 -0
- package/template/workspace/learnings/.gitkeep +3 -0
- package/template/workspace/orchestrator/.gitkeep +0 -0
- package/template/workspace/ralph-test/COMPLETE.md +18 -0
- package/template/workspace/ralph-test/hello.txt +2 -0
- package/template/workspace/reports/.gitkeep +0 -0
- package/template/workspace/scratch/.gitkeep +0 -0
- package/template/workspace/threads/.gitkeep +3 -0
|
@@ -0,0 +1,526 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Brand Voice Analyzer (US-007)
|
|
3
|
+
* Analyzes content against brand guidelines for tone, messaging, and clarity
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import type {
|
|
7
|
+
AnalysisInput,
|
|
8
|
+
BrandGuidelines,
|
|
9
|
+
BrandAnalysis,
|
|
10
|
+
Finding,
|
|
11
|
+
Recommendation,
|
|
12
|
+
} from './types.js';
|
|
13
|
+
|
|
14
|
+
// ============================================
|
|
15
|
+
// Default Brand Attributes
|
|
16
|
+
// ============================================
|
|
17
|
+
|
|
18
|
+
/**
|
|
19
|
+
* Brand voice characteristics to check
|
|
20
|
+
*/
|
|
21
|
+
export const BRAND_ATTRIBUTES = {
|
|
22
|
+
professional: ['enterprise', 'secure', 'compliance', 'trusted', 'reliable', 'proven', 'robust'],
|
|
23
|
+
confident: ['leading', 'proven', 'powerful', 'best-in-class', 'premier', 'industry-leading', 'world-class'],
|
|
24
|
+
approachable: ['simple', 'easy', 'seamless', 'intuitive', 'straightforward', 'effortless', 'user-friendly'],
|
|
25
|
+
avoid: ['cheap', 'basic', 'try', 'maybe', 'kind of', 'sort of', 'hopefully', 'might', 'possibly'],
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
/**
|
|
29
|
+
* Default brand guidelines when none provided
|
|
30
|
+
*/
|
|
31
|
+
export const DEFAULT_BRAND_GUIDELINES: BrandGuidelines = {
|
|
32
|
+
voiceAttributes: ['professional', 'confident', 'approachable'],
|
|
33
|
+
toneDescriptors: ['authoritative', 'helpful', 'clear'],
|
|
34
|
+
approvedTerms: [
|
|
35
|
+
...BRAND_ATTRIBUTES.professional,
|
|
36
|
+
...BRAND_ATTRIBUTES.confident,
|
|
37
|
+
...BRAND_ATTRIBUTES.approachable,
|
|
38
|
+
],
|
|
39
|
+
avoidTerms: BRAND_ATTRIBUTES.avoid,
|
|
40
|
+
messagingPillars: ['security', 'efficiency', 'innovation', 'trust'],
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
// ============================================
|
|
44
|
+
// Scoring Weights
|
|
45
|
+
// ============================================
|
|
46
|
+
|
|
47
|
+
const WEIGHTS = {
|
|
48
|
+
tone: 0.30,
|
|
49
|
+
messaging: 0.35,
|
|
50
|
+
clarity: 0.35,
|
|
51
|
+
};
|
|
52
|
+
|
|
53
|
+
// ============================================
|
|
54
|
+
// Main Analysis Function
|
|
55
|
+
// ============================================
|
|
56
|
+
|
|
57
|
+
/**
|
|
58
|
+
* Analyze content for brand voice alignment
|
|
59
|
+
*/
|
|
60
|
+
export function analyzeBrandVoice(
|
|
61
|
+
content: AnalysisInput,
|
|
62
|
+
guidelines: BrandGuidelines = DEFAULT_BRAND_GUIDELINES
|
|
63
|
+
): BrandAnalysis {
|
|
64
|
+
const findings: Finding[] = [];
|
|
65
|
+
const recommendations: Recommendation[] = [];
|
|
66
|
+
|
|
67
|
+
// Analyze tone
|
|
68
|
+
const toneResult = analyzeTone(content, guidelines, findings, recommendations);
|
|
69
|
+
|
|
70
|
+
// Analyze messaging alignment
|
|
71
|
+
const messagingResult = analyzeMessaging(content, guidelines, findings, recommendations);
|
|
72
|
+
|
|
73
|
+
// Analyze clarity
|
|
74
|
+
const clarityResult = analyzeClarity(content, findings, recommendations);
|
|
75
|
+
|
|
76
|
+
// Calculate overall score
|
|
77
|
+
const overallScore = Math.round(
|
|
78
|
+
toneResult.score * WEIGHTS.tone +
|
|
79
|
+
messagingResult.score * WEIGHTS.messaging +
|
|
80
|
+
clarityResult.score * WEIGHTS.clarity
|
|
81
|
+
);
|
|
82
|
+
|
|
83
|
+
return {
|
|
84
|
+
overallScore,
|
|
85
|
+
toneScore: toneResult.score,
|
|
86
|
+
messagingScore: messagingResult.score,
|
|
87
|
+
clarityScore: clarityResult.score,
|
|
88
|
+
findings,
|
|
89
|
+
recommendations,
|
|
90
|
+
};
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
// ============================================
|
|
94
|
+
// Tone Analysis
|
|
95
|
+
// ============================================
|
|
96
|
+
|
|
97
|
+
interface AnalysisScoreResult {
|
|
98
|
+
score: number;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
function analyzeTone(
|
|
102
|
+
content: AnalysisInput,
|
|
103
|
+
guidelines: BrandGuidelines,
|
|
104
|
+
findings: Finding[],
|
|
105
|
+
recommendations: Recommendation[]
|
|
106
|
+
): AnalysisScoreResult {
|
|
107
|
+
let score = 100;
|
|
108
|
+
const allText = getAllText(content).toLowerCase();
|
|
109
|
+
|
|
110
|
+
// Check for approved terms (positive)
|
|
111
|
+
const approvedTermsFound = guidelines.approvedTerms.filter(term =>
|
|
112
|
+
allText.includes(term.toLowerCase())
|
|
113
|
+
);
|
|
114
|
+
|
|
115
|
+
if (approvedTermsFound.length > 0) {
|
|
116
|
+
findings.push({
|
|
117
|
+
severity: 'pass',
|
|
118
|
+
category: 'Brand Voice - Tone',
|
|
119
|
+
message: `Found ${approvedTermsFound.length} approved brand terms`,
|
|
120
|
+
evidence: approvedTermsFound.slice(0, 5).join(', '),
|
|
121
|
+
});
|
|
122
|
+
} else {
|
|
123
|
+
findings.push({
|
|
124
|
+
severity: 'warning',
|
|
125
|
+
category: 'Brand Voice - Tone',
|
|
126
|
+
message: 'No approved brand terms found in content',
|
|
127
|
+
});
|
|
128
|
+
score -= 15;
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
// Check for terms to avoid (negative)
|
|
132
|
+
const avoidTermsFound = guidelines.avoidTerms.filter(term =>
|
|
133
|
+
allText.includes(term.toLowerCase())
|
|
134
|
+
);
|
|
135
|
+
|
|
136
|
+
if (avoidTermsFound.length > 0) {
|
|
137
|
+
findings.push({
|
|
138
|
+
severity: 'warning',
|
|
139
|
+
category: 'Brand Voice - Tone',
|
|
140
|
+
message: `Found ${avoidTermsFound.length} terms that should be avoided`,
|
|
141
|
+
evidence: avoidTermsFound.join(', '),
|
|
142
|
+
});
|
|
143
|
+
score -= avoidTermsFound.length * 10;
|
|
144
|
+
|
|
145
|
+
for (const term of avoidTermsFound) {
|
|
146
|
+
const suggestion = getSuggestionForAvoidTerm(term);
|
|
147
|
+
recommendations.push({
|
|
148
|
+
priority: 'medium',
|
|
149
|
+
category: 'Tone',
|
|
150
|
+
current: `Uses "${term}"`,
|
|
151
|
+
suggested: suggestion,
|
|
152
|
+
rationale: 'This term undermines confidence and authority in messaging',
|
|
153
|
+
});
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
// Check for passive voice (weakens tone)
|
|
158
|
+
const passiveCount = countPassiveVoice(allText);
|
|
159
|
+
if (passiveCount > 3) {
|
|
160
|
+
findings.push({
|
|
161
|
+
severity: 'info',
|
|
162
|
+
category: 'Brand Voice - Tone',
|
|
163
|
+
message: `Detected ${passiveCount} instances of passive voice`,
|
|
164
|
+
});
|
|
165
|
+
score -= Math.min(passiveCount * 2, 10);
|
|
166
|
+
recommendations.push({
|
|
167
|
+
priority: 'low',
|
|
168
|
+
category: 'Tone',
|
|
169
|
+
current: 'Multiple passive voice constructions',
|
|
170
|
+
suggested: 'Convert to active voice for stronger messaging',
|
|
171
|
+
rationale: 'Active voice is more direct and confident',
|
|
172
|
+
});
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
// Check for hedging language
|
|
176
|
+
const hedgingTerms = ['possibly', 'perhaps', 'might', 'could be', 'tends to', 'generally'];
|
|
177
|
+
const hedgingFound = hedgingTerms.filter(term => allText.includes(term));
|
|
178
|
+
if (hedgingFound.length > 0) {
|
|
179
|
+
findings.push({
|
|
180
|
+
severity: 'info',
|
|
181
|
+
category: 'Brand Voice - Tone',
|
|
182
|
+
message: 'Hedging language detected',
|
|
183
|
+
evidence: hedgingFound.join(', '),
|
|
184
|
+
});
|
|
185
|
+
score -= hedgingFound.length * 3;
|
|
186
|
+
}
|
|
187
|
+
|
|
188
|
+
return { score: Math.max(0, Math.min(100, score)) };
|
|
189
|
+
}
|
|
190
|
+
|
|
191
|
+
// ============================================
|
|
192
|
+
// Messaging Analysis
|
|
193
|
+
// ============================================
|
|
194
|
+
|
|
195
|
+
function analyzeMessaging(
|
|
196
|
+
content: AnalysisInput,
|
|
197
|
+
guidelines: BrandGuidelines,
|
|
198
|
+
findings: Finding[],
|
|
199
|
+
recommendations: Recommendation[]
|
|
200
|
+
): AnalysisScoreResult {
|
|
201
|
+
let score = 100;
|
|
202
|
+
const allText = getAllText(content).toLowerCase();
|
|
203
|
+
|
|
204
|
+
// Check messaging pillar coverage
|
|
205
|
+
const pillarsFound = guidelines.messagingPillars.filter(pillar =>
|
|
206
|
+
allText.includes(pillar.toLowerCase())
|
|
207
|
+
);
|
|
208
|
+
|
|
209
|
+
const pillarCoverage = pillarsFound.length / guidelines.messagingPillars.length;
|
|
210
|
+
|
|
211
|
+
if (pillarCoverage >= 0.75) {
|
|
212
|
+
findings.push({
|
|
213
|
+
severity: 'pass',
|
|
214
|
+
category: 'Brand Voice - Messaging',
|
|
215
|
+
message: `Strong messaging pillar coverage: ${Math.round(pillarCoverage * 100)}%`,
|
|
216
|
+
evidence: pillarsFound.join(', '),
|
|
217
|
+
});
|
|
218
|
+
} else if (pillarCoverage >= 0.5) {
|
|
219
|
+
findings.push({
|
|
220
|
+
severity: 'info',
|
|
221
|
+
category: 'Brand Voice - Messaging',
|
|
222
|
+
message: `Moderate messaging pillar coverage: ${Math.round(pillarCoverage * 100)}%`,
|
|
223
|
+
evidence: pillarsFound.join(', '),
|
|
224
|
+
});
|
|
225
|
+
score -= 10;
|
|
226
|
+
} else {
|
|
227
|
+
const missingPillars = guidelines.messagingPillars.filter(p =>
|
|
228
|
+
!pillarsFound.includes(p)
|
|
229
|
+
);
|
|
230
|
+
findings.push({
|
|
231
|
+
severity: 'warning',
|
|
232
|
+
category: 'Brand Voice - Messaging',
|
|
233
|
+
message: `Low messaging pillar coverage: ${Math.round(pillarCoverage * 100)}%`,
|
|
234
|
+
evidence: `Missing: ${missingPillars.join(', ')}`,
|
|
235
|
+
});
|
|
236
|
+
score -= 25;
|
|
237
|
+
|
|
238
|
+
recommendations.push({
|
|
239
|
+
priority: 'high',
|
|
240
|
+
category: 'Messaging',
|
|
241
|
+
current: `Only covers ${pillarsFound.length} of ${guidelines.messagingPillars.length} messaging pillars`,
|
|
242
|
+
suggested: `Incorporate messaging around: ${missingPillars.join(', ')}`,
|
|
243
|
+
rationale: 'Core messaging pillars should be represented for brand consistency',
|
|
244
|
+
});
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
// Check for value proposition clarity
|
|
248
|
+
const valueIndicators = ['save', 'reduce', 'increase', 'improve', 'boost', 'grow', 'eliminate', 'streamline'];
|
|
249
|
+
const valueTermsFound = valueIndicators.filter(term => allText.includes(term));
|
|
250
|
+
|
|
251
|
+
if (valueTermsFound.length === 0) {
|
|
252
|
+
findings.push({
|
|
253
|
+
severity: 'warning',
|
|
254
|
+
category: 'Brand Voice - Messaging',
|
|
255
|
+
message: 'No clear value proposition language found',
|
|
256
|
+
});
|
|
257
|
+
score -= 15;
|
|
258
|
+
recommendations.push({
|
|
259
|
+
priority: 'high',
|
|
260
|
+
category: 'Messaging',
|
|
261
|
+
current: 'Value proposition unclear',
|
|
262
|
+
suggested: 'Add specific benefits using action verbs (save, reduce, improve, etc.)',
|
|
263
|
+
rationale: 'Clear value propositions drive engagement and conversion',
|
|
264
|
+
});
|
|
265
|
+
} else {
|
|
266
|
+
findings.push({
|
|
267
|
+
severity: 'pass',
|
|
268
|
+
category: 'Brand Voice - Messaging',
|
|
269
|
+
message: `Value proposition language present: ${valueTermsFound.length} indicators`,
|
|
270
|
+
evidence: valueTermsFound.join(', '),
|
|
271
|
+
});
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
// Check for consistent terminology
|
|
275
|
+
const inconsistencies = checkTerminologyConsistency(content);
|
|
276
|
+
if (inconsistencies.length > 0) {
|
|
277
|
+
findings.push({
|
|
278
|
+
severity: 'info',
|
|
279
|
+
category: 'Brand Voice - Messaging',
|
|
280
|
+
message: 'Potential terminology inconsistencies detected',
|
|
281
|
+
evidence: inconsistencies.join('; '),
|
|
282
|
+
});
|
|
283
|
+
score -= inconsistencies.length * 3;
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
return { score: Math.max(0, Math.min(100, score)) };
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
// ============================================
|
|
290
|
+
// Clarity Analysis
|
|
291
|
+
// ============================================
|
|
292
|
+
|
|
293
|
+
function analyzeClarity(
|
|
294
|
+
content: AnalysisInput,
|
|
295
|
+
findings: Finding[],
|
|
296
|
+
recommendations: Recommendation[]
|
|
297
|
+
): AnalysisScoreResult {
|
|
298
|
+
let score = 100;
|
|
299
|
+
|
|
300
|
+
// Analyze readability
|
|
301
|
+
const readabilityResult = analyzeReadability(content);
|
|
302
|
+
|
|
303
|
+
if (readabilityResult.grade > 12) {
|
|
304
|
+
findings.push({
|
|
305
|
+
severity: 'warning',
|
|
306
|
+
category: 'Brand Voice - Clarity',
|
|
307
|
+
message: `Content reading level too high: Grade ${readabilityResult.grade}`,
|
|
308
|
+
evidence: `Average sentence length: ${readabilityResult.avgSentenceLength} words`,
|
|
309
|
+
});
|
|
310
|
+
score -= 20;
|
|
311
|
+
recommendations.push({
|
|
312
|
+
priority: 'medium',
|
|
313
|
+
category: 'Clarity',
|
|
314
|
+
current: `Grade ${readabilityResult.grade} reading level`,
|
|
315
|
+
suggested: 'Simplify sentences to Grade 8-10 level for broader accessibility',
|
|
316
|
+
rationale: 'B2B content should be professional but accessible',
|
|
317
|
+
});
|
|
318
|
+
} else if (readabilityResult.grade >= 8) {
|
|
319
|
+
findings.push({
|
|
320
|
+
severity: 'pass',
|
|
321
|
+
category: 'Brand Voice - Clarity',
|
|
322
|
+
message: `Appropriate reading level: Grade ${readabilityResult.grade}`,
|
|
323
|
+
});
|
|
324
|
+
} else {
|
|
325
|
+
findings.push({
|
|
326
|
+
severity: 'info',
|
|
327
|
+
category: 'Brand Voice - Clarity',
|
|
328
|
+
message: `Reading level may be too simple: Grade ${readabilityResult.grade}`,
|
|
329
|
+
});
|
|
330
|
+
score -= 5;
|
|
331
|
+
}
|
|
332
|
+
|
|
333
|
+
// Check for jargon overuse
|
|
334
|
+
const jargonTerms = [
|
|
335
|
+
'synergy', 'leverage', 'paradigm', 'bandwidth', 'circle back',
|
|
336
|
+
'low-hanging fruit', 'move the needle', 'drill down', 'holistic',
|
|
337
|
+
];
|
|
338
|
+
const allText = getAllText(content).toLowerCase();
|
|
339
|
+
const jargonFound = jargonTerms.filter(term => allText.includes(term));
|
|
340
|
+
|
|
341
|
+
if (jargonFound.length > 2) {
|
|
342
|
+
findings.push({
|
|
343
|
+
severity: 'warning',
|
|
344
|
+
category: 'Brand Voice - Clarity',
|
|
345
|
+
message: 'Excessive business jargon detected',
|
|
346
|
+
evidence: jargonFound.join(', '),
|
|
347
|
+
});
|
|
348
|
+
score -= jargonFound.length * 5;
|
|
349
|
+
recommendations.push({
|
|
350
|
+
priority: 'medium',
|
|
351
|
+
category: 'Clarity',
|
|
352
|
+
current: `Uses jargon: ${jargonFound.join(', ')}`,
|
|
353
|
+
suggested: 'Replace with clearer, more specific language',
|
|
354
|
+
rationale: 'Jargon reduces clarity and can alienate readers',
|
|
355
|
+
});
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
// Check for long paragraphs
|
|
359
|
+
const longParagraphs = content.paragraphs.filter(p => p.split(/\s+/).length > 100);
|
|
360
|
+
if (longParagraphs.length > 0) {
|
|
361
|
+
findings.push({
|
|
362
|
+
severity: 'info',
|
|
363
|
+
category: 'Brand Voice - Clarity',
|
|
364
|
+
message: `${longParagraphs.length} paragraph(s) exceed 100 words`,
|
|
365
|
+
});
|
|
366
|
+
score -= longParagraphs.length * 3;
|
|
367
|
+
recommendations.push({
|
|
368
|
+
priority: 'low',
|
|
369
|
+
category: 'Clarity',
|
|
370
|
+
current: 'Long paragraphs reduce scanability',
|
|
371
|
+
suggested: 'Break into shorter paragraphs (3-5 sentences max)',
|
|
372
|
+
rationale: 'Shorter paragraphs improve readability on screens',
|
|
373
|
+
});
|
|
374
|
+
}
|
|
375
|
+
|
|
376
|
+
// Check heading clarity
|
|
377
|
+
const unclearHeadings = content.headings.filter(h => {
|
|
378
|
+
const words = h.split(/\s+/);
|
|
379
|
+
return words.length < 2 || words.length > 12;
|
|
380
|
+
});
|
|
381
|
+
|
|
382
|
+
if (unclearHeadings.length > 0) {
|
|
383
|
+
findings.push({
|
|
384
|
+
severity: 'info',
|
|
385
|
+
category: 'Brand Voice - Clarity',
|
|
386
|
+
message: `${unclearHeadings.length} heading(s) may need refinement`,
|
|
387
|
+
evidence: unclearHeadings.slice(0, 3).join(', '),
|
|
388
|
+
});
|
|
389
|
+
score -= unclearHeadings.length * 2;
|
|
390
|
+
}
|
|
391
|
+
|
|
392
|
+
return { score: Math.max(0, Math.min(100, score)) };
|
|
393
|
+
}
|
|
394
|
+
|
|
395
|
+
// ============================================
|
|
396
|
+
// Helper Functions
|
|
397
|
+
// ============================================
|
|
398
|
+
|
|
399
|
+
/**
|
|
400
|
+
* Get all text from content as single string
|
|
401
|
+
*/
|
|
402
|
+
function getAllText(content: AnalysisInput): string {
|
|
403
|
+
return [
|
|
404
|
+
content.title,
|
|
405
|
+
...content.headings,
|
|
406
|
+
...content.paragraphs,
|
|
407
|
+
...content.ctas.map(c => c.text),
|
|
408
|
+
].join(' ');
|
|
409
|
+
}
|
|
410
|
+
|
|
411
|
+
/**
|
|
412
|
+
* Count passive voice instances
|
|
413
|
+
*/
|
|
414
|
+
function countPassiveVoice(text: string): number {
|
|
415
|
+
// Simplified passive voice detection
|
|
416
|
+
const passivePatterns = [
|
|
417
|
+
/\b(is|are|was|were|been|being)\s+\w+ed\b/gi,
|
|
418
|
+
/\b(is|are|was|were|been|being)\s+\w+en\b/gi,
|
|
419
|
+
];
|
|
420
|
+
|
|
421
|
+
let count = 0;
|
|
422
|
+
for (const pattern of passivePatterns) {
|
|
423
|
+
const matches = text.match(pattern);
|
|
424
|
+
if (matches) count += matches.length;
|
|
425
|
+
}
|
|
426
|
+
return count;
|
|
427
|
+
}
|
|
428
|
+
|
|
429
|
+
/**
|
|
430
|
+
* Get suggestion for avoided term
|
|
431
|
+
*/
|
|
432
|
+
function getSuggestionForAvoidTerm(term: string): string {
|
|
433
|
+
const suggestions: Record<string, string> = {
|
|
434
|
+
'cheap': 'Use "cost-effective" or "affordable"',
|
|
435
|
+
'basic': 'Use "essential" or "foundational"',
|
|
436
|
+
'try': 'Use "experience" or "discover"',
|
|
437
|
+
'maybe': 'Remove hedging or state confidently',
|
|
438
|
+
'kind of': 'Be specific or remove qualifier',
|
|
439
|
+
'sort of': 'Be specific or remove qualifier',
|
|
440
|
+
'hopefully': 'State with confidence or specify conditions',
|
|
441
|
+
'might': 'Use "will" or "can" when appropriate',
|
|
442
|
+
'possibly': 'Be specific about conditions or remove',
|
|
443
|
+
};
|
|
444
|
+
return suggestions[term.toLowerCase()] ?? 'Replace with more confident language';
|
|
445
|
+
}
|
|
446
|
+
|
|
447
|
+
/**
|
|
448
|
+
* Check for terminology consistency
|
|
449
|
+
*/
|
|
450
|
+
function checkTerminologyConsistency(content: AnalysisInput): string[] {
|
|
451
|
+
const inconsistencies: string[] = [];
|
|
452
|
+
const allText = getAllText(content).toLowerCase();
|
|
453
|
+
|
|
454
|
+
// Check for inconsistent product/feature naming
|
|
455
|
+
const termPairs = [
|
|
456
|
+
['customer', 'client'],
|
|
457
|
+
['user', 'member'],
|
|
458
|
+
['platform', 'solution'],
|
|
459
|
+
['dashboard', 'portal'],
|
|
460
|
+
];
|
|
461
|
+
|
|
462
|
+
for (const [term1, term2] of termPairs) {
|
|
463
|
+
const has1 = allText.includes(term1);
|
|
464
|
+
const has2 = allText.includes(term2);
|
|
465
|
+
if (has1 && has2) {
|
|
466
|
+
inconsistencies.push(`Mixed usage: "${term1}" and "${term2}"`);
|
|
467
|
+
}
|
|
468
|
+
}
|
|
469
|
+
|
|
470
|
+
return inconsistencies;
|
|
471
|
+
}
|
|
472
|
+
|
|
473
|
+
/**
|
|
474
|
+
* Analyze readability metrics
|
|
475
|
+
*/
|
|
476
|
+
function analyzeReadability(content: AnalysisInput): {
|
|
477
|
+
grade: number;
|
|
478
|
+
avgSentenceLength: number;
|
|
479
|
+
avgWordLength: number;
|
|
480
|
+
} {
|
|
481
|
+
const allText = getAllText(content);
|
|
482
|
+
const sentences = allText.split(/[.!?]+/).filter(s => s.trim().length > 0);
|
|
483
|
+
const words = allText.split(/\s+/).filter(w => w.length > 0);
|
|
484
|
+
|
|
485
|
+
const avgSentenceLength = sentences.length > 0 ? words.length / sentences.length : 0;
|
|
486
|
+
const avgWordLength = words.length > 0
|
|
487
|
+
? words.reduce((sum, w) => sum + w.length, 0) / words.length
|
|
488
|
+
: 0;
|
|
489
|
+
|
|
490
|
+
// Simplified Flesch-Kincaid grade level approximation
|
|
491
|
+
const syllables = words.reduce((sum, w) => sum + countSyllables(w), 0);
|
|
492
|
+
const avgSyllablesPerWord = words.length > 0 ? syllables / words.length : 0;
|
|
493
|
+
|
|
494
|
+
// Flesch-Kincaid Grade Level formula (simplified)
|
|
495
|
+
const grade = Math.round(
|
|
496
|
+
0.39 * avgSentenceLength + 11.8 * avgSyllablesPerWord - 15.59
|
|
497
|
+
);
|
|
498
|
+
|
|
499
|
+
return {
|
|
500
|
+
grade: Math.max(1, Math.min(20, grade)),
|
|
501
|
+
avgSentenceLength: Math.round(avgSentenceLength * 10) / 10,
|
|
502
|
+
avgWordLength: Math.round(avgWordLength * 10) / 10,
|
|
503
|
+
};
|
|
504
|
+
}
|
|
505
|
+
|
|
506
|
+
/**
|
|
507
|
+
* Count syllables in a word (approximation)
|
|
508
|
+
*/
|
|
509
|
+
function countSyllables(word: string): number {
|
|
510
|
+
word = word.toLowerCase().replace(/[^a-z]/g, '');
|
|
511
|
+
if (word.length <= 3) return 1;
|
|
512
|
+
|
|
513
|
+
// Count vowel groups
|
|
514
|
+
const matches = word.match(/[aeiouy]+/g);
|
|
515
|
+
let count = matches ? matches.length : 1;
|
|
516
|
+
|
|
517
|
+
// Adjust for silent e
|
|
518
|
+
if (word.endsWith('e')) count--;
|
|
519
|
+
|
|
520
|
+
// Adjust for -le endings
|
|
521
|
+
if (word.endsWith('le') && word.length > 2 && !/[aeiouy]/.test(word[word.length - 3])) {
|
|
522
|
+
count++;
|
|
523
|
+
}
|
|
524
|
+
|
|
525
|
+
return Math.max(1, count);
|
|
526
|
+
}
|