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.
- package/CHANGELOG.md +50 -0
- package/README.md +285 -0
- package/adapters/antigravity.js +15 -0
- package/adapters/claude-code.js +17 -0
- package/adapters/cursor.js +16 -0
- package/commands/bootstrap.md +49 -0
- package/commands/build.md +48 -0
- package/commands/content.md +48 -0
- package/commands/continuity.md +60 -0
- package/commands/debug.md +51 -0
- package/commands/demo.md +96 -0
- package/commands/deploy.md +51 -0
- package/commands/plan.md +42 -0
- package/commands/review.md +55 -0
- package/commands/track.md +46 -0
- package/commands/ux.md +46 -0
- package/dist/agent-dispatch.js +161 -0
- package/dist/chains/builtin.js +85 -0
- package/dist/continuity.js +385 -0
- package/dist/dashboard.js +926 -0
- package/dist/data.js +122 -0
- package/dist/index.js +2434 -0
- package/dist/judge.js +252 -0
- package/dist/parallel-dispatch.js +359 -0
- package/dist/parallel-quality.js +172 -0
- package/dist/skill-chain.js +258 -0
- package/install.sh +513 -0
- package/package.json +79 -0
- package/skills/.content-factory-state.json +132 -0
- package/skills/.git 2/logs/refs/heads/main +1 -0
- package/skills/.git 2/logs/refs/remotes/origin/main +1 -0
- package/skills/.git 2/objects/02/fb0956734b5f8ba3f918b7defd04a89cfe0076 +0 -0
- package/skills/.git 2/objects/08/1e129d75dc6feac6c02037272e6bd1a04e3324 +0 -0
- package/skills/.git 2/objects/0c/5393416f3c5e01c9a655a802bff0dd52f76f0a +0 -0
- package/skills/.git 2/objects/10/0b9be46978a946a77188f68be725098a122001 +0 -0
- package/skills/.git 2/objects/10/cf041167fc9843610eb3d90259ef3396315fdc +0 -0
- package/skills/.git 2/objects/12/5e19538dd6e1338ffe74f6c4c165b00435bf48 +0 -0
- package/skills/.git 2/objects/16/a9b9d0088d5c1347628b45a2620b479d8ad57c +0 -0
- package/skills/.git 2/objects/17/8c2a9ef93c33ae4eec9d58e82321f9229843a1 +0 -0
- package/skills/.git 2/objects/25/397ae41d09104d763bdcac2695209d85cdea89 +0 -0
- package/skills/.git 2/objects/2f/a836b7947f2d458e1f639788bf4bb0983a3305 +0 -0
- package/skills/.git 2/objects/3a/baaaf0a1c0909c0828335791557125fba911e0 +0 -0
- package/skills/.git 2/objects/42/2924221b81f5ce3c4e4daac9a64a24f9b01f9a +0 -0
- package/skills/.git 2/objects/42/ec0ce707447dc11446a34c9995fb8533801731 +0 -0
- package/skills/.git 2/objects/46/e43ce92866d56ce74b1d750db307cfe6154a15 +0 -0
- package/skills/.git 2/objects/48/5e41b633c63f55b8277bcc59f44f67681f671a +0 -0
- package/skills/.git 2/objects/49/49c596a3a89fa240642acd95dd3258e261eb09 +0 -0
- package/skills/.git 2/objects/50/9d42d8412ef8eaf7f7e138476bac2e4d10ce60 +0 -0
- package/skills/.git 2/objects/55/0c8c389d981b463ef849aeb792d8be3ccb6ec8 +0 -0
- package/skills/.git 2/objects/5d/82d3b18410cdda3ace3677436f0cb599dbe2d2 +0 -0
- package/skills/.git 2/objects/60/0617c58e871a38b33bf29e282d132bb3c381ad +0 -0
- package/skills/.git 2/objects/6a/8369a99c687b7245c92ffaf0e0f0dab9014504 +0 -0
- package/skills/.git 2/objects/79/bea435d40ab531c1aaf6be0432c6a5b7aaed21 +0 -0
- package/skills/.git 2/objects/7e/5ebd79251c2f14e4aceb86c74b6b6daae6b500 +0 -0
- package/skills/.git 2/objects/81/98a822a60178d6d5023ddb3e222cddf048742e +0 -0
- package/skills/.git 2/objects/86/0a0e1943dfe53411d2e499a1f16f46a96ef758 +0 -0
- package/skills/.git 2/objects/86/971fb55fdc081fdbae52376f0f13e57a4e9b04 +0 -0
- package/skills/.git 2/objects/88/b89dd609a0a03f8d4fe8bfde20d5b8fc1d326d +0 -0
- package/skills/.git 2/objects/90/8737edb6b7809e32cc01590b4e08ba42a9d40d +0 -0
- package/skills/.git 2/objects/93/d5a8a9a7d4fb7f11491cb596a6880528725118 +0 -0
- package/skills/.git 2/objects/98/46a2ab81d0c3b3eb00ef88fc56989aa7e9f316 +0 -0
- package/skills/.git 2/objects/9b/d8dd1e49cf274eaf9c555f3ab39dce7af5715e +0 -0
- package/skills/.git 2/objects/a1/13329fb0cec96ae78b222d33a24c3b5bc7fa1f +0 -0
- package/skills/.git 2/objects/a9/e6effe626e8a3aea3a8fc3364b492191c6e7d0 +0 -0
- package/skills/.git 2/objects/ad/6de7e48d9782cca9353d1ff0aa1aab7fe1df85 +0 -0
- package/skills/.git 2/objects/af/54ae316f771ff692e299ffcd8bf2f06b413b59 +0 -0
- package/skills/.git 2/objects/b0/4cb8b0b00dad633e731c1472161419e738d674 +0 -0
- package/skills/.git 2/objects/b3/094abb0b9ed46419b269e4a4e36a459690e3b0 +0 -0
- package/skills/.git 2/objects/b9/435c5d4baac2cfc5c83009ddd27b46b60db5f1 +0 -0
- package/skills/.git 2/objects/ba/5da17dbaec5ec2dcfdfd126aead518d1171d5c +0 -0
- package/skills/.git 2/objects/c0/bf58703aa258ba5dd63083bebaec8f223d844c +0 -0
- package/skills/.git 2/objects/c4/701a34edf1fc1bad58ccc57bd03f9426acb59a +0 -0
- package/skills/.git 2/objects/c7/5ccce9a4e5cc74d9b3174550cf6d993ca43638 +0 -0
- package/skills/.git 2/objects/c7/710d59b5a35b0f1f0a0399386643a0bd94c929 +0 -0
- package/skills/.git 2/objects/d1/fe58237112e953e5fec52da22cf38e08be3df9 +5 -0
- package/skills/.git 2/objects/d2/2bbe9fd2f74c95bc5583e803f5e435f1e2cd86 +0 -0
- package/skills/.git 2/objects/d7/e72852ea2bff74581dbf247d400120086229f4 +0 -0
- package/skills/.git 2/objects/d8/d4c3b5553e4fd72807e1d4b49ef07d9ef3ac35 +0 -0
- package/skills/.git 2/objects/dc/75050c2876f6a02ae2a53a3c886f395b622977 +0 -0
- package/skills/.git 2/objects/ee/e8546f95acec500187c08a28a8b9ee02db0dec +0 -0
- package/skills/.git 2/objects/ef/263c059208b416c2146434f10cb2b9fabcba16 +0 -0
- package/skills/.git 2/objects/f3/ae597e84d9a59b88acd21c99bde2eaf686d785 +0 -0
- package/skills/.git 2/objects/f3/f6f5673c821d3d8e76fa267a9e882e7a5387ea +0 -0
- package/skills/.git 2/objects/f9/6e6d0ad02624dd11d5848594d056caef7a5e8b +0 -0
- package/skills/.git 2/objects/ff/278988fc1edf0db3abcf18de795f4cc0b4f3e1 +0 -0
- package/skills/.git 2/refs/heads/main +1 -0
- package/skills/.git 2/refs/remotes/origin/main +1 -0
- package/skills/.pytest_cache 2/v/cache/nodeids +76 -0
- package/skills/.pytest_cache 2/v/cache/stepwise +1 -0
- package/skills/_shared/helpers.md +123 -0
- package/skills/_shared/outputs-convention.md +24 -0
- package/skills/cm-ads-tracker/SKILL.md +109 -0
- package/skills/cm-ads-tracker/evals/evals.json +55 -0
- package/skills/cm-ads-tracker/references/gtm-architecture.md +321 -0
- package/skills/cm-ads-tracker/references/industry-events.md +294 -0
- package/skills/cm-ads-tracker/references/platforms-api.md +238 -0
- package/skills/cm-ads-tracker/templates/capi-payload.md +79 -0
- package/skills/cm-ads-tracker/templates/datalayer-push.js +104 -0
- package/skills/cm-ads-tracker/templates/gtm-variables.js +56 -0
- package/skills/cm-brainstorm-idea/SKILL.md +423 -0
- package/skills/cm-code-review/SKILL.md +151 -0
- package/skills/cm-content-factory/SKILL.md +416 -0
- package/skills/cm-continuity/SKILL.md +399 -0
- package/skills/cm-dashboard/SKILL.md +533 -0
- package/skills/cm-dashboard/ui/app.js +1270 -0
- package/skills/cm-dashboard/ui/index.html +206 -0
- package/skills/cm-dashboard/ui/style.css +440 -0
- package/skills/cm-debugging/SKILL.md +412 -0
- package/skills/cm-deep-search/SKILL.md +242 -0
- package/skills/cm-design-system/SKILL.md +97 -0
- package/skills/cm-design-system/resources/halo-modern.md +40 -0
- package/skills/cm-design-system/resources/lunaris-advanced.md +40 -0
- package/skills/cm-design-system/resources/nitro-enterprise.md +39 -0
- package/skills/cm-design-system/resources/shadcn-default.md +37 -0
- package/skills/cm-dockit/README.md +100 -0
- package/skills/cm-dockit/SKILL.md +302 -0
- package/skills/cm-dockit/index.html +443 -0
- package/skills/cm-dockit/package-lock.json +1850 -0
- package/skills/cm-dockit/package.json +14 -0
- package/skills/cm-dockit/prompts/analysis.md +34 -0
- package/skills/cm-dockit/prompts/api-reference.md +24 -0
- package/skills/cm-dockit/prompts/architecture.md +21 -0
- package/skills/cm-dockit/prompts/data-flow.md +20 -0
- package/skills/cm-dockit/prompts/database.md +21 -0
- package/skills/cm-dockit/prompts/deployment.md +22 -0
- package/skills/cm-dockit/prompts/flows.md +21 -0
- package/skills/cm-dockit/prompts/jtbd.md +20 -0
- package/skills/cm-dockit/prompts/personas.md +24 -0
- package/skills/cm-dockit/prompts/sop-modules.md +40 -0
- package/skills/cm-dockit/scripts/doc-gen.sh +121 -0
- package/skills/cm-dockit/scripts/dockit-dashboard.sh +142 -0
- package/skills/cm-dockit/scripts/dockit-runner.sh +607 -0
- package/skills/cm-dockit/scripts/dockit-task.sh +166 -0
- package/skills/cm-dockit/skills/analyze-codebase.md +174 -0
- package/skills/cm-dockit/skills/api-reference.md +237 -0
- package/skills/cm-dockit/skills/changelog-guide.md +195 -0
- package/skills/cm-dockit/skills/content-guidelines.md +190 -0
- package/skills/cm-dockit/skills/sop-guide.md +184 -0
- package/skills/cm-dockit/skills/tech-docs.md +287 -0
- package/skills/cm-dockit/templates/markdown/structure.md +60 -0
- package/skills/cm-dockit/templates/vitepress-premium/.vitepress/config.mts +110 -0
- package/skills/cm-dockit/templates/vitepress-premium/.vitepress/theme/custom.css +189 -0
- package/skills/cm-dockit/templates/vitepress-premium/.vitepress/theme/index.ts +4 -0
- package/skills/cm-dockit/templates/vitepress-premium/package.json +19 -0
- package/skills/cm-dockit/templates/vitepress-premium/tests/frontend.test.ts +45 -0
- package/skills/cm-dockit/tests/runner.test.ts +66 -0
- package/skills/cm-dockit/workflows/export-markdown.md +82 -0
- package/skills/cm-dockit/workflows/generate-docs.md +68 -0
- package/skills/cm-dockit/workflows/setup-vitepress.md +181 -0
- package/skills/cm-example/SKILL.md +26 -0
- package/skills/cm-execution/SKILL.md +268 -0
- package/skills/cm-git-worktrees/SKILL.md +164 -0
- package/skills/cm-how-it-work/SKILL.md +189 -0
- package/skills/cm-identity-guard/SKILL.md +412 -0
- package/skills/cm-jtbd/SKILL.md +98 -0
- package/skills/cm-planning/SKILL.md +130 -0
- package/skills/cm-project-bootstrap/SKILL.md +161 -0
- package/skills/cm-project-bootstrap/templates/AGENTS.md +42 -0
- package/skills/cm-project-bootstrap/templates/frontend-safety.test.js +51 -0
- package/skills/cm-project-bootstrap/templates/i18n-sync.test.js +38 -0
- package/skills/cm-project-bootstrap/templates/pr-template.md +12 -0
- package/skills/cm-project-bootstrap/templates/project-identity.json +29 -0
- package/skills/cm-project-bootstrap/templates/vitest.config.js +10 -0
- package/skills/cm-quality-gate/SKILL.md +218 -0
- package/skills/cm-readit/SKILL.md +289 -0
- package/skills/cm-readit/audio-player.md +206 -0
- package/skills/cm-readit/examples/blog-reader.js +352 -0
- package/skills/cm-readit/examples/voice-cro.js +390 -0
- package/skills/cm-readit/tts-engine.md +262 -0
- package/skills/cm-readit/ui-patterns.md +362 -0
- package/skills/cm-readit/voice-cro.md +223 -0
- package/skills/cm-safe-deploy/SKILL.md +120 -0
- package/skills/cm-safe-deploy/templates/deploy.sh +89 -0
- package/skills/cm-safe-i18n/SKILL.md +473 -0
- package/skills/cm-secret-shield/SKILL.md +580 -0
- package/skills/cm-skill-chain/SKILL.md +78 -0
- package/skills/cm-skill-index/SKILL.md +318 -0
- package/skills/cm-skill-mastery/SKILL.md +169 -0
- package/skills/cm-start/SKILL.md +65 -0
- package/skills/cm-status/SKILL.md +12 -0
- package/skills/cm-tdd/SKILL.md +370 -0
- package/skills/cm-terminal/SKILL.md +177 -0
- package/skills/cm-test-gate/SKILL.md +242 -0
- package/skills/cm-ui-preview/SKILL.md +291 -0
- package/skills/cm-ux-master/DESIGN_STANDARD_TEMPLATE.md +54 -0
- package/skills/cm-ux-master/SKILL.md +114 -0
- package/skills/cro-methodology/SKILL.md +98 -0
- package/skills/cro-methodology/references/COPYWRITING.md +178 -0
- package/skills/cro-methodology/references/OBJECTIONS.md +135 -0
- package/skills/cro-methodology/references/PERSUASION.md +158 -0
- package/skills/cro-methodology/references/RESEARCH.md +220 -0
- package/skills/cro-methodology/references/funnel-analysis.md +365 -0
- 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 |
|