codymaster 4.8.0 → 7.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (416) hide show
  1. package/CHANGELOG.md +331 -7
  2. package/README.md +226 -296
  3. package/dist/advisory-handoff.js +89 -0
  4. package/dist/advisory-report.js +105 -0
  5. package/dist/agent/antigravity.js +152 -0
  6. package/dist/agent/backend.js +2 -0
  7. package/dist/agent/claude.js +196 -0
  8. package/dist/agent/codex.js +204 -0
  9. package/dist/agent/copilot.js +284 -0
  10. package/dist/agent/cursor.js +211 -0
  11. package/dist/agent/factory.js +30 -0
  12. package/dist/agent/gemini.js +142 -0
  13. package/dist/agent/opencode.js +205 -0
  14. package/dist/agent/spawn-helper.js +237 -0
  15. package/dist/agent/version.js +25 -0
  16. package/dist/browse/adapter-factory.js +69 -0
  17. package/dist/browse/adapters/agent-browser-adapter.js +305 -0
  18. package/dist/browse/adapters/playwright-adapter.js +309 -0
  19. package/dist/browse/adapters/types.js +6 -0
  20. package/dist/browse/error-collector.js +132 -0
  21. package/dist/browse/event-log.js +109 -0
  22. package/dist/browse/index.js +17 -0
  23. package/dist/browse-server.js +204 -120
  24. package/dist/cli/command-registry.js +20 -0
  25. package/dist/cli/commands/bench.js +69 -0
  26. package/dist/cli/commands/brain.js +108 -0
  27. package/dist/cli/commands/dashboard.js +76 -2
  28. package/dist/cli/commands/engineering.js +326 -4
  29. package/dist/cli/commands/evolve.js +123 -0
  30. package/dist/cli/commands/install.js +160 -0
  31. package/dist/cli/commands/learn.js +181 -0
  32. package/dist/cli/commands/mcp-serve.js +104 -0
  33. package/dist/cli/commands/parallel.js +138 -0
  34. package/dist/cli/commands/quality.js +105 -0
  35. package/dist/cli/commands/stack.js +49 -0
  36. package/dist/cli/commands/update.js +159 -0
  37. package/dist/cli/update-check.js +94 -10
  38. package/dist/cm-config.js +0 -18
  39. package/dist/codybench/judges/automated.js +31 -0
  40. package/dist/codybench/runners/claude-code.js +32 -0
  41. package/dist/codybench/suites/memory-retention.js +85 -0
  42. package/dist/codybench/suites/tdd-regression.js +35 -0
  43. package/dist/codybench/suites/token-efficiency.js +55 -0
  44. package/dist/codybench/types.js +2 -0
  45. package/dist/context-db.js +157 -0
  46. package/dist/continuity.js +5 -7
  47. package/dist/dashboard.js +47 -6
  48. package/dist/data.js +35 -0
  49. package/dist/execution/tdd-gate.js +113 -0
  50. package/dist/execution-analyzer.js +138 -0
  51. package/dist/executor/cancel.js +34 -0
  52. package/dist/executor/gc.js +74 -0
  53. package/dist/executor/index.js +14 -0
  54. package/dist/executor/runner.js +70 -0
  55. package/dist/executor/workdir.js +31 -0
  56. package/dist/handoff/contracts.js +22 -0
  57. package/dist/handoff/index.js +18 -0
  58. package/dist/handoff/io.js +121 -0
  59. package/dist/index.js +7 -3
  60. package/dist/indexer/skills-lib.js +533 -0
  61. package/dist/indexer/skills-map.js +1374 -0
  62. package/dist/indexer/skills.js +16 -0
  63. package/dist/indexer/stack-detect.js +219 -0
  64. package/dist/install/copy.js +98 -0
  65. package/dist/install/engine.js +42 -0
  66. package/dist/install/paths.js +70 -0
  67. package/dist/install/platforms/_simple.js +85 -0
  68. package/dist/install/platforms/antigravity.js +91 -0
  69. package/dist/install/platforms/claude-code.js +107 -0
  70. package/dist/install/platforms/cursor.js +77 -0
  71. package/dist/install/platforms/index.js +27 -0
  72. package/dist/install/platforms/simple.js +163 -0
  73. package/dist/install/profiles.js +75 -0
  74. package/dist/install/types.js +2 -0
  75. package/dist/learning-promoter.js +246 -0
  76. package/dist/learnings.js +208 -0
  77. package/dist/mcp-context-server.js +230 -1
  78. package/dist/middleware/metrics.js +30 -0
  79. package/dist/middleware/security-headers.js +14 -0
  80. package/dist/realtime/event-bus.js +29 -0
  81. package/dist/realtime/ws-hub.js +91 -0
  82. package/dist/schemas/task-schema.js +48 -0
  83. package/dist/schemas/validate.js +18 -0
  84. package/dist/skill-chain.js +63 -1
  85. package/dist/skill-evolver.js +456 -0
  86. package/dist/skill-execution-cache.js +254 -0
  87. package/dist/skills-lock.js +96 -0
  88. package/dist/smart-brain-router.js +184 -0
  89. package/dist/sprint-pipeline.js +26 -0
  90. package/dist/storage/index.js +21 -0
  91. package/dist/storage/repos/activity-repo.js +46 -0
  92. package/dist/storage/repos/message-repo.js +39 -0
  93. package/dist/storage/repos/project-repo.js +56 -0
  94. package/dist/storage/repos/task-repo.js +142 -0
  95. package/dist/storage/services/project-service.js +49 -0
  96. package/dist/storage/services/task-service.js +97 -0
  97. package/dist/storage/sqlite.js +113 -0
  98. package/dist/storage-backend.js +10 -8
  99. package/dist/tier-classify.js +131 -0
  100. package/dist/token-budget.js +88 -0
  101. package/dist/ui/onboarding.js +51 -15
  102. package/dist/utils/cli-utils.js +7 -2
  103. package/dist/utils/design-taste.js +108 -0
  104. package/dist/utils/output-compress.js +143 -0
  105. package/dist/vibecoding-index.js +126 -0
  106. package/package.json +20 -6
  107. package/public/dashboard/app.js +52 -1
  108. package/scripts/build-skills-lock.mjs +88 -0
  109. package/scripts/build-skills.mjs +187 -28
  110. package/scripts/compress-skill.mjs +73 -0
  111. package/scripts/deprecate-skill.mjs +72 -0
  112. package/scripts/install.sh +170 -0
  113. package/scripts/mcp-bridge.js +2 -2
  114. package/scripts/postinstall.js +53 -335
  115. package/scripts/update-changelog.sh +88 -0
  116. package/scripts/validate-skills.mjs +101 -4
  117. package/skills/CLAUDE.md +0 -5
  118. package/skills/_shared/SKILL_TEMPLATE.md +62 -0
  119. package/skills/_shared/helpers.md +2 -8
  120. package/skills/cm-autopilot/scripts/autopilot.py +19 -2
  121. package/skills/cm-brainstorm-idea/SKILL.md +9 -0
  122. package/skills/cm-browse/SKILL.md +6 -0
  123. package/skills/cm-clean-code/SKILL.md +20 -0
  124. package/skills/cm-code-review/SKILL.md +21 -0
  125. package/skills/cm-codeintell/SKILL.md +9 -0
  126. package/skills/cm-conductor-worktrees/SKILL.archive.md +28 -0
  127. package/skills/cm-conductor-worktrees/SKILL.md +20 -18
  128. package/skills/cm-continuity/SKILL.md +41 -33
  129. package/skills/cm-dashboard/SKILL.archive.md +15 -0
  130. package/skills/cm-dashboard/SKILL.md +20 -9
  131. package/skills/cm-dashboard/ui/app.js +9 -1
  132. package/skills/cm-debugging/SKILL.md +9 -0
  133. package/skills/cm-design-studio/SKILL.archive.md +34 -0
  134. package/skills/cm-design-studio/SKILL.md +20 -24
  135. package/skills/cm-design-system/SKILL.md +1 -0
  136. package/skills/cm-ecosystem-roadmap/SKILL.md +4 -0
  137. package/skills/cm-engineering-meta/SKILL.archive.md +73 -0
  138. package/skills/cm-engineering-meta/SKILL.md +19 -62
  139. package/skills/cm-execution/SKILL.md +98 -0
  140. package/skills/cm-git-worktrees/SKILL.archive.md +157 -0
  141. package/skills/cm-git-worktrees/SKILL.md +15 -146
  142. package/skills/cm-guardian-runtime/SKILL.md +5 -1
  143. package/skills/cm-identity-guard/SKILL.md +8 -0
  144. package/skills/cm-mcp-engineering/SKILL.md +4 -0
  145. package/skills/cm-planning/SKILL.md +63 -92
  146. package/skills/cm-post-deploy-canary/SKILL.archive.md +22 -0
  147. package/skills/cm-post-deploy-canary/SKILL.md +20 -12
  148. package/skills/cm-project-bootstrap/SKILL.md +11 -0
  149. package/skills/cm-qa-visual-cli/SKILL.archive.md +22 -0
  150. package/skills/cm-qa-visual-cli/SKILL.md +19 -11
  151. package/skills/cm-quality-gate/SKILL.md +38 -0
  152. package/skills/cm-retro-cli/SKILL.md +4 -0
  153. package/skills/cm-safe-deploy/SKILL.md +9 -0
  154. package/skills/cm-second-opinion-cli/SKILL.archive.md +23 -0
  155. package/skills/cm-second-opinion-cli/SKILL.md +20 -13
  156. package/skills/cm-secret-shield/SKILL.archive.md +580 -0
  157. package/skills/cm-secret-shield/SKILL.md +15 -569
  158. package/skills/cm-security-gate/SKILL.archive.md +239 -0
  159. package/skills/cm-security-gate/SKILL.md +16 -228
  160. package/skills/cm-skill-chain/SKILL.md +25 -4
  161. package/skills/cm-skill-evolution/SKILL.md +83 -0
  162. package/skills/cm-skill-health/SKILL.archive.md +83 -0
  163. package/skills/cm-skill-health/SKILL.md +26 -0
  164. package/skills/cm-skill-index/SKILL.md +19 -3
  165. package/skills/cm-skill-mastery/SKILL.archive.md +156 -0
  166. package/skills/cm-skill-mastery/SKILL.md +16 -146
  167. package/skills/cm-skill-search/SKILL.archive.md +49 -0
  168. package/skills/cm-skill-search/SKILL.md +26 -0
  169. package/skills/cm-skill-share/SKILL.archive.md +58 -0
  170. package/skills/cm-skill-share/SKILL.md +26 -0
  171. package/skills/cm-sprint-bus/SKILL.md +13 -0
  172. package/skills/cm-start/SKILL.md +17 -10
  173. package/skills/cm-tdd/SKILL.md +21 -2
  174. package/skills/cm-terminal/SKILL.md +15 -0
  175. package/skills/cm-test-gate/SKILL.archive.md +245 -0
  176. package/skills/cm-test-gate/SKILL.md +15 -234
  177. package/skills/cm-ui-preview/SKILL.archive.md +153 -0
  178. package/skills/cm-ui-preview/SKILL.md +16 -143
  179. package/skills/cm-ux-master/cli/uxmaster/commands/mcp.py +1 -1
  180. package/skills/cm-ux-master/mcp/mcp-config.json +1 -1
  181. package/skills/cm-ux-master/mcp/server.py +2 -2
  182. package/skills/profiles/design.txt +1 -1
  183. package/skills/profiles/full.txt +4 -10
  184. package/skills/profiles/growth.txt +8 -8
  185. package/skills/profiles/knowledge.txt +1 -1
  186. package/skills/profiles/top35.json +41 -0
  187. package/adapters/antigravity.js +0 -15
  188. package/adapters/claude-code.js +0 -17
  189. package/adapters/cursor.js +0 -16
  190. package/install.sh +0 -1125
  191. package/scripts/viking-demo.ts +0 -105
  192. package/skills/cm-ads-tracker/SKILL.md +0 -401
  193. package/skills/cm-ads-tracker/evals/evals.json +0 -55
  194. package/skills/cm-ads-tracker/references/gtm-architecture.md +0 -321
  195. package/skills/cm-ads-tracker/references/industry-events.md +0 -294
  196. package/skills/cm-ads-tracker/references/platforms-api.md +0 -238
  197. package/skills/cm-ads-tracker/templates/capi-payload.md +0 -79
  198. package/skills/cm-ads-tracker/templates/datalayer-push.js +0 -104
  199. package/skills/cm-ads-tracker/templates/gtm-variables.js +0 -56
  200. package/skills/cm-auto-publisher/SKILL.md +0 -81
  201. package/skills/cm-booking-calendar/SKILL.md +0 -521
  202. package/skills/cm-booking-calendar/references/industry-patterns.md +0 -527
  203. package/skills/cm-booking-calendar/templates/booking-form.css +0 -626
  204. package/skills/cm-booking-calendar/templates/booking-form.html +0 -477
  205. package/skills/cm-booking-calendar/templates/calendar-engine.js +0 -419
  206. package/skills/cm-booking-calendar/templates/calendar-export.js +0 -395
  207. package/skills/cm-booking-calendar/templates/reminder-config.js +0 -629
  208. package/skills/cm-content-factory/.content-factory-state.json +0 -132
  209. package/skills/cm-content-factory/.git 2/logs/refs/heads/main +0 -1
  210. package/skills/cm-content-factory/.git 2/logs/refs/remotes/origin/main +0 -1
  211. package/skills/cm-content-factory/.git 2/objects/02/fb0956734b5f8ba3f918b7defd04a89cfe0076 +0 -0
  212. package/skills/cm-content-factory/.git 2/objects/08/1e129d75dc6feac6c02037272e6bd1a04e3324 +0 -0
  213. package/skills/cm-content-factory/.git 2/objects/0c/5393416f3c5e01c9a655a802bff0dd52f76f0a +0 -0
  214. package/skills/cm-content-factory/.git 2/objects/10/0b9be46978a946a77188f68be725098a122001 +0 -0
  215. package/skills/cm-content-factory/.git 2/objects/10/cf041167fc9843610eb3d90259ef3396315fdc +0 -0
  216. package/skills/cm-content-factory/.git 2/objects/12/5e19538dd6e1338ffe74f6c4c165b00435bf48 +0 -0
  217. package/skills/cm-content-factory/.git 2/objects/16/a9b9d0088d5c1347628b45a2620b479d8ad57c +0 -0
  218. package/skills/cm-content-factory/.git 2/objects/17/8c2a9ef93c33ae4eec9d58e82321f9229843a1 +0 -0
  219. package/skills/cm-content-factory/.git 2/objects/25/397ae41d09104d763bdcac2695209d85cdea89 +0 -0
  220. package/skills/cm-content-factory/.git 2/objects/2f/a836b7947f2d458e1f639788bf4bb0983a3305 +0 -0
  221. package/skills/cm-content-factory/.git 2/objects/3a/baaaf0a1c0909c0828335791557125fba911e0 +0 -0
  222. package/skills/cm-content-factory/.git 2/objects/42/2924221b81f5ce3c4e4daac9a64a24f9b01f9a +0 -0
  223. package/skills/cm-content-factory/.git 2/objects/42/ec0ce707447dc11446a34c9995fb8533801731 +0 -0
  224. package/skills/cm-content-factory/.git 2/objects/46/e43ce92866d56ce74b1d750db307cfe6154a15 +0 -0
  225. package/skills/cm-content-factory/.git 2/objects/48/5e41b633c63f55b8277bcc59f44f67681f671a +0 -0
  226. package/skills/cm-content-factory/.git 2/objects/49/49c596a3a89fa240642acd95dd3258e261eb09 +0 -0
  227. package/skills/cm-content-factory/.git 2/objects/50/9d42d8412ef8eaf7f7e138476bac2e4d10ce60 +0 -0
  228. package/skills/cm-content-factory/.git 2/objects/55/0c8c389d981b463ef849aeb792d8be3ccb6ec8 +0 -0
  229. package/skills/cm-content-factory/.git 2/objects/5d/82d3b18410cdda3ace3677436f0cb599dbe2d2 +0 -0
  230. package/skills/cm-content-factory/.git 2/objects/60/0617c58e871a38b33bf29e282d132bb3c381ad +0 -0
  231. package/skills/cm-content-factory/.git 2/objects/6a/8369a99c687b7245c92ffaf0e0f0dab9014504 +0 -0
  232. package/skills/cm-content-factory/.git 2/objects/79/bea435d40ab531c1aaf6be0432c6a5b7aaed21 +0 -0
  233. package/skills/cm-content-factory/.git 2/objects/7e/5ebd79251c2f14e4aceb86c74b6b6daae6b500 +0 -0
  234. package/skills/cm-content-factory/.git 2/objects/81/98a822a60178d6d5023ddb3e222cddf048742e +0 -0
  235. package/skills/cm-content-factory/.git 2/objects/86/0a0e1943dfe53411d2e499a1f16f46a96ef758 +0 -0
  236. package/skills/cm-content-factory/.git 2/objects/86/971fb55fdc081fdbae52376f0f13e57a4e9b04 +0 -0
  237. package/skills/cm-content-factory/.git 2/objects/88/b89dd609a0a03f8d4fe8bfde20d5b8fc1d326d +0 -0
  238. package/skills/cm-content-factory/.git 2/objects/90/8737edb6b7809e32cc01590b4e08ba42a9d40d +0 -0
  239. package/skills/cm-content-factory/.git 2/objects/93/d5a8a9a7d4fb7f11491cb596a6880528725118 +0 -0
  240. package/skills/cm-content-factory/.git 2/objects/98/46a2ab81d0c3b3eb00ef88fc56989aa7e9f316 +0 -0
  241. package/skills/cm-content-factory/.git 2/objects/9b/d8dd1e49cf274eaf9c555f3ab39dce7af5715e +0 -0
  242. package/skills/cm-content-factory/.git 2/objects/a1/13329fb0cec96ae78b222d33a24c3b5bc7fa1f +0 -0
  243. package/skills/cm-content-factory/.git 2/objects/a9/e6effe626e8a3aea3a8fc3364b492191c6e7d0 +0 -0
  244. package/skills/cm-content-factory/.git 2/objects/ad/6de7e48d9782cca9353d1ff0aa1aab7fe1df85 +0 -0
  245. package/skills/cm-content-factory/.git 2/objects/af/54ae316f771ff692e299ffcd8bf2f06b413b59 +0 -0
  246. package/skills/cm-content-factory/.git 2/objects/b0/4cb8b0b00dad633e731c1472161419e738d674 +0 -0
  247. package/skills/cm-content-factory/.git 2/objects/b3/094abb0b9ed46419b269e4a4e36a459690e3b0 +0 -0
  248. package/skills/cm-content-factory/.git 2/objects/b9/435c5d4baac2cfc5c83009ddd27b46b60db5f1 +0 -0
  249. package/skills/cm-content-factory/.git 2/objects/ba/5da17dbaec5ec2dcfdfd126aead518d1171d5c +0 -0
  250. package/skills/cm-content-factory/.git 2/objects/c0/bf58703aa258ba5dd63083bebaec8f223d844c +0 -0
  251. package/skills/cm-content-factory/.git 2/objects/c4/701a34edf1fc1bad58ccc57bd03f9426acb59a +0 -0
  252. package/skills/cm-content-factory/.git 2/objects/c7/5ccce9a4e5cc74d9b3174550cf6d993ca43638 +0 -0
  253. package/skills/cm-content-factory/.git 2/objects/c7/710d59b5a35b0f1f0a0399386643a0bd94c929 +0 -0
  254. package/skills/cm-content-factory/.git 2/objects/d1/fe58237112e953e5fec52da22cf38e08be3df9 +0 -5
  255. package/skills/cm-content-factory/.git 2/objects/d2/2bbe9fd2f74c95bc5583e803f5e435f1e2cd86 +0 -0
  256. package/skills/cm-content-factory/.git 2/objects/d7/e72852ea2bff74581dbf247d400120086229f4 +0 -0
  257. package/skills/cm-content-factory/.git 2/objects/d8/d4c3b5553e4fd72807e1d4b49ef07d9ef3ac35 +0 -0
  258. package/skills/cm-content-factory/.git 2/objects/dc/75050c2876f6a02ae2a53a3c886f395b622977 +0 -0
  259. package/skills/cm-content-factory/.git 2/objects/ee/e8546f95acec500187c08a28a8b9ee02db0dec +0 -0
  260. package/skills/cm-content-factory/.git 2/objects/ef/263c059208b416c2146434f10cb2b9fabcba16 +0 -0
  261. package/skills/cm-content-factory/.git 2/objects/f3/ae597e84d9a59b88acd21c99bde2eaf686d785 +0 -0
  262. package/skills/cm-content-factory/.git 2/objects/f3/f6f5673c821d3d8e76fa267a9e882e7a5387ea +0 -0
  263. package/skills/cm-content-factory/.git 2/objects/f9/6e6d0ad02624dd11d5848594d056caef7a5e8b +0 -0
  264. package/skills/cm-content-factory/.git 2/objects/ff/278988fc1edf0db3abcf18de795f4cc0b4f3e1 +0 -0
  265. package/skills/cm-content-factory/.git 2/refs/heads/main +0 -1
  266. package/skills/cm-content-factory/.git 2/refs/remotes/origin/main +0 -1
  267. package/skills/cm-content-factory/.pytest_cache 2/v/cache/nodeids +0 -76
  268. package/skills/cm-content-factory/.pytest_cache 2/v/cache/stepwise +0 -1
  269. package/skills/cm-content-factory/AGENTS.md +0 -61
  270. package/skills/cm-content-factory/CLAUDE.md +0 -63
  271. package/skills/cm-content-factory/CURSOR.md +0 -43
  272. package/skills/cm-content-factory/Content Factory.zip +0 -0
  273. package/skills/cm-content-factory/SKILL.md +0 -416
  274. package/skills/cm-content-factory/cf +0 -313
  275. package/skills/cm-content-factory/config.schema.json +0 -397
  276. package/skills/cm-content-factory/dashboard/app.js +0 -556
  277. package/skills/cm-content-factory/dashboard/index.html +0 -397
  278. package/skills/cm-content-factory/dashboard/style.css +0 -1211
  279. package/skills/cm-content-factory/examples/01-real-estate.config.json +0 -146
  280. package/skills/cm-content-factory/examples/02-personal-finance.config.json +0 -146
  281. package/skills/cm-content-factory/examples/03-health-wellness.config.json +0 -147
  282. package/skills/cm-content-factory/examples/04-saas-software.config.json +0 -147
  283. package/skills/cm-content-factory/examples/05-legal-services.config.json +0 -147
  284. package/skills/cm-content-factory/examples/06-insurance.config.json +0 -146
  285. package/skills/cm-content-factory/examples/07-ecommerce-dropship.config.json +0 -146
  286. package/skills/cm-content-factory/examples/08-online-education.config.json +0 -147
  287. package/skills/cm-content-factory/examples/09-crypto-defi.config.json +0 -147
  288. package/skills/cm-content-factory/examples/10-beauty-skincare.config.json +0 -147
  289. package/skills/cm-content-factory/examples/11-home-services.config.json +0 -146
  290. package/skills/cm-content-factory/examples/12-dental-clinic.config.json +0 -147
  291. package/skills/cm-content-factory/examples/13-pet-care.config.json +0 -147
  292. package/skills/cm-content-factory/examples/14-travel-hospitality.config.json +0 -147
  293. package/skills/cm-content-factory/examples/15-ai-automation.config.json +0 -147
  294. package/skills/cm-content-factory/examples/16-wedding-events.config.json +0 -147
  295. package/skills/cm-content-factory/examples/17-fitness-coaching.config.json +0 -148
  296. package/skills/cm-content-factory/examples/18-cybersecurity.config.json +0 -147
  297. package/skills/cm-content-factory/examples/19-food-restaurant.config.json +0 -148
  298. package/skills/cm-content-factory/examples/20-solar-energy.config.json +0 -147
  299. package/skills/cm-content-factory/examples/fitness-blog.config.json +0 -116
  300. package/skills/cm-content-factory/examples/tech-blog.config.json +0 -107
  301. package/skills/cm-content-factory/extensions/EXTENSION_GUIDE.md +0 -72
  302. package/skills/cm-content-factory/extensions/hooks.py +0 -126
  303. package/skills/cm-content-factory/extensions/openclaw_adapter.py +0 -132
  304. package/skills/cm-content-factory/landing/docs/content/changelog.md +0 -36
  305. package/skills/cm-content-factory/landing/docs/content/deployment.md +0 -46
  306. package/skills/cm-content-factory/landing/docs/content/execution-flow.md +0 -67
  307. package/skills/cm-content-factory/landing/docs/content/openspace.md +0 -27
  308. package/skills/cm-content-factory/landing/docs/content/openviking.md +0 -33
  309. package/skills/cm-content-factory/landing/docs/content/use-cases.md +0 -26
  310. package/skills/cm-content-factory/landing/docs/content/v5-intro.md +0 -28
  311. package/skills/cm-content-factory/landing/docs/index.html +0 -240
  312. package/skills/cm-content-factory/landing/index.html +0 -680
  313. package/skills/cm-content-factory/landing/script.js +0 -143
  314. package/skills/cm-content-factory/landing/style.css +0 -1216
  315. package/skills/cm-content-factory/landing/translations.js +0 -508
  316. package/skills/cm-content-factory/logs/events.jsonl +0 -11
  317. package/skills/cm-content-factory/profiles/_template.profile.json +0 -231
  318. package/skills/cm-content-factory/profiles/finance.profile.json +0 -278
  319. package/skills/cm-content-factory/profiles/legal.profile.json +0 -263
  320. package/skills/cm-content-factory/profiles/medical-research.profile.json +0 -321
  321. package/skills/cm-content-factory/profiles/technology.profile.json +0 -275
  322. package/skills/cm-content-factory/scripts/agent_dispatcher.py +0 -266
  323. package/skills/cm-content-factory/scripts/audit.py +0 -106
  324. package/skills/cm-content-factory/scripts/dashboard_server.py +0 -225
  325. package/skills/cm-content-factory/scripts/deploy.py +0 -146
  326. package/skills/cm-content-factory/scripts/extract.py +0 -132
  327. package/skills/cm-content-factory/scripts/landing_generator.py +0 -459
  328. package/skills/cm-content-factory/scripts/memory.py +0 -521
  329. package/skills/cm-content-factory/scripts/monetize.py +0 -239
  330. package/skills/cm-content-factory/scripts/pipeline.py +0 -357
  331. package/skills/cm-content-factory/scripts/plan.py +0 -163
  332. package/skills/cm-content-factory/scripts/publish.py +0 -145
  333. package/skills/cm-content-factory/scripts/research.py +0 -337
  334. package/skills/cm-content-factory/scripts/scaffold.py +0 -464
  335. package/skills/cm-content-factory/scripts/scoreboard.py +0 -336
  336. package/skills/cm-content-factory/scripts/seo.py +0 -90
  337. package/skills/cm-content-factory/scripts/state_manager.py +0 -320
  338. package/skills/cm-content-factory/scripts/token_manager.py +0 -268
  339. package/skills/cm-content-factory/scripts/validate.py +0 -221
  340. package/skills/cm-content-factory/scripts/wizard.py +0 -329
  341. package/skills/cm-content-factory/scripts/write.py +0 -93
  342. package/skills/cm-content-factory/sites/docs-site/src/assets/houston.webp +0 -0
  343. package/skills/cm-content-factory/sites/docs-site/src/content/docs/architecture.md +0 -90
  344. package/skills/cm-content-factory/sites/docs-site/src/content/docs/data-flow.md +0 -54
  345. package/skills/cm-content-factory/sites/docs-site/src/content/docs/deployment.md +0 -38
  346. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/index.md +0 -65
  347. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/lc-content-lifecycle.md +0 -48
  348. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/seq-write-mode.md +0 -39
  349. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/uj-first-batch.md +0 -42
  350. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/wf-content-pipeline.md +0 -51
  351. package/skills/cm-content-factory/sites/docs-site/src/content/docs/flows/wf-learning-cycle.md +0 -52
  352. package/skills/cm-content-factory/sites/docs-site/src/content/docs/getting-started/configuration.md +0 -86
  353. package/skills/cm-content-factory/sites/docs-site/src/content/docs/getting-started/installation.md +0 -80
  354. package/skills/cm-content-factory/sites/docs-site/src/content/docs/getting-started/intro.md +0 -58
  355. package/skills/cm-content-factory/sites/docs-site/src/content/docs/index.md +0 -102
  356. package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/index.md +0 -45
  357. package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/optimize-seo.md +0 -29
  358. package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/scale-content-production.md +0 -55
  359. package/skills/cm-content-factory/sites/docs-site/src/content/docs/jtbd/standardize-quality.md +0 -29
  360. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/buyer-cmo-huong.md +0 -41
  361. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/buyer-content-lead-khoa.md +0 -40
  362. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/index.md +0 -56
  363. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/user-content-manager-lan.md +0 -46
  364. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/user-seo-minh.md +0 -45
  365. package/skills/cm-content-factory/sites/docs-site/src/content/docs/personas/user-writer-tu.md +0 -45
  366. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/content-pipeline.md +0 -108
  367. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/index.md +0 -22
  368. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/memory-system.md +0 -52
  369. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/seo-optimization.md +0 -58
  370. package/skills/cm-content-factory/sites/docs-site/src/content/docs/sop/troubleshooting-guide.md +0 -92
  371. package/skills/cm-content-factory/sites/docs-site/src/styles/custom.css +0 -575
  372. package/skills/cm-content-factory/tests/conftest.py +0 -66
  373. package/skills/cm-content-factory/tests/test_agent_dispatcher.py +0 -125
  374. package/skills/cm-content-factory/tests/test_memory.py +0 -128
  375. package/skills/cm-content-factory/tests/test_pipeline.py +0 -107
  376. package/skills/cm-content-factory/tests/test_research.py +0 -56
  377. package/skills/cm-content-factory/tests/test_state_manager.py +0 -131
  378. package/skills/cm-content-factory/tests/test_token_manager.py +0 -110
  379. package/skills/cm-content-factory/tests/test_wizard.py +0 -121
  380. package/skills/cm-cro-methodology/SKILL.md +0 -290
  381. package/skills/cm-cro-methodology/references/COPYWRITING.md +0 -178
  382. package/skills/cm-cro-methodology/references/OBJECTIONS.md +0 -135
  383. package/skills/cm-cro-methodology/references/PERSUASION.md +0 -158
  384. package/skills/cm-cro-methodology/references/RESEARCH.md +0 -220
  385. package/skills/cm-cro-methodology/references/funnel-analysis.md +0 -365
  386. package/skills/cm-cro-methodology/references/testing-methodology.md +0 -330
  387. package/skills/cm-google-form/SKILL.md +0 -266
  388. package/skills/cm-google-form/templates/apps-script.js +0 -55
  389. package/skills/cm-google-form/templates/form-markup.html +0 -110
  390. package/skills/cm-google-form/templates/form-submit.js +0 -201
  391. package/skills/cm-google-form/templates/toast.css +0 -152
  392. package/skills/cm-growth-hacking/SKILL.md +0 -282
  393. package/skills/cm-growth-hacking/bottom-sheet-engine.md +0 -261
  394. package/skills/cm-growth-hacking/calendar-integration.md +0 -264
  395. package/skills/cm-growth-hacking/references/engagement-patterns.md +0 -346
  396. package/skills/cm-growth-hacking/templates/bottom-sheet.css +0 -528
  397. package/skills/cm-growth-hacking/templates/bottom-sheet.js +0 -269
  398. package/skills/cm-growth-hacking/templates/calendar-cta.js +0 -213
  399. package/skills/cm-growth-hacking/templates/tracking-events.js +0 -211
  400. package/skills/cm-growth-hacking/templates/trigger-manager.js +0 -254
  401. package/skills/cm-growth-hacking/tracking-events.md +0 -246
  402. package/skills/cm-growth-hacking/trigger-system.md +0 -342
  403. package/skills/cm-jtbd/SKILL.md +0 -98
  404. package/skills/cm-notebooklm/SKILL.md +0 -156
  405. package/skills/cm-notebooklm/references/command_reference.md +0 -94
  406. package/skills/cm-notebooklm/references/workflows.md +0 -60
  407. package/skills/cm-notebooklm/resources/knowledge_sources.md +0 -106
  408. package/skills/cm-notebooklm/scripts/brain-sync.sh +0 -453
  409. package/skills/cm-notebooklm/scripts/graduate_wisdom.py +0 -101
  410. package/skills/cm-readit/SKILL.md +0 -289
  411. package/skills/cm-readit/audio-player.md +0 -206
  412. package/skills/cm-readit/examples/blog-reader.js +0 -352
  413. package/skills/cm-readit/examples/voice-cro.js +0 -390
  414. package/skills/cm-readit/tts-engine.md +0 -262
  415. package/skills/cm-readit/ui-patterns.md +0 -362
  416. package/skills/cm-readit/voice-cro.md +0 -223
@@ -1,390 +0,0 @@
1
- /**
2
- * Voice CRO System — AI Sales Consultant (1-chiều)
3
- * Trigger-based bottom sheet + pre-generated MP3 + TTS fallback
4
- * Config-driven, per-page voice scripts & triggers
5
- */
6
-
7
- (function () {
8
- 'use strict';
9
-
10
- /* ─── Page Config ─── */
11
- const CONFIG = {
12
- '/': {
13
- delay: 20000, scroll: 0.30,
14
- audio: ['/audio/homepage-intro.mp3', '/audio/homepage-full.mp3'],
15
- sheetIcon: '🎧',
16
- sheetText: 'Bạn muốn nghe giới thiệu nhanh về dịch vụ?',
17
- ctaText: 'Đặt Lịch Ngay', ctaHref: '#uu-dai'
18
- },
19
- '/index.html': null, // alias → use '/'
20
- '/tri-lieu-co-vai-gay.html': {
21
- delay: 15000, scroll: 0.25,
22
- audio: ['/audio/vai-gay-intro.mp3', '/audio/vai-gay-full.mp3'],
23
- sheetIcon: '💆',
24
- sheetText: 'Đau vai gáy lâu rồi? Nghe tư vấn nhanh từ chuyên gia',
25
- ctaText: 'Đặt Lịch Ngay', ctaHref: '#dat-lich'
26
- },
27
- '/tri-mat-ngu.html': {
28
- delay: 18000, scroll: 0.30,
29
- audio: ['/audio/mat-ngu-intro.mp3', '/audio/mat-ngu-full.mp3'],
30
- sheetIcon: '🌙',
31
- sheetText: 'Mất ngủ kiệt sức? Dưỡng sinh tạng giúp lấy lại giấc ngủ sâu',
32
- ctaText: 'Đặt Lịch Dưỡng Sinh', ctaHref: '#dat-lich',
33
- checkboxTrigger: { selector: '.check-item input[type=checkbox]', minChecked: 2 }
34
- },
35
- '/tri-lieu-dau-lung-xuong-khop.html': {
36
- delay: 20000, scroll: 0.35,
37
- audio: ['/audio/dau-lung-intro.mp3', '/audio/dau-lung-full.mp3'],
38
- sheetIcon: '🦴',
39
- sheetText: 'Tìm hiểu cho bố mẹ? Nghe tư vấn an toàn từ chuyên gia',
40
- ctaText: 'Tư Vấn Miễn Phí', ctaHref: '#dat-lich'
41
- },
42
- '/phuc-hoi-the-thao.html': {
43
- delay: 12000, scroll: 0.20,
44
- audio: ['/audio/the-thao-intro.mp3', '/audio/the-thao-full.mp3'],
45
- sheetIcon: '💪',
46
- sheetText: 'DOMS kéo dài? Massage Đông y giúp recovery thật sự',
47
- ctaText: 'Đặt Lịch Massage', ctaHref: '#dat-lich'
48
- },
49
- '/cham-soc-me-sau-sinh.html': {
50
- delay: 18000, scroll: 0.25,
51
- audio: ['/audio/me-sau-sinh-intro.mp3', '/audio/me-sau-sinh-full.mp3'],
52
- sheetIcon: '💝',
53
- sheetText: 'Mẹ ơi, mẹ xứng đáng được chăm sóc',
54
- ctaText: 'Đặt Lịch Cho Mẹ', ctaHref: '#dat-lich'
55
- },
56
- '/thu-gian-premium.html': {
57
- delay: 22000, scroll: 0.35,
58
- audio: ['/audio/premium-intro.mp3', '/audio/premium-full.mp3'],
59
- sheetIcon: '✦',
60
- sheetText: 'Tìm kiếm trải nghiệm dưỡng sinh toàn diện?',
61
- ctaText: 'Đặt Lịch Dưỡng Sinh', ctaHref: '#dat-lich'
62
- },
63
- '/khoa-hoc-bam-huyet.html': {
64
- delay: 25000, scroll: 0.40,
65
- audio: ['/audio/khoa-hoc-intro.mp3', '/audio/khoa-hoc-full.mp3'],
66
- sheetIcon: '📚',
67
- sheetText: 'Muốn chuyển nghề? Nghe chia sẻ từ học viên',
68
- ctaText: 'Đăng Ký Tư Vấn', ctaHref: '#dat-lich'
69
- }
70
- };
71
-
72
- /* ─── State ─── */
73
- const STATE_KEY = 'voiceCroDismissed';
74
- const STATS_KEY = 'voiceCroStats';
75
- let currentAudio = null;
76
- let sheetEl = null;
77
- let playerEl = null;
78
- let isPlaying = false;
79
- let playingPart = 0; // 0=none, 1=intro, 2=full
80
-
81
- /* ─── Helpers ─── */
82
- function getPageConfig() {
83
- const path = window.location.pathname;
84
- // Try exact match, then alias, then root
85
- let cfg = CONFIG[path];
86
- if (cfg === null && CONFIG['/']) cfg = CONFIG['/']; // alias
87
- if (!cfg && path === '/') cfg = CONFIG['/'];
88
- return cfg || null;
89
- }
90
-
91
- function isDismissed() {
92
- try { return sessionStorage.getItem(STATE_KEY) === '1'; } catch { return false; }
93
- }
94
-
95
- function setDismissed() {
96
- try { sessionStorage.setItem(STATE_KEY, '1'); } catch { /* noop */ }
97
- }
98
-
99
- function trackStat(event) {
100
- try {
101
- const stats = JSON.parse(localStorage.getItem(STATS_KEY) || '{}');
102
- const page = window.location.pathname;
103
- if (!stats[page]) stats[page] = {};
104
- stats[page][event] = (stats[page][event] || 0) + 1;
105
- stats[page].lastSeen = Date.now();
106
- localStorage.setItem(STATS_KEY, JSON.stringify(stats));
107
- } catch { /* noop */ }
108
- }
109
-
110
- function getScrollPercent() {
111
- const h = document.documentElement.scrollHeight - window.innerHeight;
112
- return h > 0 ? window.scrollY / h : 0;
113
- }
114
-
115
- /* ─── Audio Playback ─── */
116
- function playAudio(src, onEnd) {
117
- stopAudio();
118
- currentAudio = new Audio(src);
119
- currentAudio.volume = 0.85;
120
- currentAudio.addEventListener('ended', () => {
121
- isPlaying = false;
122
- if (onEnd) onEnd();
123
- updatePlayerUI();
124
- });
125
- currentAudio.addEventListener('error', () => {
126
- isPlaying = false;
127
- if (onEnd) onEnd();
128
- updatePlayerUI();
129
- });
130
- const playPromise = currentAudio.play();
131
- if (playPromise) {
132
- playPromise.then(() => {
133
- isPlaying = true;
134
- updatePlayerUI();
135
- }).catch(() => {
136
- isPlaying = false;
137
- updatePlayerUI();
138
- });
139
- }
140
- }
141
-
142
- function stopAudio() {
143
- if (currentAudio) {
144
- currentAudio.pause();
145
- currentAudio.currentTime = 0;
146
- currentAudio = null;
147
- }
148
- isPlaying = false;
149
- }
150
-
151
- function togglePause() {
152
- if (!currentAudio) return;
153
- if (currentAudio.paused) {
154
- currentAudio.play();
155
- isPlaying = true;
156
- } else {
157
- currentAudio.pause();
158
- isPlaying = false;
159
- }
160
- updatePlayerUI();
161
- }
162
-
163
- /* ─── Bottom Sheet ─── */
164
- function createSheet(cfg) {
165
- const el = document.createElement('div');
166
- el.className = 'vcro-sheet';
167
- el.innerHTML = `
168
- <div class="vcro-sheet-inner">
169
- <button class="vcro-sheet-close" aria-label="Đóng">✕</button>
170
- <div class="vcro-sheet-icon">${cfg.sheetIcon}</div>
171
- <p class="vcro-sheet-text">${cfg.sheetText}</p>
172
- <div class="vcro-sheet-actions">
173
- <button class="vcro-sheet-listen btn btn-primary btn-sm">🔊 Nghe Ngay</button>
174
- <button class="vcro-sheet-dismiss btn btn-outline btn-sm">Để Sau</button>
175
- </div>
176
- </div>
177
- `;
178
-
179
- // Close
180
- el.querySelector('.vcro-sheet-close').addEventListener('click', () => dismiss(el));
181
- el.querySelector('.vcro-sheet-dismiss').addEventListener('click', () => dismiss(el));
182
-
183
- // Listen
184
- el.querySelector('.vcro-sheet-listen').addEventListener('click', () => {
185
- trackStat('listen');
186
- el.classList.remove('active');
187
- setTimeout(() => el.remove(), 400);
188
- playIntro(cfg);
189
- });
190
-
191
- // Swipe down to dismiss (mobile)
192
- let startY = 0;
193
- el.addEventListener('touchstart', (e) => { startY = e.touches[0].clientY; }, { passive: true });
194
- el.addEventListener('touchmove', (e) => {
195
- const dy = e.touches[0].clientY - startY;
196
- if (dy > 60) dismiss(el);
197
- }, { passive: true });
198
-
199
- document.body.appendChild(el);
200
- // Animate in
201
- requestAnimationFrame(() => requestAnimationFrame(() => el.classList.add('active')));
202
- sheetEl = el;
203
- trackStat('shown');
204
- }
205
-
206
- function dismiss(el) {
207
- setDismissed();
208
- trackStat('dismissed');
209
- el.classList.remove('active');
210
- setTimeout(() => el.remove(), 400);
211
- }
212
-
213
- /* ─── Mini Player Bar ─── */
214
- function createPlayer(cfg) {
215
- const el = document.createElement('div');
216
- el.className = 'vcro-player';
217
- el.innerHTML = `
218
- <div class="vcro-player-inner">
219
- <button class="vcro-player-toggle" aria-label="Play/Pause">
220
- <span class="vcro-player-icon">⏸</span>
221
- </button>
222
- <div class="vcro-player-info">
223
- <span class="vcro-player-label">Đang phát...</span>
224
- <div class="vcro-player-progress"><div class="vcro-player-bar"></div></div>
225
- </div>
226
- <button class="vcro-player-next btn btn-primary btn-sm" style="display:none;">Nghe Tiếp</button>
227
- <a href="${cfg.ctaHref}" class="vcro-player-cta btn btn-primary btn-sm">${cfg.ctaText}</a>
228
- <button class="vcro-player-close" aria-label="Đóng">✕</button>
229
- </div>
230
- `;
231
-
232
- el.querySelector('.vcro-player-toggle').addEventListener('click', togglePause);
233
- el.querySelector('.vcro-player-close').addEventListener('click', () => {
234
- stopAudio();
235
- el.classList.remove('active');
236
- setTimeout(() => el.remove(), 400);
237
- playerEl = null;
238
- });
239
-
240
- // Next button → play full audio
241
- el.querySelector('.vcro-player-next').addEventListener('click', () => {
242
- trackStat('listenFull');
243
- playingPart = 2;
244
- el.querySelector('.vcro-player-next').style.display = 'none';
245
- el.querySelector('.vcro-player-label').textContent = 'Nghe tư vấn chi tiết...';
246
- playAudio(cfg.audio[1], () => {
247
- el.querySelector('.vcro-player-label').textContent = 'Cảm ơn bạn đã lắng nghe!';
248
- el.querySelector('.vcro-player-icon').textContent = '✓';
249
- el.querySelector('.vcro-player-cta').style.display = 'inline-flex';
250
- });
251
- });
252
-
253
- document.body.appendChild(el);
254
- requestAnimationFrame(() => requestAnimationFrame(() => el.classList.add('active')));
255
- playerEl = el;
256
-
257
- // Progress tracking
258
- startProgressTracking();
259
- }
260
-
261
- function updatePlayerUI() {
262
- if (!playerEl) return;
263
- const icon = playerEl.querySelector('.vcro-player-icon');
264
- if (isPlaying) {
265
- icon.textContent = '⏸';
266
- } else if (currentAudio && !currentAudio.ended) {
267
- icon.textContent = '▶';
268
- }
269
- }
270
-
271
- function startProgressTracking() {
272
- const update = () => {
273
- if (!currentAudio || !playerEl) return;
274
- const bar = playerEl.querySelector('.vcro-player-bar');
275
- if (bar && currentAudio.duration) {
276
- bar.style.width = (currentAudio.currentTime / currentAudio.duration * 100) + '%';
277
- }
278
- if (isPlaying) requestAnimationFrame(update);
279
- };
280
- requestAnimationFrame(update);
281
- // Also re-trigger on play
282
- if (currentAudio) {
283
- currentAudio.addEventListener('play', () => requestAnimationFrame(update));
284
- }
285
- }
286
-
287
- /* ─── Play Flow ─── */
288
- function playIntro(cfg) {
289
- playingPart = 1;
290
- createPlayer(cfg);
291
- playerEl.querySelector('.vcro-player-label').textContent = 'Nghe giới thiệu...';
292
- playerEl.querySelector('.vcro-player-cta').style.display = 'none';
293
-
294
- playAudio(cfg.audio[0], () => {
295
- // Intro done → show "Nghe Tiếp" button
296
- if (playerEl) {
297
- playerEl.querySelector('.vcro-player-label').textContent = 'Bạn muốn nghe thêm?';
298
- playerEl.querySelector('.vcro-player-next').style.display = 'inline-flex';
299
- playerEl.querySelector('.vcro-player-icon').textContent = '✓';
300
- playerEl.querySelector('.vcro-player-cta').style.display = 'inline-flex';
301
- }
302
- });
303
- }
304
-
305
- /* ─── Trigger Logic ─── */
306
- function initTrigger(cfg) {
307
- if (isDismissed()) return;
308
- if (window.matchMedia('(prefers-reduced-motion: reduce)').matches) return;
309
-
310
- let timeReady = false;
311
- let scrollReady = false;
312
- let checkboxReady = false;
313
- let triggered = false;
314
-
315
- const tryTrigger = () => {
316
- if (triggered) return;
317
- // Normal path: time + scroll
318
- if (timeReady && scrollReady) {
319
- triggered = true;
320
- createSheet(cfg);
321
- return;
322
- }
323
- // Checkbox shortcut (mất ngủ page)
324
- if (checkboxReady) {
325
- triggered = true;
326
- createSheet(cfg);
327
- }
328
- };
329
-
330
- // Time trigger
331
- setTimeout(() => {
332
- timeReady = true;
333
- tryTrigger();
334
- }, cfg.delay);
335
-
336
- // Scroll trigger
337
- const onScroll = () => {
338
- if (getScrollPercent() >= cfg.scroll) {
339
- scrollReady = true;
340
- window.removeEventListener('scroll', onScroll);
341
- tryTrigger();
342
- }
343
- };
344
- window.addEventListener('scroll', onScroll, { passive: true });
345
- // Check immediately in case page is already scrolled
346
- onScroll();
347
-
348
- // Checkbox trigger (if configured)
349
- if (cfg.checkboxTrigger) {
350
- const checkboxes = document.querySelectorAll(cfg.checkboxTrigger.selector);
351
- if (checkboxes.length) {
352
- checkboxes.forEach(cb => {
353
- cb.addEventListener('change', () => {
354
- const checked = document.querySelectorAll(cfg.checkboxTrigger.selector + ':checked').length;
355
- if (checked >= cfg.checkboxTrigger.minChecked) {
356
- checkboxReady = true;
357
- tryTrigger();
358
- }
359
- });
360
- });
361
- }
362
- }
363
- }
364
-
365
- /* ─── Init ─── */
366
- function init() {
367
- const cfg = getPageConfig();
368
- if (!cfg) return;
369
-
370
- // Verify audio files exist (preload)
371
- if (cfg.audio && cfg.audio.length) {
372
- const preload = new Audio();
373
- preload.preload = 'none'; // Just check existence later
374
- preload.src = cfg.audio[0];
375
- }
376
-
377
- // Don't compete with blog reader
378
- if (document.querySelector('.reader-player-bar.active')) return;
379
-
380
- initTrigger(cfg);
381
- }
382
-
383
- // Wait for DOM
384
- if (document.readyState === 'loading') {
385
- document.addEventListener('DOMContentLoaded', init);
386
- } else {
387
- init();
388
- }
389
-
390
- })();
@@ -1,262 +0,0 @@
1
- # TTS Engine — SpeechSynthesis API Reference
2
-
3
- > The browser's built-in text-to-speech engine. Zero dependencies, works offline on most devices.
4
-
5
- ---
6
-
7
- ## API Quick Reference
8
-
9
- ```javascript
10
- const synth = window.speechSynthesis;
11
-
12
- // Core methods
13
- synth.speak(utterance) // Start speaking
14
- synth.pause() // Pause
15
- synth.resume() // Resume
16
- synth.cancel() // Stop all (⚠️ triggers onerror!)
17
- synth.getVoices() // Returns SpeechSynthesisVoice[]
18
-
19
- // Properties
20
- synth.speaking // boolean — currently speaking?
21
- synth.paused // boolean — currently paused?
22
- synth.pending // boolean — utterances in queue?
23
-
24
- // Events
25
- synth.onvoiceschanged // Fires when voice list changes
26
- ```
27
-
28
- ### SpeechSynthesisUtterance
29
-
30
- ```javascript
31
- const u = new SpeechSynthesisUtterance(text);
32
- u.lang = 'vi-VN'; // BCP-47 language tag
33
- u.voice = voiceObject; // SpeechSynthesisVoice
34
- u.rate = 1.0; // 0.1 — 10 (normal = 1.0)
35
- u.pitch = 1.0; // 0 — 2 (normal = 1.0)
36
- u.volume = 1.0; // 0 — 1
37
-
38
- // Events
39
- u.onstart = fn; // Speaking started
40
- u.onend = fn; // Finished speaking this utterance
41
- u.onerror = fn(event); // Error occurred (event.error = string)
42
- u.onpause = fn; // Paused
43
- u.onresume = fn; // Resumed
44
- u.onboundary = fn(event); // Word/sentence boundary (event.charIndex)
45
- ```
46
-
47
- ---
48
-
49
- ## Content Extraction Pattern
50
-
51
- ```javascript
52
- function extractCoreText(containerSelector) {
53
- const container = document.querySelector(containerSelector);
54
- if (!container || !container.textContent.trim()) return '';
55
-
56
- const clone = container.cloneNode(true);
57
-
58
- // STRIP non-content elements
59
- clone.querySelectorAll([
60
- // Navigation & structure
61
- 'nav, footer, header, aside',
62
- // Media (cannot be spoken)
63
- 'img, video, audio, iframe, svg, figure, canvas',
64
- // Scripts & styles
65
- 'script, style, noscript',
66
- // Hidden elements
67
- '[aria-hidden="true"], [hidden], .sr-only',
68
- // Site-specific noise (customize per project)
69
- '.cta-box, .tags, .related-posts, .comments',
70
- '.advertisement, .promo, .popup'
71
- ].join(', ')).forEach(el => el.remove());
72
-
73
- return clone.innerText
74
- .replace(/\n{3,}/g, '\n\n') // collapse excess newlines
75
- .replace(/\s{2,}/g, ' ') // collapse excess spaces
76
- .trim();
77
- }
78
- ```
79
-
80
- **Customization:** The strip selector should be adapted per project. Common additions:
81
- - Blog: `.blog-cta-box, .blog-tags, .blog-related`
82
- - E-commerce: `.add-to-cart, .price-compare, .reviews-count`
83
- - Docs: `.sidebar-nav, .edit-on-github, .breadcrumb`
84
-
85
- ---
86
-
87
- ## Chunking Strategy
88
-
89
- ### Why Chunk?
90
-
91
- Browsers have hard limits on utterance text length:
92
- - **Chrome**: ~5000 chars (silently fails beyond)
93
- - **Safari**: ~4000 chars
94
- - **Firefox**: ~3000 chars
95
- - **Safe maximum**: **2500 chars**
96
-
97
- ### Sentence-Aware Splitting
98
-
99
- ```javascript
100
- function splitIntoChunks(text, maxLen = 2500) {
101
- const chunks = [];
102
- const sentences = text.split(/(?<=[.!?。?!\n])\s+/);
103
- let current = '';
104
-
105
- for (const sentence of sentences) {
106
- if ((current + ' ' + sentence).length > maxLen && current) {
107
- chunks.push(current.trim());
108
- current = sentence;
109
- } else {
110
- current += (current ? ' ' : '') + sentence;
111
- }
112
- }
113
- if (current.trim()) chunks.push(current.trim());
114
- return chunks;
115
- }
116
- ```
117
-
118
- ### Chunk Chaining
119
-
120
- ```javascript
121
- function buildUtterances(chunks, voice, rate = 1.0) {
122
- return chunks.map((chunk, i) => {
123
- const u = new SpeechSynthesisUtterance(chunk);
124
- u.lang = 'vi-VN'; // set per project
125
- u.rate = rate;
126
- u.pitch = 1.0;
127
- if (voice) u.voice = voice;
128
-
129
- u.onend = () => {
130
- if (i < chunks.length - 1) {
131
- currentIdx = i + 1;
132
- synth.speak(utterances[currentIdx]);
133
- updateProgress();
134
- } else {
135
- stopReading(); // last chunk done
136
- }
137
- };
138
-
139
- u.onerror = (e) => {
140
- // synth.cancel() fires onerror with 'canceled'
141
- if (e.error === 'canceled' || e.error === 'interrupted') return;
142
- stopReading();
143
- };
144
-
145
- return u;
146
- });
147
- }
148
- ```
149
-
150
- ---
151
-
152
- ## Voice Selection
153
-
154
- ### Strategy
155
-
156
- ```javascript
157
- function getVoice(langCode = 'vi-VN') {
158
- const voices = speechSynthesis.getVoices();
159
- const prefix = langCode.split('-')[0]; // 'vi'
160
-
161
- // 1. Exact match + local (fastest, works offline)
162
- const local = voices.find(v => v.lang === langCode && v.localService);
163
- if (local) return local;
164
-
165
- // 2. Exact match (may be network/higher quality)
166
- const exact = voices.find(v => v.lang === langCode);
167
- if (exact) return exact;
168
-
169
- // 3. Language prefix match
170
- const prefix_match = voices.find(v => v.lang.startsWith(prefix));
171
- if (prefix_match) return prefix_match;
172
-
173
- // 4. null = browser default
174
- return null;
175
- }
176
- ```
177
-
178
- ### Voices Loading (Async)
179
-
180
- On some browsers (Chrome), `getVoices()` returns empty until loaded:
181
-
182
- ```javascript
183
- if (synth.getVoices().length === 0) {
184
- synth.addEventListener('voiceschanged', function onVoices() {
185
- synth.removeEventListener('voiceschanged', onVoices);
186
- // Now voices are available
187
- });
188
- }
189
- ```
190
-
191
- ---
192
-
193
- ## Chrome Keep-Alive
194
-
195
- > **The #1 TTS bug:** Chrome stops `SpeechSynthesis` after ~15 seconds of continuous speech. No error is thrown. Audio simply stops.
196
-
197
- ### Fix
198
-
199
- ```javascript
200
- let keepAliveTimer = null;
201
-
202
- function startKeepAlive() {
203
- stopKeepAlive();
204
- keepAliveTimer = setInterval(() => {
205
- if (synth.speaking && !synth.paused) {
206
- synth.pause();
207
- synth.resume();
208
- }
209
- }, 10000); // every 10 seconds
210
- }
211
-
212
- function stopKeepAlive() {
213
- if (keepAliveTimer) {
214
- clearInterval(keepAliveTimer);
215
- keepAliveTimer = null;
216
- }
217
- }
218
- ```
219
-
220
- **Always start keep-alive when speaking, stop when done.**
221
-
222
- ---
223
-
224
- ## Rate & Pitch Guide
225
-
226
- | Rate | Feel | Use Case |
227
- |------|------|----------|
228
- | 0.7 | Very slow | Elderly users, complex medical content |
229
- | 0.85 | Slow | Educational, foreign language |
230
- | 1.0 | Normal | General content |
231
- | 1.2 | Slightly fast | News, light reading |
232
- | 1.5 | Fast | Familiar content, power users |
233
- | 2.0 | Very fast | Scanning, speed reading |
234
-
235
- **Default recommendation:** `1.0` for general, `0.9` for specialized/medical content.
236
-
237
- ---
238
-
239
- ## Browser Support
240
-
241
- | Browser | SpeechSynthesis | Voices | Keep-Alive Needed |
242
- |---------|----------------|--------|-------------------|
243
- | Chrome | ✅ | Network + Local | ✅ Yes (pause/resume) |
244
- | Safari | ✅ | Local only | ❌ No |
245
- | Firefox | ✅ | Local only | ❌ No |
246
- | Edge | ✅ | Network + Local | ✅ Yes (same as Chrome) |
247
- | Mobile Chrome | ✅ | Limited | ✅ Yes |
248
- | Mobile Safari | ✅ | Good vi-VN | ❌ No |
249
-
250
- ---
251
-
252
- ## Error Handling Matrix
253
-
254
- | Error | Cause | Recovery |
255
- |-------|-------|----------|
256
- | `canceled` | `synth.cancel()` called | Ignore (expected) |
257
- | `interrupted` | New utterance replaced current | Ignore (expected) |
258
- | `audio-busy` | Another utterance playing | Wait and retry |
259
- | `network` | Network voice unavailable | Fall back to local voice |
260
- | `synthesis-unavailable` | TTS engine not available | Show "Not supported" message |
261
- | `synthesis-failed` | Voice failed to synthesize | Try different voice or skip chunk |
262
- | `not-allowed` | User hasn't interacted | Require button click first |