codymaster 4.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 (193) hide show
  1. package/CHANGELOG.md +50 -0
  2. package/README.md +285 -0
  3. package/adapters/antigravity.js +15 -0
  4. package/adapters/claude-code.js +17 -0
  5. package/adapters/cursor.js +16 -0
  6. package/commands/bootstrap.md +49 -0
  7. package/commands/build.md +48 -0
  8. package/commands/content.md +48 -0
  9. package/commands/continuity.md +60 -0
  10. package/commands/debug.md +51 -0
  11. package/commands/demo.md +96 -0
  12. package/commands/deploy.md +51 -0
  13. package/commands/plan.md +42 -0
  14. package/commands/review.md +55 -0
  15. package/commands/track.md +46 -0
  16. package/commands/ux.md +46 -0
  17. package/dist/agent-dispatch.js +161 -0
  18. package/dist/chains/builtin.js +85 -0
  19. package/dist/continuity.js +385 -0
  20. package/dist/dashboard.js +926 -0
  21. package/dist/data.js +122 -0
  22. package/dist/index.js +2434 -0
  23. package/dist/judge.js +252 -0
  24. package/dist/parallel-dispatch.js +359 -0
  25. package/dist/parallel-quality.js +172 -0
  26. package/dist/skill-chain.js +258 -0
  27. package/install.sh +513 -0
  28. package/package.json +79 -0
  29. package/skills/.content-factory-state.json +132 -0
  30. package/skills/.git 2/logs/refs/heads/main +1 -0
  31. package/skills/.git 2/logs/refs/remotes/origin/main +1 -0
  32. package/skills/.git 2/objects/02/fb0956734b5f8ba3f918b7defd04a89cfe0076 +0 -0
  33. package/skills/.git 2/objects/08/1e129d75dc6feac6c02037272e6bd1a04e3324 +0 -0
  34. package/skills/.git 2/objects/0c/5393416f3c5e01c9a655a802bff0dd52f76f0a +0 -0
  35. package/skills/.git 2/objects/10/0b9be46978a946a77188f68be725098a122001 +0 -0
  36. package/skills/.git 2/objects/10/cf041167fc9843610eb3d90259ef3396315fdc +0 -0
  37. package/skills/.git 2/objects/12/5e19538dd6e1338ffe74f6c4c165b00435bf48 +0 -0
  38. package/skills/.git 2/objects/16/a9b9d0088d5c1347628b45a2620b479d8ad57c +0 -0
  39. package/skills/.git 2/objects/17/8c2a9ef93c33ae4eec9d58e82321f9229843a1 +0 -0
  40. package/skills/.git 2/objects/25/397ae41d09104d763bdcac2695209d85cdea89 +0 -0
  41. package/skills/.git 2/objects/2f/a836b7947f2d458e1f639788bf4bb0983a3305 +0 -0
  42. package/skills/.git 2/objects/3a/baaaf0a1c0909c0828335791557125fba911e0 +0 -0
  43. package/skills/.git 2/objects/42/2924221b81f5ce3c4e4daac9a64a24f9b01f9a +0 -0
  44. package/skills/.git 2/objects/42/ec0ce707447dc11446a34c9995fb8533801731 +0 -0
  45. package/skills/.git 2/objects/46/e43ce92866d56ce74b1d750db307cfe6154a15 +0 -0
  46. package/skills/.git 2/objects/48/5e41b633c63f55b8277bcc59f44f67681f671a +0 -0
  47. package/skills/.git 2/objects/49/49c596a3a89fa240642acd95dd3258e261eb09 +0 -0
  48. package/skills/.git 2/objects/50/9d42d8412ef8eaf7f7e138476bac2e4d10ce60 +0 -0
  49. package/skills/.git 2/objects/55/0c8c389d981b463ef849aeb792d8be3ccb6ec8 +0 -0
  50. package/skills/.git 2/objects/5d/82d3b18410cdda3ace3677436f0cb599dbe2d2 +0 -0
  51. package/skills/.git 2/objects/60/0617c58e871a38b33bf29e282d132bb3c381ad +0 -0
  52. package/skills/.git 2/objects/6a/8369a99c687b7245c92ffaf0e0f0dab9014504 +0 -0
  53. package/skills/.git 2/objects/79/bea435d40ab531c1aaf6be0432c6a5b7aaed21 +0 -0
  54. package/skills/.git 2/objects/7e/5ebd79251c2f14e4aceb86c74b6b6daae6b500 +0 -0
  55. package/skills/.git 2/objects/81/98a822a60178d6d5023ddb3e222cddf048742e +0 -0
  56. package/skills/.git 2/objects/86/0a0e1943dfe53411d2e499a1f16f46a96ef758 +0 -0
  57. package/skills/.git 2/objects/86/971fb55fdc081fdbae52376f0f13e57a4e9b04 +0 -0
  58. package/skills/.git 2/objects/88/b89dd609a0a03f8d4fe8bfde20d5b8fc1d326d +0 -0
  59. package/skills/.git 2/objects/90/8737edb6b7809e32cc01590b4e08ba42a9d40d +0 -0
  60. package/skills/.git 2/objects/93/d5a8a9a7d4fb7f11491cb596a6880528725118 +0 -0
  61. package/skills/.git 2/objects/98/46a2ab81d0c3b3eb00ef88fc56989aa7e9f316 +0 -0
  62. package/skills/.git 2/objects/9b/d8dd1e49cf274eaf9c555f3ab39dce7af5715e +0 -0
  63. package/skills/.git 2/objects/a1/13329fb0cec96ae78b222d33a24c3b5bc7fa1f +0 -0
  64. package/skills/.git 2/objects/a9/e6effe626e8a3aea3a8fc3364b492191c6e7d0 +0 -0
  65. package/skills/.git 2/objects/ad/6de7e48d9782cca9353d1ff0aa1aab7fe1df85 +0 -0
  66. package/skills/.git 2/objects/af/54ae316f771ff692e299ffcd8bf2f06b413b59 +0 -0
  67. package/skills/.git 2/objects/b0/4cb8b0b00dad633e731c1472161419e738d674 +0 -0
  68. package/skills/.git 2/objects/b3/094abb0b9ed46419b269e4a4e36a459690e3b0 +0 -0
  69. package/skills/.git 2/objects/b9/435c5d4baac2cfc5c83009ddd27b46b60db5f1 +0 -0
  70. package/skills/.git 2/objects/ba/5da17dbaec5ec2dcfdfd126aead518d1171d5c +0 -0
  71. package/skills/.git 2/objects/c0/bf58703aa258ba5dd63083bebaec8f223d844c +0 -0
  72. package/skills/.git 2/objects/c4/701a34edf1fc1bad58ccc57bd03f9426acb59a +0 -0
  73. package/skills/.git 2/objects/c7/5ccce9a4e5cc74d9b3174550cf6d993ca43638 +0 -0
  74. package/skills/.git 2/objects/c7/710d59b5a35b0f1f0a0399386643a0bd94c929 +0 -0
  75. package/skills/.git 2/objects/d1/fe58237112e953e5fec52da22cf38e08be3df9 +5 -0
  76. package/skills/.git 2/objects/d2/2bbe9fd2f74c95bc5583e803f5e435f1e2cd86 +0 -0
  77. package/skills/.git 2/objects/d7/e72852ea2bff74581dbf247d400120086229f4 +0 -0
  78. package/skills/.git 2/objects/d8/d4c3b5553e4fd72807e1d4b49ef07d9ef3ac35 +0 -0
  79. package/skills/.git 2/objects/dc/75050c2876f6a02ae2a53a3c886f395b622977 +0 -0
  80. package/skills/.git 2/objects/ee/e8546f95acec500187c08a28a8b9ee02db0dec +0 -0
  81. package/skills/.git 2/objects/ef/263c059208b416c2146434f10cb2b9fabcba16 +0 -0
  82. package/skills/.git 2/objects/f3/ae597e84d9a59b88acd21c99bde2eaf686d785 +0 -0
  83. package/skills/.git 2/objects/f3/f6f5673c821d3d8e76fa267a9e882e7a5387ea +0 -0
  84. package/skills/.git 2/objects/f9/6e6d0ad02624dd11d5848594d056caef7a5e8b +0 -0
  85. package/skills/.git 2/objects/ff/278988fc1edf0db3abcf18de795f4cc0b4f3e1 +0 -0
  86. package/skills/.git 2/refs/heads/main +1 -0
  87. package/skills/.git 2/refs/remotes/origin/main +1 -0
  88. package/skills/.pytest_cache 2/v/cache/nodeids +76 -0
  89. package/skills/.pytest_cache 2/v/cache/stepwise +1 -0
  90. package/skills/_shared/helpers.md +123 -0
  91. package/skills/_shared/outputs-convention.md +24 -0
  92. package/skills/cm-ads-tracker/SKILL.md +109 -0
  93. package/skills/cm-ads-tracker/evals/evals.json +55 -0
  94. package/skills/cm-ads-tracker/references/gtm-architecture.md +321 -0
  95. package/skills/cm-ads-tracker/references/industry-events.md +294 -0
  96. package/skills/cm-ads-tracker/references/platforms-api.md +238 -0
  97. package/skills/cm-ads-tracker/templates/capi-payload.md +79 -0
  98. package/skills/cm-ads-tracker/templates/datalayer-push.js +104 -0
  99. package/skills/cm-ads-tracker/templates/gtm-variables.js +56 -0
  100. package/skills/cm-brainstorm-idea/SKILL.md +423 -0
  101. package/skills/cm-code-review/SKILL.md +151 -0
  102. package/skills/cm-content-factory/SKILL.md +416 -0
  103. package/skills/cm-continuity/SKILL.md +399 -0
  104. package/skills/cm-dashboard/SKILL.md +533 -0
  105. package/skills/cm-dashboard/ui/app.js +1270 -0
  106. package/skills/cm-dashboard/ui/index.html +206 -0
  107. package/skills/cm-dashboard/ui/style.css +440 -0
  108. package/skills/cm-debugging/SKILL.md +412 -0
  109. package/skills/cm-deep-search/SKILL.md +242 -0
  110. package/skills/cm-design-system/SKILL.md +97 -0
  111. package/skills/cm-design-system/resources/halo-modern.md +40 -0
  112. package/skills/cm-design-system/resources/lunaris-advanced.md +40 -0
  113. package/skills/cm-design-system/resources/nitro-enterprise.md +39 -0
  114. package/skills/cm-design-system/resources/shadcn-default.md +37 -0
  115. package/skills/cm-dockit/README.md +100 -0
  116. package/skills/cm-dockit/SKILL.md +302 -0
  117. package/skills/cm-dockit/index.html +443 -0
  118. package/skills/cm-dockit/package-lock.json +1850 -0
  119. package/skills/cm-dockit/package.json +14 -0
  120. package/skills/cm-dockit/prompts/analysis.md +34 -0
  121. package/skills/cm-dockit/prompts/api-reference.md +24 -0
  122. package/skills/cm-dockit/prompts/architecture.md +21 -0
  123. package/skills/cm-dockit/prompts/data-flow.md +20 -0
  124. package/skills/cm-dockit/prompts/database.md +21 -0
  125. package/skills/cm-dockit/prompts/deployment.md +22 -0
  126. package/skills/cm-dockit/prompts/flows.md +21 -0
  127. package/skills/cm-dockit/prompts/jtbd.md +20 -0
  128. package/skills/cm-dockit/prompts/personas.md +24 -0
  129. package/skills/cm-dockit/prompts/sop-modules.md +40 -0
  130. package/skills/cm-dockit/scripts/doc-gen.sh +121 -0
  131. package/skills/cm-dockit/scripts/dockit-dashboard.sh +142 -0
  132. package/skills/cm-dockit/scripts/dockit-runner.sh +607 -0
  133. package/skills/cm-dockit/scripts/dockit-task.sh +166 -0
  134. package/skills/cm-dockit/skills/analyze-codebase.md +174 -0
  135. package/skills/cm-dockit/skills/api-reference.md +237 -0
  136. package/skills/cm-dockit/skills/changelog-guide.md +195 -0
  137. package/skills/cm-dockit/skills/content-guidelines.md +190 -0
  138. package/skills/cm-dockit/skills/sop-guide.md +184 -0
  139. package/skills/cm-dockit/skills/tech-docs.md +287 -0
  140. package/skills/cm-dockit/templates/markdown/structure.md +60 -0
  141. package/skills/cm-dockit/templates/vitepress-premium/.vitepress/config.mts +110 -0
  142. package/skills/cm-dockit/templates/vitepress-premium/.vitepress/theme/custom.css +189 -0
  143. package/skills/cm-dockit/templates/vitepress-premium/.vitepress/theme/index.ts +4 -0
  144. package/skills/cm-dockit/templates/vitepress-premium/package.json +19 -0
  145. package/skills/cm-dockit/templates/vitepress-premium/tests/frontend.test.ts +45 -0
  146. package/skills/cm-dockit/tests/runner.test.ts +66 -0
  147. package/skills/cm-dockit/workflows/export-markdown.md +82 -0
  148. package/skills/cm-dockit/workflows/generate-docs.md +68 -0
  149. package/skills/cm-dockit/workflows/setup-vitepress.md +181 -0
  150. package/skills/cm-example/SKILL.md +26 -0
  151. package/skills/cm-execution/SKILL.md +268 -0
  152. package/skills/cm-git-worktrees/SKILL.md +164 -0
  153. package/skills/cm-how-it-work/SKILL.md +189 -0
  154. package/skills/cm-identity-guard/SKILL.md +412 -0
  155. package/skills/cm-jtbd/SKILL.md +98 -0
  156. package/skills/cm-planning/SKILL.md +130 -0
  157. package/skills/cm-project-bootstrap/SKILL.md +161 -0
  158. package/skills/cm-project-bootstrap/templates/AGENTS.md +42 -0
  159. package/skills/cm-project-bootstrap/templates/frontend-safety.test.js +51 -0
  160. package/skills/cm-project-bootstrap/templates/i18n-sync.test.js +38 -0
  161. package/skills/cm-project-bootstrap/templates/pr-template.md +12 -0
  162. package/skills/cm-project-bootstrap/templates/project-identity.json +29 -0
  163. package/skills/cm-project-bootstrap/templates/vitest.config.js +10 -0
  164. package/skills/cm-quality-gate/SKILL.md +218 -0
  165. package/skills/cm-readit/SKILL.md +289 -0
  166. package/skills/cm-readit/audio-player.md +206 -0
  167. package/skills/cm-readit/examples/blog-reader.js +352 -0
  168. package/skills/cm-readit/examples/voice-cro.js +390 -0
  169. package/skills/cm-readit/tts-engine.md +262 -0
  170. package/skills/cm-readit/ui-patterns.md +362 -0
  171. package/skills/cm-readit/voice-cro.md +223 -0
  172. package/skills/cm-safe-deploy/SKILL.md +120 -0
  173. package/skills/cm-safe-deploy/templates/deploy.sh +89 -0
  174. package/skills/cm-safe-i18n/SKILL.md +473 -0
  175. package/skills/cm-secret-shield/SKILL.md +580 -0
  176. package/skills/cm-skill-chain/SKILL.md +78 -0
  177. package/skills/cm-skill-index/SKILL.md +318 -0
  178. package/skills/cm-skill-mastery/SKILL.md +169 -0
  179. package/skills/cm-start/SKILL.md +65 -0
  180. package/skills/cm-status/SKILL.md +12 -0
  181. package/skills/cm-tdd/SKILL.md +370 -0
  182. package/skills/cm-terminal/SKILL.md +177 -0
  183. package/skills/cm-test-gate/SKILL.md +242 -0
  184. package/skills/cm-ui-preview/SKILL.md +291 -0
  185. package/skills/cm-ux-master/DESIGN_STANDARD_TEMPLATE.md +54 -0
  186. package/skills/cm-ux-master/SKILL.md +114 -0
  187. package/skills/cro-methodology/SKILL.md +98 -0
  188. package/skills/cro-methodology/references/COPYWRITING.md +178 -0
  189. package/skills/cro-methodology/references/OBJECTIONS.md +135 -0
  190. package/skills/cro-methodology/references/PERSUASION.md +158 -0
  191. package/skills/cro-methodology/references/RESEARCH.md +220 -0
  192. package/skills/cro-methodology/references/funnel-analysis.md +365 -0
  193. package/skills/cro-methodology/references/testing-methodology.md +330 -0
@@ -0,0 +1,362 @@
1
+ # UI Patterns — Player Bars & Bottom Sheets
2
+
3
+ > CSS and HTML patterns for audio player UI. Responsive, accessible, animated.
4
+
5
+ ---
6
+
7
+ ## Player Bar (Bottom-Fixed)
8
+
9
+ ### HTML Structure
10
+
11
+ ```html
12
+ <div class="reader-bar">
13
+ <div class="reader-bar-inner">
14
+ <div class="reader-bar-info">
15
+ <span class="reader-bar-icon">🔊</span>
16
+ <span class="reader-bar-title">Article Title</span>
17
+ </div>
18
+ <div class="reader-bar-controls">
19
+ <button class="reader-bar-btn reader-btn-toggle" aria-label="Pause">
20
+ <!-- Pause/Play SVG icons -->
21
+ </button>
22
+ <button class="reader-bar-btn reader-btn-close" aria-label="Close">
23
+ <!-- Close SVG icon -->
24
+ </button>
25
+ </div>
26
+ </div>
27
+ <div class="reader-bar-progress">
28
+ <div class="reader-bar-progress-fill"></div>
29
+ </div>
30
+ </div>
31
+ ```
32
+
33
+ ### CSS
34
+
35
+ ```css
36
+ .reader-bar {
37
+ position: fixed;
38
+ bottom: 0;
39
+ left: 0;
40
+ right: 0;
41
+ z-index: 1000;
42
+ transform: translateY(100%);
43
+ transition: transform 0.35s cubic-bezier(0.4, 0, 0.2, 1);
44
+ /* safe-area for notched devices */
45
+ padding-bottom: env(safe-area-inset-bottom, 0);
46
+ }
47
+
48
+ .reader-bar.active {
49
+ transform: translateY(0);
50
+ }
51
+
52
+ .reader-bar-inner {
53
+ display: flex;
54
+ align-items: center;
55
+ justify-content: space-between;
56
+ padding: 10px 16px;
57
+ background: rgba(255, 255, 255, 0.95);
58
+ backdrop-filter: blur(20px);
59
+ -webkit-backdrop-filter: blur(20px);
60
+ border-top: 1px solid rgba(0, 0, 0, 0.08);
61
+ box-shadow: 0 -4px 20px rgba(0, 0, 0, 0.08);
62
+ }
63
+
64
+ /* Dark mode variant */
65
+ @media (prefers-color-scheme: dark) {
66
+ .reader-bar-inner {
67
+ background: rgba(30, 30, 30, 0.95);
68
+ border-top-color: rgba(255, 255, 255, 0.08);
69
+ }
70
+ }
71
+
72
+ .reader-bar-info {
73
+ display: flex;
74
+ align-items: center;
75
+ gap: 8px;
76
+ flex: 1;
77
+ min-width: 0;
78
+ overflow: hidden;
79
+ }
80
+
81
+ .reader-bar-title {
82
+ font-size: 14px;
83
+ font-weight: 500;
84
+ white-space: nowrap;
85
+ overflow: hidden;
86
+ text-overflow: ellipsis;
87
+ }
88
+
89
+ .reader-bar-controls {
90
+ display: flex;
91
+ align-items: center;
92
+ gap: 4px;
93
+ flex-shrink: 0;
94
+ }
95
+
96
+ .reader-bar-btn {
97
+ display: flex;
98
+ align-items: center;
99
+ justify-content: center;
100
+ width: 40px;
101
+ height: 40px;
102
+ border-radius: 50%;
103
+ border: none;
104
+ background: transparent;
105
+ cursor: pointer;
106
+ color: inherit;
107
+ transition: background 0.15s ease;
108
+ }
109
+
110
+ .reader-bar-btn:hover {
111
+ background: rgba(0, 0, 0, 0.06);
112
+ }
113
+
114
+ /* Progress bar */
115
+ .reader-bar-progress {
116
+ height: 3px;
117
+ background: rgba(0, 0, 0, 0.08);
118
+ }
119
+
120
+ .reader-bar-progress-fill {
121
+ height: 100%;
122
+ width: 0%;
123
+ background: #c9a84c; /* gold accent */
124
+ border-radius: 0 3px 3px 0;
125
+ transition: width 0.3s linear;
126
+ }
127
+ ```
128
+
129
+ ### Body Padding (Prevent Content Overlap)
130
+
131
+ ```javascript
132
+ // When showing bar
133
+ document.body.style.paddingBottom = '72px';
134
+
135
+ // When hiding bar
136
+ document.body.style.paddingBottom = '';
137
+ ```
138
+
139
+ ---
140
+
141
+ ## Bottom Sheet
142
+
143
+ ### HTML Structure
144
+
145
+ ```html
146
+ <div class="audio-sheet">
147
+ <div class="audio-sheet-inner">
148
+ <button class="audio-sheet-close" aria-label="Close">✕</button>
149
+ <div class="audio-sheet-icon">🎧</div>
150
+ <p class="audio-sheet-text">Want to hear a quick introduction?</p>
151
+ <div class="audio-sheet-actions">
152
+ <button class="btn btn-primary">🔊 Listen Now</button>
153
+ <button class="btn btn-outline">Later</button>
154
+ </div>
155
+ </div>
156
+ </div>
157
+ ```
158
+
159
+ ### CSS
160
+
161
+ ```css
162
+ .audio-sheet {
163
+ position: fixed;
164
+ bottom: 0;
165
+ left: 0;
166
+ right: 0;
167
+ z-index: 1100;
168
+ transform: translateY(100%);
169
+ transition: transform 0.4s cubic-bezier(0.32, 0.72, 0, 1);
170
+ pointer-events: none;
171
+ }
172
+
173
+ .audio-sheet.active {
174
+ transform: translateY(0);
175
+ pointer-events: auto;
176
+ }
177
+
178
+ .audio-sheet-inner {
179
+ position: relative;
180
+ max-width: 480px;
181
+ margin: 0 auto;
182
+ padding: 24px 20px 20px;
183
+ background: rgba(255, 255, 255, 0.92);
184
+ backdrop-filter: blur(24px);
185
+ -webkit-backdrop-filter: blur(24px);
186
+ border-radius: 20px 20px 0 0;
187
+ box-shadow: 0 -8px 40px rgba(0, 0, 0, 0.12);
188
+ text-align: center;
189
+ }
190
+
191
+ /* Drag handle indicator */
192
+ .audio-sheet-inner::before {
193
+ content: '';
194
+ display: block;
195
+ width: 36px;
196
+ height: 4px;
197
+ background: rgba(0, 0, 0, 0.15);
198
+ border-radius: 2px;
199
+ margin: 0 auto 16px;
200
+ }
201
+
202
+ .audio-sheet-close {
203
+ position: absolute;
204
+ top: 12px;
205
+ right: 12px;
206
+ width: 32px;
207
+ height: 32px;
208
+ border-radius: 50%;
209
+ border: none;
210
+ background: transparent;
211
+ cursor: pointer;
212
+ font-size: 14px;
213
+ color: rgba(0, 0, 0, 0.4);
214
+ display: flex;
215
+ align-items: center;
216
+ justify-content: center;
217
+ }
218
+
219
+ .audio-sheet-icon {
220
+ font-size: 40px;
221
+ margin-bottom: 12px;
222
+ line-height: 1;
223
+ }
224
+
225
+ .audio-sheet-text {
226
+ font-size: 16px;
227
+ font-weight: 500;
228
+ margin-bottom: 16px;
229
+ line-height: 1.5;
230
+ }
231
+
232
+ .audio-sheet-actions {
233
+ display: flex;
234
+ gap: 12px;
235
+ justify-content: center;
236
+ }
237
+
238
+ .audio-sheet-actions .btn {
239
+ flex: 1;
240
+ max-width: 180px;
241
+ }
242
+ ```
243
+
244
+ ### Swipe-to-Dismiss (Mobile)
245
+
246
+ ```javascript
247
+ let startY = 0;
248
+
249
+ sheet.addEventListener('touchstart', (e) => {
250
+ startY = e.touches[0].clientY;
251
+ }, { passive: true });
252
+
253
+ sheet.addEventListener('touchmove', (e) => {
254
+ const dy = e.touches[0].clientY - startY;
255
+ if (dy > 60) dismiss(sheet);
256
+ }, { passive: true });
257
+ ```
258
+
259
+ ---
260
+
261
+ ## Animation Pattern
262
+
263
+ ### Enter (slide up)
264
+
265
+ ```javascript
266
+ // Create element → append to DOM → next frame add active class
267
+ document.body.appendChild(element);
268
+ requestAnimationFrame(() => {
269
+ requestAnimationFrame(() => {
270
+ element.classList.add('active');
271
+ });
272
+ });
273
+ ```
274
+
275
+ **Double `requestAnimationFrame`:** Ensures the browser has painted the initial state (translateY 100%) before transitioning to the active state.
276
+
277
+ ### Exit (slide down)
278
+
279
+ ```javascript
280
+ element.classList.remove('active');
281
+ setTimeout(() => element.remove(), 400); // match transition duration
282
+ ```
283
+
284
+ ---
285
+
286
+ ## Trigger Button (Topbar)
287
+
288
+ ```css
289
+ .topbar-audio-btn {
290
+ display: flex;
291
+ align-items: center;
292
+ justify-content: center;
293
+ width: 40px;
294
+ height: 40px;
295
+ border-radius: 50%;
296
+ border: none;
297
+ background: transparent;
298
+ color: inherit;
299
+ cursor: pointer;
300
+ transition: background 0.15s ease, color 0.15s ease;
301
+ }
302
+
303
+ .topbar-audio-btn:hover {
304
+ background: rgba(0, 0, 0, 0.06);
305
+ }
306
+
307
+ /* Active state when reading */
308
+ .topbar-audio-btn.reading {
309
+ color: #2d5016; /* moss green or brand color */
310
+ animation: pulse 2s infinite;
311
+ }
312
+
313
+ @keyframes pulse {
314
+ 0%, 100% { opacity: 1; }
315
+ 50% { opacity: 0.6; }
316
+ }
317
+ ```
318
+
319
+ ---
320
+
321
+ ## Responsive Considerations
322
+
323
+ ```css
324
+ @media (max-width: 768px) {
325
+ .reader-bar-inner {
326
+ padding: 8px 12px;
327
+ gap: 8px;
328
+ }
329
+
330
+ .reader-bar-btn {
331
+ width: 36px;
332
+ height: 36px;
333
+ }
334
+
335
+ .reader-bar-title {
336
+ font-size: 13px;
337
+ }
338
+
339
+ .audio-sheet-inner {
340
+ padding: 20px 16px 16px;
341
+ border-radius: 16px 16px 0 0;
342
+ }
343
+
344
+ .audio-sheet-icon {
345
+ font-size: 32px;
346
+ }
347
+
348
+ .audio-sheet-text {
349
+ font-size: 14px;
350
+ }
351
+ }
352
+ ```
353
+
354
+ ---
355
+
356
+ ## Accessibility
357
+
358
+ - All buttons must have `aria-label`
359
+ - Toggle button must update label between "Play" / "Pause"
360
+ - Respect `prefers-reduced-motion`: skip animations
361
+ - Ensure sufficient color contrast for controls
362
+ - Player bar should not trap focus
@@ -0,0 +1,223 @@
1
+ # Voice CRO — Conversion Rate Optimization via Audio
2
+
3
+ > Use audio strategically to increase engagement and conversions. Trigger-based, per-page, config-driven.
4
+
5
+ ---
6
+
7
+ ## What is Voice CRO?
8
+
9
+ Voice CRO uses pre-recorded audio (sales pitch, social proof, expertise showcase) to:
10
+ 1. **Interrupt scroll patterns** with a bottom sheet offer
11
+ 2. **Build trust** through human voice (more personal than text)
12
+ 3. **Guide users to CTA** after listening
13
+
14
+ ### Flow
15
+
16
+ ```
17
+ User lands on page
18
+
19
+ ├─ Time trigger (e.g., 15s) ─┐
20
+ │ ├─ BOTH met → Show Bottom Sheet
21
+ └─ Scroll trigger (e.g., 25%) ─┘
22
+
23
+ "🔊 Nghe Ngay" or "Để Sau"
24
+
25
+ ┌─────┴─────┐
26
+ │ │
27
+ Play Intro Dismiss
28
+ (30s MP3) (sessionStorage)
29
+
30
+ Intro Done
31
+
32
+ "Nghe Tiếp?"
33
+
34
+ ┌─────┴─────┐
35
+ │ │
36
+ Play Full Show CTA
37
+ (2min MP3)
38
+
39
+ Full Done
40
+
41
+ Show CTA + Thank You
42
+ ```
43
+
44
+ ---
45
+
46
+ ## Config Pattern
47
+
48
+ ```javascript
49
+ const CONFIG = {
50
+ '/': {
51
+ delay: 20000, // ms before time trigger
52
+ scroll: 0.30, // scroll % threshold
53
+ audio: [
54
+ '/audio/homepage-intro.mp3', // Part 1: short introduction
55
+ '/audio/homepage-full.mp3' // Part 2: full pitch
56
+ ],
57
+ sheetIcon: '🎧', // Bottom sheet emoji
58
+ sheetText: 'Bạn muốn nghe giới thiệu nhanh?',
59
+ ctaText: 'Đặt Lịch Ngay', // CTA button text
60
+ ctaHref: '#dat-lich' // CTA link
61
+ },
62
+ '/product-page.html': {
63
+ delay: 15000,
64
+ scroll: 0.25,
65
+ audio: ['/audio/product-intro.mp3', '/audio/product-full.mp3'],
66
+ sheetIcon: '💆',
67
+ sheetText: 'Nghe tư vấn nhanh từ chuyên gia',
68
+ ctaText: 'Mua Ngay',
69
+ ctaHref: '#buy',
70
+ // Optional: interaction trigger
71
+ checkboxTrigger: {
72
+ selector: '.check-item input[type=checkbox]',
73
+ minChecked: 2
74
+ }
75
+ }
76
+ };
77
+ ```
78
+
79
+ ### Config Fields
80
+
81
+ | Field | Type | Required | Description |
82
+ |-------|------|----------|-------------|
83
+ | `delay` | number | ✅ | Milliseconds before time trigger fires |
84
+ | `scroll` | number | ✅ | Scroll percentage (0-1) threshold |
85
+ | `audio` | string[] | ✅ | [intro.mp3, full.mp3] URLs |
86
+ | `sheetIcon` | string | ✅ | Emoji for bottom sheet |
87
+ | `sheetText` | string | ✅ | Bottom sheet message |
88
+ | `ctaText` | string | ✅ | CTA button label |
89
+ | `ctaHref` | string | ✅ | CTA link target |
90
+ | `checkboxTrigger` | object | ❌ | Interaction-based trigger |
91
+
92
+ ---
93
+
94
+ ## Trigger System
95
+
96
+ ### Dual-Condition Trigger (Time + Scroll)
97
+
98
+ ```javascript
99
+ function initTrigger(cfg) {
100
+ if (isDismissed()) return;
101
+ if (window.matchMedia('(prefers-reduced-motion: reduce)').matches) return;
102
+
103
+ let timeReady = false;
104
+ let scrollReady = false;
105
+ let triggered = false;
106
+
107
+ const tryTrigger = () => {
108
+ if (triggered) return;
109
+ if (timeReady && scrollReady) {
110
+ triggered = true;
111
+ showBottomSheet(cfg);
112
+ }
113
+ };
114
+
115
+ // Time trigger
116
+ setTimeout(() => { timeReady = true; tryTrigger(); }, cfg.delay);
117
+
118
+ // Scroll trigger
119
+ const onScroll = () => {
120
+ const pct = window.scrollY / (document.documentElement.scrollHeight - window.innerHeight);
121
+ if (pct >= cfg.scroll) {
122
+ scrollReady = true;
123
+ window.removeEventListener('scroll', onScroll);
124
+ tryTrigger();
125
+ }
126
+ };
127
+ window.addEventListener('scroll', onScroll, { passive: true });
128
+ onScroll(); // check immediately
129
+ }
130
+ ```
131
+
132
+ ### Interaction Trigger (Checkbox)
133
+
134
+ ```javascript
135
+ if (cfg.checkboxTrigger) {
136
+ const checkboxes = document.querySelectorAll(cfg.checkboxTrigger.selector);
137
+ checkboxes.forEach(cb => {
138
+ cb.addEventListener('change', () => {
139
+ const checked = document.querySelectorAll(
140
+ cfg.checkboxTrigger.selector + ':checked'
141
+ ).length;
142
+ if (checked >= cfg.checkboxTrigger.minChecked) {
143
+ triggered = true;
144
+ showBottomSheet(cfg);
145
+ }
146
+ });
147
+ });
148
+ }
149
+ ```
150
+
151
+ ---
152
+
153
+ ## Session Management
154
+
155
+ ```javascript
156
+ const STATE_KEY = 'voiceCroDismissed';
157
+
158
+ function isDismissed() {
159
+ try { return sessionStorage.getItem(STATE_KEY) === '1'; }
160
+ catch { return false; }
161
+ }
162
+
163
+ function setDismissed() {
164
+ try { sessionStorage.setItem(STATE_KEY, '1'); }
165
+ catch { /* noop */ }
166
+ }
167
+ ```
168
+
169
+ **Using `sessionStorage`:** Dismissed state resets when tab closes. User sees the offer again next session.
170
+ **Using `localStorage`:** Persistent. Use if you want "show once per user" behavior.
171
+
172
+ ---
173
+
174
+ ## Analytics Tracking
175
+
176
+ ```javascript
177
+ const STATS_KEY = 'voiceCroStats';
178
+
179
+ function trackStat(event) {
180
+ try {
181
+ const stats = JSON.parse(localStorage.getItem(STATS_KEY) || '{}');
182
+ const page = window.location.pathname;
183
+ if (!stats[page]) stats[page] = {};
184
+ stats[page][event] = (stats[page][event] || 0) + 1;
185
+ stats[page].lastSeen = Date.now();
186
+ localStorage.setItem(STATS_KEY, JSON.stringify(stats));
187
+ } catch { /* noop */ }
188
+ }
189
+
190
+ // Track: 'shown', 'listen', 'listenFull', 'dismissed', 'ctaClick'
191
+ ```
192
+
193
+ **For production:** Replace localStorage with server-side analytics (GA4 events, PostHog, etc.)
194
+
195
+ ---
196
+
197
+ ## Best Practices
198
+
199
+ ### Trigger Timing
200
+
201
+ | Page Type | Recommended Delay | Recommended Scroll |
202
+ |-----------|------------------|--------------------|
203
+ | Homepage | 20-25s | 30% |
204
+ | Service page | 12-18s | 20-30% |
205
+ | Blog/article | 25-30s | 40% |
206
+ | Course/product | 15-20s | 25% |
207
+ | Checkout | ❌ Don't trigger | ❌ Don't trigger |
208
+
209
+ ### Audio Script Guidelines
210
+
211
+ | Part | Duration | Content |
212
+ |------|----------|---------|
213
+ | **Intro** | 15-30s | Hook + value prop + "nghe thêm?" |
214
+ | **Full** | 60-120s | Problem → Solution → Social proof → CTA |
215
+
216
+ ### Don'ts
217
+
218
+ - ❌ Don't autoplay audio
219
+ - ❌ Don't show on checkout/payment pages
220
+ - ❌ Don't block page interaction while playing
221
+ - ❌ Don't trigger if another audio system is active (TTS reader)
222
+ - ❌ Don't trigger for `prefers-reduced-motion` users
223
+ - ❌ Don't trigger more than once per session
@@ -0,0 +1,120 @@
1
+ ---
2
+ name: cm-safe-deploy
3
+ description: Use when setting up deployment infrastructure for any project - establishes multi-gate deploy pipeline with test gates, build verification, frontend safety checks, version bump, changelog generation, and rollback strategy before code reaches production
4
+ ---
5
+
6
+ # Safe Deploy Pipeline v2
7
+
8
+ > **Role: Release Engineer** — You manage the 9-gate pipeline from secret hygiene to changelog. No shortcuts.
9
+
10
+ ## The Iron Law
11
+
12
+ **NO DEPLOY WITHOUT PASSING ALL GATES. GATES ARE SEQUENTIAL. EACH MUST PASS BEFORE THE NEXT.**
13
+
14
+ ## When to Use
15
+
16
+ - Setting up a new project's deployment infrastructure
17
+ - Project has no test gate before deploy
18
+ - After a production incident caused by untested code
19
+ - Adding CI/CD to an existing project
20
+
21
+ ## The 9-Gate Pipeline
22
+
23
+ ```
24
+ Gate 0: Secret Hygiene → Gate 1: Syntax → Gate 2: Test Suite → Gate 3: i18n Parity →
25
+ Gate 4: Build → Gate 5: Dist Verify → Gate 6: Deploy + Smoke → Gate 7: Version Bump → Gate 8: Changelog
26
+ ```
27
+
28
+ Each gate MUST pass. On failure → STOP, fix, retry from failed gate.
29
+
30
+ ---
31
+
32
+ ### Gate 0: Secret Hygiene (<0.5s)
33
+
34
+ Check wrangler config for secret values (SERVICE_KEY, ANON_KEY, DB_PASSWORD). Verify `.gitignore` has `.env` + `.dev.vars`. Verify no secret files tracked by git. Calls `cm-secret-shield` Layer 4.
35
+
36
+ **Rule:** `SERVICE_KEY`/`ANON_KEY` → Cloudflare Secrets (`wrangler secret put`), NEVER in `wrangler.jsonc`. Local dev secrets → `.dev.vars` (gitignored).
37
+
38
+ ### Gate 1: Syntax Validation (<1s)
39
+
40
+ | Stack | Command |
41
+ |-------|---------|
42
+ | Vanilla JS | `node -c path/to/app.js` |
43
+ | TypeScript | `npx tsc --noEmit` |
44
+ | Python | `python -m py_compile app.py` |
45
+
46
+ Gives exact line number of error instantly, before the slower test suite.
47
+
48
+ ### Gate 2: Test Suite
49
+
50
+ Must include: frontend safety, backend API, business logic, i18n sync, integration tests. Run via `npm run test:gate` (`vitest run --reporter=verbose`). 0 failures → proceed.
51
+
52
+ ### Gate 3: i18n Parity Check
53
+
54
+ All language files must have identical key counts and no null/empty values. Skip if project has no i18n.
55
+
56
+ ### Gate 4: Build Verification
57
+
58
+ `npm run build` must succeed. Catches: import failures, tree-shaking errors, missing env vars, bundle size issues.
59
+
60
+ ### Gate 5: Dist Asset Verification
61
+
62
+ Verify critical files exist in `dist/`: worker entry, frontend JS/CSS, translation files, critical assets.
63
+
64
+ ### Gate 6: Deploy + Smoke Test
65
+
66
+ Deploy command by platform: Cloudflare (`wrangler pages deploy`), Vercel (`vercel --prod`), Netlify (`netlify deploy --prod`). Post-deploy: `curl` deployed URL, must return HTTP 200.
67
+
68
+ ### Gate 7: Version Bump (POST-DEPLOY)
69
+
70
+ Detect bump type from commits: `fix:`/`bug:` → patch, `feat:`/`improve:` → minor. Update `package.json` version, commit.
71
+
72
+ ### Gate 8: Changelog Generation (POST-DEPLOY)
73
+
74
+ Collect commits since last tag, categorize into **🚀 Improvements** and **🐛 Bug Fixes**. Write to `CHANGELOG.md`, git tag release.
75
+ **CRITICAL:** The `CHANGELOG.md` file dynamically powers the public `/docs/changelog` page. You MUST ensure this gate passes and updates the changelog accurately so the documentation site reflects the latest version history.
76
+
77
+ ---
78
+
79
+ ## Composing the Deploy Script
80
+
81
+ ```json
82
+ {
83
+ "scripts": {
84
+ "deploy": "node -c src/app.js && npm run test:gate && npm run build && YOUR_DEPLOY_COMMAND && npm run release:version && npm run release:changelog"
85
+ }
86
+ }
87
+ ```
88
+
89
+ Chain with `&&` — any gate failure stops the chain.
90
+
91
+ ## Rollback Protocol
92
+
93
+ | Severity | Action |
94
+ |----------|--------|
95
+ | White screen | `git revert HEAD && npm run deploy` |
96
+ | Broken translations | `git checkout HEAD~1 -- i18n/*.json && npm run deploy` |
97
+ | API error | `git revert HEAD && npm run deploy` |
98
+ | Cloudflare | `wrangler pages deployment rollback <id>` |
99
+
100
+ ## Post-Release Memory (cm-continuity)
101
+
102
+ **On SUCCESS:** Update CONTINUITY.md: "Released v[version] — all 9 gates passed"
103
+ **On FAILURE:** Write to `learnings.json`: gate, error, root cause, prevention. Update CONTINUITY.md blockers.
104
+
105
+ ## Template Files (load on-demand with view_file)
106
+
107
+ | Template | Use When |
108
+ |----------|----------|
109
+ | `templates/deploy.sh` | Setting up the full 9-gate deploy script |
110
+
111
+ ## Integration
112
+
113
+ | Skill | When |
114
+ |-------|------|
115
+ | `cm-quality-gate` | Gate 2 frontend tests |
116
+ | `cm-secret-shield` | Gate 0 deep scanning |
117
+ | `cm-safe-i18n` | i18n-specific gates |
118
+ | `cm-identity-guard` | Gate 0 verifies deploy identity |
119
+ | `cm-continuity` | Post-release memory |
120
+ | `cm-dockit` | Gate 8 changelog → docs |