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,390 @@
|
|
|
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
|
+
})();
|
|
@@ -0,0 +1,262 @@
|
|
|
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 |
|