@mison/ag-kit-cn 2.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.
Files changed (237) hide show
  1. package/.agent/.shared/ui-ux-pro-max/data/charts.csv +26 -0
  2. package/.agent/.shared/ui-ux-pro-max/data/colors.csv +97 -0
  3. package/.agent/.shared/ui-ux-pro-max/data/icons.csv +101 -0
  4. package/.agent/.shared/ui-ux-pro-max/data/landing.csv +31 -0
  5. package/.agent/.shared/ui-ux-pro-max/data/products.csv +97 -0
  6. package/.agent/.shared/ui-ux-pro-max/data/prompts.csv +24 -0
  7. package/.agent/.shared/ui-ux-pro-max/data/react-performance.csv +45 -0
  8. package/.agent/.shared/ui-ux-pro-max/data/stacks/flutter.csv +53 -0
  9. package/.agent/.shared/ui-ux-pro-max/data/stacks/html-tailwind.csv +56 -0
  10. package/.agent/.shared/ui-ux-pro-max/data/stacks/jetpack-compose.csv +53 -0
  11. package/.agent/.shared/ui-ux-pro-max/data/stacks/nextjs.csv +53 -0
  12. package/.agent/.shared/ui-ux-pro-max/data/stacks/nuxt-ui.csv +51 -0
  13. package/.agent/.shared/ui-ux-pro-max/data/stacks/nuxtjs.csv +59 -0
  14. package/.agent/.shared/ui-ux-pro-max/data/stacks/react-native.csv +52 -0
  15. package/.agent/.shared/ui-ux-pro-max/data/stacks/react.csv +54 -0
  16. package/.agent/.shared/ui-ux-pro-max/data/stacks/shadcn.csv +61 -0
  17. package/.agent/.shared/ui-ux-pro-max/data/stacks/svelte.csv +54 -0
  18. package/.agent/.shared/ui-ux-pro-max/data/stacks/swiftui.csv +51 -0
  19. package/.agent/.shared/ui-ux-pro-max/data/stacks/vue.csv +50 -0
  20. package/.agent/.shared/ui-ux-pro-max/data/styles.csv +59 -0
  21. package/.agent/.shared/ui-ux-pro-max/data/typography.csv +58 -0
  22. package/.agent/.shared/ui-ux-pro-max/data/ui-reasoning.csv +101 -0
  23. package/.agent/.shared/ui-ux-pro-max/data/ux-guidelines.csv +100 -0
  24. package/.agent/.shared/ui-ux-pro-max/data/web-interface.csv +31 -0
  25. package/.agent/.shared/ui-ux-pro-max/scripts/core.py +258 -0
  26. package/.agent/.shared/ui-ux-pro-max/scripts/design_system.py +1067 -0
  27. package/.agent/.shared/ui-ux-pro-max/scripts/search.py +106 -0
  28. package/.agent/ARCHITECTURE.md +285 -0
  29. package/.agent/agents/backend-specialist.md +268 -0
  30. package/.agent/agents/code-archaeologist.md +106 -0
  31. package/.agent/agents/database-architect.md +225 -0
  32. package/.agent/agents/debugger.md +225 -0
  33. package/.agent/agents/devops-engineer.md +242 -0
  34. package/.agent/agents/documentation-writer.md +104 -0
  35. package/.agent/agents/explorer-agent.md +73 -0
  36. package/.agent/agents/frontend-specialist.md +618 -0
  37. package/.agent/agents/game-developer.md +162 -0
  38. package/.agent/agents/mobile-developer.md +382 -0
  39. package/.agent/agents/orchestrator.md +438 -0
  40. package/.agent/agents/penetration-tester.md +188 -0
  41. package/.agent/agents/performance-optimizer.md +187 -0
  42. package/.agent/agents/product-manager.md +112 -0
  43. package/.agent/agents/product-owner.md +95 -0
  44. package/.agent/agents/project-planner.md +405 -0
  45. package/.agent/agents/qa-automation-engineer.md +103 -0
  46. package/.agent/agents/security-auditor.md +170 -0
  47. package/.agent/agents/seo-specialist.md +111 -0
  48. package/.agent/agents/test-engineer.md +158 -0
  49. package/.agent/mcp_config.json +12 -0
  50. package/.agent/rules/GEMINI.md +273 -0
  51. package/.agent/scripts/auto_preview.py +148 -0
  52. package/.agent/scripts/checklist.py +217 -0
  53. package/.agent/scripts/session_manager.py +120 -0
  54. package/.agent/scripts/verify_all.py +327 -0
  55. package/.agent/skills/api-patterns/SKILL.md +84 -0
  56. package/.agent/skills/api-patterns/api-style.md +42 -0
  57. package/.agent/skills/api-patterns/auth.md +24 -0
  58. package/.agent/skills/api-patterns/documentation.md +26 -0
  59. package/.agent/skills/api-patterns/graphql.md +41 -0
  60. package/.agent/skills/api-patterns/rate-limiting.md +31 -0
  61. package/.agent/skills/api-patterns/response.md +37 -0
  62. package/.agent/skills/api-patterns/rest.md +40 -0
  63. package/.agent/skills/api-patterns/scripts/api_validator.py +211 -0
  64. package/.agent/skills/api-patterns/security-testing.md +122 -0
  65. package/.agent/skills/api-patterns/trpc.md +41 -0
  66. package/.agent/skills/api-patterns/versioning.md +22 -0
  67. package/.agent/skills/app-builder/SKILL.md +75 -0
  68. package/.agent/skills/app-builder/agent-coordination.md +74 -0
  69. package/.agent/skills/app-builder/feature-building.md +53 -0
  70. package/.agent/skills/app-builder/project-detection.md +34 -0
  71. package/.agent/skills/app-builder/scaffolding.md +118 -0
  72. package/.agent/skills/app-builder/tech-stack.md +40 -0
  73. package/.agent/skills/app-builder/templates/SKILL.md +39 -0
  74. package/.agent/skills/app-builder/templates/astro-static/TEMPLATE.md +76 -0
  75. package/.agent/skills/app-builder/templates/chrome-extension/TEMPLATE.md +92 -0
  76. package/.agent/skills/app-builder/templates/cli-tool/TEMPLATE.md +88 -0
  77. package/.agent/skills/app-builder/templates/electron-desktop/TEMPLATE.md +88 -0
  78. package/.agent/skills/app-builder/templates/express-api/TEMPLATE.md +83 -0
  79. package/.agent/skills/app-builder/templates/flutter-app/TEMPLATE.md +90 -0
  80. package/.agent/skills/app-builder/templates/monorepo-turborepo/TEMPLATE.md +90 -0
  81. package/.agent/skills/app-builder/templates/nextjs-fullstack/TEMPLATE.md +122 -0
  82. package/.agent/skills/app-builder/templates/nextjs-saas/TEMPLATE.md +122 -0
  83. package/.agent/skills/app-builder/templates/nextjs-static/TEMPLATE.md +169 -0
  84. package/.agent/skills/app-builder/templates/nuxt-app/TEMPLATE.md +134 -0
  85. package/.agent/skills/app-builder/templates/python-fastapi/TEMPLATE.md +83 -0
  86. package/.agent/skills/app-builder/templates/react-native-app/TEMPLATE.md +119 -0
  87. package/.agent/skills/architecture/SKILL.md +57 -0
  88. package/.agent/skills/architecture/context-discovery.md +43 -0
  89. package/.agent/skills/architecture/examples.md +94 -0
  90. package/.agent/skills/architecture/pattern-selection.md +68 -0
  91. package/.agent/skills/architecture/patterns-reference.md +50 -0
  92. package/.agent/skills/architecture/trade-off-analysis.md +77 -0
  93. package/.agent/skills/bash-linux/SKILL.md +201 -0
  94. package/.agent/skills/behavioral-modes/SKILL.md +264 -0
  95. package/.agent/skills/brainstorming/SKILL.md +164 -0
  96. package/.agent/skills/brainstorming/dynamic-questioning.md +359 -0
  97. package/.agent/skills/clean-code/SKILL.md +200 -0
  98. package/.agent/skills/code-review-checklist/SKILL.md +125 -0
  99. package/.agent/skills/database-design/SKILL.md +54 -0
  100. package/.agent/skills/database-design/database-selection.md +43 -0
  101. package/.agent/skills/database-design/indexing.md +39 -0
  102. package/.agent/skills/database-design/migrations.md +50 -0
  103. package/.agent/skills/database-design/optimization.md +36 -0
  104. package/.agent/skills/database-design/orm-selection.md +30 -0
  105. package/.agent/skills/database-design/schema-design.md +56 -0
  106. package/.agent/skills/database-design/scripts/schema_validator.py +172 -0
  107. package/.agent/skills/deployment-procedures/SKILL.md +241 -0
  108. package/.agent/skills/doc.md +177 -0
  109. package/.agent/skills/documentation-templates/SKILL.md +194 -0
  110. package/.agent/skills/frontend-design/SKILL.md +418 -0
  111. package/.agent/skills/frontend-design/animation-guide.md +331 -0
  112. package/.agent/skills/frontend-design/color-system.md +307 -0
  113. package/.agent/skills/frontend-design/decision-trees.md +418 -0
  114. package/.agent/skills/frontend-design/motion-graphics.md +306 -0
  115. package/.agent/skills/frontend-design/scripts/accessibility_checker.py +183 -0
  116. package/.agent/skills/frontend-design/scripts/ux_audit.py +727 -0
  117. package/.agent/skills/frontend-design/typography-system.md +345 -0
  118. package/.agent/skills/frontend-design/ux-psychology.md +1118 -0
  119. package/.agent/skills/frontend-design/visual-effects.md +383 -0
  120. package/.agent/skills/game-development/2d-games/SKILL.md +119 -0
  121. package/.agent/skills/game-development/3d-games/SKILL.md +135 -0
  122. package/.agent/skills/game-development/SKILL.md +167 -0
  123. package/.agent/skills/game-development/game-art/SKILL.md +185 -0
  124. package/.agent/skills/game-development/game-audio/SKILL.md +190 -0
  125. package/.agent/skills/game-development/game-design/SKILL.md +129 -0
  126. package/.agent/skills/game-development/mobile-games/SKILL.md +108 -0
  127. package/.agent/skills/game-development/multiplayer/SKILL.md +132 -0
  128. package/.agent/skills/game-development/pc-games/SKILL.md +144 -0
  129. package/.agent/skills/game-development/vr-ar/SKILL.md +123 -0
  130. package/.agent/skills/game-development/web-games/SKILL.md +150 -0
  131. package/.agent/skills/geo-fundamentals/SKILL.md +155 -0
  132. package/.agent/skills/geo-fundamentals/scripts/geo_checker.py +289 -0
  133. package/.agent/skills/i18n-localization/SKILL.md +154 -0
  134. package/.agent/skills/i18n-localization/scripts/i18n_checker.py +241 -0
  135. package/.agent/skills/intelligent-routing/SKILL.md +335 -0
  136. package/.agent/skills/lint-and-validate/SKILL.md +44 -0
  137. package/.agent/skills/lint-and-validate/scripts/lint_runner.py +184 -0
  138. package/.agent/skills/lint-and-validate/scripts/type_coverage.py +173 -0
  139. package/.agent/skills/mcp-builder/SKILL.md +176 -0
  140. package/.agent/skills/mobile-design/SKILL.md +394 -0
  141. package/.agent/skills/mobile-design/decision-trees.md +516 -0
  142. package/.agent/skills/mobile-design/mobile-backend.md +491 -0
  143. package/.agent/skills/mobile-design/mobile-color-system.md +420 -0
  144. package/.agent/skills/mobile-design/mobile-debugging.md +122 -0
  145. package/.agent/skills/mobile-design/mobile-design-thinking.md +355 -0
  146. package/.agent/skills/mobile-design/mobile-navigation.md +458 -0
  147. package/.agent/skills/mobile-design/mobile-performance.md +767 -0
  148. package/.agent/skills/mobile-design/mobile-testing.md +356 -0
  149. package/.agent/skills/mobile-design/mobile-typography.md +432 -0
  150. package/.agent/skills/mobile-design/platform-android.md +666 -0
  151. package/.agent/skills/mobile-design/platform-ios.md +561 -0
  152. package/.agent/skills/mobile-design/scripts/mobile_audit.py +670 -0
  153. package/.agent/skills/mobile-design/touch-psychology.md +537 -0
  154. package/.agent/skills/nextjs-react-expert/1-async-eliminating-waterfalls.md +311 -0
  155. package/.agent/skills/nextjs-react-expert/2-bundle-bundle-size-optimization.md +241 -0
  156. package/.agent/skills/nextjs-react-expert/3-server-server-side-performance.md +489 -0
  157. package/.agent/skills/nextjs-react-expert/4-client-client-side-data-fetching.md +263 -0
  158. package/.agent/skills/nextjs-react-expert/5-rerender-re-render-optimization.md +581 -0
  159. package/.agent/skills/nextjs-react-expert/6-rendering-rendering-performance.md +431 -0
  160. package/.agent/skills/nextjs-react-expert/7-js-javascript-performance.md +683 -0
  161. package/.agent/skills/nextjs-react-expert/8-advanced-advanced-patterns.md +149 -0
  162. package/.agent/skills/nextjs-react-expert/SKILL.md +286 -0
  163. package/.agent/skills/nextjs-react-expert/scripts/convert_rules.py +222 -0
  164. package/.agent/skills/nextjs-react-expert/scripts/react_performance_checker.py +252 -0
  165. package/.agent/skills/nodejs-best-practices/SKILL.md +333 -0
  166. package/.agent/skills/parallel-agents/SKILL.md +194 -0
  167. package/.agent/skills/performance-profiling/SKILL.md +149 -0
  168. package/.agent/skills/performance-profiling/scripts/lighthouse_audit.py +76 -0
  169. package/.agent/skills/plan-writing/SKILL.md +152 -0
  170. package/.agent/skills/powershell-windows/SKILL.md +166 -0
  171. package/.agent/skills/python-patterns/SKILL.md +441 -0
  172. package/.agent/skills/red-team-tactics/SKILL.md +203 -0
  173. package/.agent/skills/rust-pro/SKILL.md +190 -0
  174. package/.agent/skills/seo-fundamentals/SKILL.md +135 -0
  175. package/.agent/skills/seo-fundamentals/scripts/seo_checker.py +215 -0
  176. package/.agent/skills/server-management/SKILL.md +161 -0
  177. package/.agent/skills/systematic-debugging/SKILL.md +114 -0
  178. package/.agent/skills/tailwind-patterns/SKILL.md +269 -0
  179. package/.agent/skills/tdd-workflow/SKILL.md +149 -0
  180. package/.agent/skills/testing-patterns/SKILL.md +178 -0
  181. package/.agent/skills/testing-patterns/scripts/test_runner.py +219 -0
  182. package/.agent/skills/vulnerability-scanner/SKILL.md +276 -0
  183. package/.agent/skills/vulnerability-scanner/checklists.md +131 -0
  184. package/.agent/skills/vulnerability-scanner/scripts/security_scan.py +459 -0
  185. package/.agent/skills/web-design-guidelines/SKILL.md +57 -0
  186. package/.agent/skills/webapp-testing/SKILL.md +187 -0
  187. package/.agent/skills/webapp-testing/scripts/playwright_runner.py +173 -0
  188. package/.agent/workflows/brainstorm.md +113 -0
  189. package/.agent/workflows/create.md +59 -0
  190. package/.agent/workflows/debug.md +103 -0
  191. package/.agent/workflows/deploy.md +176 -0
  192. package/.agent/workflows/enhance.md +63 -0
  193. package/.agent/workflows/orchestrate.md +242 -0
  194. package/.agent/workflows/plan.md +89 -0
  195. package/.agent/workflows/preview.md +80 -0
  196. package/.agent/workflows/restore-localize-compat.md +525 -0
  197. package/.agent/workflows/status.md +86 -0
  198. package/.agent/workflows/test.md +144 -0
  199. package/.agent/workflows/ui-ux-pro-max.md +295 -0
  200. package/AGENT_FLOW.md +609 -0
  201. package/CHANGELOG.md +68 -0
  202. package/LICENSE +21 -0
  203. package/README.md +260 -0
  204. package/bin/adapters/base.js +63 -0
  205. package/bin/adapters/codex.js +391 -0
  206. package/bin/adapters/gemini.js +137 -0
  207. package/bin/ag-kit.js +1336 -0
  208. package/bin/core/builder.js +80 -0
  209. package/bin/core/generator.js +59 -0
  210. package/bin/core/resource-loader.js +64 -0
  211. package/bin/core/transformer.js +208 -0
  212. package/bin/interactive.js +65 -0
  213. package/bin/utils/atomic-writer.js +97 -0
  214. package/bin/utils/git-helper.js +68 -0
  215. package/bin/utils/managed-block.js +65 -0
  216. package/bin/utils/manifest.js +241 -0
  217. package/bin/utils.js +82 -0
  218. package/docs/codex-rules-template.md +36 -0
  219. package/docs/mapping-spec.md +68 -0
  220. package/docs/multi-target-adapter.md +80 -0
  221. package/docs/official/README.md +53 -0
  222. package/docs/official/antigravity/agent-modes-settings.md +64 -0
  223. package/docs/official/antigravity/rules-workflows.md +96 -0
  224. package/docs/official/antigravity/skills.md +147 -0
  225. package/docs/official/codex/agents-md.md +119 -0
  226. package/docs/official/codex/config-advanced.md +358 -0
  227. package/docs/official/codex/config-basic.md +141 -0
  228. package/docs/official/codex/config-reference.md +223 -0
  229. package/docs/official/codex/config-sample.md +216 -0
  230. package/docs/official/codex/mcp.md +107 -0
  231. package/docs/official/codex/rules.md +79 -0
  232. package/docs/official/codex/skills.md +114 -0
  233. package/docs/official/sources-index.md +32 -0
  234. package/docs/operations.md +145 -0
  235. package/docs/terminology-style-guide.md +69 -0
  236. package/package.json +51 -0
  237. package/scripts/postinstall-check.js +112 -0
@@ -0,0 +1,727 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ UX Audit Script - Full Frontend Design Coverage
4
+
5
+ Analyzes code for compliance with:
6
+
7
+ 1. CORE PSYCHOLOGY LAWS:
8
+ - Hick's Law (nav items, form complexity)
9
+ - Fitts' Law (target sizes, touch targets)
10
+ - Miller's Law (chunking, memory limits)
11
+ - Von Restorff Effect (primary CTA visibility)
12
+ - Serial Position Effect (important items at start/end)
13
+
14
+ 2. EMOTIONAL DESIGN (Don Norman):
15
+ - Visceral (first impressions, gradients, animations)
16
+ - Behavioral (feedback, usability, performance)
17
+ - Reflective (brand story, values, identity)
18
+
19
+ 3. TRUST BUILDING:
20
+ - Security signals (SSL, encryption on forms)
21
+ - Social proof (testimonials, reviews, logos)
22
+ - Authority indicators (certifications, awards, media)
23
+
24
+ 4. COGNITIVE LOAD MANAGEMENT:
25
+ - Progressive disclosure (accordion, tabs, "Advanced")
26
+ - Visual noise (too many colors/borders)
27
+ - Familiar patterns (labels, standard conventions)
28
+
29
+ 5. PERSUASIVE DESIGN (Ethical):
30
+ - Smart defaults (pre-selected options)
31
+ - Anchoring (original vs discount price)
32
+ - Social proof (live indicators, numbers)
33
+ - Progress indicators (progress bars, steps)
34
+
35
+ 6. TYPOGRAPHY SYSTEM (9 sections):
36
+ - Font Pairing (max 3 families)
37
+ - Line Length (45-75ch)
38
+ - Line Height (proper ratios)
39
+ - Letter Spacing (uppercase, display text)
40
+ - Weight and Emphasis (contrast levels)
41
+ - Responsive Typography (clamp())
42
+ - Hierarchy (sequential headings)
43
+ - Modular Scale (consistent ratios)
44
+ - Readability (chunking, subheadings)
45
+
46
+ 7. VISUAL EFFECTS (10 sections):
47
+ - Glassmorphism (blur + transparency)
48
+ - Neomorphism (dual shadows, inset)
49
+ - Shadow Hierarchy (elevation levels)
50
+ - Gradients (usage, overuse)
51
+ - Border Effects (complexity check)
52
+ - Glow Effects (text-shadow, box-shadow)
53
+ - Overlay Techniques (image text readability)
54
+ - GPU Acceleration (transform/opacity vs layout)
55
+ - Performance (will-change usage)
56
+ - Effect Selection (purpose over decoration)
57
+
58
+ 8. COLOR SYSTEM (7 sections):
59
+ - PURPLE BAN (Critical Maestro rule - #8B5CF6, #A855F7, etc.)
60
+ - 60-30-10 Rule (dominant, secondary, accent)
61
+ - Color Scheme Patterns (monochromatic, analogous)
62
+ - Dark Mode Compliance (no pure black/white)
63
+ - WCAG Contrast (low-contrast detection)
64
+ - Color Psychology Context (food + blue = bad)
65
+ - HSL-Based Palettes (recommended approach)
66
+
67
+ 9. ANIMATION GUIDE (6 sections):
68
+ - Duration Appropriateness (50ms minimum, 1s max transitions)
69
+ - Easing Functions (ease-out for entry, ease-in for exit)
70
+ - Micro-interactions (hover/focus feedback)
71
+ - Loading States (skeleton, spinner, progress)
72
+ - Page Transitions (fade/slide for routing)
73
+ - Scroll Animation Performance (no layout properties)
74
+
75
+ 10. MOTION GRAPHICS (7 sections):
76
+ - Lottie Animations (reduced motion fallbacks)
77
+ - GSAP Memory Leaks (kill/revert on unmount)
78
+ - SVG Animation Performance (stroke-dashoffset sparingly)
79
+ - 3D Transforms (perspective parent, mobile warning)
80
+ - Particle Effects (mobile fallback)
81
+ - Scroll-Driven Animations (throttle with rAF)
82
+ - Motion Decision Tree (functional vs decorative)
83
+
84
+ 11. ACCESSIBILITY:
85
+ - Alt text for images
86
+ - Reduced motion checks
87
+ - Form labels
88
+
89
+ Total: 80+ checks across all design principles
90
+ """
91
+
92
+ import sys
93
+ import os
94
+ import re
95
+ import json
96
+ from pathlib import Path
97
+
98
+ class UXAuditor:
99
+ def __init__(self):
100
+ self.issues = []
101
+ self.warnings = []
102
+ self.passed_count = 0
103
+ self.files_checked = 0
104
+
105
+ def audit_file(self, filepath: str) -> None:
106
+ try:
107
+ with open(filepath, 'r', encoding='utf-8', errors='replace') as f:
108
+ content = f.read()
109
+ except: return
110
+
111
+ self.files_checked += 1
112
+ filename = os.path.basename(filepath)
113
+
114
+ # Pre-calculate common flags
115
+ has_long_text = bool(re.search(r'<p|<div.*class=.*text|article|<span.*text', content, re.IGNORECASE))
116
+ has_form = bool(re.search(r'<form|<input|password|credit|card|payment', content, re.IGNORECASE))
117
+ complex_elements = len(re.findall(r'<input|<select|<textarea|<option', content, re.IGNORECASE))
118
+
119
+ # --- 1. PSYCHOLOGY LAWS ---
120
+ # Hick's Law
121
+ is_nav_heavy_area = any(d in filepath.lower() for d in ['footer', 'ui', 'components', 'docs'])
122
+ nav_items = len(re.findall(r'<NavLink|<Link|<a\s+href|nav-item', content, re.IGNORECASE))
123
+ if nav_items > 7 and not is_nav_heavy_area:
124
+ self.issues.append(f"[Hick's Law] {filename}: {nav_items} nav items (Max 7)")
125
+
126
+ # Fitts' Law
127
+ if re.search(r'height:\s*([0-3]\d)px', content) or re.search(r'h-[1-9]\b|h-10\b', content):
128
+ self.warnings.append(f"[Fitts' Law] {filename}: Small targets (< 44px)")
129
+
130
+ # Miller's Law
131
+ form_fields = len(re.findall(r'<input|<select|<textarea', content, re.IGNORECASE))
132
+ if form_fields > 7 and not re.search(r'step|wizard|stage', content, re.IGNORECASE):
133
+ self.warnings.append(f"[Miller's Law] {filename}: Complex form ({form_fields} fields)")
134
+
135
+ # Von Restorff
136
+ if 'button' in content.lower() and not re.search(r'primary|bg-primary|Button.*primary|variant=["\']primary', content, re.IGNORECASE):
137
+ self.warnings.append(f"[Von Restorff] {filename}: No primary CTA")
138
+
139
+ # Serial Position Effect - Important items at beginning/end
140
+ if nav_items > 3:
141
+ # Check if last nav item is important (contact, login, etc.)
142
+ nav_content = re.findall(r'<NavLink|<Link|<a\s+href[^>]*>([^<]+)</a>', content, re.IGNORECASE)
143
+ if nav_content and len(nav_content) > 2:
144
+ last_item = nav_content[-1].lower() if nav_content else ''
145
+ if not any(x in last_item for x in ['contact', 'login', 'sign', 'get started', 'cta', 'button']):
146
+ self.warnings.append(f"[Serial Position] {filename}: Last nav item may not be important. Place key actions at start/end.")
147
+
148
+ # --- 1.5 EMOTIONAL DESIGN (Don Norman) ---
149
+
150
+ # Visceral: First impressions (aesthetics, gradients, animations)
151
+ has_hero = bool(re.search(r'hero|<h1|banner', content, re.IGNORECASE))
152
+ if has_hero:
153
+ # Check for visual appeal elements
154
+ has_gradient = bool(re.search(r'gradient|linear-gradient|radial-gradient', content))
155
+ has_animation = bool(re.search(r'@keyframes|transition:|animate-', content))
156
+ has_visual_interest = has_gradient or has_animation
157
+
158
+ if not has_visual_interest and not re.search(r'background:|bg-', content):
159
+ self.warnings.append(f"[Visceral] {filename}: Hero section lacks visual appeal. Consider gradients or subtle animations.")
160
+
161
+ # Behavioral: Instant feedback and usability
162
+ if 'onClick' in content or '@click' in content or 'onclick' in content:
163
+ has_feedback = re.search(r'transition|animate|hover:|focus:|disabled|loading|spinner', content, re.IGNORECASE)
164
+ has_state_change = re.search(r'setState|useState|disabled|loading', content)
165
+
166
+ if not has_feedback and not has_state_change:
167
+ self.warnings.append(f"[Behavioral] {filename}: Interactive elements lack immediate feedback. Add hover/focus/disabled states.")
168
+
169
+ # Reflective: Brand story, values, identity
170
+ has_reflective = bool(re.search(r'about|story|mission|values|why we|our journey|testimonials', content, re.IGNORECASE))
171
+ if has_long_text and not has_reflective:
172
+ self.warnings.append(f"[Reflective] {filename}: Long-form content without brand story/values. Add 'About' or 'Why We Exist' section.")
173
+
174
+ # --- 1.6 TRUST BUILDING (Enhanced) ---
175
+
176
+ # Security signals
177
+ if has_form:
178
+ security_signals = re.findall(r'ssl|secure|encrypt|lock|padlock|https', content, re.IGNORECASE)
179
+ if len(security_signals) == 0 and not re.search(r'checkout|payment', content, re.IGNORECASE):
180
+ self.warnings.append(f"[Trust] {filename}: Form without security indicators. Add 'SSL Secure' or lock icon.")
181
+
182
+ # Social proof elements
183
+ social_proof = re.findall(r'review|testimonial|rating|star|trust|trusted by|customer|logo', content, re.IGNORECASE)
184
+ if len(social_proof) > 0:
185
+ self.passed_count += 1
186
+ else:
187
+ if has_long_text:
188
+ self.warnings.append(f"[Trust] {filename}: No social proof detected. Consider adding testimonials, ratings, or 'Trusted by' logos.")
189
+
190
+ # Authority indicators
191
+ has_footer = bool(re.search(r'footer|<footer', content, re.IGNORECASE))
192
+ if has_footer:
193
+ authority = re.findall(r'certif|award|media|press|featured|as seen in', content, re.IGNORECASE)
194
+ if len(authority) == 0:
195
+ self.warnings.append(f"[Trust] {filename}: Footer lacks authority signals. Add certifications, awards, or media mentions.")
196
+
197
+ # --- 1.7 COGNITIVE LOAD MANAGEMENT ---
198
+
199
+ # Progressive disclosure
200
+ if complex_elements > 5:
201
+ has_progressive = re.search(r'step|wizard|stage|accordion|collapsible|tab|more\.\.\.|advanced|show more', content, re.IGNORECASE)
202
+ if not has_progressive:
203
+ self.warnings.append(f"[Cognitive Load] {filename}: Many form elements without progressive disclosure. Consider accordion, tabs, or 'Advanced' toggle.")
204
+
205
+ # Visual noise check
206
+ has_many_colors = len(re.findall(r'#[0-9a-fA-F]{3,6}|rgb|hsl', content)) > 15
207
+ has_many_borders = len(re.findall(r'border:|border-', content)) > 10
208
+ if has_many_colors and has_many_borders:
209
+ self.warnings.append(f"[Cognitive Load] {filename}: High visual noise detected. Many colors and borders increase cognitive load.")
210
+
211
+ # Familiar patterns
212
+ if has_form and not filepath.endswith('.css'):
213
+ has_standard_labels = bool(re.search(r'<label|placeholder|aria-label', content, re.IGNORECASE))
214
+ if not has_standard_labels:
215
+ self.issues.append(f"[Cognitive Load] {filename}: Form inputs without labels. Use <label> for accessibility and clarity.")
216
+
217
+ # --- 1.8 PERSUASIVE DESIGN (Ethical) ---
218
+
219
+ # Smart defaults
220
+ if has_form:
221
+ has_defaults = bool(re.search(r'checked|selected|default|value=["\'].*["\']', content))
222
+ radio_inputs = len(re.findall(r'type=["\']radio', content, re.IGNORECASE))
223
+ if radio_inputs > 0 and not has_defaults:
224
+ self.warnings.append(f"[Persuasion] {filename}: Radio buttons without default selection. Pre-select recommended option.")
225
+
226
+ # Anchoring (showing original price)
227
+ if re.search(r'price|pricing|cost|\$\d+', content, re.IGNORECASE):
228
+ has_anchor = bool(re.search(r'original|was|strike|del|save \d+%', content, re.IGNORECASE))
229
+ if not has_anchor:
230
+ self.warnings.append(f"[Persuasion] {filename}: Prices without anchoring. Show original price to frame discount value.")
231
+
232
+ # Social proof live indicators
233
+ has_social = bool(re.search(r'join|subscriber|member|user', content, re.IGNORECASE))
234
+ if has_social:
235
+ has_count = bool(re.findall(r'\d+[+kmb]|\d+,\d+', content))
236
+ if not has_count:
237
+ self.warnings.append(f"[Persuasion] {filename}: Social proof without specific numbers. Use 'Join 10,000+' format.")
238
+
239
+ # Progress indicators
240
+ if has_form:
241
+ has_progress = bool(re.search(r'progress|step \d+|complete|%|bar', content, re.IGNORECASE))
242
+ if complex_elements > 5 and not has_progress:
243
+ self.warnings.append(f"[Persuasion] {filename}: Long form without progress indicator. Add progress bar or 'Step X of Y'.")
244
+
245
+ # --- 2. TYPOGRAPHY SYSTEM (Complete Coverage) ---
246
+
247
+ # 2.1 Font Pairing - Too many font families
248
+ font_families = set()
249
+ # Check for @font-face, Google Fonts, font-family declarations
250
+ font_faces = re.findall(r'@font-face\s*\{[^}]*family:\s*["\']?([^;"\'\s}]+)', content, re.IGNORECASE)
251
+ google_fonts = re.findall(r'fonts\.googleapis\.com[^"\']*family=([^"&]+)', content, re.IGNORECASE)
252
+ font_family_css = re.findall(r'font-family:\s*([^;]+)', content, re.IGNORECASE)
253
+
254
+ for font in font_faces: font_families.add(font.strip().lower())
255
+ for font in google_fonts:
256
+ for f in font.replace('+', ' ').split('|'):
257
+ font_families.add(f.split(':')[0].strip().lower())
258
+ for family in font_family_css:
259
+ # Extract first font from stack
260
+ first_font = family.split(',')[0].strip().strip('"\'')
261
+
262
+ if first_font.lower() not in {'sans-serif', 'serif', 'monospace', 'cursive', 'fantasy', 'system-ui', 'inherit', 'arial', 'georgia', 'times new roman', 'courier new', 'verdana', 'helvetica', 'tahoma'}:
263
+ font_families.add(first_font.lower())
264
+
265
+ if len(font_families) > 3:
266
+ self.issues.append(f"[Typography] {filename}: {len(font_families)} font families detected. Limit to 2-3 for cohesion.")
267
+
268
+ # 2.2 Line Length - Character-based width
269
+ if has_long_text and not re.search(r'max-w-(?:prose|[\[\\]?\d+ch[\]\\]?)|max-width:\s*\d+ch', content):
270
+ self.warnings.append(f"[Typography] {filename}: No line length constraint (45-75ch). Use max-w-prose or max-w-[65ch].")
271
+
272
+ # 2.3 Line Height - Proper leading ratios
273
+ # Check for text without proper line-height
274
+ text_elements = len(re.findall(r'<p|<span|<div.*text|<h[1-6]', content, re.IGNORECASE))
275
+ if text_elements > 0 and not re.search(r'leading-|line-height:', content):
276
+ self.warnings.append(f"[Typography] {filename}: Text elements found without line-height. Body: 1.4-1.6, Headings: 1.1-1.3")
277
+
278
+ # Check for heading-specific line height issues
279
+ if re.search(r'<h[1-6]|text-(?:xl|2xl|3xl|4xl|5xl|6xl)', content, re.IGNORECASE):
280
+ # Extract line-height values
281
+ line_heights = re.findall(r'(?:leading-|line-height:\s*)([\d.]+)', content)
282
+ for lh in line_heights:
283
+ if float(lh) > 1.5:
284
+ self.warnings.append(f"[Typography] {filename}: Heading has line-height {lh} (>1.3). Headings should be tighter (1.1-1.3).")
285
+
286
+ # 2.4 Letter Spacing (Tracking)
287
+ # Uppercase without tracking
288
+ if re.search(r'uppercase|text-transform:\s*uppercase', content, re.IGNORECASE):
289
+ if not re.search(r'tracking-|letter-spacing:', content):
290
+ self.warnings.append(f"[Typography] {filename}: Uppercase text without tracking. ALL CAPS needs +5-10% spacing.")
291
+
292
+ # Large text (display/hero) should have negative tracking
293
+ if re.search(r'text-(?:4xl|5xl|6xl|7xl|8xl|9xl)|font-size:\s*[3-9]\dpx', content):
294
+ if not re.search(r'tracking-tight|letter-spacing:\s*-[0-9]', content):
295
+ self.warnings.append(f"[Typography] {filename}: Large display text without tracking-tight. Big text needs -1% to -4% spacing.")
296
+
297
+ # 2.5 Weight and Emphasis - Contrast levels
298
+ # Check for adjacent weight levels (poor contrast)
299
+ weights = re.findall(r'font-weight:\s*(\d+)|font-(?:thin|extralight|light|normal|medium|semibold|bold|extrabold|black)|fw-(\d+)', content, re.IGNORECASE)
300
+ weight_values = []
301
+ for w in weights:
302
+ val = w[0] or w[1]
303
+ if val:
304
+ # Map named weights to numbers
305
+ weight_map = {'thin': '100', 'extralight': '200', 'light': '300', 'normal': '400', 'medium': '500', 'semibold': '600', 'bold': '700', 'extrabold': '800', 'black': '900'}
306
+ val = weight_map.get(val.lower(), val)
307
+ try:
308
+ weight_values.append(int(val))
309
+ except: pass
310
+
311
+ # Check for adjacent weights (400/500, 500/600, etc.)
312
+ for i in range(len(weight_values) - 1):
313
+ diff = abs(weight_values[i] - weight_values[i+1])
314
+ if diff == 100:
315
+ self.warnings.append(f"[Typography] {filename}: Adjacent font weights ({weight_values[i]}/{weight_values[i+1]}). Skip at least 2 levels for contrast.")
316
+
317
+ # Too many weight levels
318
+ unique_weights = set(weight_values)
319
+ if len(unique_weights) > 4:
320
+ self.warnings.append(f"[Typography] {filename}: {len(unique_weights)} font weights. Limit to 3-4 per page.")
321
+
322
+ # 2.6 Responsive Typography - Fluid sizing with clamp()
323
+ has_font_sizes = bool(re.search(r'font-size:|text-(?:xs|sm|base|lg|xl|2xl)', content))
324
+ if has_font_sizes and not re.search(r'clamp\(|responsive:', content):
325
+ self.warnings.append(f"[Typography] {filename}: Fixed font sizes without clamp(). Consider fluid typography: clamp(MIN, PREFERRED, MAX)")
326
+
327
+ # 2.7 Hierarchy - Heading structure
328
+ headings = re.findall(r'<(h[1-6])', content, re.IGNORECASE)
329
+ if headings:
330
+ # Check for skipped levels (h1 -> h3)
331
+ for i in range(len(headings) - 1):
332
+ curr = int(headings[i][1])
333
+ next_h = int(headings[i+1][1])
334
+ if next_h > curr + 1:
335
+ self.warnings.append(f"[Typography] {filename}: Skipped heading level (h{curr} -> h{next_h}). Maintain sequential hierarchy.")
336
+
337
+ # Check if h1 exists for main content
338
+ if 'h1' not in [h.lower() for h in headings] and has_long_text:
339
+ self.warnings.append(f"[Typography] {filename}: No h1 found. Each page should have one primary heading.")
340
+
341
+ # 2.8 Modular Scale - Consistent sizing
342
+ # Extract font-size values
343
+ font_sizes = re.findall(r'font-size:\s*(\d+(?:\.\d+)?)(px|rem|em)', content)
344
+ size_values = []
345
+ for size, unit in font_sizes:
346
+ if unit == 'rem' or unit == 'em':
347
+ size_values.append(float(size))
348
+ elif unit == 'px':
349
+ size_values.append(float(size) / 16) # Normalize to rem
350
+
351
+ if len(size_values) > 2:
352
+ # Check if sizes follow a modular scale roughly
353
+ sorted_sizes = sorted(set(size_values))
354
+ ratios = []
355
+ for i in range(1, len(sorted_sizes)):
356
+ if sorted_sizes[i-1] > 0:
357
+ ratios.append(sorted_sizes[i] / sorted_sizes[i-1])
358
+
359
+ # Common scale ratios: 1.067, 1.125, 1.2, 1.25, 1.333, 1.5, 1.618
360
+ common_ratios = {1.067, 1.125, 1.2, 1.25, 1.333, 1.5, 1.618}
361
+ for ratio in ratios[:3]: # Check first 3 ratios
362
+ if not any(abs(ratio - cr) < 0.05 for cr in common_ratios):
363
+ self.warnings.append(f"[Typography] {filename}: Font sizes may not follow modular scale (ratio: {ratio:.2f}). Consider consistent ratio like 1.25 (Major Third).")
364
+ break
365
+
366
+ # 2.9 Readability - Content chunking
367
+ # Check for very long paragraphs (>5 lines estimated)
368
+ paragraphs = re.findall(r'<p[^>]*>([^<]+)</p>', content, re.IGNORECASE)
369
+ for p in paragraphs:
370
+ word_count = len(p.split())
371
+ if word_count > 100: # ~5-6 lines
372
+ self.warnings.append(f"[Typography] {filename}: Long paragraph detected ({word_count} words). Break into 3-4 line chunks for readability.")
373
+
374
+ # Check for missing subheadings in long content
375
+ if len(paragraphs) > 5:
376
+ subheadings = len(re.findall(r'<h[2-6]', content, re.IGNORECASE))
377
+ if subheadings == 0:
378
+ self.warnings.append(f"[Typography] {filename}: Long content without subheadings. Add h2/h3 to break up text.")
379
+
380
+ # --- 3. VISUAL EFFECTS (visual-effects.md) ---
381
+
382
+ # Glassmorphism Check
383
+ if 'backdrop-filter' in content or 'blur(' in content:
384
+ if not re.search(r'background:\s*rgba|bg-opacity|bg-[a-z0-9]+\/\d+', content):
385
+ self.warnings.append(f"[Visual] {filename}: Blur used without semi-transparent background (Glassmorphism fail)")
386
+
387
+ # GPU Acceleration / Performance
388
+ if re.search(r'@keyframes|transition:', content):
389
+ expensive_props = re.findall(r'width|height|top|left|right|bottom|margin|padding', content)
390
+ if expensive_props:
391
+ self.warnings.append(f"[Performance] {filename}: Animating expensive properties ({', '.join(set(expensive_props))}). Use transform/opacity where possible.")
392
+
393
+ # Reduced Motion
394
+ if not re.search(r'prefers-reduced-motion', content):
395
+ self.warnings.append(f"[Accessibility] {filename}: Animations found without prefers-reduced-motion check")
396
+
397
+ # Natural Shadows
398
+ shadows = re.findall(r'box-shadow:\s*([^;]+)', content)
399
+ for shadow in shadows:
400
+ # Check if natural (Y > X) or multiple layers
401
+ if ',' not in shadow and not re.search(r'\d+px\s+[1-9]\d*px', shadow): # Simple heuristic for Y-offset
402
+ self.warnings.append(f"[Visual] {filename}: Simple/Unnatural shadow detected. Consider multiple layers or Y > X offset for realism.")
403
+
404
+ # --- 3.1 NEOMORPHISM CHECK ---
405
+ # Check for neomorphism patterns (dual shadows with opposite directions)
406
+ neo_shadows = re.findall(r'box-shadow:\s*([^;]+)', content)
407
+ for shadow in neo_shadows:
408
+ # Neomorphism has two shadows: positive offset + negative offset
409
+ if ',' in shadow and '-' in shadow:
410
+ # Check for inset pattern (pressed state)
411
+ if 'inset' in shadow:
412
+ self.warnings.append(f"[Visual] {filename}: Neomorphism inset detected. Ensure adequate contrast for accessibility.")
413
+
414
+ # --- 3.2 SHADOW HIERARCHY ---
415
+ # Count shadow levels to check for elevation consistency
416
+ shadow_count = len(shadows)
417
+ if shadow_count > 0:
418
+ # Check for shadow opacity levels (should indicate hierarchy)
419
+ opacities = re.findall(r'rgba?\([^)]+,\s*([\d.]+)\)', content)
420
+ shadow_opacities = [float(o) for o in opacities if float(o) < 0.5]
421
+ if shadow_count >= 3 and len(shadow_opacities) > 0:
422
+ # Check if there's variety in shadow opacities for different elevations
423
+ unique_opacities = len(set(shadow_opacities))
424
+ if unique_opacities < 2:
425
+ self.warnings.append(f"[Visual] {filename}: All shadows at same opacity level. Vary shadow intensity for elevation hierarchy.")
426
+
427
+ # --- 3.3 GRADIENT CHECKS ---
428
+ # Check for gradient usage
429
+ has_gradient = bool(re.search(r'gradient|linear-gradient|radial-gradient|conic-gradient', content))
430
+ if has_gradient:
431
+ # Warn about mesh/aurora gradients (can be overused)
432
+ gradient_count = len(re.findall(r'gradient', content, re.IGNORECASE))
433
+ if gradient_count > 5:
434
+ self.warnings.append(f"[Visual] {filename}: Many gradients detected ({gradient_count}). Ensure this serves purpose, not decoration.")
435
+ else:
436
+ # Check if hero section exists without gradient
437
+ if has_hero and not re.search(r'background:|bg-', content):
438
+ self.warnings.append(f"[Visual] {filename}: Hero section without visual interest. Consider gradient for depth.")
439
+
440
+ # --- 3.4 BORDER EFFECTS ---
441
+ # Check for gradient borders or animated borders
442
+ has_border = bool(re.search(r'border:|border-', content))
443
+ if has_border:
444
+ # Check for overly complex borders
445
+ border_count = len(re.findall(r'border:', content))
446
+ if border_count > 8:
447
+ self.warnings.append(f"[Visual] {filename}: Many border declarations ({border_count}). Simplify for cleaner look.")
448
+
449
+ # --- 3.5 GLOW EFFECTS ---
450
+ # Check for text-shadow or multiple box-shadow layers (glow effects)
451
+ text_shadows = re.findall(r'text-shadow:', content)
452
+ for ts in text_shadows:
453
+ # Multiple text-shadow layers indicate glow
454
+ if ',' in ts:
455
+ self.warnings.append(f"[Visual] {filename}: Text glow effect detected. Ensure readability is maintained.")
456
+
457
+ # Check for box-shadow glow (multiple layers with 0 offset)
458
+ glow_shadows = re.findall(r'box-shadow:\s*[^;]*0\s+0\s+', content)
459
+ if len(glow_shadows) > 2:
460
+ self.warnings.append(f"[Visual] {filename}: Multiple glow effects detected. Use sparingly for emphasis only.")
461
+
462
+ # --- 3.6 OVERLAY TECHNIQUES ---
463
+ # Check for image overlays (for readability)
464
+ has_images = bool(re.search(r'<img|background-image:|bg-\[url', content))
465
+ if has_images and has_long_text:
466
+ has_overlay = bool(re.search(r'overlay|rgba\(0|gradient.*transparent|::after|::before', content))
467
+ if not has_overlay:
468
+ self.warnings.append(f"[Visual] {filename}: Text over image without overlay. Add gradient overlay for readability.")
469
+
470
+ # --- 3.7 PERFORMANCE: will-change ---
471
+ # Check for will-change usage
472
+ if re.search(r'will-change:', content):
473
+ will_change_props = re.findall(r'will-change:\s*([^;]+)', content)
474
+ for prop in will_change_props:
475
+ prop = prop.strip().lower()
476
+ if prop in ['width', 'height', 'top', 'left', 'right', 'bottom', 'margin', 'padding']:
477
+ self.issues.append(f"[Performance] {filename}: will-change on '{prop}' (layout property). Use only for transform/opacity.")
478
+
479
+ # Check for excessive will-change usage
480
+ will_change_count = len(re.findall(r'will-change:', content))
481
+ if will_change_count > 3:
482
+ self.warnings.append(f"[Performance] {filename}: Many will-change declarations ({will_change_count}). Use sparingly, only for heavy animations.")
483
+
484
+ # --- 3.8 EFFECT SELECTION ---
485
+ # Check for effect overuse (too many visual effects)
486
+ effect_count = (
487
+ (1 if has_gradient else 0) +
488
+ shadow_count +
489
+ len(re.findall(r'backdrop-filter|blur\(', content)) +
490
+ len(re.findall(r'text-shadow:', content))
491
+ )
492
+ if effect_count > 10:
493
+ self.warnings.append(f"[Visual] {filename}: Many visual effects ({effect_count}). Ensure effects serve purpose, not decoration.")
494
+
495
+ # Check for static/flat design (no depth)
496
+ if has_long_text and effect_count == 0:
497
+ self.warnings.append(f"[Visual] {filename}: Flat design with no depth. Consider shadows or subtle gradients for hierarchy.")
498
+
499
+ # --- 4. COLOR SYSTEM (color-system.md) ---
500
+
501
+ # 4.1 PURPLE BAN - Critical check from color-system.md
502
+ purple_hexes = ['#8B5CF6', '#A855F7', '#9333EA', '#7C3AED', '#6D28D9',
503
+ '#8B5CF6', '#A78BFA', '#C4B5FD', '#DDD6FE', '#EDE9FE',
504
+ '#8b5cf6', '#a855f7', '#9333ea', '#7c3aed', '#6d28d9',
505
+ 'purple', 'violet', 'fuchsia', 'magenta', 'lavender']
506
+ for purple in purple_hexes:
507
+ if purple.lower() in content.lower():
508
+ self.issues.append(f"[Color] {filename}: PURPLE DETECTED ('{purple}'). Banned by Maestro rules. Use Teal/Cyan/Emerald instead.")
509
+ break
510
+
511
+ # 4.2 60-30-10 Rule check
512
+ # Count color usage to estimate ratio
513
+ color_hex_count = len(re.findall(r'#[0-9a-fA-F]{3,6}', content))
514
+ hsl_count = len(re.findall(r'hsl\(', content))
515
+ total_colors = color_hex_count + hsl_count
516
+ if total_colors > 3:
517
+ # Check for dominant colors (should be ~60%)
518
+ bg_declarations = re.findall(r'(?:background|bg-|bg\[)([^;}\s]+)', content)
519
+ text_declarations = re.findall(r'(?:color|text-)([^;}\s]+)', content)
520
+ if len(bg_declarations) > 0 and len(text_declarations) > 0:
521
+ # Just warn if too many distinct colors
522
+ unique_hexes = set(re.findall(r'#[0-9a-fA-F]{6}', content))
523
+ if len(unique_hexes) > 5:
524
+ self.warnings.append(f"[Color] {filename}: {len(unique_hexes)} distinct colors. Consider 60-30-10 rule: dominant (60%), secondary (30%), accent (10%).")
525
+
526
+ # 4.3 Color Scheme Pattern Detection
527
+ # Detect monochromatic (same hue, different lightness)
528
+ hsl_matches = re.findall(r'hsl\((\d+),\s*\d+%,\s*\d+%\)', content)
529
+ if len(hsl_matches) >= 3:
530
+ hues = [int(h) for h in hsl_matches]
531
+ hue_range = max(hues) - min(hues)
532
+ if hue_range < 10:
533
+ self.warnings.append(f"[Color] {filename}: Monochromatic palette detected (hue variance: {hue_range}deg). Ensure adequate contrast.")
534
+
535
+ # 4.4 Dark Mode Compliance
536
+ # Check for pure black (#000000) or pure white (#FFFFFF) text (forbidden)
537
+ if re.search(r'color:\s*#000000|#000\b', content):
538
+ self.warnings.append(f"[Color] {filename}: Pure black (#000000) detected. Use #1a1a1a or darker grays for better dark mode.")
539
+ if re.search(r'background:\s*#ffffff|#fff\b', content) and re.search(r'dark:\s*|dark:', content):
540
+ self.warnings.append(f"[Color] {filename}: Pure white background in dark mode context. Use slight off-white (#f9fafb) for reduced eye strain.")
541
+
542
+ # 4.5 WCAG Contrast Pattern Check
543
+ # Look for potential low-contrast combinations
544
+ light_bg_light_text = bool(re.search(r'bg-(?:gray|slate|zinc)-50|bg-white.*text-(?:gray|slate)-[12]', content))
545
+ dark_bg_dark_text = bool(re.search(r'bg-(?:gray|slate|zinct)-9|bg-black.*text-(?:gray|slate)-[89]', content))
546
+ if light_bg_light_text or dark_bg_dark_text:
547
+ self.warnings.append(f"[Color] {filename}: Possible low-contrast combination detected. Verify WCAG AA (4.5:1 for text).")
548
+
549
+ # 4.6 Color Psychology Context Check
550
+ # Warn if blue used for food/restaurant context
551
+ has_blue = bool(re.search(r'bg-blue|text-blue|from-blue|#[0-9a-fA-F]*00[0-9A-Fa-f]{2}|#[0-9a-fA-F]*1[0-9A-Fa-f]{2}', content))
552
+ has_food_context = bool(re.search(r'restaurant|food|cooking|recipe|menu|dish|meal', content, re.IGNORECASE))
553
+ if has_blue and has_food_context:
554
+ self.warnings.append(f"[Color] {filename}: Blue color in food context. Blue suppresses appetite; consider warm colors (red, orange, yellow).")
555
+
556
+ # 4.7 HSL-Based Palette Detection
557
+ # Check if using HSL for palette (recommended in color-system.md)
558
+ has_color_vars = bool(re.search(r'--color-|color-|primary-|secondary-', content))
559
+ if has_color_vars and not re.search(r'hsl\(', content):
560
+ self.warnings.append(f"[Color] {filename}: Color variables without HSL. Consider HSL for easier palette adjustment (Hue, Saturation, Lightness).")
561
+
562
+ # --- 5. ANIMATION GUIDE (animation-guide.md) ---
563
+
564
+ # 5.1 Duration Appropriateness
565
+ # Check for excessively long or short animations
566
+ durations = re.findall(r'(?:duration|animation-duration|transition-duration):\s*([\d.]+)(s|ms)', content)
567
+ for duration, unit in durations:
568
+ duration_ms = float(duration) * (1000 if unit == 's' else 1)
569
+ if duration_ms < 50:
570
+ self.warnings.append(f"[Animation] {filename}: Very fast animation ({duration}{unit}). Minimum 50ms for visibility.")
571
+ elif duration_ms > 1000 and 'transition' in content.lower():
572
+ self.warnings.append(f"[Animation] {filename}: Long transition ({duration}{unit}). Transitions should be 100-300ms for responsiveness.")
573
+
574
+ # 5.2 Easing Function Correctness
575
+ # Check for incorrect easing patterns
576
+ if re.search(r'ease-in\s+.*entry|fade-in.*ease-in', content):
577
+ self.warnings.append(f"[Animation] {filename}: Entry animation with ease-in. Entry should use ease-out for snappy feel.")
578
+ if re.search(r'ease-out\s+.*exit|fade-out.*ease-out', content):
579
+ self.warnings.append(f"[Animation] {filename}: Exit animation with ease-out. Exit should use ease-in for natural feel.")
580
+
581
+ # 5.3 Micro-interaction Feedback Patterns
582
+ # Check for interactive elements without hover/focus states
583
+ interactive_elements = len(re.findall(r'<button|<a\s+href|onClick|@click', content))
584
+ has_hover_focus = bool(re.search(r'hover:|focus:|:hover|:focus', content))
585
+ if interactive_elements > 2 and not has_hover_focus:
586
+ self.warnings.append(f"[Animation] {filename}: Interactive elements without hover/focus states. Add micro-interactions for feedback.")
587
+
588
+ # 5.4 Loading State Indicators
589
+ # Check for loading patterns
590
+ has_async = bool(re.search(r'async|await|fetch|axios|loading|isLoading', content))
591
+ has_loading_indicator = bool(re.search(r'skeleton|spinner|progress|loading|<circle.*animate', content))
592
+ if has_async and not has_loading_indicator:
593
+ self.warnings.append(f"[Animation] {filename}: Async operations without loading indicator. Add skeleton or spinner for perceived performance.")
594
+
595
+ # 5.5 Page Transition Patterns
596
+ # Check for page/view transitions
597
+ has_routing = bool(re.search(r'router|navigate|Link.*to|useHistory', content))
598
+ has_page_transition = bool(re.search(r'AnimatePresence|motion\.|transition.*page|fade.*route', content))
599
+ if has_routing and not has_page_transition:
600
+ self.warnings.append(f"[Animation] {filename}: Routing detected without page transitions. Consider fade/slide for context continuity.")
601
+
602
+ # 5.6 Scroll Animation Performance
603
+ # Check for scroll-driven animations
604
+ has_scroll_anim = bool(re.search(r'onScroll|scroll.*trigger|IntersectionObserver', content))
605
+ if has_scroll_anim:
606
+ # Check if using expensive properties in scroll handlers
607
+ if re.search(r'onScroll.*[^\w](width|height|top|left)', content):
608
+ self.issues.append(f"[Animation] {filename}: Scroll handler animating layout properties. Use transform/opacity for 60fps.")
609
+
610
+ # --- 6. MOTION GRAPHICS (motion-graphics.md) ---
611
+
612
+ # 6.1 Lottie Animation Checks
613
+ has_lottie = bool(re.search(r'lottie|Lottie|@lottie-react', content))
614
+ if has_lottie:
615
+ # Check for reduced motion fallback
616
+ has_lottie_fallback = bool(re.search(r'prefers-reduced-motion.*lottie|lottie.*isPaused|lottie.*stop', content))
617
+ if not has_lottie_fallback:
618
+ self.warnings.append(f"[Motion] {filename}: Lottie animation without reduced-motion fallback. Add pause/stop for accessibility.")
619
+
620
+ # 6.2 GSAP Memory Leak Risks
621
+ has_gsap = bool(re.search(r'gsap|ScrollTrigger|from\(.*gsap', content))
622
+ if has_gsap:
623
+ # Check for cleanup patterns
624
+ has_gsap_cleanup = bool(re.search(r'kill\(|revert\(|useEffect.*return.*gsap', content))
625
+ if not has_gsap_cleanup:
626
+ self.issues.append(f"[Motion] {filename}: GSAP animation without cleanup (kill/revert). Memory leak risk on unmount.")
627
+
628
+ # 6.3 SVG Animation Performance
629
+ svg_animations = re.findall(r'<animate|<animateTransform|stroke-dasharray|stroke-dashoffset', content)
630
+ if len(svg_animations) > 3:
631
+ self.warnings.append(f"[Motion] {filename}: Multiple SVG animations detected. Ensure stroke-dashoffset is used sparingly for mobile performance.")
632
+
633
+ # 6.4 3D Transform Performance
634
+ has_3d_transform = bool(re.search(r'transform3d|perspective\(|rotate3d|translate3d', content))
635
+ if has_3d_transform:
636
+ # Check for perspective on parent
637
+ has_perspective_parent = bool(re.search(r'perspective:\s*\d+px|perspective\s*\(', content))
638
+ if not has_perspective_parent:
639
+ self.warnings.append(f"[Motion] {filename}: 3D transform without perspective parent. Add perspective: 1000px for realistic depth.")
640
+
641
+ # Warn about mobile performance
642
+ self.warnings.append(f"[Motion] {filename}: 3D transforms detected. Test on mobile; can impact performance on low-end devices.")
643
+
644
+ # 6.5 Particle Effect Warnings
645
+ # Check for canvas/WebGL particle systems
646
+ has_particles = bool(re.search(r'particle|canvas.*loop|requestAnimationFrame.*draw|Three\.js', content))
647
+ if has_particles:
648
+ self.warnings.append(f"[Motion] {filename}: Particle effects detected. Ensure fallback or reduced-quality option for mobile devices.")
649
+
650
+ # 6.6 Scroll-Driven Animation Performance
651
+ has_scroll_driven = bool(re.search(r'IntersectionObserver.*animate|scroll.*progress|view-timeline', content))
652
+ if has_scroll_driven:
653
+ # Check for throttling/debouncing
654
+ has_throttle = bool(re.search(r'throttle|debounce|requestAnimationFrame', content))
655
+ if not has_throttle:
656
+ self.issues.append(f"[Motion] {filename}: Scroll-driven animation without throttling. Add requestAnimationFrame for 60fps.")
657
+
658
+ # 6.7 Motion Decision Tree - Context Check
659
+ # Check if animation serves purpose (not just decoration)
660
+ total_animations = (
661
+ len(re.findall(r'@keyframes|transition:|animate-', content)) +
662
+ (1 if has_lottie else 0) +
663
+ (1 if has_gsap else 0)
664
+ )
665
+ if total_animations > 5:
666
+ # Check if animations are functional
667
+ functional_animations = len(re.findall(r'hover:|focus:|disabled|loading|error|success', content))
668
+ if functional_animations < total_animations / 2:
669
+ self.warnings.append(f"[Motion] {filename}: Many animations ({total_animations}). Ensure majority serve functional purpose (feedback, guidance), not decoration.")
670
+
671
+ # --- 7. ACCESSIBILITY ---
672
+ if re.search(r'<img(?![^>]*alt=)[^>]*>', content):
673
+ self.issues.append(f"[Accessibility] {filename}: Missing img alt text")
674
+
675
+ def audit_directory(self, directory: str) -> None:
676
+ extensions = {'.tsx', '.jsx', '.html', '.vue', '.svelte', '.css'}
677
+ for root, dirs, files in os.walk(directory):
678
+ dirs[:] = [d for d in dirs if d not in {'node_modules', '.git', 'dist', 'build', '.next', 'reference'}]
679
+ for file in files:
680
+ if Path(file).suffix in extensions:
681
+ # Skip files in Footer or non-page directories for general Hick's Law
682
+ if 'footer' in root.lower() or 'ui' in root.lower():
683
+ # Still audit but with more context? For now just audit file.
684
+ pass
685
+ self.audit_file(os.path.join(root, file))
686
+
687
+ def get_report(self):
688
+ return {
689
+ "files_checked": self.files_checked,
690
+ "issues": self.issues,
691
+ "warnings": self.warnings,
692
+ "passed_checks": self.passed_count,
693
+ "compliant": len(self.issues) == 0
694
+ }
695
+
696
+ def main():
697
+ if len(sys.argv) < 2: sys.exit(1)
698
+
699
+ path = sys.argv[1]
700
+ is_json = "--json" in sys.argv
701
+
702
+ auditor = UXAuditor()
703
+ if os.path.isfile(path): auditor.audit_file(path)
704
+ else: auditor.audit_directory(path)
705
+
706
+ report = auditor.get_report()
707
+
708
+ if is_json:
709
+ print(json.dumps(report))
710
+ else:
711
+ # Use ASCII-safe output for Windows console compatibility
712
+ print(f"\n[UX AUDIT] {report['files_checked']} files checked")
713
+ print("-" * 50)
714
+ if report['issues']:
715
+ print(f"[!] ISSUES ({len(report['issues'])}):")
716
+ for i in report['issues'][:10]: print(f" - {i}")
717
+ if report['warnings']:
718
+ print(f"[*] WARNINGS ({len(report['warnings'])}):")
719
+ for w in report['warnings'][:15]: print(f" - {w}")
720
+ print(f"[+] PASSED CHECKS: {report['passed_checks']}")
721
+ status = "PASS" if report['compliant'] else "FAIL"
722
+ print(f"STATUS: {status}")
723
+
724
+ sys.exit(0 if report['compliant'] else 1)
725
+
726
+ if __name__ == "__main__":
727
+ main()