@heyai-rules/pilo-masterkit 2.1.0 → 3.1.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.
Files changed (739) hide show
  1. package/.agent/agents/PILO_MASTER.md +77 -77
  2. package/.agent/agents/architect.md +211 -211
  3. package/.agent/agents/backend-specialist.md +263 -263
  4. package/.agent/agents/build-error-resolver.md +114 -114
  5. package/.agent/agents/chief-of-staff.md +151 -151
  6. package/.agent/agents/code-archaeologist.md +106 -106
  7. package/.agent/agents/code-reviewer.md +237 -237
  8. package/.agent/agents/cpp-build-resolver.md +90 -90
  9. package/.agent/agents/cpp-reviewer.md +72 -72
  10. package/.agent/agents/csharp-reviewer.md +101 -101
  11. package/.agent/agents/dart-build-resolver.md +201 -201
  12. package/.agent/agents/database-architect.md +226 -226
  13. package/.agent/agents/database-reviewer.md +91 -91
  14. package/.agent/agents/debugger.md +225 -225
  15. package/.agent/agents/devops-engineer.md +242 -242
  16. package/.agent/agents/doc-updater.md +107 -107
  17. package/.agent/agents/docs-lookup.md +68 -68
  18. package/.agent/agents/documentation-writer.md +104 -104
  19. package/.agent/agents/e2e-runner.md +107 -107
  20. package/.agent/agents/explorer-agent.md +73 -73
  21. package/.agent/agents/flutter-reviewer.md +243 -243
  22. package/.agent/agents/frontend-specialist.md +593 -593
  23. package/.agent/agents/game-developer.md +162 -162
  24. package/.agent/agents/gan-evaluator.md +209 -209
  25. package/.agent/agents/gan-generator.md +131 -131
  26. package/.agent/agents/gan-planner.md +99 -99
  27. package/.agent/agents/go-build-resolver.md +94 -94
  28. package/.agent/agents/go-reviewer.md +76 -76
  29. package/.agent/agents/harness-optimizer.md +35 -35
  30. package/.agent/agents/healthcare-reviewer.md +83 -83
  31. package/.agent/agents/java-build-resolver.md +153 -153
  32. package/.agent/agents/java-reviewer.md +92 -92
  33. package/.agent/agents/kotlin-build-resolver.md +118 -118
  34. package/.agent/agents/kotlin-reviewer.md +159 -159
  35. package/.agent/agents/loop-operator.md +36 -36
  36. package/.agent/agents/mobile-developer.md +377 -377
  37. package/.agent/agents/opensource-forker.md +198 -198
  38. package/.agent/agents/opensource-packager.md +249 -249
  39. package/.agent/agents/opensource-sanitizer.md +188 -188
  40. package/.agent/agents/orchestrator.md +416 -416
  41. package/.agent/agents/penetration-tester.md +188 -188
  42. package/.agent/agents/performance-optimizer.md +446 -446
  43. package/.agent/agents/personas/athena-agent/agent.json +10 -10
  44. package/.agent/agents/personas/athena-agent/athena-backend-logic-architecture-profile.md +3 -3
  45. package/.agent/agents/personas/athena-agent/context-files/agents.md +1 -1
  46. package/.agent/agents/personas/athena-agent/context-files/identity.md +1 -1
  47. package/.agent/agents/personas/athena-agent/context-files/soul.md +1 -1
  48. package/.agent/agents/personas/athena-agent/context-files/user-predefined.md +1 -1
  49. package/.agent/agents/personas/athena-agent/user-context-files/system/bootstrap.md +1 -1
  50. package/.agent/agents/personas/athena-agent/user-context-files/system/user.md +1 -1
  51. package/.agent/agents/personas/da-vinci-agent/agent.json +10 -10
  52. package/.agent/agents/personas/da-vinci-agent/context-files/agents.md +1 -1
  53. package/.agent/agents/personas/da-vinci-agent/context-files/identity.md +1 -1
  54. package/.agent/agents/personas/da-vinci-agent/context-files/soul.md +1 -1
  55. package/.agent/agents/personas/da-vinci-agent/context-files/user-predefined.md +1 -1
  56. package/.agent/agents/personas/da-vinci-agent/da-vinci-frontend-ui-ux-design-profile.md +3 -3
  57. package/.agent/agents/personas/da-vinci-agent/user-context-files/system/bootstrap.md +1 -1
  58. package/.agent/agents/personas/da-vinci-agent/user-context-files/system/user.md +1 -1
  59. package/.agent/agents/personas/duong-tang-agent/agent.json +10 -10
  60. package/.agent/agents/personas/duong-tang-agent/context-files/agents.md +1 -1
  61. package/.agent/agents/personas/duong-tang-agent/context-files/identity.md +1 -1
  62. package/.agent/agents/personas/duong-tang-agent/context-files/soul.md +1 -1
  63. package/.agent/agents/personas/duong-tang-agent/context-files/user-predefined.md +1 -1
  64. package/.agent/agents/personas/duong-tang-agent/tang-monk-quality-testing-documentation-profile.md +3 -3
  65. package/.agent/agents/personas/duong-tang-agent/user-context-files/system/bootstrap.md +1 -1
  66. package/.agent/agents/personas/duong-tang-agent/user-context-files/system/user.md +1 -1
  67. package/.agent/agents/personas/gia-cat-luong-agent/agent.json +10 -10
  68. package/.agent/agents/personas/gia-cat-luong-agent/context-files/agents.md +1 -1
  69. package/.agent/agents/personas/gia-cat-luong-agent/context-files/identity.md +1 -1
  70. package/.agent/agents/personas/gia-cat-luong-agent/context-files/soul.md +1 -1
  71. package/.agent/agents/personas/gia-cat-luong-agent/context-files/user-predefined.md +1 -1
  72. package/.agent/agents/personas/gia-cat-luong-agent/kongming-research-strategy-analysis-profile.md +3 -3
  73. package/.agent/agents/personas/gia-cat-luong-agent/user-context-files/system/bootstrap.md +1 -1
  74. package/.agent/agents/personas/gia-cat-luong-agent/user-context-files/system/user.md +1 -1
  75. package/.agent/agents/personas/mihata-agent/agent.json +10 -10
  76. package/.agent/agents/personas/mihata-agent/context-files/agents.md +1 -1
  77. package/.agent/agents/personas/mihata-agent/context-files/identity.md +1 -1
  78. package/.agent/agents/personas/mihata-agent/context-files/soul.md +1 -1
  79. package/.agent/agents/personas/mihata-agent/context-files/user-predefined.md +1 -1
  80. package/.agent/agents/personas/mihata-agent/mihata-multi-agent-orchestration-profile.md +3 -3
  81. package/.agent/agents/personas/mihata-agent/user-context-files/system/bootstrap.md +1 -1
  82. package/.agent/agents/personas/mihata-agent/user-context-files/system/user.md +1 -1
  83. package/.agent/agents/personas/tesla-agent/agent.json +10 -10
  84. package/.agent/agents/personas/tesla-agent/context-files/agents.md +1 -1
  85. package/.agent/agents/personas/tesla-agent/context-files/identity.md +1 -1
  86. package/.agent/agents/personas/tesla-agent/context-files/soul.md +1 -1
  87. package/.agent/agents/personas/tesla-agent/context-files/user-predefined.md +1 -1
  88. package/.agent/agents/personas/tesla-agent/tesla-fullstack-system-optimization-profile.md +3 -3
  89. package/.agent/agents/personas/tesla-agent/user-context-files/system/bootstrap.md +1 -1
  90. package/.agent/agents/personas/tesla-agent/user-context-files/system/user.md +1 -1
  91. package/.agent/agents/personas/tu-ma-y-agent/agent.json +10 -10
  92. package/.agent/agents/personas/tu-ma-y-agent/context-files/agents.md +1 -1
  93. package/.agent/agents/personas/tu-ma-y-agent/context-files/identity.md +1 -1
  94. package/.agent/agents/personas/tu-ma-y-agent/context-files/soul.md +1 -1
  95. package/.agent/agents/personas/tu-ma-y-agent/context-files/user-predefined.md +1 -1
  96. package/.agent/agents/personas/tu-ma-y-agent/simayi-feasibility-risk-control-profile.md +3 -3
  97. package/.agent/agents/personas/tu-ma-y-agent/user-context-files/system/bootstrap.md +1 -1
  98. package/.agent/agents/personas/tu-ma-y-agent/user-context-files/system/user.md +1 -1
  99. package/.agent/agents/personas/venti-agent/agent.json +10 -10
  100. package/.agent/agents/personas/venti-agent/context-files/agents.md +1 -1
  101. package/.agent/agents/personas/venti-agent/context-files/identity.md +1 -1
  102. package/.agent/agents/personas/venti-agent/context-files/soul.md +1 -1
  103. package/.agent/agents/personas/venti-agent/context-files/user-predefined.md +1 -1
  104. package/.agent/agents/personas/venti-agent/user-context-files/system/bootstrap.md +1 -1
  105. package/.agent/agents/personas/venti-agent/user-context-files/system/user.md +1 -1
  106. package/.agent/agents/personas/venti-agent/venti-learning-communication-mentoring-profile.md +3 -3
  107. package/.agent/agents/planner.md +212 -212
  108. package/.agent/agents/product-manager.md +112 -112
  109. package/.agent/agents/product-owner.md +95 -95
  110. package/.agent/agents/project-planner.md +406 -406
  111. package/.agent/agents/python-reviewer.md +98 -98
  112. package/.agent/agents/pytorch-build-resolver.md +120 -120
  113. package/.agent/agents/qa-automation-engineer.md +103 -103
  114. package/.agent/agents/refactor-cleaner.md +85 -85
  115. package/.agent/agents/rust-build-resolver.md +148 -148
  116. package/.agent/agents/rust-reviewer.md +94 -94
  117. package/.agent/agents/security-auditor.md +170 -170
  118. package/.agent/agents/security-reviewer.md +108 -108
  119. package/.agent/agents/seo-specialist.md +111 -111
  120. package/.agent/agents/tdd-guide.md +91 -91
  121. package/.agent/agents/test-engineer.md +158 -158
  122. package/.agent/agents/typescript-reviewer.md +112 -112
  123. package/.agent/contexts/dev.md +20 -20
  124. package/.agent/contexts/research.md +26 -26
  125. package/.agent/contexts/review.md +22 -22
  126. package/.agent/hooks/hooks.json +395 -395
  127. package/.agent/hooks/readme.md +222 -222
  128. package/.agent/mcp-configs/mcp-servers.json +181 -181
  129. package/.agent/rules/ARCHITECTURAL_BLUEPRINTS.md +62 -62
  130. package/.agent/rules/CODE_CRAFTSMANSHIP.md +69 -69
  131. package/.agent/rules/CORE_RULES.md +72 -72
  132. package/.agent/rules/PROJECT_MAP.md +58 -58
  133. package/.agent/rules/QUALITY_ASSURANCE.md +54 -54
  134. package/.agent/rules/SECURITY_ARMOR.md +44 -44
  135. package/.agent/rules/VERSION_ORCHESTRATION.md +64 -64
  136. package/.agent/rules/WORKFLOW_ORCHESTRATION.md +55 -55
  137. package/.agent/rules/common/agents.md +50 -50
  138. package/.agent/rules/common/code-review.md +124 -124
  139. package/.agent/rules/common/coding-style.md +48 -48
  140. package/.agent/rules/common/development-workflow.md +44 -44
  141. package/.agent/rules/common/git-workflow.md +24 -24
  142. package/.agent/rules/common/hooks.md +30 -30
  143. package/.agent/rules/common/patterns.md +31 -31
  144. package/.agent/rules/common/performance.md +55 -55
  145. package/.agent/rules/common/security.md +29 -29
  146. package/.agent/rules/common/testing.md +29 -29
  147. package/.agent/rules/cpp/coding-style.md +44 -44
  148. package/.agent/rules/cpp/hooks.md +39 -39
  149. package/.agent/rules/cpp/patterns.md +51 -51
  150. package/.agent/rules/cpp/security.md +51 -51
  151. package/.agent/rules/cpp/testing.md +44 -44
  152. package/.agent/rules/csharp/coding-style.md +72 -72
  153. package/.agent/rules/csharp/hooks.md +25 -25
  154. package/.agent/rules/csharp/patterns.md +50 -50
  155. package/.agent/rules/csharp/security.md +58 -58
  156. package/.agent/rules/csharp/testing.md +46 -46
  157. package/.agent/rules/dart/coding-style.md +159 -159
  158. package/.agent/rules/dart/hooks.md +66 -66
  159. package/.agent/rules/dart/patterns.md +261 -261
  160. package/.agent/rules/dart/security.md +135 -135
  161. package/.agent/rules/dart/testing.md +215 -215
  162. package/.agent/rules/golang/coding-style.md +32 -32
  163. package/.agent/rules/golang/hooks.md +17 -17
  164. package/.agent/rules/golang/patterns.md +45 -45
  165. package/.agent/rules/golang/security.md +34 -34
  166. package/.agent/rules/golang/testing.md +31 -31
  167. package/.agent/rules/java/coding-style.md +114 -114
  168. package/.agent/rules/java/hooks.md +18 -18
  169. package/.agent/rules/java/patterns.md +146 -146
  170. package/.agent/rules/java/security.md +100 -100
  171. package/.agent/rules/java/testing.md +131 -131
  172. package/.agent/rules/kotlin/coding-style.md +86 -86
  173. package/.agent/rules/kotlin/hooks.md +17 -17
  174. package/.agent/rules/kotlin/patterns.md +146 -146
  175. package/.agent/rules/kotlin/security.md +82 -82
  176. package/.agent/rules/kotlin/testing.md +128 -128
  177. package/.agent/rules/perl/coding-style.md +46 -46
  178. package/.agent/rules/perl/hooks.md +22 -22
  179. package/.agent/rules/perl/patterns.md +76 -76
  180. package/.agent/rules/perl/security.md +69 -69
  181. package/.agent/rules/perl/testing.md +54 -54
  182. package/.agent/rules/php/coding-style.md +40 -40
  183. package/.agent/rules/php/hooks.md +24 -24
  184. package/.agent/rules/php/patterns.md +33 -33
  185. package/.agent/rules/php/security.md +37 -37
  186. package/.agent/rules/php/testing.md +39 -39
  187. package/.agent/rules/python/coding-style.md +42 -42
  188. package/.agent/rules/python/hooks.md +19 -19
  189. package/.agent/rules/python/patterns.md +39 -39
  190. package/.agent/rules/python/security.md +30 -30
  191. package/.agent/rules/python/testing.md +38 -38
  192. package/.agent/rules/readme.md +111 -111
  193. package/.agent/rules/rust/coding-style.md +151 -151
  194. package/.agent/rules/rust/hooks.md +16 -16
  195. package/.agent/rules/rust/patterns.md +168 -168
  196. package/.agent/rules/rust/security.md +141 -141
  197. package/.agent/rules/rust/testing.md +154 -154
  198. package/.agent/rules/swift/coding-style.md +47 -47
  199. package/.agent/rules/swift/hooks.md +20 -20
  200. package/.agent/rules/swift/patterns.md +66 -66
  201. package/.agent/rules/swift/security.md +33 -33
  202. package/.agent/rules/swift/testing.md +45 -45
  203. package/.agent/rules/typescript/coding-style.md +199 -199
  204. package/.agent/rules/typescript/hooks.md +22 -22
  205. package/.agent/rules/typescript/patterns.md +52 -52
  206. package/.agent/rules/typescript/security.md +28 -28
  207. package/.agent/rules/typescript/testing.md +18 -18
  208. package/.agent/rules/web/coding-style.md +96 -96
  209. package/.agent/rules/web/design-quality.md +63 -63
  210. package/.agent/rules/web/hooks.md +120 -120
  211. package/.agent/rules/web/patterns.md +79 -79
  212. package/.agent/rules/web/performance.md +64 -64
  213. package/.agent/rules/web/security.md +57 -57
  214. package/.agent/rules/web/testing.md +55 -55
  215. package/.agent/rules/zh/agents.md +50 -50
  216. package/.agent/rules/zh/code-review.md +124 -124
  217. package/.agent/rules/zh/coding-style.md +48 -48
  218. package/.agent/rules/zh/development-workflow.md +44 -44
  219. package/.agent/rules/zh/git-workflow.md +24 -24
  220. package/.agent/rules/zh/hooks.md +30 -30
  221. package/.agent/rules/zh/patterns.md +31 -31
  222. package/.agent/rules/zh/performance.md +55 -55
  223. package/.agent/rules/zh/readme.md +108 -108
  224. package/.agent/rules/zh/security.md +29 -29
  225. package/.agent/rules/zh/testing.md +29 -29
  226. package/.agent/scripts/auto_preview.py +148 -148
  227. package/.agent/scripts/checklist.py +217 -217
  228. package/.agent/scripts/session_manager.py +120 -120
  229. package/.agent/scripts/verify_all.py +327 -327
  230. package/.agent/skills/agent-eval/SKILL.md +145 -145
  231. package/.agent/skills/agent-harness-construction/SKILL.md +73 -73
  232. package/.agent/skills/agent-payment-x402/SKILL.md +178 -178
  233. package/.agent/skills/agentic-engineering/SKILL.md +63 -63
  234. package/.agent/skills/ai-first-engineering/SKILL.md +51 -51
  235. package/.agent/skills/ai-regression-testing/SKILL.md +385 -385
  236. package/.agent/skills/android-clean-architecture/SKILL.md +339 -339
  237. package/.agent/skills/api-design/SKILL.md +523 -523
  238. package/.agent/skills/api-patterns/SKILL.md +81 -81
  239. package/.agent/skills/api-patterns/api-style.md +42 -42
  240. package/.agent/skills/api-patterns/auth.md +24 -24
  241. package/.agent/skills/api-patterns/documentation.md +26 -26
  242. package/.agent/skills/api-patterns/graphql.md +41 -41
  243. package/.agent/skills/api-patterns/rate-limiting.md +31 -31
  244. package/.agent/skills/api-patterns/response.md +37 -37
  245. package/.agent/skills/api-patterns/rest.md +40 -40
  246. package/.agent/skills/api-patterns/scripts/api_validator.py +211 -211
  247. package/.agent/skills/api-patterns/security-testing.md +122 -122
  248. package/.agent/skills/api-patterns/trpc.md +41 -41
  249. package/.agent/skills/api-patterns/versioning.md +22 -22
  250. package/.agent/skills/app-builder/SKILL.md +75 -75
  251. package/.agent/skills/app-builder/agent-coordination.md +71 -71
  252. package/.agent/skills/app-builder/feature-building.md +53 -53
  253. package/.agent/skills/app-builder/project-detection.md +34 -34
  254. package/.agent/skills/app-builder/scaffolding.md +118 -118
  255. package/.agent/skills/app-builder/tech-stack.md +41 -41
  256. package/.agent/skills/app-builder/templates/SKILL.md +39 -39
  257. package/.agent/skills/app-builder/templates/astro-static/TEMPLATE.md +76 -76
  258. package/.agent/skills/app-builder/templates/chrome-extension/TEMPLATE.md +92 -92
  259. package/.agent/skills/app-builder/templates/cli-tool/TEMPLATE.md +88 -88
  260. package/.agent/skills/app-builder/templates/electron-desktop/TEMPLATE.md +88 -88
  261. package/.agent/skills/app-builder/templates/express-api/TEMPLATE.md +83 -83
  262. package/.agent/skills/app-builder/templates/flutter-app/TEMPLATE.md +90 -90
  263. package/.agent/skills/app-builder/templates/monorepo-turborepo/TEMPLATE.md +90 -90
  264. package/.agent/skills/app-builder/templates/nextjs-fullstack/TEMPLATE.md +122 -122
  265. package/.agent/skills/app-builder/templates/nextjs-saas/TEMPLATE.md +122 -122
  266. package/.agent/skills/app-builder/templates/nextjs-static/TEMPLATE.md +169 -169
  267. package/.agent/skills/app-builder/templates/nuxt-app/TEMPLATE.md +134 -134
  268. package/.agent/skills/app-builder/templates/python-fastapi/TEMPLATE.md +83 -83
  269. package/.agent/skills/app-builder/templates/react-native-app/TEMPLATE.md +119 -119
  270. package/.agent/skills/architecture/SKILL.md +55 -55
  271. package/.agent/skills/architecture/context-discovery.md +43 -43
  272. package/.agent/skills/architecture/examples.md +94 -94
  273. package/.agent/skills/architecture/pattern-selection.md +68 -68
  274. package/.agent/skills/architecture/patterns-reference.md +50 -50
  275. package/.agent/skills/architecture/trade-off-analysis.md +77 -77
  276. package/.agent/skills/architecture-decision-records/SKILL.md +179 -179
  277. package/.agent/skills/article-writing/SKILL.md +79 -79
  278. package/.agent/skills/autonomous-agent-harness/SKILL.md +267 -267
  279. package/.agent/skills/autonomous-loops/SKILL.md +610 -610
  280. package/.agent/skills/backend-patterns/SKILL.md +598 -598
  281. package/.agent/skills/bash-linux/SKILL.md +199 -199
  282. package/.agent/skills/behavioral-modes/SKILL.md +242 -242
  283. package/.agent/skills/benchmark/SKILL.md +93 -93
  284. package/.agent/skills/blueprint/SKILL.md +105 -105
  285. package/.agent/skills/brainstorming/SKILL.md +163 -163
  286. package/.agent/skills/brainstorming/dynamic-questioning.md +350 -350
  287. package/.agent/skills/brand-voice/SKILL.md +97 -97
  288. package/.agent/skills/brand-voice/references/voice-profile-schema.md +55 -55
  289. package/.agent/skills/browser-qa/SKILL.md +87 -87
  290. package/.agent/skills/bun-runtime/SKILL.md +84 -84
  291. package/.agent/skills/canary-watch/SKILL.md +99 -99
  292. package/.agent/skills/carrier-relationship-management/SKILL.md +212 -212
  293. package/.agent/skills/ck/SKILL.md +147 -147
  294. package/.agent/skills/ck/commands/forget.mjs +44 -44
  295. package/.agent/skills/ck/commands/info.mjs +24 -24
  296. package/.agent/skills/ck/commands/init.mjs +143 -143
  297. package/.agent/skills/ck/commands/list.mjs +40 -40
  298. package/.agent/skills/ck/commands/migrate.mjs +202 -202
  299. package/.agent/skills/ck/commands/resume.mjs +36 -36
  300. package/.agent/skills/ck/commands/save.mjs +210 -210
  301. package/.agent/skills/ck/commands/shared.mjs +387 -387
  302. package/.agent/skills/ck/hooks/session-start.mjs +224 -224
  303. package/.agent/skills/claude-api/SKILL.md +337 -337
  304. package/.agent/skills/claude-devfleet/SKILL.md +103 -103
  305. package/.agent/skills/clean-code/SKILL.md +201 -201
  306. package/.agent/skills/click-path-audit/SKILL.md +244 -244
  307. package/.agent/skills/clickhouse-io/SKILL.md +439 -439
  308. package/.agent/skills/code-review-checklist/SKILL.md +109 -109
  309. package/.agent/skills/codebase-onboarding/SKILL.md +233 -233
  310. package/.agent/skills/coding-standards/SKILL.md +530 -530
  311. package/.agent/skills/compose-multiplatform-patterns/SKILL.md +299 -299
  312. package/.agent/skills/configure-ecc/SKILL.md +367 -367
  313. package/.agent/skills/connections-optimizer/SKILL.md +189 -189
  314. package/.agent/skills/content-engine/SKILL.md +131 -131
  315. package/.agent/skills/content-hash-cache-pattern/SKILL.md +161 -161
  316. package/.agent/skills/context-budget/SKILL.md +135 -135
  317. package/.agent/skills/continuous-agent-loop/SKILL.md +45 -45
  318. package/.agent/skills/continuous-learning/SKILL.md +119 -119
  319. package/.agent/skills/continuous-learning/config.json +18 -18
  320. package/.agent/skills/continuous-learning/evaluate-session.sh +69 -69
  321. package/.agent/skills/continuous-learning-v2/SKILL.md +365 -365
  322. package/.agent/skills/continuous-learning-v2/agents/observer-loop.sh +271 -271
  323. package/.agent/skills/continuous-learning-v2/agents/observer.md +198 -198
  324. package/.agent/skills/continuous-learning-v2/agents/session-guardian.sh +150 -150
  325. package/.agent/skills/continuous-learning-v2/agents/start-observer.sh +244 -244
  326. package/.agent/skills/continuous-learning-v2/config.json +8 -8
  327. package/.agent/skills/continuous-learning-v2/hooks/observe.sh +428 -428
  328. package/.agent/skills/continuous-learning-v2/scripts/detect-project.sh +228 -228
  329. package/.agent/skills/continuous-learning-v2/scripts/instinct-cli.py +1426 -1426
  330. package/.agent/skills/continuous-learning-v2/scripts/test-parse-instinct.py +984 -984
  331. package/.agent/skills/cost-aware-llm-pipeline/SKILL.md +183 -183
  332. package/.agent/skills/cpp-coding-standards/SKILL.md +723 -723
  333. package/.agent/skills/cpp-testing/SKILL.md +324 -324
  334. package/.agent/skills/crosspost/SKILL.md +111 -111
  335. package/.agent/skills/csharp-testing/SKILL.md +321 -321
  336. package/.agent/skills/customer-billing-ops/SKILL.md +140 -140
  337. package/.agent/skills/customs-trade-compliance/SKILL.md +263 -263
  338. package/.agent/skills/dart-flutter-patterns/SKILL.md +563 -563
  339. package/.agent/skills/data-scraper-agent/SKILL.md +764 -764
  340. package/.agent/skills/database-design/SKILL.md +52 -52
  341. package/.agent/skills/database-design/database-selection.md +43 -43
  342. package/.agent/skills/database-design/indexing.md +39 -39
  343. package/.agent/skills/database-design/migrations.md +48 -48
  344. package/.agent/skills/database-design/optimization.md +36 -36
  345. package/.agent/skills/database-design/orm-selection.md +30 -30
  346. package/.agent/skills/database-design/schema-design.md +56 -56
  347. package/.agent/skills/database-design/scripts/schema_validator.py +172 -172
  348. package/.agent/skills/database-migrations/SKILL.md +429 -429
  349. package/.agent/skills/deep-research/SKILL.md +155 -155
  350. package/.agent/skills/deployment-patterns/SKILL.md +427 -427
  351. package/.agent/skills/deployment-procedures/SKILL.md +241 -241
  352. package/.agent/skills/design-system/SKILL.md +82 -82
  353. package/.agent/skills/django-patterns/SKILL.md +734 -734
  354. package/.agent/skills/django-security/SKILL.md +593 -593
  355. package/.agent/skills/django-tdd/SKILL.md +729 -729
  356. package/.agent/skills/django-verification/SKILL.md +469 -469
  357. package/.agent/skills/dmux-workflows/SKILL.md +191 -191
  358. package/.agent/skills/doc.md +177 -177
  359. package/.agent/skills/docker-patterns/SKILL.md +364 -364
  360. package/.agent/skills/documentation-lookup/SKILL.md +90 -90
  361. package/.agent/skills/documentation-templates/SKILL.md +194 -194
  362. package/.agent/skills/dotnet-patterns/SKILL.md +321 -321
  363. package/.agent/skills/e2e-testing/SKILL.md +326 -326
  364. package/.agent/skills/energy-procurement/SKILL.md +228 -228
  365. package/.agent/skills/enterprise-agent-ops/SKILL.md +50 -50
  366. package/.agent/skills/eval-harness/SKILL.md +270 -270
  367. package/.agent/skills/exa-search/SKILL.md +103 -103
  368. package/.agent/skills/fal-ai-media/SKILL.md +284 -284
  369. package/.agent/skills/flutter-dart-code-review/SKILL.md +435 -435
  370. package/.agent/skills/foundation-models-on-device/SKILL.md +243 -243
  371. package/.agent/skills/frontend-design/SKILL.md +452 -452
  372. package/.agent/skills/frontend-design/animation-guide.md +331 -331
  373. package/.agent/skills/frontend-design/color-system.md +311 -311
  374. package/.agent/skills/frontend-design/decision-trees.md +418 -418
  375. package/.agent/skills/frontend-design/motion-graphics.md +306 -306
  376. package/.agent/skills/frontend-design/scripts/accessibility_checker.py +183 -183
  377. package/.agent/skills/frontend-design/scripts/ux_audit.py +722 -722
  378. package/.agent/skills/frontend-design/typography-system.md +345 -345
  379. package/.agent/skills/frontend-design/ux-psychology.md +1116 -1116
  380. package/.agent/skills/frontend-design/visual-effects.md +383 -383
  381. package/.agent/skills/frontend-patterns/SKILL.md +642 -642
  382. package/.agent/skills/frontend-slides/SKILL.md +184 -184
  383. package/.agent/skills/frontend-slides/style-presets.md +330 -330
  384. package/.agent/skills/game-development/2d-games/SKILL.md +119 -119
  385. package/.agent/skills/game-development/3d-games/SKILL.md +135 -135
  386. package/.agent/skills/game-development/SKILL.md +167 -167
  387. package/.agent/skills/game-development/game-art/SKILL.md +185 -185
  388. package/.agent/skills/game-development/game-audio/SKILL.md +190 -190
  389. package/.agent/skills/game-development/game-design/SKILL.md +129 -129
  390. package/.agent/skills/game-development/mobile-games/SKILL.md +108 -108
  391. package/.agent/skills/game-development/multiplayer/SKILL.md +132 -132
  392. package/.agent/skills/game-development/pc-games/SKILL.md +144 -144
  393. package/.agent/skills/game-development/vr-ar/SKILL.md +123 -123
  394. package/.agent/skills/game-development/web-games/SKILL.md +150 -150
  395. package/.agent/skills/gan-style-harness/SKILL.md +278 -278
  396. package/.agent/skills/geo-fundamentals/SKILL.md +156 -156
  397. package/.agent/skills/geo-fundamentals/scripts/geo_checker.py +289 -289
  398. package/.agent/skills/git-workflow/SKILL.md +715 -715
  399. package/.agent/skills/golang-patterns/SKILL.md +674 -674
  400. package/.agent/skills/golang-testing/SKILL.md +720 -720
  401. package/.agent/skills/google-workspace-ops/SKILL.md +95 -95
  402. package/.agent/skills/healthcare-cdss-patterns/SKILL.md +245 -245
  403. package/.agent/skills/healthcare-emr-patterns/SKILL.md +159 -159
  404. package/.agent/skills/healthcare-eval-harness/SKILL.md +207 -207
  405. package/.agent/skills/healthcare-phi-compliance/SKILL.md +145 -145
  406. package/.agent/skills/hexagonal-architecture/SKILL.md +276 -276
  407. package/.agent/skills/i18n-localization/SKILL.md +154 -154
  408. package/.agent/skills/i18n-localization/scripts/i18n_checker.py +241 -241
  409. package/.agent/skills/intelligent-routing/SKILL.md +335 -335
  410. package/.agent/skills/inventory-demand-planning/SKILL.md +247 -247
  411. package/.agent/skills/investor-materials/SKILL.md +96 -96
  412. package/.agent/skills/investor-outreach/SKILL.md +91 -91
  413. package/.agent/skills/iterative-retrieval/SKILL.md +211 -211
  414. package/.agent/skills/java-coding-standards/SKILL.md +147 -147
  415. package/.agent/skills/jira-integration/SKILL.md +293 -293
  416. package/.agent/skills/jpa-patterns/SKILL.md +151 -151
  417. package/.agent/skills/kotlin-coroutines-flows/SKILL.md +284 -284
  418. package/.agent/skills/kotlin-exposed-patterns/SKILL.md +719 -719
  419. package/.agent/skills/kotlin-ktor-patterns/SKILL.md +689 -689
  420. package/.agent/skills/kotlin-patterns/SKILL.md +711 -711
  421. package/.agent/skills/kotlin-testing/SKILL.md +824 -824
  422. package/.agent/skills/laravel-patterns/SKILL.md +415 -415
  423. package/.agent/skills/laravel-plugin-discovery/SKILL.md +229 -229
  424. package/.agent/skills/laravel-security/SKILL.md +285 -285
  425. package/.agent/skills/laravel-tdd/SKILL.md +283 -283
  426. package/.agent/skills/laravel-verification/SKILL.md +179 -179
  427. package/.agent/skills/lead-intelligence/SKILL.md +321 -321
  428. package/.agent/skills/lead-intelligence/agents/enrichment-agent.md +85 -85
  429. package/.agent/skills/lead-intelligence/agents/mutual-mapper.md +75 -75
  430. package/.agent/skills/lead-intelligence/agents/outreach-drafter.md +98 -98
  431. package/.agent/skills/lead-intelligence/agents/signal-scorer.md +60 -60
  432. package/.agent/skills/lint-and-validate/SKILL.md +45 -45
  433. package/.agent/skills/lint-and-validate/scripts/lint_runner.py +184 -184
  434. package/.agent/skills/lint-and-validate/scripts/type_coverage.py +173 -173
  435. package/.agent/skills/liquid-glass-design/SKILL.md +279 -279
  436. package/.agent/skills/logistics-exception-management/SKILL.md +222 -222
  437. package/.agent/skills/manim-video/SKILL.md +89 -89
  438. package/.agent/skills/manim-video/assets/network-graph-scene.py +52 -52
  439. package/.agent/skills/market-research/SKILL.md +75 -75
  440. package/.agent/skills/mcp-server-patterns/SKILL.md +67 -67
  441. package/.agent/skills/mobile-design/SKILL.md +394 -394
  442. package/.agent/skills/mobile-design/decision-trees.md +516 -516
  443. package/.agent/skills/mobile-design/mobile-backend.md +491 -491
  444. package/.agent/skills/mobile-design/mobile-color-system.md +420 -420
  445. package/.agent/skills/mobile-design/mobile-debugging.md +122 -122
  446. package/.agent/skills/mobile-design/mobile-design-thinking.md +357 -357
  447. package/.agent/skills/mobile-design/mobile-navigation.md +458 -458
  448. package/.agent/skills/mobile-design/mobile-performance.md +767 -767
  449. package/.agent/skills/mobile-design/mobile-testing.md +356 -356
  450. package/.agent/skills/mobile-design/mobile-typography.md +433 -433
  451. package/.agent/skills/mobile-design/platform-android.md +666 -666
  452. package/.agent/skills/mobile-design/platform-ios.md +561 -561
  453. package/.agent/skills/mobile-design/scripts/mobile_audit.py +670 -670
  454. package/.agent/skills/mobile-design/touch-psychology.md +537 -537
  455. package/.agent/skills/nanoclaw-repl/SKILL.md +33 -33
  456. package/.agent/skills/nestjs-patterns/SKILL.md +230 -230
  457. package/.agent/skills/nextjs-react-expert/1-async-eliminating-waterfalls.md +351 -351
  458. package/.agent/skills/nextjs-react-expert/2-bundle-bundle-size-optimization.md +240 -240
  459. package/.agent/skills/nextjs-react-expert/3-server-server-side-performance.md +490 -490
  460. package/.agent/skills/nextjs-react-expert/4-client-client-side-data-fetching.md +264 -264
  461. package/.agent/skills/nextjs-react-expert/5-rerender-re-render-optimization.md +581 -581
  462. package/.agent/skills/nextjs-react-expert/6-rendering-rendering-performance.md +432 -432
  463. package/.agent/skills/nextjs-react-expert/7-js-javascript-performance.md +684 -684
  464. package/.agent/skills/nextjs-react-expert/8-advanced-advanced-patterns.md +150 -150
  465. package/.agent/skills/nextjs-react-expert/9-cache-components.md +103 -103
  466. package/.agent/skills/nextjs-react-expert/SKILL.md +293 -293
  467. package/.agent/skills/nextjs-react-expert/scripts/convert_rules.py +222 -222
  468. package/.agent/skills/nextjs-react-expert/scripts/react_performance_checker.py +252 -252
  469. package/.agent/skills/nextjs-turbopack/SKILL.md +44 -44
  470. package/.agent/skills/nodejs-best-practices/SKILL.md +333 -333
  471. package/.agent/skills/nutrient-document-processing/SKILL.md +167 -167
  472. package/.agent/skills/nuxt4-patterns/SKILL.md +100 -100
  473. package/.agent/skills/openclaw-persona-forge/SKILL.md +296 -296
  474. package/.agent/skills/openclaw-persona-forge/gacha.py +224 -224
  475. package/.agent/skills/openclaw-persona-forge/gacha.sh +5 -5
  476. package/.agent/skills/openclaw-persona-forge/references/avatar-style.md +124 -124
  477. package/.agent/skills/openclaw-persona-forge/references/boundary-rules.md +53 -53
  478. package/.agent/skills/openclaw-persona-forge/references/error-handling.md +53 -53
  479. package/.agent/skills/openclaw-persona-forge/references/identity-tension.md +48 -48
  480. package/.agent/skills/openclaw-persona-forge/references/naming-system.md +39 -39
  481. package/.agent/skills/openclaw-persona-forge/references/output-template.md +166 -166
  482. package/.agent/skills/opensource-pipeline/SKILL.md +255 -255
  483. package/.agent/skills/parallel-agents/SKILL.md +175 -175
  484. package/.agent/skills/performance-profiling/SKILL.md +143 -143
  485. package/.agent/skills/performance-profiling/scripts/lighthouse_audit.py +76 -76
  486. package/.agent/skills/perl-patterns/SKILL.md +504 -504
  487. package/.agent/skills/perl-security/SKILL.md +503 -503
  488. package/.agent/skills/perl-testing/SKILL.md +475 -475
  489. package/.agent/skills/plan-writing/SKILL.md +152 -152
  490. package/.agent/skills/plankton-code-quality/SKILL.md +236 -236
  491. package/.agent/skills/postgres-patterns/SKILL.md +147 -147
  492. package/.agent/skills/powershell-windows/SKILL.md +167 -167
  493. package/.agent/skills/product-lens/SKILL.md +85 -85
  494. package/.agent/skills/production-scheduling/SKILL.md +238 -238
  495. package/.agent/skills/project-flow-ops/SKILL.md +111 -111
  496. package/.agent/skills/project-guidelines-example/SKILL.md +349 -349
  497. package/.agent/skills/prompt-optimizer/SKILL.md +397 -397
  498. package/.agent/skills/python-patterns/SKILL.md +750 -750
  499. package/.agent/skills/python-testing/SKILL.md +816 -816
  500. package/.agent/skills/pytorch-patterns/SKILL.md +396 -396
  501. package/.agent/skills/quality-nonconformance/SKILL.md +260 -260
  502. package/.agent/skills/ralphinho-rfc-pipeline/SKILL.md +67 -67
  503. package/.agent/skills/red-team-tactics/SKILL.md +199 -199
  504. package/.agent/skills/regex-vs-llm-structured-text/SKILL.md +220 -220
  505. package/.agent/skills/remotion-video-creation/SKILL.md +43 -43
  506. package/.agent/skills/remotion-video-creation/rules/3d.md +86 -86
  507. package/.agent/skills/remotion-video-creation/rules/animations.md +29 -29
  508. package/.agent/skills/remotion-video-creation/rules/assets/charts-bar-chart.tsx +173 -173
  509. package/.agent/skills/remotion-video-creation/rules/assets/text-animations-typewriter.tsx +100 -100
  510. package/.agent/skills/remotion-video-creation/rules/assets/text-animations-word-highlight.tsx +108 -108
  511. package/.agent/skills/remotion-video-creation/rules/assets.md +78 -78
  512. package/.agent/skills/remotion-video-creation/rules/audio.md +172 -172
  513. package/.agent/skills/remotion-video-creation/rules/calculate-metadata.md +104 -104
  514. package/.agent/skills/remotion-video-creation/rules/can-decode.md +75 -75
  515. package/.agent/skills/remotion-video-creation/rules/charts.md +58 -58
  516. package/.agent/skills/remotion-video-creation/rules/compositions.md +146 -146
  517. package/.agent/skills/remotion-video-creation/rules/display-captions.md +126 -126
  518. package/.agent/skills/remotion-video-creation/rules/extract-frames.md +229 -229
  519. package/.agent/skills/remotion-video-creation/rules/fonts.md +152 -152
  520. package/.agent/skills/remotion-video-creation/rules/get-audio-duration.md +58 -58
  521. package/.agent/skills/remotion-video-creation/rules/get-video-dimensions.md +68 -68
  522. package/.agent/skills/remotion-video-creation/rules/get-video-duration.md +58 -58
  523. package/.agent/skills/remotion-video-creation/rules/gifs.md +138 -138
  524. package/.agent/skills/remotion-video-creation/rules/images.md +130 -130
  525. package/.agent/skills/remotion-video-creation/rules/import-srt-captions.md +67 -67
  526. package/.agent/skills/remotion-video-creation/rules/lottie.md +67 -67
  527. package/.agent/skills/remotion-video-creation/rules/measuring-dom-nodes.md +34 -34
  528. package/.agent/skills/remotion-video-creation/rules/measuring-text.md +143 -143
  529. package/.agent/skills/remotion-video-creation/rules/sequencing.md +106 -106
  530. package/.agent/skills/remotion-video-creation/rules/tailwind.md +11 -11
  531. package/.agent/skills/remotion-video-creation/rules/text-animations.md +20 -20
  532. package/.agent/skills/remotion-video-creation/rules/timing.md +179 -179
  533. package/.agent/skills/remotion-video-creation/rules/transcribe-captions.md +19 -19
  534. package/.agent/skills/remotion-video-creation/rules/transitions.md +122 -122
  535. package/.agent/skills/remotion-video-creation/rules/trimming.md +52 -52
  536. package/.agent/skills/remotion-video-creation/rules/videos.md +171 -171
  537. package/.agent/skills/repo-scan/SKILL.md +63 -63
  538. package/.agent/skills/returns-reverse-logistics/SKILL.md +240 -240
  539. package/.agent/skills/rules-distill/SKILL.md +264 -264
  540. package/.agent/skills/rules-distill/scripts/scan-rules.sh +58 -58
  541. package/.agent/skills/rules-distill/scripts/scan-skills.sh +129 -129
  542. package/.agent/skills/rust-patterns/SKILL.md +499 -499
  543. package/.agent/skills/rust-pro/SKILL.md +175 -175
  544. package/.agent/skills/rust-testing/SKILL.md +500 -500
  545. package/.agent/skills/safety-guard/SKILL.md +75 -75
  546. package/.agent/skills/santa-method/SKILL.md +306 -306
  547. package/.agent/skills/search-first/SKILL.md +161 -161
  548. package/.agent/skills/security-review/SKILL.md +495 -495
  549. package/.agent/skills/security-review/cloud-infrastructure-security.md +361 -361
  550. package/.agent/skills/security-scan/SKILL.md +165 -165
  551. package/.agent/skills/seo-fundamentals/SKILL.md +129 -129
  552. package/.agent/skills/seo-fundamentals/scripts/seo_checker.py +219 -219
  553. package/.agent/skills/server-management/SKILL.md +161 -161
  554. package/.agent/skills/skill-comply/SKILL.md +58 -58
  555. package/.agent/skills/skill-comply/fixtures/compliant-trace.jsonl +5 -5
  556. package/.agent/skills/skill-comply/fixtures/noncompliant-trace.jsonl +3 -3
  557. package/.agent/skills/skill-comply/fixtures/tdd-spec.yaml +44 -44
  558. package/.agent/skills/skill-comply/prompts/classifier.md +24 -24
  559. package/.agent/skills/skill-comply/prompts/scenario-generator.md +62 -62
  560. package/.agent/skills/skill-comply/prompts/spec-generator.md +42 -42
  561. package/.agent/skills/skill-comply/pyproject.toml +15 -15
  562. package/.agent/skills/skill-comply/scripts/classifier.py +85 -85
  563. package/.agent/skills/skill-comply/scripts/grader.py +122 -122
  564. package/.agent/skills/skill-comply/scripts/parser.py +107 -107
  565. package/.agent/skills/skill-comply/scripts/report.py +170 -170
  566. package/.agent/skills/skill-comply/scripts/run.py +127 -127
  567. package/.agent/skills/skill-comply/scripts/runner.py +161 -161
  568. package/.agent/skills/skill-comply/scripts/scenario-generator.py +70 -70
  569. package/.agent/skills/skill-comply/scripts/spec-generator.py +72 -72
  570. package/.agent/skills/skill-comply/scripts/utils.py +13 -13
  571. package/.agent/skills/skill-comply/tests/test-grader.py +137 -137
  572. package/.agent/skills/skill-comply/tests/test-parser.py +90 -90
  573. package/.agent/skills/skill-stocktake/SKILL.md +193 -193
  574. package/.agent/skills/skill-stocktake/scripts/quick-diff.sh +87 -87
  575. package/.agent/skills/skill-stocktake/scripts/save-results.sh +56 -56
  576. package/.agent/skills/skill-stocktake/scripts/scan.sh +170 -170
  577. package/.agent/skills/social-graph-ranker/SKILL.md +154 -154
  578. package/.agent/skills/springboot-patterns/SKILL.md +314 -314
  579. package/.agent/skills/springboot-security/SKILL.md +272 -272
  580. package/.agent/skills/springboot-tdd/SKILL.md +158 -158
  581. package/.agent/skills/springboot-verification/SKILL.md +231 -231
  582. package/.agent/skills/strategic-compact/SKILL.md +131 -131
  583. package/.agent/skills/strategic-compact/suggest-compact.sh +54 -54
  584. package/.agent/skills/swift-actor-persistence/SKILL.md +143 -143
  585. package/.agent/skills/swift-concurrency-6-2/SKILL.md +216 -216
  586. package/.agent/skills/swift-protocol-di-testing/SKILL.md +190 -190
  587. package/.agent/skills/swiftui-patterns/SKILL.md +259 -259
  588. package/.agent/skills/systematic-debugging/SKILL.md +109 -109
  589. package/.agent/skills/tailwind-patterns/SKILL.md +269 -269
  590. package/.agent/skills/tdd-workflow/SKILL.md +463 -463
  591. package/.agent/skills/team-builder/SKILL.md +168 -168
  592. package/.agent/skills/testing-patterns/SKILL.md +178 -178
  593. package/.agent/skills/testing-patterns/scripts/test_runner.py +219 -219
  594. package/.agent/skills/token-budget-advisor/SKILL.md +133 -133
  595. package/.agent/skills/ui-demo/SKILL.md +465 -465
  596. package/.agent/skills/ui-ux-pro-max/SKILL.md +292 -292
  597. package/.agent/skills/ui-ux-pro-max/data/charts.csv +26 -26
  598. package/.agent/skills/ui-ux-pro-max/data/colors.csv +97 -97
  599. package/.agent/skills/ui-ux-pro-max/data/icons.csv +101 -101
  600. package/.agent/skills/ui-ux-pro-max/data/landing.csv +31 -31
  601. package/.agent/skills/ui-ux-pro-max/data/products.csv +96 -96
  602. package/.agent/skills/ui-ux-pro-max/data/react-performance.csv +45 -45
  603. package/.agent/skills/ui-ux-pro-max/data/stacks/astro.csv +54 -54
  604. package/.agent/skills/ui-ux-pro-max/data/stacks/flutter.csv +53 -53
  605. package/.agent/skills/ui-ux-pro-max/data/stacks/html-tailwind.csv +56 -56
  606. package/.agent/skills/ui-ux-pro-max/data/stacks/jetpack-compose.csv +53 -53
  607. package/.agent/skills/ui-ux-pro-max/data/stacks/nextjs.csv +53 -53
  608. package/.agent/skills/ui-ux-pro-max/data/stacks/nuxt-ui.csv +51 -51
  609. package/.agent/skills/ui-ux-pro-max/data/stacks/nuxtjs.csv +59 -59
  610. package/.agent/skills/ui-ux-pro-max/data/stacks/react-native.csv +52 -52
  611. package/.agent/skills/ui-ux-pro-max/data/stacks/react.csv +54 -54
  612. package/.agent/skills/ui-ux-pro-max/data/stacks/shadcn.csv +61 -61
  613. package/.agent/skills/ui-ux-pro-max/data/stacks/svelte.csv +54 -54
  614. package/.agent/skills/ui-ux-pro-max/data/stacks/swiftui.csv +51 -51
  615. package/.agent/skills/ui-ux-pro-max/data/stacks/vue.csv +50 -50
  616. package/.agent/skills/ui-ux-pro-max/data/styles.csv +68 -68
  617. package/.agent/skills/ui-ux-pro-max/data/typography.csv +57 -57
  618. package/.agent/skills/ui-ux-pro-max/data/ui-reasoning.csv +101 -101
  619. package/.agent/skills/ui-ux-pro-max/data/ux-guidelines.csv +99 -99
  620. package/.agent/skills/ui-ux-pro-max/data/web-interface.csv +31 -31
  621. package/.agent/skills/ui-ux-pro-max/scripts/core.py +253 -253
  622. package/.agent/skills/ui-ux-pro-max/scripts/design_system.py +1067 -1067
  623. package/.agent/skills/ui-ux-pro-max/scripts/search.py +114 -114
  624. package/.agent/skills/verification-loop/SKILL.md +126 -126
  625. package/.agent/skills/video-editing/SKILL.md +310 -310
  626. package/.agent/skills/videodb/SKILL.md +374 -374
  627. package/.agent/skills/videodb/reference/api-reference.md +550 -550
  628. package/.agent/skills/videodb/reference/capture-reference.md +407 -407
  629. package/.agent/skills/videodb/reference/capture.md +101 -101
  630. package/.agent/skills/videodb/reference/editor.md +443 -443
  631. package/.agent/skills/videodb/reference/generative.md +331 -331
  632. package/.agent/skills/videodb/reference/rtstream-reference.md +564 -564
  633. package/.agent/skills/videodb/reference/rtstream.md +65 -65
  634. package/.agent/skills/videodb/reference/search.md +230 -230
  635. package/.agent/skills/videodb/reference/streaming.md +406 -406
  636. package/.agent/skills/videodb/reference/use-cases.md +118 -118
  637. package/.agent/skills/videodb/scripts/ws-listener.py +282 -282
  638. package/.agent/skills/visa-doc-translate/SKILL.md +117 -117
  639. package/.agent/skills/visa-doc-translate/readme.md +86 -86
  640. package/.agent/skills/vulnerability-scanner/SKILL.md +276 -276
  641. package/.agent/skills/vulnerability-scanner/checklists.md +121 -121
  642. package/.agent/skills/vulnerability-scanner/scripts/security_scan.py +458 -458
  643. package/.agent/skills/web-design-guidelines/SKILL.md +57 -57
  644. package/.agent/skills/webapp-testing/SKILL.md +187 -187
  645. package/.agent/skills/webapp-testing/scripts/playwright_runner.py +173 -173
  646. package/.agent/skills/workspace-surface-audit/SKILL.md +125 -125
  647. package/.agent/skills/x-api/SKILL.md +230 -230
  648. package/.agent/tasks/lessons.md +40 -40
  649. package/.agent/tasks/todo.md +33 -33
  650. package/.agent/tasks/two-track-merge-contract.md +1 -1
  651. package/.agent/workflows/aside.md +164 -164
  652. package/.agent/workflows/brainstorm.md +113 -113
  653. package/.agent/workflows/build-fix.md +62 -62
  654. package/.agent/workflows/checkpoint.md +74 -74
  655. package/.agent/workflows/claw.md +23 -23
  656. package/.agent/workflows/clean-memory.md +34 -34
  657. package/.agent/workflows/code-review.md +289 -289
  658. package/.agent/workflows/context-budget.md +23 -23
  659. package/.agent/workflows/cpp-build.md +173 -173
  660. package/.agent/workflows/cpp-review.md +132 -132
  661. package/.agent/workflows/cpp-test.md +251 -251
  662. package/.agent/workflows/create.md +59 -59
  663. package/.agent/workflows/debug.md +103 -103
  664. package/.agent/workflows/deploy.md +176 -176
  665. package/.agent/workflows/devfleet.md +23 -23
  666. package/.agent/workflows/docs.md +23 -23
  667. package/.agent/workflows/e2e.md +268 -268
  668. package/.agent/workflows/enhance.md +63 -63
  669. package/.agent/workflows/eval.md +23 -23
  670. package/.agent/workflows/evolve.md +178 -178
  671. package/.agent/workflows/flutter-build.md +164 -164
  672. package/.agent/workflows/flutter-review.md +116 -116
  673. package/.agent/workflows/flutter-test.md +144 -144
  674. package/.agent/workflows/gan-build.md +99 -99
  675. package/.agent/workflows/gan-design.md +35 -35
  676. package/.agent/workflows/go-build.md +183 -183
  677. package/.agent/workflows/go-review.md +148 -148
  678. package/.agent/workflows/go-test.md +268 -268
  679. package/.agent/workflows/gradle-build.md +70 -70
  680. package/.agent/workflows/harness-audit.md +73 -73
  681. package/.agent/workflows/init-docs.md +46 -46
  682. package/.agent/workflows/instinct-export.md +66 -66
  683. package/.agent/workflows/instinct-import.md +114 -114
  684. package/.agent/workflows/instinct-status.md +59 -59
  685. package/.agent/workflows/jira.md +106 -106
  686. package/.agent/workflows/kotlin-build.md +174 -174
  687. package/.agent/workflows/kotlin-review.md +140 -140
  688. package/.agent/workflows/kotlin-test.md +312 -312
  689. package/.agent/workflows/learn-eval.md +116 -116
  690. package/.agent/workflows/learn.md +70 -70
  691. package/.agent/workflows/loop-start.md +32 -32
  692. package/.agent/workflows/loop-status.md +24 -24
  693. package/.agent/workflows/model-route.md +26 -26
  694. package/.agent/workflows/multi-backend.md +158 -158
  695. package/.agent/workflows/multi-execute.md +315 -315
  696. package/.agent/workflows/multi-frontend.md +158 -158
  697. package/.agent/workflows/multi-plan.md +268 -268
  698. package/.agent/workflows/multi-workflow.md +191 -191
  699. package/.agent/workflows/orchestrate.md +135 -135
  700. package/.agent/workflows/plan.md +117 -117
  701. package/.agent/workflows/pm2.md +272 -272
  702. package/.agent/workflows/preview.md +81 -81
  703. package/.agent/workflows/projects.md +39 -39
  704. package/.agent/workflows/promote.md +41 -41
  705. package/.agent/workflows/prompt-optimize.md +23 -23
  706. package/.agent/workflows/prp-commit.md +112 -112
  707. package/.agent/workflows/prp-implement.md +385 -385
  708. package/.agent/workflows/prp-plan.md +502 -502
  709. package/.agent/workflows/prp-pr.md +184 -184
  710. package/.agent/workflows/prp-prd.md +447 -447
  711. package/.agent/workflows/prune.md +31 -31
  712. package/.agent/workflows/python-review.md +297 -297
  713. package/.agent/workflows/quality-gate.md +29 -29
  714. package/.agent/workflows/refactor-clean.md +80 -80
  715. package/.agent/workflows/resume-session.md +156 -156
  716. package/.agent/workflows/rules-distill.md +20 -20
  717. package/.agent/workflows/rust-build.md +187 -187
  718. package/.agent/workflows/rust-review.md +142 -142
  719. package/.agent/workflows/rust-test.md +308 -308
  720. package/.agent/workflows/santa-loop.md +175 -175
  721. package/.agent/workflows/save-session.md +275 -275
  722. package/.agent/workflows/sessions.md +333 -333
  723. package/.agent/workflows/setup-pm.md +80 -80
  724. package/.agent/workflows/skill-create.md +174 -174
  725. package/.agent/workflows/skill-health.md +54 -54
  726. package/.agent/workflows/status.md +86 -86
  727. package/.agent/workflows/tdd.md +231 -231
  728. package/.agent/workflows/test-coverage.md +69 -69
  729. package/.agent/workflows/test.md +144 -144
  730. package/.agent/workflows/ui-ux-pro-max.md +295 -295
  731. package/.agent/workflows/update-codemaps.md +72 -72
  732. package/.agent/workflows/update-docs.md +84 -84
  733. package/.agent/workflows/verify.md +23 -23
  734. package/LICENSE +176 -176
  735. package/README.md +144 -144
  736. package/package.json +1 -1
  737. package/scripts/release-check.js +55 -55
  738. package/src/bin/cli.js +424 -354
  739. package/src/lib/installer.js +223 -11
@@ -1,670 +1,670 @@
1
- #!/usr/bin/env python3
2
- """
3
- Mobile UX Audit Script - Full Mobile Design Coverage
4
-
5
- Analyzes React Native / Flutter code for compliance with:
6
-
7
- 1. TOUCH PSYCHOLOGY (touch-psychology.md):
8
- - Touch Target Sizes (44pt iOS, 48dp Android, 44px WCAG)
9
- - Touch Target Spacing (8px minimum gap)
10
- - Thumb Zone Placement (primary CTAs at bottom)
11
- - Gesture Alternatives (visible buttons for swipe)
12
- - Haptic Feedback Patterns
13
- - Touch Feedback Timing (<50ms)
14
- - Touch Accessibility (motor impairment support)
15
-
16
- 2. MOBILE PERFORMANCE (mobile-performance.md):
17
- - ScrollView vs FlatList (CRITICAL)
18
- - React.memo for List Items
19
- - useCallback for renderItem
20
- - Stable keyExtractor (NOT index)
21
- - useNativeDriver for Animations
22
- - Memory Leak Prevention (cleanup)
23
- - Console.log Detection
24
- - Inline Function Detection
25
- - Animation Performance (transform/opacity only)
26
-
27
- 3. MOBILE NAVIGATION (mobile-navigation.md):
28
- - Tab Bar Max Items (5)
29
- - Tab State Preservation
30
- - Proper Back Handling
31
- - Deep Link Support
32
- - Navigation Structure
33
-
34
- 4. MOBILE TYPOGRAPHY (mobile-typography.md):
35
- - System Font Usage
36
- - Dynamic Type Support (iOS)
37
- - Text Scaling Constraints
38
- - Mobile Line Height
39
- - Font Size Limits
40
-
41
- 5. MOBILE COLOR SYSTEM (mobile-color-system.md):
42
- - Pure Black Avoidance (#000000)
43
- - OLED Optimization
44
- - Dark Mode Support
45
- - Contrast Ratios
46
-
47
- 6. PLATFORM iOS (platform-ios.md):
48
- - SF Symbols Usage
49
- - iOS Navigation Patterns
50
- - iOS Haptic Types
51
- - iOS-Specific Components
52
-
53
- 7. PLATFORM ANDROID (platform-android.md):
54
- - Material Icons Usage
55
- - Android Navigation Patterns
56
- - Ripple Effects
57
- - Android-Specific Components
58
-
59
- 8. MOBILE BACKEND (mobile-backend.md):
60
- - Secure Storage (NOT AsyncStorage)
61
- - Offline Handling
62
- - Push Notification Support
63
- - API Response Caching
64
-
65
- Total: 50+ mobile-specific checks
66
- """
67
-
68
- import sys
69
- import os
70
- import re
71
- import json
72
- from pathlib import Path
73
-
74
- class MobileAuditor:
75
- def __init__(self):
76
- self.issues = []
77
- self.warnings = []
78
- self.passed_count = 0
79
- self.files_checked = 0
80
-
81
- def audit_file(self, filepath: str) -> None:
82
- try:
83
- with open(filepath, 'r', encoding='utf-8', errors='replace') as f:
84
- content = f.read()
85
- except:
86
- return
87
-
88
- self.files_checked += 1
89
- filename = os.path.basename(filepath)
90
-
91
- # Detect framework
92
- is_react_native = bool(re.search(r'react-native|@react-navigation|React\.Native', content))
93
- is_flutter = bool(re.search(r'import \'package:flutter|MaterialApp|Widget\.build', content))
94
-
95
- if not (is_react_native or is_flutter):
96
- return # Skip non-mobile files
97
-
98
- # --- 1. TOUCH PSYCHOLOGY CHECKS ---
99
-
100
- # 1.1 Touch Target Size Check
101
- # Look for small touch targets
102
- small_sizes = re.findall(r'(?:width|height|size):\s*([0-3]\d)', content)
103
- for size in small_sizes:
104
- if int(size) < 44:
105
- self.issues.append(f"[Touch Target] {filename}: Touch target size {size}px < 44px minimum (iOS: 44pt, Android: 48dp)")
106
-
107
- # 1.2 Touch Target Spacing Check
108
- # Look for inadequate spacing between touchable elements
109
- small_gaps = re.findall(r'(?:margin|gap):\s*([0-7])\s*(?:px|dp)', content)
110
- for gap in small_gaps:
111
- if int(gap) < 8:
112
- self.warnings.append(f"[Touch Spacing] {filename}: Touch target spacing {gap}px < 8px minimum. Accidental taps risk.")
113
-
114
- # 1.3 Thumb Zone Placement Check
115
- # Primary CTAs should be at bottom (easy thumb reach)
116
- primary_buttons = re.findall(r'(?:testID|id):\s*["\'](?:.*(?:primary|cta|submit|confirm)[^"\']*)["\']', content, re.IGNORECASE)
117
- has_bottom_placement = bool(re.search(r'position:\s*["\']?absolute["\']?|bottom:\s*\d+|style.*bottom|justifyContent:\s*["\']?flex-end', content))
118
- if primary_buttons and not has_bottom_placement:
119
- self.warnings.append(f"[Thumb Zone] {filename}: Primary CTA may not be in thumb zone (bottom). Place primary actions at bottom for easy reach.")
120
-
121
- # 1.4 Gesture Alternatives Check
122
- # Swipe actions should have visible button alternatives
123
- has_swipe_gestures = bool(re.search(r'Swipeable|onSwipe|PanGestureHandler|swipe', content))
124
- has_visible_buttons = bool(re.search(r'Button.*(?:delete|archive|more)|TouchableOpacity|Pressable', content))
125
- if has_swipe_gestures and not has_visible_buttons:
126
- self.warnings.append(f"[Gestures] {filename}: Swipe gestures detected without visible button alternatives. Motor impaired users need alternatives.")
127
-
128
- # 1.5 Haptic Feedback Check
129
- # Important actions should have haptic feedback
130
- has_important_actions = bool(re.search(r'(?:onPress|onSubmit|delete|remove|confirm|purchase)', content))
131
- has_haptics = bool(re.search(r'Haptics|Vibration|react-native-haptic-feedback|FeedbackManager', content))
132
- if has_important_actions and not has_haptics:
133
- self.warnings.append(f"[Haptics] {filename}: Important actions without haptic feedback. Consider adding haptic confirmation.")
134
-
135
- # 1.6 Touch Feedback Timing Check
136
- # Touch feedback should be immediate (<50ms)
137
- if is_react_native:
138
- has_pressable = bool(re.search(r'Pressable|TouchableOpacity', content))
139
- has_feedback_state = bool(re.search(r'pressed|style.*opacity|underlay', content))
140
- if has_pressable and not has_feedback_state:
141
- self.warnings.append(f"[Touch Feedback] {filename}: Pressable without visual feedback state. Add opacity/scale change for tap confirmation.")
142
-
143
- # --- 2. MOBILE PERFORMANCE CHECKS ---
144
-
145
- # 2.1 CRITICAL: ScrollView vs FlatList
146
- has_scrollview = bool(re.search(r'<ScrollView|ScrollView\.', content))
147
- has_map_in_scrollview = bool(re.search(r'ScrollView.*\.map\(|ScrollView.*\{.*\.map', content))
148
- if has_scrollview and has_map_in_scrollview:
149
- self.issues.append(f"[Performance CRITICAL] {filename}: ScrollView with .map() detected. Use FlatList for lists to prevent memory explosion.")
150
-
151
- # 2.2 React.memo Check
152
- if is_react_native:
153
- has_list = bool(re.search(r'FlatList|FlashList|SectionList', content))
154
- has_react_memo = bool(re.search(r'React\.memo|memo\(', content))
155
- if has_list and not has_react_memo:
156
- self.warnings.append(f"[Performance] {filename}: FlatList without React.memo on list items. Items will re-render on every parent update.")
157
-
158
- # 2.3 useCallback Check
159
- if is_react_native:
160
- has_flatlist = bool(re.search(r'FlatList|FlashList', content))
161
- has_use_callback = bool(re.search(r'useCallback', content))
162
- if has_flatlist and not has_use_callback:
163
- self.warnings.append(f"[Performance] {filename}: FlatList renderItem without useCallback. New function created every render.")
164
-
165
- # 2.4 keyExtractor Check (CRITICAL)
166
- if is_react_native:
167
- has_flatlist = bool(re.search(r'FlatList', content))
168
- has_key_extractor = bool(re.search(r'keyExtractor', content))
169
- uses_index_key = bool(re.search(r'key=\{.*index.*\}|key:\s*index', content))
170
- if has_flatlist and not has_key_extractor:
171
- self.issues.append(f"[Performance CRITICAL] {filename}: FlatList without keyExtractor. Index-based keys cause bugs on reorder/delete.")
172
- if uses_index_key:
173
- self.issues.append(f"[Performance CRITICAL] {filename}: Using index as key. This causes bugs when list changes. Use unique ID from data.")
174
-
175
- # 2.5 useNativeDriver Check
176
- if is_react_native:
177
- has_animated = bool(re.search(r'Animated\.', content))
178
- has_native_driver = bool(re.search(r'useNativeDriver:\s*true', content))
179
- has_native_driver_false = bool(re.search(r'useNativeDriver:\s*false', content))
180
- if has_animated and has_native_driver_false:
181
- self.warnings.append(f"[Performance] {filename}: Animation with useNativeDriver: false. Use true for 60fps (only supports transform/opacity).")
182
- if has_animated and not has_native_driver:
183
- self.warnings.append(f"[Performance] {filename}: Animated component without useNativeDriver. Add useNativeDriver: true for 60fps.")
184
-
185
- # 2.6 Memory Leak Check
186
- if is_react_native:
187
- has_effect = bool(re.search(r'useEffect', content))
188
- has_cleanup = bool(re.search(r'return\s*\(\)\s*=>|return\s+function', content))
189
- has_subscriptions = bool(re.search(r'addEventListener|subscribe|\.focus\(\)|\.off\(', content))
190
- if has_effect and has_subscriptions and not has_cleanup:
191
- self.issues.append(f"[Memory Leak] {filename}: useEffect with subscriptions but no cleanup function. Memory leak on unmount.")
192
-
193
- # 2.7 Console.log Detection
194
- console_logs = len(re.findall(r'console\.log|console\.warn|console\.error|console\.debug', content))
195
- if console_logs > 5:
196
- self.warnings.append(f"[Performance] {filename}: {console_logs} console.log statements detected. Remove before production (blocks JS thread).")
197
-
198
- # 2.8 Inline Function Detection
199
- if is_react_native:
200
- inline_functions = re.findall(r'(?:onPress|onPressIn|onPressOut|renderItem):\s*\([^)]*\)\s*=>', content)
201
- if len(inline_functions) > 3:
202
- self.warnings.append(f"[Performance] {filename}: {len(inline_functions)} inline arrow functions in props. Creates new function every render. Use useCallback.")
203
-
204
- # 2.9 Animation Properties Check
205
- # Warn if animating expensive properties
206
- animating_layout = bool(re.search(r'Animated\.timing.*(?:width|height|margin|padding)', content))
207
- if animating_layout:
208
- self.issues.append(f"[Performance] {filename}: Animating layout properties (width/height/margin). Use transform/opacity for 60fps.")
209
-
210
- # --- 3. MOBILE NAVIGATION CHECKS ---
211
-
212
- # 3.1 Tab Bar Max Items Check
213
- tab_bar_items = len(re.findall(r'Tab\.Screen|createBottomTabNavigator|BottomTab', content))
214
- if tab_bar_items > 5:
215
- self.warnings.append(f"[Navigation] {filename}: {tab_bar_items} tab bar items (max 5 recommended). More than 5 becomes hard to tap.")
216
-
217
- # 3.2 Tab State Preservation Check
218
- has_tab_nav = bool(re.search(r'createBottomTabNavigator|Tab\.Navigator', content))
219
- if has_tab_nav:
220
- # Look for lazy prop (false preserves state)
221
- has_lazy_false = bool(re.search(r'lazy:\s*false', content))
222
- if not has_lazy_false:
223
- self.warnings.append(f"[Navigation] {filename}: Tab navigation without lazy: false. Tabs may lose state on switch.")
224
-
225
- # 3.3 Back Handling Check
226
- has_back_listener = bool(re.search(r'BackHandler|useFocusEffect|navigation\.addListener', content))
227
- has_custom_back = bool(re.search(r'onBackPress|handleBackPress', content))
228
- if has_custom_back and not has_back_listener:
229
- self.warnings.append(f"[Navigation] {filename}: Custom back handling without BackHandler listener. May not work correctly.")
230
-
231
- # 3.4 Deep Link Support Check
232
- has_linking = bool(re.search(r'Linking\.|Linking\.openURL|deepLink|universalLink', content))
233
- has_config = bool(re.search(r'apollo-link|react-native-screens|navigation\.link', content))
234
- if not has_linking and not has_config:
235
- self.passed_count += 1
236
- else:
237
- if has_linking and not has_config:
238
- self.warnings.append(f"[Navigation] {filename}: Deep linking detected but may lack proper configuration. Test notification/share flows.")
239
-
240
- # --- 4. MOBILE TYPOGRAPHY CHECKS ---
241
-
242
- # 4.1 System Font Check
243
- if is_react_native:
244
- has_custom_font = bool(re.search(r"fontFamily:\s*[\"'][^\"']+", content))
245
- has_system_font = bool(re.search(r"fontFamily:\s*[\"']?(?:System|San Francisco|Roboto|-apple-system)", content))
246
- if has_custom_font and not has_system_font:
247
- self.warnings.append(f"[Typography] {filename}: Custom font detected. Consider system fonts (iOS: SF Pro, Android: Roboto) for native feel.")
248
-
249
- # 4.2 Text Scaling Check (iOS Dynamic Type)
250
- if is_react_native:
251
- has_font_sizes = bool(re.search(r'fontSize:', content))
252
- has_scaling = bool(re.search(r'allowFontScaling:\s*true|responsiveFontSize|useWindowDimensions', content))
253
- if has_font_sizes and not has_scaling:
254
- self.warnings.append(f"[Typography] {filename}: Fixed font sizes without scaling support. Consider allowFontScaling for accessibility.")
255
-
256
- # 4.3 Mobile Line Height Check
257
- line_heights = re.findall(r'lineHeight:\s*([\d.]+)', content)
258
- for lh in line_heights:
259
- if float(lh) > 1.8:
260
- self.warnings.append(f"[Typography] {filename}: lineHeight {lh} too high for mobile. Mobile text needs tighter spacing (1.3-1.5).")
261
-
262
- # 4.4 Font Size Limits
263
- font_sizes = re.findall(r'fontSize:\s*([\d.]+)', content)
264
- for fs in font_sizes:
265
- size = float(fs)
266
- if size < 12:
267
- self.warnings.append(f"[Typography] {filename}: fontSize {size}px below 12px minimum readability.")
268
- elif size > 32:
269
- self.warnings.append(f"[Typography] {filename}: fontSize {size}px very large. Consider using responsive scaling.")
270
-
271
- # --- 5. MOBILE COLOR SYSTEM CHECKS ---
272
-
273
- # 5.1 Pure Black Avoidance
274
- if re.search(r'#000000|color:\s*black|backgroundColor:\s*["\']?black', content):
275
- self.warnings.append(f"[Color] {filename}: Pure black (#000000) detected. Use dark gray (#1C1C1E iOS, #121212 Android) for better OLED/battery.")
276
-
277
- # 5.2 Dark Mode Support
278
- has_color_schemes = bool(re.search(r'useColorScheme|colorScheme|appearance:\s*["\']?dark', content))
279
- has_dark_mode_style = bool(re.search(r'\\\?.*dark|style:\s*.*dark|isDark', content))
280
- if not has_color_schemes and not has_dark_mode_style:
281
- self.warnings.append(f"[Color] {filename}: No dark mode support detected. Consider useColorScheme for system dark mode.")
282
-
283
- # --- 6. PLATFORM iOS CHECKS ---
284
-
285
- if is_react_native:
286
- # 6.1 SF Symbols Check
287
- has_ios_icons = bool(re.search(r'@expo/vector-icons|ionicons', content))
288
- has_sf_symbols = bool(re.search(r'sf-symbol|SF Symbols', content))
289
- if has_ios_icons and not has_sf_symbols:
290
- self.passed_count += 1
291
-
292
- # 6.2 iOS Haptic Types
293
- has_haptic_import = bool(re.search(r'expo-haptics|react-native-haptic-feedback', content))
294
- has_haptic_types = bool(re.search(r'ImpactFeedback|NotificationFeedback|SelectionFeedback', content))
295
- if has_haptic_import and not has_haptic_types:
296
- self.warnings.append(f"[iOS Haptics] {filename}: Haptic library imported but not using typed haptics (Impact/Notification/Selection).")
297
-
298
- # 6.3 iOS Safe Area
299
- has_safe_area = bool(re.search(r'SafeAreaView|useSafeAreaInsets|safeArea', content))
300
- if not has_safe_area:
301
- self.warnings.append(f"[iOS] {filename}: No SafeArea detected. Content may be hidden by notch/home indicator.")
302
-
303
- # --- 7. PLATFORM ANDROID CHECKS ---
304
-
305
- if is_react_native:
306
- # 7.1 Material Icons Check
307
- has_material_icons = bool(re.search(r'@expo/vector-icons|MaterialIcons', content))
308
- if has_material_icons:
309
- self.passed_count += 1
310
-
311
- # 7.2 Ripple Effect
312
- has_ripple = bool(re.search(r'ripple|android_ripple|foregroundRipple', content))
313
- has_pressable = bool(re.search(r'Pressable|Touchable', content))
314
- if has_pressable and not has_ripple:
315
- self.warnings.append(f"[Android] {filename}: Touchable without ripple effect. Android users expect ripple feedback.")
316
-
317
- # 7.3 Hardware Back Button
318
- if is_react_native:
319
- has_back_button = bool(re.search(r'BackHandler|useBackHandler', content))
320
- has_navigation = bool(re.search(r'@react-navigation', content))
321
- if has_navigation and not has_back_button:
322
- self.warnings.append(f"[Android] {filename}: React Navigation detected without BackHandler listener. Android hardware back may not work correctly.")
323
-
324
- # --- 8. MOBILE BACKEND CHECKS ---
325
-
326
- # 8.1 Secure Storage Check
327
- has_async_storage = bool(re.search(r'AsyncStorage|@react-native-async-storage', content))
328
- has_secure_storage = bool(re.search(r'SecureStore|Keychain|EncryptedSharedPreferences', content))
329
- has_token_storage = bool(re.search(r'token|jwt|auth.*storage', content, re.IGNORECASE))
330
- if has_token_storage and has_async_storage and not has_secure_storage:
331
- self.issues.append(f"[Security] {filename}: Storing auth tokens in AsyncStorage (insecure). Use SecureStore (iOS) / EncryptedSharedPreferences (Android).")
332
-
333
- # 8.2 Offline Handling Check
334
- has_network = bool(re.search(r'fetch|axios|netinfo|@react-native-community/netinfo', content))
335
- has_offline = bool(re.search(r'offline|isConnected|netInfo|cache.*offline', content))
336
- if has_network and not has_offline:
337
- self.warnings.append(f"[Offline] {filename}: Network requests detected without offline handling. Consider NetInfo for connection status.")
338
-
339
- # 8.3 Push Notification Support
340
- has_push = bool(re.search(r'Notifications|pushNotification|Firebase\.messaging|PushNotificationIOS', content))
341
- has_push_handler = bool(re.search(r'onNotification|addNotificationListener|notification\.open', content))
342
- if has_push and not has_push_handler:
343
- self.warnings.append(f"[Push] {filename}: Push notifications imported but no handler found. May miss notifications.")
344
-
345
- # --- 9. EXTENDED MOBILE TYPOGRAPHY CHECKS ---
346
-
347
- # 9.1 iOS Type Scale Check
348
- if is_react_native:
349
- # Check for iOS text styles that match HIG
350
- has_large_title = bool(re.search(r'fontSize:\s*34|largeTitle|font-weight:\s*["\']?bold', content))
351
- has_title_1 = bool(re.search(r'fontSize:\s*28', content))
352
- has_headline = bool(re.search(r'fontSize:\s*17.*semibold|headline', content))
353
- has_body = bool(re.search(r'fontSize:\s*17.*regular|body', content))
354
-
355
- # Check if following iOS scale roughly
356
- font_sizes = re.findall(r'fontSize:\s*([\d.]+)', content)
357
- ios_scale_sizes = [34, 28, 22, 20, 17, 16, 15, 13, 12, 11]
358
- matching_ios = sum(1 for size in font_sizes if any(abs(float(size) - ios_size) < 1 for ios_size in ios_scale_sizes))
359
-
360
- if len(font_sizes) > 3 and matching_ios < len(font_sizes) / 2:
361
- self.warnings.append(f"[iOS Typography] {filename}: Font sizes don't match iOS type scale. Consider iOS text styles for native feel.")
362
-
363
- # 9.2 Android Material Type Scale Check
364
- if is_react_native:
365
- # Check for Material 3 text styles
366
- has_display = bool(re.search(r'fontSize:\s*[456][0-9]|display', content))
367
- has_headline_material = bool(re.search(r'fontSize:\s*[23][0-9]|headline', content))
368
- has_title_material = bool(re.search(r'fontSize:\s*2[12][0-9].*medium|title', content))
369
- has_body_material = bool(re.search(r'fontSize:\s*1[456].*regular|body', content))
370
- has_label = bool(re.search(r'fontSize:\s*1[1234].*medium|label', content))
371
-
372
- # Check if using sp (scale-independent pixels)
373
- uses_sp = bool(re.search(r'\d+\s*sp\b', content))
374
- if has_display or has_headline_material:
375
- if not uses_sp:
376
- self.warnings.append(f"[Android Typography] {filename}: Material typography detected without sp units. Use sp for text to respect user font size preferences.")
377
-
378
- # 9.3 Modular Scale Check
379
- # Check if font sizes follow modular scale
380
- font_sizes = re.findall(r'fontSize:\s*(\d+(?:\.\d+)?)', content)
381
- if len(font_sizes) > 3:
382
- sorted_sizes = sorted(set([float(s) for s in font_sizes]))
383
- ratios = []
384
- for i in range(1, len(sorted_sizes)):
385
- if sorted_sizes[i-1] > 0:
386
- ratios.append(sorted_sizes[i] / sorted_sizes[i-1])
387
-
388
- # Common ratios: 1.125, 1.2, 1.25, 1.333, 1.5
389
- common_ratios = {1.125, 1.2, 1.25, 1.333, 1.5}
390
- for ratio in ratios[:3]:
391
- if not any(abs(ratio - cr) < 0.03 for cr in common_ratios):
392
- self.warnings.append(f"[Typography] {filename}: Font sizes may not follow modular scale (ratio: {ratio:.2f}). Consider consistent ratio.")
393
- break
394
-
395
- # 9.4 Line Length Check (Mobile-specific)
396
- # Mobile text should be 40-60 characters max
397
- if is_react_native:
398
- has_long_text = bool(re.search(r'<Text[^>]*>[^<]{40,}', content))
399
- has_max_width = bool(re.search(r'maxWidth|max-w-\d+|width:\s*["\']?\d+', content))
400
- if has_long_text and not has_max_width:
401
- self.warnings.append(f"[Mobile Typography] {filename}: Text without max-width constraint. Mobile text should be 40-60 characters per line for readability.")
402
-
403
- # 9.5 Font Weight Pattern Check
404
- # Check for font weight distribution
405
- if is_react_native:
406
- font_weights = re.findall(r'fontWeight:\s*["\']?(\d+|normal|bold|medium|light)', content)
407
- weight_map = {'normal': '400', 'light': '300', 'medium': '500', 'bold': '700'}
408
- numeric_weights = []
409
- for w in font_weights:
410
- val = weight_map.get(w.lower(), w)
411
- try:
412
- numeric_weights.append(int(val))
413
- except:
414
- pass
415
-
416
- # Check if overusing bold (mobile should be regular-dominant)
417
- bold_count = sum(1 for w in numeric_weights if w >= 700)
418
- regular_count = sum(1 for w in numeric_weights if 400 <= w < 500)
419
- if bold_count > regular_count:
420
- self.warnings.append(f"[Mobile Typography] {filename}: More bold weights than regular. Mobile typography should be regular-dominant for readability.")
421
-
422
- # --- 10. EXTENDED MOBILE COLOR SYSTEM CHECKS ---
423
-
424
- # 10.1 OLED Optimization Check
425
- # Check for near-black colors instead of pure black
426
- if re.search(r'#121212|#1A1A1A|#0D0D0D', content):
427
- self.passed_count += 1 # Good OLED optimization
428
- elif re.search(r'backgroundColor:\s*["\']?#000000', content):
429
- # Using pure black for background is OK for OLED
430
- pass
431
- elif re.search(r'backgroundColor:\s*["\']?#[0-9A-Fa-f]{6}', content):
432
- # Check if using light colors in dark mode (bad for OLED)
433
- self.warnings.append(f"[Mobile Color] {filename}: Consider OLED-optimized dark backgrounds (#121212 Android, #000000 iOS) for battery savings.")
434
-
435
- # 10.2 Saturated Color Detection (Battery)
436
- # Highly saturated colors consume more power on OLED
437
- hex_colors = re.findall(r'#([0-9A-Fa-f]{2})([0-9A-Fa-f]{2})([0-9A-Fa-f]{2})', content)
438
- saturated_count = 0
439
- for r, g, b in hex_colors:
440
- # Convert to RGB 0-255
441
- try:
442
- r_val, g_val, b_val = int(r, 16), int(g, 16), int(b, 16)
443
- max_val = max(r_val, g_val, b_val)
444
- min_val = min(r_val, g_val, b_val)
445
- # Saturation = (max - min) / max
446
- if max_val > 0:
447
- saturation = (max_val - min_val) / max_val
448
- if saturation > 0.8: # Highly saturated
449
- saturated_count += 1
450
- except:
451
- pass
452
-
453
- if saturated_count > 10:
454
- self.warnings.append(f"[Mobile Color] {filename}: {saturated_count} highly saturated colors detected. Desaturated colors save battery on OLED screens.")
455
-
456
- # 10.3 Outdoor Visibility Check
457
- # Low contrast combinations fail in outdoor sunlight
458
- light_colors = re.findall(r'#[0-9A-Fa-f]{6}|rgba?\([^)]+\)', content)
459
- # Check for potential low contrast (light gray on white, dark gray on black)
460
- potential_low_contrast = bool(re.search(r'#[EeEeEeEe].*#ffffff|#999999.*#ffffff|#333333.*#000000|#666666.*#000000', content))
461
- if potential_low_contrast:
462
- self.warnings.append(f"[Mobile Color] {filename}: Possible low contrast combination detected. Critical for outdoor visibility. Ensure WCAG AAA (7:1) for mobile.")
463
-
464
- # 10.4 Dark Mode Text Color Check
465
- # In dark mode, text should not be pure white
466
- has_dark_mode = bool(re.search(r'dark:\s*|isDark|useColorScheme|colorScheme:\s*["\']?dark', content))
467
- if has_dark_mode:
468
- has_pure_white_text = bool(re.search(r'color:\s*["\']?#ffffff|#fff["\']?\}|textColor:\s*["\']?white', content))
469
- if has_pure_white_text:
470
- self.warnings.append(f"[Mobile Color] {filename}: Pure white text (#FFFFFF) in dark mode. Use #E8E8E8 or light gray for better readability.")
471
-
472
- # --- 11. EXTENDED PLATFORM IOS CHECKS ---
473
-
474
- if is_react_native:
475
- # 11.1 SF Pro Font Detection
476
- has_sf_pro = bool(re.search(r'SF Pro|SFPro|fontFamily:\s*["\']?[-\s]*SF', content))
477
- has_custom_font = bool(re.search(r'fontFamily:\s*["\'][^"\']+', content))
478
- if has_custom_font and not has_sf_pro:
479
- self.warnings.append(f"[iOS] {filename}: Custom font without SF Pro fallback. Consider SF Pro Text for body, SF Pro Display for headings.")
480
-
481
- # 11.2 iOS System Colors Check
482
- # Check for semantic color usage
483
- has_label = bool(re.search(r'color:\s*["\']?label|\.label', content))
484
- has_secondaryLabel = bool(re.search(r'secondaryLabel|\.secondaryLabel', content))
485
- has_systemBackground = bool(re.search(r'systemBackground|\.systemBackground', content))
486
-
487
- has_hardcoded_gray = bool(re.search(r'#[78]0{4}', content))
488
- if has_hardcoded_gray and not (has_label or has_secondaryLabel):
489
- self.warnings.append(f"[iOS] {filename}: Hardcoded gray colors detected. Consider iOS semantic colors (label, secondaryLabel) for automatic dark mode.")
490
-
491
- # 11.3 iOS Accent Colors Check
492
- ios_blue = bool(re.search(r'#007AFF|#0A84FF|systemBlue', content))
493
- ios_green = bool(re.search(r'#34C759|#30D158|systemGreen', content))
494
- ios_red = bool(re.search(r'#FF3B30|#FF453A|systemRed', content))
495
-
496
- has_custom_primary = bool(re.search(r'primaryColor|theme.*primary|colors\.primary', content))
497
- if has_custom_primary and not (ios_blue or ios_green or ios_red):
498
- self.warnings.append(f"[iOS] {filename}: Custom primary color without iOS system color fallback. Consider systemBlue for consistent iOS feel.")
499
-
500
- # 11.4 iOS Navigation Patterns Check
501
- has_navigation_bar = bool(re.search(r'navigationOptions|headerStyle|cardStyle', content))
502
- has_header_title = bool(re.search(r'title:\s*["\']|headerTitle|navigation\.setOptions', content))
503
- if has_navigation_bar and not has_header_title:
504
- self.warnings.append(f"[iOS] {filename}: Navigation bar detected without title. iOS apps should have clear context in nav bar.")
505
-
506
- # 11.5 iOS Component Patterns Check
507
- # Check for iOS-specific components
508
- has_alert = bool(re.search(r'Alert\.alert|showAlert', content))
509
- has_action_sheet = bool(re.search(r'ActionSheet|ActionSheetIOS|showActionSheetWithOptions', content))
510
- has_activity_indicator = bool(re.search(r'ActivityIndicator|ActivityIndic', content))
511
-
512
- if has_alert or has_action_sheet or has_activity_indicator:
513
- self.passed_count += 1 # Good iOS component usage
514
-
515
- # --- 12. EXTENDED PLATFORM ANDROID CHECKS ---
516
-
517
- if is_react_native:
518
- # 12.1 Roboto Font Detection
519
- has_roboto = bool(re.search(r'Roboto|fontFamily:\s*["\']?[-\s]*Roboto', content))
520
- has_custom_font = bool(re.search(r'fontFamily:\s*["\'][^"\']+', content))
521
- if has_custom_font and not has_roboto:
522
- self.warnings.append(f"[Android] {filename}: Custom font without Roboto fallback. Roboto is optimized for Android displays.")
523
-
524
- # 12.2 Material 3 Dynamic Color Check
525
- has_material_colors = bool(re.search(r'MD3|MaterialYou|dynamicColor|useColorScheme', content))
526
- has_theme_provider = bool(re.search(r'MaterialTheme|ThemeProvider|PaperProvider|ThemeProvider', content))
527
- if not has_material_colors and not has_theme_provider:
528
- self.warnings.append(f"[Android] {filename}: No Material 3 dynamic color detected. Consider Material 3 theming for personalized feel.")
529
-
530
- # 12.3 Material Elevation Check
531
- # Check for elevation values (Material 3 uses elevation for depth)
532
- has_elevation = bool(re.search(r'elevation:\s*\d+|shadowOpacity|shadowRadius|android:elevation', content))
533
- has_box_shadow = bool(re.search(r'boxShadow:', content))
534
- if has_box_shadow and not has_elevation:
535
- self.warnings.append(f"[Android] {filename}: CSS box-shadow detected without elevation. Consider Material elevation system for consistent depth.")
536
-
537
- # 12.4 Material Component Patterns Check
538
- # Check for Material components
539
- has_ripple = bool(re.search(r'ripple|android_ripple|foregroundRipple', content))
540
- has_card = bool(re.search(r'Card|Paper|elevation.*\d+', content))
541
- has_fab = bool(re.search(r'FAB|FloatingActionButton|fab', content))
542
- has_snackbar = bool(re.search(r'Snackbar|showSnackBar|Toast', content))
543
-
544
- material_component_count = sum([has_ripple, has_card, has_fab, has_snackbar])
545
- if material_component_count >= 2:
546
- self.passed_count += 1 # Good Material design usage
547
-
548
- # 12.5 Android Navigation Patterns Check
549
- has_top_app_bar = bool(re.search(r'TopAppBar|AppBar|CollapsingToolbar', content))
550
- has_bottom_nav = bool(re.search(r'BottomNavigation|BottomNav', content))
551
- has_navigation_rail = bool(re.search(r'NavigationRail', content))
552
-
553
- if has_bottom_nav:
554
- self.passed_count += 1 # Good Android pattern
555
- elif has_top_app_bar and not (has_bottom_nav or has_navigation_rail):
556
- self.warnings.append(f"[Android] {filename}: TopAppBar without bottom navigation. Consider BottomNavigation for thumb-friendly access.")
557
-
558
- # --- 13. MOBILE TESTING CHECKS ---
559
-
560
- # 13.1 Testing Tool Detection
561
- has_rntl = bool(re.search(r'react-native-testing-library|@testing-library', content))
562
- has_detox = bool(re.search(r'detox|element\(|by\.text|by\.id', content))
563
- has_maestro = bool(re.search(r'maestro|\.yaml$', content))
564
- has_jest = bool(re.search(r'jest|describe\(|test\(|it\(', content))
565
-
566
- testing_tools = []
567
- if has_jest: testing_tools.append('Jest')
568
- if has_rntl: testing_tools.append('RNTL')
569
- if has_detox: testing_tools.append('Detox')
570
- if has_maestro: testing_tools.append('Maestro')
571
-
572
- if len(testing_tools) == 0:
573
- self.warnings.append(f"[Testing] {filename}: No testing framework detected. Consider Jest (unit) + Detox/Maestro (E2E) for mobile.")
574
-
575
- # 13.2 Test Pyramid Balance Check
576
- test_files = len(re.findall(r'\.test\.(tsx|ts|js|jsx)|\.spec\.', content))
577
- e2e_tests = len(re.findall(r'detox|maestro|e2e|spec\.e2e', content.lower()))
578
-
579
- if test_files > 0 and e2e_tests == 0:
580
- self.warnings.append(f"[Testing] {filename}: Unit tests found but no E2E tests. Mobile needs E2E on real devices for complete coverage.")
581
-
582
- # 13.3 Accessibility Label Check (Mobile-specific)
583
- if is_react_native:
584
- has_pressable = bool(re.search(r'Pressable|TouchableOpacity|TouchableHighlight', content))
585
- has_a11y_label = bool(re.search(r'accessibilityLabel|aria-label|testID', content))
586
- if has_pressable and not has_a11y_label:
587
- self.warnings.append(f"[A11y Mobile] {filename}: Touchable element without accessibilityLabel. Screen readers need labels for all interactive elements.")
588
-
589
- # --- 14. MOBILE DEBUGGING CHECKS ---
590
-
591
- # 14.1 Performance Profiling Check
592
- has_performance = bool(re.search(r'Performance|systrace|profile|Flipper', content))
593
- has_console_log = len(re.findall(r'console\.(log|warn|error|debug|info)', content))
594
- has_debugger = bool(re.search(r'debugger|__DEV__|React\.DevTools', content))
595
-
596
- if has_console_log > 10:
597
- self.warnings.append(f"[Debugging] {filename}: {has_console_log} console.log statements. Remove before production; they block JS thread.")
598
-
599
- if has_performance:
600
- self.passed_count += 1 # Good performance monitoring
601
-
602
- # 14.2 Error Boundary Check
603
- has_error_boundary = bool(re.search(r'ErrorBoundary|componentDidCatch|getDerivedStateFromError', content))
604
- if not has_error_boundary and is_react_native:
605
- self.warnings.append(f"[Debugging] {filename}: No ErrorBoundary detected. Consider adding ErrorBoundary to prevent app crashes.")
606
-
607
- # 14.3 Hermes Check (React Native specific)
608
- if is_react_native:
609
- # Check if using Hermes engine (should be default in modern RN)
610
- # This is more of a configuration check, not code pattern
611
- self.passed_count += 1 # Hermes is default in RN 0.70+
612
-
613
- def audit_directory(self, directory: str) -> None:
614
- extensions = {'.tsx', '.ts', '.jsx', '.js', '.dart'}
615
- for root, dirs, files in os.walk(directory):
616
- dirs[:] = [d for d in dirs if d not in {'node_modules', '.git', 'dist', 'build', '.next', 'ios', 'android', 'build', '.idea'}]
617
- for file in files:
618
- if Path(file).suffix in extensions:
619
- self.audit_file(os.path.join(root, file))
620
-
621
- def get_report(self):
622
- return {
623
- "files_checked": self.files_checked,
624
- "issues": self.issues,
625
- "warnings": self.warnings,
626
- "passed_checks": self.passed_count,
627
- "compliant": len(self.issues) == 0
628
- }
629
-
630
-
631
- def main():
632
- if len(sys.argv) < 2:
633
- print("Usage: python mobile_audit.py <directory>")
634
- sys.exit(1)
635
-
636
- path = sys.argv[1]
637
- is_json = "--json" in sys.argv
638
-
639
- auditor = MobileAuditor()
640
- if os.path.isfile(path):
641
- auditor.audit_file(path)
642
- else:
643
- auditor.audit_directory(path)
644
-
645
- report = auditor.get_report()
646
-
647
- if is_json:
648
- print(json.dumps(report, indent=2))
649
- else:
650
- print(f"\n[MOBILE AUDIT] {report['files_checked']} mobile files checked")
651
- print("-" * 50)
652
- if report['issues']:
653
- print(f"[!] ISSUES ({len(report['issues'])}):")
654
- for i in report['issues'][:10]:
655
- print(f" - {i}")
656
- if report['warnings']:
657
- print(f"[*] WARNINGS ({len(report['warnings'])}):")
658
- for w in report['warnings'][:15]:
659
- print(f" - {w}")
660
- print(f"[+] PASSED CHECKS: {report['passed_checks']}")
661
- status = "PASS" if report['compliant'] else "FAIL"
662
- print(f"STATUS: {status}")
663
-
664
- sys.exit(0 if report['compliant'] else 1)
665
-
666
-
667
- if __name__ == "__main__":
668
- # Fix missing import
669
- import re
670
- main()
1
+ #!/usr/bin/env python3
2
+ """
3
+ Mobile UX Audit Script - Full Mobile Design Coverage
4
+
5
+ Analyzes React Native / Flutter code for compliance with:
6
+
7
+ 1. TOUCH PSYCHOLOGY (touch-psychology.md):
8
+ - Touch Target Sizes (44pt iOS, 48dp Android, 44px WCAG)
9
+ - Touch Target Spacing (8px minimum gap)
10
+ - Thumb Zone Placement (primary CTAs at bottom)
11
+ - Gesture Alternatives (visible buttons for swipe)
12
+ - Haptic Feedback Patterns
13
+ - Touch Feedback Timing (<50ms)
14
+ - Touch Accessibility (motor impairment support)
15
+
16
+ 2. MOBILE PERFORMANCE (mobile-performance.md):
17
+ - ScrollView vs FlatList (CRITICAL)
18
+ - React.memo for List Items
19
+ - useCallback for renderItem
20
+ - Stable keyExtractor (NOT index)
21
+ - useNativeDriver for Animations
22
+ - Memory Leak Prevention (cleanup)
23
+ - Console.log Detection
24
+ - Inline Function Detection
25
+ - Animation Performance (transform/opacity only)
26
+
27
+ 3. MOBILE NAVIGATION (mobile-navigation.md):
28
+ - Tab Bar Max Items (5)
29
+ - Tab State Preservation
30
+ - Proper Back Handling
31
+ - Deep Link Support
32
+ - Navigation Structure
33
+
34
+ 4. MOBILE TYPOGRAPHY (mobile-typography.md):
35
+ - System Font Usage
36
+ - Dynamic Type Support (iOS)
37
+ - Text Scaling Constraints
38
+ - Mobile Line Height
39
+ - Font Size Limits
40
+
41
+ 5. MOBILE COLOR SYSTEM (mobile-color-system.md):
42
+ - Pure Black Avoidance (#000000)
43
+ - OLED Optimization
44
+ - Dark Mode Support
45
+ - Contrast Ratios
46
+
47
+ 6. PLATFORM iOS (platform-ios.md):
48
+ - SF Symbols Usage
49
+ - iOS Navigation Patterns
50
+ - iOS Haptic Types
51
+ - iOS-Specific Components
52
+
53
+ 7. PLATFORM ANDROID (platform-android.md):
54
+ - Material Icons Usage
55
+ - Android Navigation Patterns
56
+ - Ripple Effects
57
+ - Android-Specific Components
58
+
59
+ 8. MOBILE BACKEND (mobile-backend.md):
60
+ - Secure Storage (NOT AsyncStorage)
61
+ - Offline Handling
62
+ - Push Notification Support
63
+ - API Response Caching
64
+
65
+ Total: 50+ mobile-specific checks
66
+ """
67
+
68
+ import sys
69
+ import os
70
+ import re
71
+ import json
72
+ from pathlib import Path
73
+
74
+ class MobileAuditor:
75
+ def __init__(self):
76
+ self.issues = []
77
+ self.warnings = []
78
+ self.passed_count = 0
79
+ self.files_checked = 0
80
+
81
+ def audit_file(self, filepath: str) -> None:
82
+ try:
83
+ with open(filepath, 'r', encoding='utf-8', errors='replace') as f:
84
+ content = f.read()
85
+ except:
86
+ return
87
+
88
+ self.files_checked += 1
89
+ filename = os.path.basename(filepath)
90
+
91
+ # Detect framework
92
+ is_react_native = bool(re.search(r'react-native|@react-navigation|React\.Native', content))
93
+ is_flutter = bool(re.search(r'import \'package:flutter|MaterialApp|Widget\.build', content))
94
+
95
+ if not (is_react_native or is_flutter):
96
+ return # Skip non-mobile files
97
+
98
+ # --- 1. TOUCH PSYCHOLOGY CHECKS ---
99
+
100
+ # 1.1 Touch Target Size Check
101
+ # Look for small touch targets
102
+ small_sizes = re.findall(r'(?:width|height|size):\s*([0-3]\d)', content)
103
+ for size in small_sizes:
104
+ if int(size) < 44:
105
+ self.issues.append(f"[Touch Target] {filename}: Touch target size {size}px < 44px minimum (iOS: 44pt, Android: 48dp)")
106
+
107
+ # 1.2 Touch Target Spacing Check
108
+ # Look for inadequate spacing between touchable elements
109
+ small_gaps = re.findall(r'(?:margin|gap):\s*([0-7])\s*(?:px|dp)', content)
110
+ for gap in small_gaps:
111
+ if int(gap) < 8:
112
+ self.warnings.append(f"[Touch Spacing] {filename}: Touch target spacing {gap}px < 8px minimum. Accidental taps risk.")
113
+
114
+ # 1.3 Thumb Zone Placement Check
115
+ # Primary CTAs should be at bottom (easy thumb reach)
116
+ primary_buttons = re.findall(r'(?:testID|id):\s*["\'](?:.*(?:primary|cta|submit|confirm)[^"\']*)["\']', content, re.IGNORECASE)
117
+ has_bottom_placement = bool(re.search(r'position:\s*["\']?absolute["\']?|bottom:\s*\d+|style.*bottom|justifyContent:\s*["\']?flex-end', content))
118
+ if primary_buttons and not has_bottom_placement:
119
+ self.warnings.append(f"[Thumb Zone] {filename}: Primary CTA may not be in thumb zone (bottom). Place primary actions at bottom for easy reach.")
120
+
121
+ # 1.4 Gesture Alternatives Check
122
+ # Swipe actions should have visible button alternatives
123
+ has_swipe_gestures = bool(re.search(r'Swipeable|onSwipe|PanGestureHandler|swipe', content))
124
+ has_visible_buttons = bool(re.search(r'Button.*(?:delete|archive|more)|TouchableOpacity|Pressable', content))
125
+ if has_swipe_gestures and not has_visible_buttons:
126
+ self.warnings.append(f"[Gestures] {filename}: Swipe gestures detected without visible button alternatives. Motor impaired users need alternatives.")
127
+
128
+ # 1.5 Haptic Feedback Check
129
+ # Important actions should have haptic feedback
130
+ has_important_actions = bool(re.search(r'(?:onPress|onSubmit|delete|remove|confirm|purchase)', content))
131
+ has_haptics = bool(re.search(r'Haptics|Vibration|react-native-haptic-feedback|FeedbackManager', content))
132
+ if has_important_actions and not has_haptics:
133
+ self.warnings.append(f"[Haptics] {filename}: Important actions without haptic feedback. Consider adding haptic confirmation.")
134
+
135
+ # 1.6 Touch Feedback Timing Check
136
+ # Touch feedback should be immediate (<50ms)
137
+ if is_react_native:
138
+ has_pressable = bool(re.search(r'Pressable|TouchableOpacity', content))
139
+ has_feedback_state = bool(re.search(r'pressed|style.*opacity|underlay', content))
140
+ if has_pressable and not has_feedback_state:
141
+ self.warnings.append(f"[Touch Feedback] {filename}: Pressable without visual feedback state. Add opacity/scale change for tap confirmation.")
142
+
143
+ # --- 2. MOBILE PERFORMANCE CHECKS ---
144
+
145
+ # 2.1 CRITICAL: ScrollView vs FlatList
146
+ has_scrollview = bool(re.search(r'<ScrollView|ScrollView\.', content))
147
+ has_map_in_scrollview = bool(re.search(r'ScrollView.*\.map\(|ScrollView.*\{.*\.map', content))
148
+ if has_scrollview and has_map_in_scrollview:
149
+ self.issues.append(f"[Performance CRITICAL] {filename}: ScrollView with .map() detected. Use FlatList for lists to prevent memory explosion.")
150
+
151
+ # 2.2 React.memo Check
152
+ if is_react_native:
153
+ has_list = bool(re.search(r'FlatList|FlashList|SectionList', content))
154
+ has_react_memo = bool(re.search(r'React\.memo|memo\(', content))
155
+ if has_list and not has_react_memo:
156
+ self.warnings.append(f"[Performance] {filename}: FlatList without React.memo on list items. Items will re-render on every parent update.")
157
+
158
+ # 2.3 useCallback Check
159
+ if is_react_native:
160
+ has_flatlist = bool(re.search(r'FlatList|FlashList', content))
161
+ has_use_callback = bool(re.search(r'useCallback', content))
162
+ if has_flatlist and not has_use_callback:
163
+ self.warnings.append(f"[Performance] {filename}: FlatList renderItem without useCallback. New function created every render.")
164
+
165
+ # 2.4 keyExtractor Check (CRITICAL)
166
+ if is_react_native:
167
+ has_flatlist = bool(re.search(r'FlatList', content))
168
+ has_key_extractor = bool(re.search(r'keyExtractor', content))
169
+ uses_index_key = bool(re.search(r'key=\{.*index.*\}|key:\s*index', content))
170
+ if has_flatlist and not has_key_extractor:
171
+ self.issues.append(f"[Performance CRITICAL] {filename}: FlatList without keyExtractor. Index-based keys cause bugs on reorder/delete.")
172
+ if uses_index_key:
173
+ self.issues.append(f"[Performance CRITICAL] {filename}: Using index as key. This causes bugs when list changes. Use unique ID from data.")
174
+
175
+ # 2.5 useNativeDriver Check
176
+ if is_react_native:
177
+ has_animated = bool(re.search(r'Animated\.', content))
178
+ has_native_driver = bool(re.search(r'useNativeDriver:\s*true', content))
179
+ has_native_driver_false = bool(re.search(r'useNativeDriver:\s*false', content))
180
+ if has_animated and has_native_driver_false:
181
+ self.warnings.append(f"[Performance] {filename}: Animation with useNativeDriver: false. Use true for 60fps (only supports transform/opacity).")
182
+ if has_animated and not has_native_driver:
183
+ self.warnings.append(f"[Performance] {filename}: Animated component without useNativeDriver. Add useNativeDriver: true for 60fps.")
184
+
185
+ # 2.6 Memory Leak Check
186
+ if is_react_native:
187
+ has_effect = bool(re.search(r'useEffect', content))
188
+ has_cleanup = bool(re.search(r'return\s*\(\)\s*=>|return\s+function', content))
189
+ has_subscriptions = bool(re.search(r'addEventListener|subscribe|\.focus\(\)|\.off\(', content))
190
+ if has_effect and has_subscriptions and not has_cleanup:
191
+ self.issues.append(f"[Memory Leak] {filename}: useEffect with subscriptions but no cleanup function. Memory leak on unmount.")
192
+
193
+ # 2.7 Console.log Detection
194
+ console_logs = len(re.findall(r'console\.log|console\.warn|console\.error|console\.debug', content))
195
+ if console_logs > 5:
196
+ self.warnings.append(f"[Performance] {filename}: {console_logs} console.log statements detected. Remove before production (blocks JS thread).")
197
+
198
+ # 2.8 Inline Function Detection
199
+ if is_react_native:
200
+ inline_functions = re.findall(r'(?:onPress|onPressIn|onPressOut|renderItem):\s*\([^)]*\)\s*=>', content)
201
+ if len(inline_functions) > 3:
202
+ self.warnings.append(f"[Performance] {filename}: {len(inline_functions)} inline arrow functions in props. Creates new function every render. Use useCallback.")
203
+
204
+ # 2.9 Animation Properties Check
205
+ # Warn if animating expensive properties
206
+ animating_layout = bool(re.search(r'Animated\.timing.*(?:width|height|margin|padding)', content))
207
+ if animating_layout:
208
+ self.issues.append(f"[Performance] {filename}: Animating layout properties (width/height/margin). Use transform/opacity for 60fps.")
209
+
210
+ # --- 3. MOBILE NAVIGATION CHECKS ---
211
+
212
+ # 3.1 Tab Bar Max Items Check
213
+ tab_bar_items = len(re.findall(r'Tab\.Screen|createBottomTabNavigator|BottomTab', content))
214
+ if tab_bar_items > 5:
215
+ self.warnings.append(f"[Navigation] {filename}: {tab_bar_items} tab bar items (max 5 recommended). More than 5 becomes hard to tap.")
216
+
217
+ # 3.2 Tab State Preservation Check
218
+ has_tab_nav = bool(re.search(r'createBottomTabNavigator|Tab\.Navigator', content))
219
+ if has_tab_nav:
220
+ # Look for lazy prop (false preserves state)
221
+ has_lazy_false = bool(re.search(r'lazy:\s*false', content))
222
+ if not has_lazy_false:
223
+ self.warnings.append(f"[Navigation] {filename}: Tab navigation without lazy: false. Tabs may lose state on switch.")
224
+
225
+ # 3.3 Back Handling Check
226
+ has_back_listener = bool(re.search(r'BackHandler|useFocusEffect|navigation\.addListener', content))
227
+ has_custom_back = bool(re.search(r'onBackPress|handleBackPress', content))
228
+ if has_custom_back and not has_back_listener:
229
+ self.warnings.append(f"[Navigation] {filename}: Custom back handling without BackHandler listener. May not work correctly.")
230
+
231
+ # 3.4 Deep Link Support Check
232
+ has_linking = bool(re.search(r'Linking\.|Linking\.openURL|deepLink|universalLink', content))
233
+ has_config = bool(re.search(r'apollo-link|react-native-screens|navigation\.link', content))
234
+ if not has_linking and not has_config:
235
+ self.passed_count += 1
236
+ else:
237
+ if has_linking and not has_config:
238
+ self.warnings.append(f"[Navigation] {filename}: Deep linking detected but may lack proper configuration. Test notification/share flows.")
239
+
240
+ # --- 4. MOBILE TYPOGRAPHY CHECKS ---
241
+
242
+ # 4.1 System Font Check
243
+ if is_react_native:
244
+ has_custom_font = bool(re.search(r"fontFamily:\s*[\"'][^\"']+", content))
245
+ has_system_font = bool(re.search(r"fontFamily:\s*[\"']?(?:System|San Francisco|Roboto|-apple-system)", content))
246
+ if has_custom_font and not has_system_font:
247
+ self.warnings.append(f"[Typography] {filename}: Custom font detected. Consider system fonts (iOS: SF Pro, Android: Roboto) for native feel.")
248
+
249
+ # 4.2 Text Scaling Check (iOS Dynamic Type)
250
+ if is_react_native:
251
+ has_font_sizes = bool(re.search(r'fontSize:', content))
252
+ has_scaling = bool(re.search(r'allowFontScaling:\s*true|responsiveFontSize|useWindowDimensions', content))
253
+ if has_font_sizes and not has_scaling:
254
+ self.warnings.append(f"[Typography] {filename}: Fixed font sizes without scaling support. Consider allowFontScaling for accessibility.")
255
+
256
+ # 4.3 Mobile Line Height Check
257
+ line_heights = re.findall(r'lineHeight:\s*([\d.]+)', content)
258
+ for lh in line_heights:
259
+ if float(lh) > 1.8:
260
+ self.warnings.append(f"[Typography] {filename}: lineHeight {lh} too high for mobile. Mobile text needs tighter spacing (1.3-1.5).")
261
+
262
+ # 4.4 Font Size Limits
263
+ font_sizes = re.findall(r'fontSize:\s*([\d.]+)', content)
264
+ for fs in font_sizes:
265
+ size = float(fs)
266
+ if size < 12:
267
+ self.warnings.append(f"[Typography] {filename}: fontSize {size}px below 12px minimum readability.")
268
+ elif size > 32:
269
+ self.warnings.append(f"[Typography] {filename}: fontSize {size}px very large. Consider using responsive scaling.")
270
+
271
+ # --- 5. MOBILE COLOR SYSTEM CHECKS ---
272
+
273
+ # 5.1 Pure Black Avoidance
274
+ if re.search(r'#000000|color:\s*black|backgroundColor:\s*["\']?black', content):
275
+ self.warnings.append(f"[Color] {filename}: Pure black (#000000) detected. Use dark gray (#1C1C1E iOS, #121212 Android) for better OLED/battery.")
276
+
277
+ # 5.2 Dark Mode Support
278
+ has_color_schemes = bool(re.search(r'useColorScheme|colorScheme|appearance:\s*["\']?dark', content))
279
+ has_dark_mode_style = bool(re.search(r'\\\?.*dark|style:\s*.*dark|isDark', content))
280
+ if not has_color_schemes and not has_dark_mode_style:
281
+ self.warnings.append(f"[Color] {filename}: No dark mode support detected. Consider useColorScheme for system dark mode.")
282
+
283
+ # --- 6. PLATFORM iOS CHECKS ---
284
+
285
+ if is_react_native:
286
+ # 6.1 SF Symbols Check
287
+ has_ios_icons = bool(re.search(r'@expo/vector-icons|ionicons', content))
288
+ has_sf_symbols = bool(re.search(r'sf-symbol|SF Symbols', content))
289
+ if has_ios_icons and not has_sf_symbols:
290
+ self.passed_count += 1
291
+
292
+ # 6.2 iOS Haptic Types
293
+ has_haptic_import = bool(re.search(r'expo-haptics|react-native-haptic-feedback', content))
294
+ has_haptic_types = bool(re.search(r'ImpactFeedback|NotificationFeedback|SelectionFeedback', content))
295
+ if has_haptic_import and not has_haptic_types:
296
+ self.warnings.append(f"[iOS Haptics] {filename}: Haptic library imported but not using typed haptics (Impact/Notification/Selection).")
297
+
298
+ # 6.3 iOS Safe Area
299
+ has_safe_area = bool(re.search(r'SafeAreaView|useSafeAreaInsets|safeArea', content))
300
+ if not has_safe_area:
301
+ self.warnings.append(f"[iOS] {filename}: No SafeArea detected. Content may be hidden by notch/home indicator.")
302
+
303
+ # --- 7. PLATFORM ANDROID CHECKS ---
304
+
305
+ if is_react_native:
306
+ # 7.1 Material Icons Check
307
+ has_material_icons = bool(re.search(r'@expo/vector-icons|MaterialIcons', content))
308
+ if has_material_icons:
309
+ self.passed_count += 1
310
+
311
+ # 7.2 Ripple Effect
312
+ has_ripple = bool(re.search(r'ripple|android_ripple|foregroundRipple', content))
313
+ has_pressable = bool(re.search(r'Pressable|Touchable', content))
314
+ if has_pressable and not has_ripple:
315
+ self.warnings.append(f"[Android] {filename}: Touchable without ripple effect. Android users expect ripple feedback.")
316
+
317
+ # 7.3 Hardware Back Button
318
+ if is_react_native:
319
+ has_back_button = bool(re.search(r'BackHandler|useBackHandler', content))
320
+ has_navigation = bool(re.search(r'@react-navigation', content))
321
+ if has_navigation and not has_back_button:
322
+ self.warnings.append(f"[Android] {filename}: React Navigation detected without BackHandler listener. Android hardware back may not work correctly.")
323
+
324
+ # --- 8. MOBILE BACKEND CHECKS ---
325
+
326
+ # 8.1 Secure Storage Check
327
+ has_async_storage = bool(re.search(r'AsyncStorage|@react-native-async-storage', content))
328
+ has_secure_storage = bool(re.search(r'SecureStore|Keychain|EncryptedSharedPreferences', content))
329
+ has_token_storage = bool(re.search(r'token|jwt|auth.*storage', content, re.IGNORECASE))
330
+ if has_token_storage and has_async_storage and not has_secure_storage:
331
+ self.issues.append(f"[Security] {filename}: Storing auth tokens in AsyncStorage (insecure). Use SecureStore (iOS) / EncryptedSharedPreferences (Android).")
332
+
333
+ # 8.2 Offline Handling Check
334
+ has_network = bool(re.search(r'fetch|axios|netinfo|@react-native-community/netinfo', content))
335
+ has_offline = bool(re.search(r'offline|isConnected|netInfo|cache.*offline', content))
336
+ if has_network and not has_offline:
337
+ self.warnings.append(f"[Offline] {filename}: Network requests detected without offline handling. Consider NetInfo for connection status.")
338
+
339
+ # 8.3 Push Notification Support
340
+ has_push = bool(re.search(r'Notifications|pushNotification|Firebase\.messaging|PushNotificationIOS', content))
341
+ has_push_handler = bool(re.search(r'onNotification|addNotificationListener|notification\.open', content))
342
+ if has_push and not has_push_handler:
343
+ self.warnings.append(f"[Push] {filename}: Push notifications imported but no handler found. May miss notifications.")
344
+
345
+ # --- 9. EXTENDED MOBILE TYPOGRAPHY CHECKS ---
346
+
347
+ # 9.1 iOS Type Scale Check
348
+ if is_react_native:
349
+ # Check for iOS text styles that match HIG
350
+ has_large_title = bool(re.search(r'fontSize:\s*34|largeTitle|font-weight:\s*["\']?bold', content))
351
+ has_title_1 = bool(re.search(r'fontSize:\s*28', content))
352
+ has_headline = bool(re.search(r'fontSize:\s*17.*semibold|headline', content))
353
+ has_body = bool(re.search(r'fontSize:\s*17.*regular|body', content))
354
+
355
+ # Check if following iOS scale roughly
356
+ font_sizes = re.findall(r'fontSize:\s*([\d.]+)', content)
357
+ ios_scale_sizes = [34, 28, 22, 20, 17, 16, 15, 13, 12, 11]
358
+ matching_ios = sum(1 for size in font_sizes if any(abs(float(size) - ios_size) < 1 for ios_size in ios_scale_sizes))
359
+
360
+ if len(font_sizes) > 3 and matching_ios < len(font_sizes) / 2:
361
+ self.warnings.append(f"[iOS Typography] {filename}: Font sizes don't match iOS type scale. Consider iOS text styles for native feel.")
362
+
363
+ # 9.2 Android Material Type Scale Check
364
+ if is_react_native:
365
+ # Check for Material 3 text styles
366
+ has_display = bool(re.search(r'fontSize:\s*[456][0-9]|display', content))
367
+ has_headline_material = bool(re.search(r'fontSize:\s*[23][0-9]|headline', content))
368
+ has_title_material = bool(re.search(r'fontSize:\s*2[12][0-9].*medium|title', content))
369
+ has_body_material = bool(re.search(r'fontSize:\s*1[456].*regular|body', content))
370
+ has_label = bool(re.search(r'fontSize:\s*1[1234].*medium|label', content))
371
+
372
+ # Check if using sp (scale-independent pixels)
373
+ uses_sp = bool(re.search(r'\d+\s*sp\b', content))
374
+ if has_display or has_headline_material:
375
+ if not uses_sp:
376
+ self.warnings.append(f"[Android Typography] {filename}: Material typography detected without sp units. Use sp for text to respect user font size preferences.")
377
+
378
+ # 9.3 Modular Scale Check
379
+ # Check if font sizes follow modular scale
380
+ font_sizes = re.findall(r'fontSize:\s*(\d+(?:\.\d+)?)', content)
381
+ if len(font_sizes) > 3:
382
+ sorted_sizes = sorted(set([float(s) for s in font_sizes]))
383
+ ratios = []
384
+ for i in range(1, len(sorted_sizes)):
385
+ if sorted_sizes[i-1] > 0:
386
+ ratios.append(sorted_sizes[i] / sorted_sizes[i-1])
387
+
388
+ # Common ratios: 1.125, 1.2, 1.25, 1.333, 1.5
389
+ common_ratios = {1.125, 1.2, 1.25, 1.333, 1.5}
390
+ for ratio in ratios[:3]:
391
+ if not any(abs(ratio - cr) < 0.03 for cr in common_ratios):
392
+ self.warnings.append(f"[Typography] {filename}: Font sizes may not follow modular scale (ratio: {ratio:.2f}). Consider consistent ratio.")
393
+ break
394
+
395
+ # 9.4 Line Length Check (Mobile-specific)
396
+ # Mobile text should be 40-60 characters max
397
+ if is_react_native:
398
+ has_long_text = bool(re.search(r'<Text[^>]*>[^<]{40,}', content))
399
+ has_max_width = bool(re.search(r'maxWidth|max-w-\d+|width:\s*["\']?\d+', content))
400
+ if has_long_text and not has_max_width:
401
+ self.warnings.append(f"[Mobile Typography] {filename}: Text without max-width constraint. Mobile text should be 40-60 characters per line for readability.")
402
+
403
+ # 9.5 Font Weight Pattern Check
404
+ # Check for font weight distribution
405
+ if is_react_native:
406
+ font_weights = re.findall(r'fontWeight:\s*["\']?(\d+|normal|bold|medium|light)', content)
407
+ weight_map = {'normal': '400', 'light': '300', 'medium': '500', 'bold': '700'}
408
+ numeric_weights = []
409
+ for w in font_weights:
410
+ val = weight_map.get(w.lower(), w)
411
+ try:
412
+ numeric_weights.append(int(val))
413
+ except:
414
+ pass
415
+
416
+ # Check if overusing bold (mobile should be regular-dominant)
417
+ bold_count = sum(1 for w in numeric_weights if w >= 700)
418
+ regular_count = sum(1 for w in numeric_weights if 400 <= w < 500)
419
+ if bold_count > regular_count:
420
+ self.warnings.append(f"[Mobile Typography] {filename}: More bold weights than regular. Mobile typography should be regular-dominant for readability.")
421
+
422
+ # --- 10. EXTENDED MOBILE COLOR SYSTEM CHECKS ---
423
+
424
+ # 10.1 OLED Optimization Check
425
+ # Check for near-black colors instead of pure black
426
+ if re.search(r'#121212|#1A1A1A|#0D0D0D', content):
427
+ self.passed_count += 1 # Good OLED optimization
428
+ elif re.search(r'backgroundColor:\s*["\']?#000000', content):
429
+ # Using pure black for background is OK for OLED
430
+ pass
431
+ elif re.search(r'backgroundColor:\s*["\']?#[0-9A-Fa-f]{6}', content):
432
+ # Check if using light colors in dark mode (bad for OLED)
433
+ self.warnings.append(f"[Mobile Color] {filename}: Consider OLED-optimized dark backgrounds (#121212 Android, #000000 iOS) for battery savings.")
434
+
435
+ # 10.2 Saturated Color Detection (Battery)
436
+ # Highly saturated colors consume more power on OLED
437
+ hex_colors = re.findall(r'#([0-9A-Fa-f]{2})([0-9A-Fa-f]{2})([0-9A-Fa-f]{2})', content)
438
+ saturated_count = 0
439
+ for r, g, b in hex_colors:
440
+ # Convert to RGB 0-255
441
+ try:
442
+ r_val, g_val, b_val = int(r, 16), int(g, 16), int(b, 16)
443
+ max_val = max(r_val, g_val, b_val)
444
+ min_val = min(r_val, g_val, b_val)
445
+ # Saturation = (max - min) / max
446
+ if max_val > 0:
447
+ saturation = (max_val - min_val) / max_val
448
+ if saturation > 0.8: # Highly saturated
449
+ saturated_count += 1
450
+ except:
451
+ pass
452
+
453
+ if saturated_count > 10:
454
+ self.warnings.append(f"[Mobile Color] {filename}: {saturated_count} highly saturated colors detected. Desaturated colors save battery on OLED screens.")
455
+
456
+ # 10.3 Outdoor Visibility Check
457
+ # Low contrast combinations fail in outdoor sunlight
458
+ light_colors = re.findall(r'#[0-9A-Fa-f]{6}|rgba?\([^)]+\)', content)
459
+ # Check for potential low contrast (light gray on white, dark gray on black)
460
+ potential_low_contrast = bool(re.search(r'#[EeEeEeEe].*#ffffff|#999999.*#ffffff|#333333.*#000000|#666666.*#000000', content))
461
+ if potential_low_contrast:
462
+ self.warnings.append(f"[Mobile Color] {filename}: Possible low contrast combination detected. Critical for outdoor visibility. Ensure WCAG AAA (7:1) for mobile.")
463
+
464
+ # 10.4 Dark Mode Text Color Check
465
+ # In dark mode, text should not be pure white
466
+ has_dark_mode = bool(re.search(r'dark:\s*|isDark|useColorScheme|colorScheme:\s*["\']?dark', content))
467
+ if has_dark_mode:
468
+ has_pure_white_text = bool(re.search(r'color:\s*["\']?#ffffff|#fff["\']?\}|textColor:\s*["\']?white', content))
469
+ if has_pure_white_text:
470
+ self.warnings.append(f"[Mobile Color] {filename}: Pure white text (#FFFFFF) in dark mode. Use #E8E8E8 or light gray for better readability.")
471
+
472
+ # --- 11. EXTENDED PLATFORM IOS CHECKS ---
473
+
474
+ if is_react_native:
475
+ # 11.1 SF Pro Font Detection
476
+ has_sf_pro = bool(re.search(r'SF Pro|SFPro|fontFamily:\s*["\']?[-\s]*SF', content))
477
+ has_custom_font = bool(re.search(r'fontFamily:\s*["\'][^"\']+', content))
478
+ if has_custom_font and not has_sf_pro:
479
+ self.warnings.append(f"[iOS] {filename}: Custom font without SF Pro fallback. Consider SF Pro Text for body, SF Pro Display for headings.")
480
+
481
+ # 11.2 iOS System Colors Check
482
+ # Check for semantic color usage
483
+ has_label = bool(re.search(r'color:\s*["\']?label|\.label', content))
484
+ has_secondaryLabel = bool(re.search(r'secondaryLabel|\.secondaryLabel', content))
485
+ has_systemBackground = bool(re.search(r'systemBackground|\.systemBackground', content))
486
+
487
+ has_hardcoded_gray = bool(re.search(r'#[78]0{4}', content))
488
+ if has_hardcoded_gray and not (has_label or has_secondaryLabel):
489
+ self.warnings.append(f"[iOS] {filename}: Hardcoded gray colors detected. Consider iOS semantic colors (label, secondaryLabel) for automatic dark mode.")
490
+
491
+ # 11.3 iOS Accent Colors Check
492
+ ios_blue = bool(re.search(r'#007AFF|#0A84FF|systemBlue', content))
493
+ ios_green = bool(re.search(r'#34C759|#30D158|systemGreen', content))
494
+ ios_red = bool(re.search(r'#FF3B30|#FF453A|systemRed', content))
495
+
496
+ has_custom_primary = bool(re.search(r'primaryColor|theme.*primary|colors\.primary', content))
497
+ if has_custom_primary and not (ios_blue or ios_green or ios_red):
498
+ self.warnings.append(f"[iOS] {filename}: Custom primary color without iOS system color fallback. Consider systemBlue for consistent iOS feel.")
499
+
500
+ # 11.4 iOS Navigation Patterns Check
501
+ has_navigation_bar = bool(re.search(r'navigationOptions|headerStyle|cardStyle', content))
502
+ has_header_title = bool(re.search(r'title:\s*["\']|headerTitle|navigation\.setOptions', content))
503
+ if has_navigation_bar and not has_header_title:
504
+ self.warnings.append(f"[iOS] {filename}: Navigation bar detected without title. iOS apps should have clear context in nav bar.")
505
+
506
+ # 11.5 iOS Component Patterns Check
507
+ # Check for iOS-specific components
508
+ has_alert = bool(re.search(r'Alert\.alert|showAlert', content))
509
+ has_action_sheet = bool(re.search(r'ActionSheet|ActionSheetIOS|showActionSheetWithOptions', content))
510
+ has_activity_indicator = bool(re.search(r'ActivityIndicator|ActivityIndic', content))
511
+
512
+ if has_alert or has_action_sheet or has_activity_indicator:
513
+ self.passed_count += 1 # Good iOS component usage
514
+
515
+ # --- 12. EXTENDED PLATFORM ANDROID CHECKS ---
516
+
517
+ if is_react_native:
518
+ # 12.1 Roboto Font Detection
519
+ has_roboto = bool(re.search(r'Roboto|fontFamily:\s*["\']?[-\s]*Roboto', content))
520
+ has_custom_font = bool(re.search(r'fontFamily:\s*["\'][^"\']+', content))
521
+ if has_custom_font and not has_roboto:
522
+ self.warnings.append(f"[Android] {filename}: Custom font without Roboto fallback. Roboto is optimized for Android displays.")
523
+
524
+ # 12.2 Material 3 Dynamic Color Check
525
+ has_material_colors = bool(re.search(r'MD3|MaterialYou|dynamicColor|useColorScheme', content))
526
+ has_theme_provider = bool(re.search(r'MaterialTheme|ThemeProvider|PaperProvider|ThemeProvider', content))
527
+ if not has_material_colors and not has_theme_provider:
528
+ self.warnings.append(f"[Android] {filename}: No Material 3 dynamic color detected. Consider Material 3 theming for personalized feel.")
529
+
530
+ # 12.3 Material Elevation Check
531
+ # Check for elevation values (Material 3 uses elevation for depth)
532
+ has_elevation = bool(re.search(r'elevation:\s*\d+|shadowOpacity|shadowRadius|android:elevation', content))
533
+ has_box_shadow = bool(re.search(r'boxShadow:', content))
534
+ if has_box_shadow and not has_elevation:
535
+ self.warnings.append(f"[Android] {filename}: CSS box-shadow detected without elevation. Consider Material elevation system for consistent depth.")
536
+
537
+ # 12.4 Material Component Patterns Check
538
+ # Check for Material components
539
+ has_ripple = bool(re.search(r'ripple|android_ripple|foregroundRipple', content))
540
+ has_card = bool(re.search(r'Card|Paper|elevation.*\d+', content))
541
+ has_fab = bool(re.search(r'FAB|FloatingActionButton|fab', content))
542
+ has_snackbar = bool(re.search(r'Snackbar|showSnackBar|Toast', content))
543
+
544
+ material_component_count = sum([has_ripple, has_card, has_fab, has_snackbar])
545
+ if material_component_count >= 2:
546
+ self.passed_count += 1 # Good Material design usage
547
+
548
+ # 12.5 Android Navigation Patterns Check
549
+ has_top_app_bar = bool(re.search(r'TopAppBar|AppBar|CollapsingToolbar', content))
550
+ has_bottom_nav = bool(re.search(r'BottomNavigation|BottomNav', content))
551
+ has_navigation_rail = bool(re.search(r'NavigationRail', content))
552
+
553
+ if has_bottom_nav:
554
+ self.passed_count += 1 # Good Android pattern
555
+ elif has_top_app_bar and not (has_bottom_nav or has_navigation_rail):
556
+ self.warnings.append(f"[Android] {filename}: TopAppBar without bottom navigation. Consider BottomNavigation for thumb-friendly access.")
557
+
558
+ # --- 13. MOBILE TESTING CHECKS ---
559
+
560
+ # 13.1 Testing Tool Detection
561
+ has_rntl = bool(re.search(r'react-native-testing-library|@testing-library', content))
562
+ has_detox = bool(re.search(r'detox|element\(|by\.text|by\.id', content))
563
+ has_maestro = bool(re.search(r'maestro|\.yaml$', content))
564
+ has_jest = bool(re.search(r'jest|describe\(|test\(|it\(', content))
565
+
566
+ testing_tools = []
567
+ if has_jest: testing_tools.append('Jest')
568
+ if has_rntl: testing_tools.append('RNTL')
569
+ if has_detox: testing_tools.append('Detox')
570
+ if has_maestro: testing_tools.append('Maestro')
571
+
572
+ if len(testing_tools) == 0:
573
+ self.warnings.append(f"[Testing] {filename}: No testing framework detected. Consider Jest (unit) + Detox/Maestro (E2E) for mobile.")
574
+
575
+ # 13.2 Test Pyramid Balance Check
576
+ test_files = len(re.findall(r'\.test\.(tsx|ts|js|jsx)|\.spec\.', content))
577
+ e2e_tests = len(re.findall(r'detox|maestro|e2e|spec\.e2e', content.lower()))
578
+
579
+ if test_files > 0 and e2e_tests == 0:
580
+ self.warnings.append(f"[Testing] {filename}: Unit tests found but no E2E tests. Mobile needs E2E on real devices for complete coverage.")
581
+
582
+ # 13.3 Accessibility Label Check (Mobile-specific)
583
+ if is_react_native:
584
+ has_pressable = bool(re.search(r'Pressable|TouchableOpacity|TouchableHighlight', content))
585
+ has_a11y_label = bool(re.search(r'accessibilityLabel|aria-label|testID', content))
586
+ if has_pressable and not has_a11y_label:
587
+ self.warnings.append(f"[A11y Mobile] {filename}: Touchable element without accessibilityLabel. Screen readers need labels for all interactive elements.")
588
+
589
+ # --- 14. MOBILE DEBUGGING CHECKS ---
590
+
591
+ # 14.1 Performance Profiling Check
592
+ has_performance = bool(re.search(r'Performance|systrace|profile|Flipper', content))
593
+ has_console_log = len(re.findall(r'console\.(log|warn|error|debug|info)', content))
594
+ has_debugger = bool(re.search(r'debugger|__DEV__|React\.DevTools', content))
595
+
596
+ if has_console_log > 10:
597
+ self.warnings.append(f"[Debugging] {filename}: {has_console_log} console.log statements. Remove before production; they block JS thread.")
598
+
599
+ if has_performance:
600
+ self.passed_count += 1 # Good performance monitoring
601
+
602
+ # 14.2 Error Boundary Check
603
+ has_error_boundary = bool(re.search(r'ErrorBoundary|componentDidCatch|getDerivedStateFromError', content))
604
+ if not has_error_boundary and is_react_native:
605
+ self.warnings.append(f"[Debugging] {filename}: No ErrorBoundary detected. Consider adding ErrorBoundary to prevent app crashes.")
606
+
607
+ # 14.3 Hermes Check (React Native specific)
608
+ if is_react_native:
609
+ # Check if using Hermes engine (should be default in modern RN)
610
+ # This is more of a configuration check, not code pattern
611
+ self.passed_count += 1 # Hermes is default in RN 0.70+
612
+
613
+ def audit_directory(self, directory: str) -> None:
614
+ extensions = {'.tsx', '.ts', '.jsx', '.js', '.dart'}
615
+ for root, dirs, files in os.walk(directory):
616
+ dirs[:] = [d for d in dirs if d not in {'node_modules', '.git', 'dist', 'build', '.next', 'ios', 'android', 'build', '.idea'}]
617
+ for file in files:
618
+ if Path(file).suffix in extensions:
619
+ self.audit_file(os.path.join(root, file))
620
+
621
+ def get_report(self):
622
+ return {
623
+ "files_checked": self.files_checked,
624
+ "issues": self.issues,
625
+ "warnings": self.warnings,
626
+ "passed_checks": self.passed_count,
627
+ "compliant": len(self.issues) == 0
628
+ }
629
+
630
+
631
+ def main():
632
+ if len(sys.argv) < 2:
633
+ print("Usage: python mobile_audit.py <directory>")
634
+ sys.exit(1)
635
+
636
+ path = sys.argv[1]
637
+ is_json = "--json" in sys.argv
638
+
639
+ auditor = MobileAuditor()
640
+ if os.path.isfile(path):
641
+ auditor.audit_file(path)
642
+ else:
643
+ auditor.audit_directory(path)
644
+
645
+ report = auditor.get_report()
646
+
647
+ if is_json:
648
+ print(json.dumps(report, indent=2))
649
+ else:
650
+ print(f"\n[MOBILE AUDIT] {report['files_checked']} mobile files checked")
651
+ print("-" * 50)
652
+ if report['issues']:
653
+ print(f"[!] ISSUES ({len(report['issues'])}):")
654
+ for i in report['issues'][:10]:
655
+ print(f" - {i}")
656
+ if report['warnings']:
657
+ print(f"[*] WARNINGS ({len(report['warnings'])}):")
658
+ for w in report['warnings'][:15]:
659
+ print(f" - {w}")
660
+ print(f"[+] PASSED CHECKS: {report['passed_checks']}")
661
+ status = "PASS" if report['compliant'] else "FAIL"
662
+ print(f"STATUS: {status}")
663
+
664
+ sys.exit(0 if report['compliant'] else 1)
665
+
666
+
667
+ if __name__ == "__main__":
668
+ # Fix missing import
669
+ import re
670
+ main()