accessify-widget 0.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 (54) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +76 -0
  3. package/dist/accessify.min.js +2 -0
  4. package/dist/accessify.min.js.map +1 -0
  5. package/dist/accessify.mjs +6 -0
  6. package/dist/accessify.mjs.map +1 -0
  7. package/dist/alt-text-CLxbmwG6.js +567 -0
  8. package/dist/alt-text-CLxbmwG6.js.map +1 -0
  9. package/dist/animation-stop-DXebPS8D.js +88 -0
  10. package/dist/animation-stop-DXebPS8D.js.map +1 -0
  11. package/dist/auto-scan-pg-09o7A.js +885 -0
  12. package/dist/auto-scan-pg-09o7A.js.map +1 -0
  13. package/dist/big-cursor-B2UKu9dQ.js +88 -0
  14. package/dist/big-cursor-B2UKu9dQ.js.map +1 -0
  15. package/dist/color-blind-0LFng55r.js +108 -0
  16. package/dist/color-blind-0LFng55r.js.map +1 -0
  17. package/dist/contrast-DCkE0NXZ.js +64 -0
  18. package/dist/contrast-DCkE0NXZ.js.map +1 -0
  19. package/dist/dyslexia-font-wONgIy2T.js +77 -0
  20. package/dist/dyslexia-font-wONgIy2T.js.map +1 -0
  21. package/dist/focus-highlight-CjERyyUF.js +93 -0
  22. package/dist/focus-highlight-CjERyyUF.js.map +1 -0
  23. package/dist/hide-images-DJwmsV2C.js +39 -0
  24. package/dist/hide-images-DJwmsV2C.js.map +1 -0
  25. package/dist/index-CUQfpnwR.js +6520 -0
  26. package/dist/index-CUQfpnwR.js.map +1 -0
  27. package/dist/keyboard-nav-BdPyLaZt.js +312 -0
  28. package/dist/keyboard-nav-BdPyLaZt.js.map +1 -0
  29. package/dist/line-height-BT98qgEF.js +54 -0
  30. package/dist/line-height-BT98qgEF.js.map +1 -0
  31. package/dist/link-highlight-DBGm067Y.js +87 -0
  32. package/dist/link-highlight-DBGm067Y.js.map +1 -0
  33. package/dist/loader.min.js +1 -0
  34. package/dist/page-structure-2X8mOSpC.js +166 -0
  35. package/dist/page-structure-2X8mOSpC.js.map +1 -0
  36. package/dist/reading-guide-VT8NciIL.js +122 -0
  37. package/dist/reading-guide-VT8NciIL.js.map +1 -0
  38. package/dist/reading-mask-BABChuCz.js +76 -0
  39. package/dist/reading-mask-BABChuCz.js.map +1 -0
  40. package/dist/saturation-D8ZXpWAN.js +59 -0
  41. package/dist/saturation-D8ZXpWAN.js.map +1 -0
  42. package/dist/spacing-DENai3JU.js +106 -0
  43. package/dist/spacing-DENai3JU.js.map +1 -0
  44. package/dist/text-align-BDRPqPvl.js +51 -0
  45. package/dist/text-align-BDRPqPvl.js.map +1 -0
  46. package/dist/text-simplify-CELklw5A.js +223 -0
  47. package/dist/text-simplify-CELklw5A.js.map +1 -0
  48. package/dist/text-size-B-uv436p.js +69 -0
  49. package/dist/text-size-B-uv436p.js.map +1 -0
  50. package/dist/tts-02b9iV0h.js +505 -0
  51. package/dist/tts-02b9iV0h.js.map +1 -0
  52. package/dist/widget.js +2 -0
  53. package/dist/widget.js.map +1 -0
  54. package/package.json +61 -0
@@ -0,0 +1,505 @@
1
+ const DEFAULT_OPTIONS = {
2
+ speed: 1
3
+ };
4
+ function createTTSModule() {
5
+ let enabled = false;
6
+ let options = { ...DEFAULT_OPTIONS };
7
+ let controlBar = null;
8
+ let highlightedEl = null;
9
+ let isPlaying = false;
10
+ let isPaused = false;
11
+ let clickToReadActive = true;
12
+ let currentUtterances = [];
13
+ let currentUtteranceIndex = 0;
14
+ let selectedVoice = null;
15
+ let detectedLang = "en";
16
+ const STORAGE_KEY = "accessify-tts";
17
+ const CONTROL_BAR_ID = "accessify-tts-controls";
18
+ const STYLE_ID = "accessify-tts-styles";
19
+ const HIGHLIGHT_CLASS = "accessify-tts-highlight";
20
+ function detectPageLanguage() {
21
+ const htmlLang = document.documentElement.lang;
22
+ if (htmlLang) {
23
+ return htmlLang.toLowerCase();
24
+ }
25
+ const meta = document.querySelector(
26
+ 'meta[http-equiv="content-language"]'
27
+ );
28
+ if (meta?.content) {
29
+ return meta.content.toLowerCase();
30
+ }
31
+ return "en";
32
+ }
33
+ function selectBestVoice(lang) {
34
+ const voices = speechSynthesis.getVoices();
35
+ if (voices.length === 0) return null;
36
+ const langBase = lang.split("-")[0];
37
+ if (langBase === "de") {
38
+ const deDENetwork = voices.find(
39
+ (v) => v.lang.toLowerCase().startsWith("de-de") && v.localService === false
40
+ );
41
+ if (deDENetwork) return deDENetwork;
42
+ const deDE = voices.find(
43
+ (v) => v.lang.toLowerCase().startsWith("de-de")
44
+ );
45
+ if (deDE) return deDE;
46
+ const deAny = voices.find(
47
+ (v) => v.lang.toLowerCase().startsWith("de")
48
+ );
49
+ if (deAny) return deAny;
50
+ }
51
+ const exact = voices.find(
52
+ (v) => v.lang.toLowerCase() === lang
53
+ );
54
+ if (exact) return exact;
55
+ const baseMatch = voices.find(
56
+ (v) => v.lang.toLowerCase().startsWith(langBase)
57
+ );
58
+ if (baseMatch) return baseMatch;
59
+ return voices[0] || null;
60
+ }
61
+ function refreshVoice() {
62
+ selectedVoice = selectBestVoice(detectedLang);
63
+ }
64
+ function splitIntoSentences(text) {
65
+ const raw = text.match(/[^.!?]*[.!?]+[\s]?|[^.!?]+$/g);
66
+ if (!raw) return [text];
67
+ const sentences = [];
68
+ let buffer = "";
69
+ for (const segment of raw) {
70
+ const trimmed = segment.trim();
71
+ if (!trimmed) continue;
72
+ buffer += (buffer ? " " : "") + trimmed;
73
+ if (buffer.length > 80 || /[.!?]$/.test(buffer)) {
74
+ sentences.push(buffer);
75
+ buffer = "";
76
+ }
77
+ }
78
+ if (buffer.trim()) {
79
+ sentences.push(buffer.trim());
80
+ }
81
+ return sentences.length > 0 ? sentences : [text];
82
+ }
83
+ function stopSpeech() {
84
+ speechSynthesis.cancel();
85
+ isPlaying = false;
86
+ isPaused = false;
87
+ currentUtterances = [];
88
+ currentUtteranceIndex = 0;
89
+ removeHighlight();
90
+ updateControlBarState();
91
+ }
92
+ function speakText(text, sourceEl) {
93
+ stopSpeech();
94
+ if (!text.trim()) return;
95
+ if (sourceEl) {
96
+ applyHighlight(sourceEl);
97
+ }
98
+ const sentences = splitIntoSentences(text);
99
+ currentUtterances = sentences.map((sentence) => {
100
+ const utterance = new SpeechSynthesisUtterance(sentence);
101
+ utterance.rate = options.speed;
102
+ utterance.lang = detectedLang;
103
+ if (selectedVoice) {
104
+ utterance.voice = selectedVoice;
105
+ }
106
+ return utterance;
107
+ });
108
+ currentUtteranceIndex = 0;
109
+ isPlaying = true;
110
+ isPaused = false;
111
+ updateControlBarState();
112
+ speakNextUtterance();
113
+ }
114
+ function speakNextUtterance() {
115
+ if (currentUtteranceIndex >= currentUtterances.length) {
116
+ isPlaying = false;
117
+ isPaused = false;
118
+ removeHighlight();
119
+ updateControlBarState();
120
+ return;
121
+ }
122
+ const utterance = currentUtterances[currentUtteranceIndex];
123
+ utterance.onend = () => {
124
+ currentUtteranceIndex++;
125
+ if (currentUtteranceIndex < currentUtterances.length && isPlaying) {
126
+ setTimeout(() => speakNextUtterance(), 50);
127
+ } else {
128
+ isPlaying = false;
129
+ isPaused = false;
130
+ removeHighlight();
131
+ updateControlBarState();
132
+ }
133
+ };
134
+ utterance.onerror = (e) => {
135
+ if (e.error !== "interrupted" && e.error !== "canceled") {
136
+ console.warn("[Accessify TTS] Speech error:", e.error);
137
+ }
138
+ isPlaying = false;
139
+ isPaused = false;
140
+ removeHighlight();
141
+ updateControlBarState();
142
+ };
143
+ speechSynthesis.speak(utterance);
144
+ }
145
+ function togglePause() {
146
+ if (!isPlaying) return;
147
+ if (isPaused) {
148
+ speechSynthesis.resume();
149
+ isPaused = false;
150
+ } else {
151
+ speechSynthesis.pause();
152
+ isPaused = true;
153
+ }
154
+ updateControlBarState();
155
+ }
156
+ function applyHighlight(el) {
157
+ removeHighlight();
158
+ el.classList.add(HIGHLIGHT_CLASS);
159
+ highlightedEl = el;
160
+ }
161
+ function removeHighlight() {
162
+ if (highlightedEl) {
163
+ highlightedEl.classList.remove(HIGHLIGHT_CLASS);
164
+ highlightedEl = null;
165
+ }
166
+ document.querySelectorAll(`.${HIGHLIGHT_CLASS}`).forEach((el) => el.classList.remove(HIGHLIGHT_CLASS));
167
+ }
168
+ function handleClick(e) {
169
+ if (!enabled || !clickToReadActive) return;
170
+ const target = e.target;
171
+ if (!target) return;
172
+ if (controlBar?.contains(target)) return;
173
+ e.preventDefault();
174
+ e.stopPropagation();
175
+ const textEl = findTextElement(target);
176
+ if (!textEl) return;
177
+ const text = extractTextContent(textEl);
178
+ if (text.trim()) {
179
+ speakText(text, textEl);
180
+ }
181
+ }
182
+ function findTextElement(el) {
183
+ const directText = extractTextContent(el);
184
+ if (directText.trim().length > 0) return el;
185
+ let current = el;
186
+ while (current && current !== document.body) {
187
+ const text = extractTextContent(current);
188
+ if (text.trim().length > 0) return current;
189
+ current = current.parentElement;
190
+ }
191
+ return null;
192
+ }
193
+ function extractTextContent(el) {
194
+ return (el.innerText || el.textContent || "").trim();
195
+ }
196
+ function createControlBar() {
197
+ const bar = document.createElement("div");
198
+ bar.id = CONTROL_BAR_ID;
199
+ bar.setAttribute("role", "toolbar");
200
+ bar.setAttribute("aria-label", "Text-to-Speech controls");
201
+ Object.assign(bar.style, {
202
+ position: "fixed",
203
+ top: "0",
204
+ left: "50%",
205
+ transform: "translateX(-50%)",
206
+ zIndex: "2147483646",
207
+ display: "flex",
208
+ alignItems: "center",
209
+ gap: "6px",
210
+ padding: "6px 14px",
211
+ background: "#1a1a2e",
212
+ color: "#f0f0f0",
213
+ borderRadius: "0 0 10px 10px",
214
+ boxShadow: "0 4px 16px rgba(0,0,0,0.3)",
215
+ fontFamily: "system-ui, -apple-system, sans-serif",
216
+ fontSize: "13px",
217
+ lineHeight: "1",
218
+ userSelect: "none",
219
+ transition: "opacity 0.2s ease"
220
+ });
221
+ const playPauseBtn = document.createElement("button");
222
+ playPauseBtn.className = "accessify-tts-btn accessify-tts-play";
223
+ playPauseBtn.setAttribute("aria-label", "Play / Pause");
224
+ playPauseBtn.title = "Play / Pause";
225
+ playPauseBtn.innerHTML = iconPlay();
226
+ playPauseBtn.addEventListener("click", () => {
227
+ if (isPlaying) {
228
+ togglePause();
229
+ }
230
+ });
231
+ const stopBtn = document.createElement("button");
232
+ stopBtn.className = "accessify-tts-btn accessify-tts-stop";
233
+ stopBtn.setAttribute("aria-label", "Stop");
234
+ stopBtn.title = "Stop";
235
+ stopBtn.innerHTML = iconStop();
236
+ stopBtn.addEventListener("click", () => {
237
+ stopSpeech();
238
+ });
239
+ const speedContainer = document.createElement("div");
240
+ Object.assign(speedContainer.style, {
241
+ display: "flex",
242
+ alignItems: "center",
243
+ gap: "4px",
244
+ marginLeft: "4px"
245
+ });
246
+ const speedLabel = document.createElement("label");
247
+ speedLabel.textContent = "Speed";
248
+ speedLabel.setAttribute("for", "accessify-tts-speed");
249
+ Object.assign(speedLabel.style, {
250
+ fontSize: "11px",
251
+ opacity: "0.8",
252
+ whiteSpace: "nowrap"
253
+ });
254
+ const speedSlider = document.createElement("input");
255
+ speedSlider.type = "range";
256
+ speedSlider.id = "accessify-tts-speed";
257
+ speedSlider.min = "0.5";
258
+ speedSlider.max = "2";
259
+ speedSlider.step = "0.1";
260
+ speedSlider.value = String(options.speed);
261
+ speedSlider.setAttribute("aria-label", "Speech speed");
262
+ Object.assign(speedSlider.style, {
263
+ width: "70px",
264
+ accentColor: "#4ea8de",
265
+ cursor: "pointer"
266
+ });
267
+ const speedValue = document.createElement("span");
268
+ speedValue.className = "accessify-tts-speed-val";
269
+ speedValue.textContent = `${options.speed.toFixed(1)}x`;
270
+ Object.assign(speedValue.style, {
271
+ fontSize: "11px",
272
+ minWidth: "30px",
273
+ textAlign: "center",
274
+ opacity: "0.8"
275
+ });
276
+ speedSlider.addEventListener("input", () => {
277
+ const newSpeed = parseFloat(speedSlider.value);
278
+ options.speed = newSpeed;
279
+ speedValue.textContent = `${newSpeed.toFixed(1)}x`;
280
+ savePreferences();
281
+ for (const u of currentUtterances) {
282
+ u.rate = newSpeed;
283
+ }
284
+ });
285
+ speedContainer.appendChild(speedLabel);
286
+ speedContainer.appendChild(speedSlider);
287
+ speedContainer.appendChild(speedValue);
288
+ const sep = document.createElement("div");
289
+ Object.assign(sep.style, {
290
+ width: "1px",
291
+ height: "18px",
292
+ background: "rgba(255,255,255,0.2)",
293
+ margin: "0 4px"
294
+ });
295
+ const hint = document.createElement("span");
296
+ hint.className = "accessify-tts-hint";
297
+ hint.textContent = "Click any text to read it aloud";
298
+ Object.assign(hint.style, {
299
+ fontSize: "11px",
300
+ opacity: "0.6",
301
+ whiteSpace: "nowrap"
302
+ });
303
+ const srNote = document.createElement("span");
304
+ Object.assign(srNote.style, {
305
+ position: "absolute",
306
+ width: "1px",
307
+ height: "1px",
308
+ padding: "0",
309
+ margin: "-1px",
310
+ overflow: "hidden",
311
+ clip: "rect(0,0,0,0)",
312
+ whiteSpace: "nowrap",
313
+ border: "0"
314
+ });
315
+ srNote.textContent = "For full screen reader support, use NVDA, JAWS, or VoiceOver";
316
+ bar.appendChild(playPauseBtn);
317
+ bar.appendChild(stopBtn);
318
+ bar.appendChild(sep);
319
+ bar.appendChild(speedContainer);
320
+ bar.appendChild(sep.cloneNode(true));
321
+ bar.appendChild(hint);
322
+ bar.appendChild(srNote);
323
+ return bar;
324
+ }
325
+ function updateControlBarState() {
326
+ if (!controlBar) return;
327
+ const playBtn = controlBar.querySelector(
328
+ ".accessify-tts-play"
329
+ );
330
+ const hintEl = controlBar.querySelector(
331
+ ".accessify-tts-hint"
332
+ );
333
+ if (playBtn) {
334
+ if (isPlaying && !isPaused) {
335
+ playBtn.innerHTML = iconPause();
336
+ playBtn.setAttribute("aria-label", "Pause");
337
+ playBtn.title = "Pause";
338
+ } else if (isPlaying && isPaused) {
339
+ playBtn.innerHTML = iconPlay();
340
+ playBtn.setAttribute("aria-label", "Resume");
341
+ playBtn.title = "Resume";
342
+ } else {
343
+ playBtn.innerHTML = iconPlay();
344
+ playBtn.setAttribute("aria-label", "Play / Pause");
345
+ playBtn.title = "Play / Pause";
346
+ }
347
+ }
348
+ if (hintEl) {
349
+ if (isPlaying && !isPaused) {
350
+ hintEl.textContent = "Speaking...";
351
+ } else if (isPaused) {
352
+ hintEl.textContent = "Paused";
353
+ } else {
354
+ hintEl.textContent = "Click any text to read it aloud";
355
+ }
356
+ }
357
+ }
358
+ function iconPlay() {
359
+ return '<svg width="16" height="16" viewBox="0 0 24 24" fill="currentColor" stroke="none" aria-hidden="true"><polygon points="6,3 20,12 6,21"/></svg>';
360
+ }
361
+ function iconPause() {
362
+ return '<svg width="16" height="16" viewBox="0 0 24 24" fill="currentColor" stroke="none" aria-hidden="true"><rect x="5" y="3" width="4" height="18"/><rect x="15" y="3" width="4" height="18"/></svg>';
363
+ }
364
+ function iconStop() {
365
+ return '<svg width="16" height="16" viewBox="0 0 24 24" fill="currentColor" stroke="none" aria-hidden="true"><rect x="4" y="4" width="16" height="16" rx="2"/></svg>';
366
+ }
367
+ function getStyles() {
368
+ return `
369
+ /* accessify TTS styles */
370
+ .${HIGHLIGHT_CLASS} {
371
+ background-color: rgba(78, 168, 222, 0.15) !important;
372
+ outline: 2px solid rgba(78, 168, 222, 0.5) !important;
373
+ outline-offset: 2px !important;
374
+ border-radius: 3px !important;
375
+ transition: background-color 0.2s ease, outline-color 0.2s ease !important;
376
+ }
377
+
378
+ #${CONTROL_BAR_ID} .accessify-tts-btn {
379
+ display: inline-flex;
380
+ align-items: center;
381
+ justify-content: center;
382
+ width: 28px;
383
+ height: 28px;
384
+ padding: 0;
385
+ margin: 0;
386
+ border: 1px solid rgba(255,255,255,0.15);
387
+ border-radius: 6px;
388
+ background: rgba(255,255,255,0.08);
389
+ color: #f0f0f0;
390
+ cursor: pointer;
391
+ transition: background 0.15s ease, border-color 0.15s ease;
392
+ line-height: 1;
393
+ }
394
+
395
+ #${CONTROL_BAR_ID} .accessify-tts-btn:hover {
396
+ background: rgba(255,255,255,0.18);
397
+ border-color: rgba(255,255,255,0.3);
398
+ }
399
+
400
+ #${CONTROL_BAR_ID} .accessify-tts-btn:focus-visible {
401
+ outline: 2px solid #4ea8de;
402
+ outline-offset: 1px;
403
+ }
404
+
405
+ #${CONTROL_BAR_ID} .accessify-tts-btn:active {
406
+ background: rgba(255,255,255,0.25);
407
+ }
408
+ `;
409
+ }
410
+ function injectStyles() {
411
+ let styleEl = document.getElementById(STYLE_ID);
412
+ if (!styleEl) {
413
+ styleEl = document.createElement("style");
414
+ styleEl.id = STYLE_ID;
415
+ document.head.appendChild(styleEl);
416
+ }
417
+ styleEl.textContent = getStyles();
418
+ }
419
+ function removeStyles() {
420
+ const styleEl = document.getElementById(STYLE_ID);
421
+ styleEl?.remove();
422
+ }
423
+ function loadPreferences() {
424
+ const saved = localStorage.getItem(STORAGE_KEY);
425
+ if (saved) {
426
+ try {
427
+ const parsed = JSON.parse(saved);
428
+ options.speed = Math.min(2, Math.max(0.5, parsed.speed ?? DEFAULT_OPTIONS.speed));
429
+ } catch {
430
+ options = { ...DEFAULT_OPTIONS };
431
+ }
432
+ }
433
+ }
434
+ function savePreferences() {
435
+ localStorage.setItem(STORAGE_KEY, JSON.stringify({ speed: options.speed }));
436
+ }
437
+ function activate() {
438
+ if (enabled) return;
439
+ enabled = true;
440
+ loadPreferences();
441
+ detectedLang = detectPageLanguage();
442
+ refreshVoice();
443
+ if (!selectedVoice && speechSynthesis.onvoiceschanged !== void 0) {
444
+ speechSynthesis.onvoiceschanged = () => {
445
+ refreshVoice();
446
+ };
447
+ }
448
+ injectStyles();
449
+ controlBar = createControlBar();
450
+ document.documentElement.appendChild(controlBar);
451
+ clickToReadActive = true;
452
+ document.addEventListener("click", handleClick, true);
453
+ }
454
+ function deactivate() {
455
+ enabled = false;
456
+ stopSpeech();
457
+ document.removeEventListener("click", handleClick, true);
458
+ clickToReadActive = false;
459
+ if (controlBar) {
460
+ controlBar.remove();
461
+ controlBar = null;
462
+ }
463
+ removeHighlight();
464
+ removeStyles();
465
+ if (speechSynthesis.onvoiceschanged !== void 0) {
466
+ speechSynthesis.onvoiceschanged = null;
467
+ }
468
+ }
469
+ return {
470
+ id: "tts",
471
+ name: () => "Text-to-Speech",
472
+ description: "Click any text to hear it read aloud with adjustable speed",
473
+ icon: "tts",
474
+ category: "cognitive",
475
+ activate,
476
+ deactivate,
477
+ getState: () => ({
478
+ id: "tts",
479
+ enabled,
480
+ value: { speed: options.speed }
481
+ }),
482
+ setState: (newState) => {
483
+ if (newState.speed !== void 0) {
484
+ options.speed = Math.min(2, Math.max(0.5, newState.speed));
485
+ savePreferences();
486
+ const slider = document.getElementById(
487
+ "accessify-tts-speed"
488
+ );
489
+ if (slider) {
490
+ slider.value = String(options.speed);
491
+ }
492
+ const valDisplay = controlBar?.querySelector(
493
+ ".accessify-tts-speed-val"
494
+ );
495
+ if (valDisplay) {
496
+ valDisplay.textContent = `${options.speed.toFixed(1)}x`;
497
+ }
498
+ }
499
+ }
500
+ };
501
+ }
502
+ export {
503
+ createTTSModule as default
504
+ };
505
+ //# sourceMappingURL=tts-02b9iV0h.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"tts-02b9iV0h.js","sources":["../src/features/tts.ts"],"sourcesContent":["import type { FeatureModule, FeatureState } from '../types';\n\ninterface TTSOptions {\n speed: number;\n}\n\nconst DEFAULT_OPTIONS: TTSOptions = {\n speed: 1.0,\n};\n\nexport default function createTTSModule(): FeatureModule {\n let enabled = false;\n let options: TTSOptions = { ...DEFAULT_OPTIONS };\n let controlBar: HTMLDivElement | null = null;\n let highlightedEl: HTMLElement | null = null;\n let isPlaying = false;\n let isPaused = false;\n let clickToReadActive = true;\n let currentUtterances: SpeechSynthesisUtterance[] = [];\n let currentUtteranceIndex = 0;\n let selectedVoice: SpeechSynthesisVoice | null = null;\n let detectedLang = 'en';\n\n const STORAGE_KEY = 'accessify-tts';\n const CONTROL_BAR_ID = 'accessify-tts-controls';\n const STYLE_ID = 'accessify-tts-styles';\n const HIGHLIGHT_CLASS = 'accessify-tts-highlight';\n\n // ---------------------------------------------------------------------------\n // Language detection\n // ---------------------------------------------------------------------------\n\n function detectPageLanguage(): string {\n const htmlLang = document.documentElement.lang;\n if (htmlLang) {\n return htmlLang.toLowerCase();\n }\n // Fallback: check <meta> content-language\n const meta = document.querySelector<HTMLMetaElement>(\n 'meta[http-equiv=\"content-language\"]'\n );\n if (meta?.content) {\n return meta.content.toLowerCase();\n }\n return 'en';\n }\n\n // ---------------------------------------------------------------------------\n // Voice selection\n // ---------------------------------------------------------------------------\n\n function selectBestVoice(lang: string): SpeechSynthesisVoice | null {\n const voices = speechSynthesis.getVoices();\n if (voices.length === 0) return null;\n\n const langBase = lang.split('-')[0]; // e.g. 'de' from 'de-DE'\n\n // For German, strongly prioritize native German voices\n if (langBase === 'de') {\n // Prefer high-quality (non-local) de-DE voices first\n const deDENetwork = voices.find(\n (v) => v.lang.toLowerCase().startsWith('de-de') && v.localService === false\n );\n if (deDENetwork) return deDENetwork;\n\n // Then any de-DE voice\n const deDE = voices.find(\n (v) => v.lang.toLowerCase().startsWith('de-de')\n );\n if (deDE) return deDE;\n\n // Then any German voice\n const deAny = voices.find((v) =>\n v.lang.toLowerCase().startsWith('de')\n );\n if (deAny) return deAny;\n }\n\n // Exact match (e.g. 'en-US' matches 'en-US')\n const exact = voices.find(\n (v) => v.lang.toLowerCase() === lang\n );\n if (exact) return exact;\n\n // Base language match (e.g. 'en' matches 'en-US', 'en-GB')\n const baseMatch = voices.find(\n (v) => v.lang.toLowerCase().startsWith(langBase)\n );\n if (baseMatch) return baseMatch;\n\n // Ultimate fallback: first available voice\n return voices[0] || null;\n }\n\n function refreshVoice() {\n selectedVoice = selectBestVoice(detectedLang);\n }\n\n // ---------------------------------------------------------------------------\n // Sentence splitting (avoids Chrome's ~15s synthesis cutoff)\n // ---------------------------------------------------------------------------\n\n function splitIntoSentences(text: string): string[] {\n // Split on sentence-ending punctuation followed by whitespace or end\n const raw = text.match(/[^.!?]*[.!?]+[\\s]?|[^.!?]+$/g);\n if (!raw) return [text];\n\n const sentences: string[] = [];\n let buffer = '';\n\n for (const segment of raw) {\n const trimmed = segment.trim();\n if (!trimmed) continue;\n\n buffer += (buffer ? ' ' : '') + trimmed;\n\n // Flush buffer if it's reasonably long (> 80 chars) or is a complete sentence\n if (buffer.length > 80 || /[.!?]$/.test(buffer)) {\n sentences.push(buffer);\n buffer = '';\n }\n }\n\n if (buffer.trim()) {\n sentences.push(buffer.trim());\n }\n\n return sentences.length > 0 ? sentences : [text];\n }\n\n // ---------------------------------------------------------------------------\n // Speech engine\n // ---------------------------------------------------------------------------\n\n function stopSpeech() {\n speechSynthesis.cancel();\n isPlaying = false;\n isPaused = false;\n currentUtterances = [];\n currentUtteranceIndex = 0;\n removeHighlight();\n updateControlBarState();\n }\n\n function speakText(text: string, sourceEl?: HTMLElement) {\n stopSpeech();\n\n if (!text.trim()) return;\n\n if (sourceEl) {\n applyHighlight(sourceEl);\n }\n\n const sentences = splitIntoSentences(text);\n currentUtterances = sentences.map((sentence) => {\n const utterance = new SpeechSynthesisUtterance(sentence);\n utterance.rate = options.speed;\n utterance.lang = detectedLang;\n if (selectedVoice) {\n utterance.voice = selectedVoice;\n }\n return utterance;\n });\n\n currentUtteranceIndex = 0;\n isPlaying = true;\n isPaused = false;\n updateControlBarState();\n\n speakNextUtterance();\n }\n\n function speakNextUtterance() {\n if (currentUtteranceIndex >= currentUtterances.length) {\n // All done\n isPlaying = false;\n isPaused = false;\n removeHighlight();\n updateControlBarState();\n return;\n }\n\n const utterance = currentUtterances[currentUtteranceIndex];\n\n utterance.onend = () => {\n currentUtteranceIndex++;\n // Small delay between sentences for naturalness\n if (currentUtteranceIndex < currentUtterances.length && isPlaying) {\n setTimeout(() => speakNextUtterance(), 50);\n } else {\n isPlaying = false;\n isPaused = false;\n removeHighlight();\n updateControlBarState();\n }\n };\n\n utterance.onerror = (e) => {\n // 'interrupted' and 'canceled' are not real errors — they fire on cancel()\n if (e.error !== 'interrupted' && e.error !== 'canceled') {\n console.warn('[Accessify TTS] Speech error:', e.error);\n }\n isPlaying = false;\n isPaused = false;\n removeHighlight();\n updateControlBarState();\n };\n\n speechSynthesis.speak(utterance);\n }\n\n function togglePause() {\n if (!isPlaying) return;\n\n if (isPaused) {\n speechSynthesis.resume();\n isPaused = false;\n } else {\n speechSynthesis.pause();\n isPaused = true;\n }\n updateControlBarState();\n }\n\n // ---------------------------------------------------------------------------\n // Highlight management\n // ---------------------------------------------------------------------------\n\n function applyHighlight(el: HTMLElement) {\n removeHighlight();\n el.classList.add(HIGHLIGHT_CLASS);\n highlightedEl = el;\n }\n\n function removeHighlight() {\n if (highlightedEl) {\n highlightedEl.classList.remove(HIGHLIGHT_CLASS);\n highlightedEl = null;\n }\n // Safety: remove from any stale elements\n document\n .querySelectorAll(`.${HIGHLIGHT_CLASS}`)\n .forEach((el) => el.classList.remove(HIGHLIGHT_CLASS));\n }\n\n // ---------------------------------------------------------------------------\n // Click-to-read handler\n // ---------------------------------------------------------------------------\n\n function handleClick(e: MouseEvent) {\n if (!enabled || !clickToReadActive) return;\n\n const target = e.target as HTMLElement;\n if (!target) return;\n\n // Ignore clicks on the control bar itself\n if (controlBar?.contains(target)) return;\n\n e.preventDefault();\n e.stopPropagation();\n\n // Walk up to find the most meaningful text container\n const textEl = findTextElement(target);\n if (!textEl) return;\n\n const text = extractTextContent(textEl);\n if (text.trim()) {\n speakText(text, textEl);\n }\n }\n\n function findTextElement(el: HTMLElement): HTMLElement | null {\n // If the element itself has substantial text, use it\n const directText = extractTextContent(el);\n if (directText.trim().length > 0) return el;\n\n // Walk up to find a parent with text\n let current: HTMLElement | null = el;\n while (current && current !== document.body) {\n const text = extractTextContent(current);\n if (text.trim().length > 0) return current;\n current = current.parentElement;\n }\n return null;\n }\n\n function extractTextContent(el: HTMLElement): string {\n // Use innerText rather than textContent to get rendered text with spacing\n return (el.innerText || el.textContent || '').trim();\n }\n\n // ---------------------------------------------------------------------------\n // Control bar UI\n // ---------------------------------------------------------------------------\n\n function createControlBar(): HTMLDivElement {\n const bar = document.createElement('div');\n bar.id = CONTROL_BAR_ID;\n bar.setAttribute('role', 'toolbar');\n bar.setAttribute('aria-label', 'Text-to-Speech controls');\n\n Object.assign(bar.style, {\n position: 'fixed',\n top: '0',\n left: '50%',\n transform: 'translateX(-50%)',\n zIndex: '2147483646',\n display: 'flex',\n alignItems: 'center',\n gap: '6px',\n padding: '6px 14px',\n background: '#1a1a2e',\n color: '#f0f0f0',\n borderRadius: '0 0 10px 10px',\n boxShadow: '0 4px 16px rgba(0,0,0,0.3)',\n fontFamily: 'system-ui, -apple-system, sans-serif',\n fontSize: '13px',\n lineHeight: '1',\n userSelect: 'none',\n transition: 'opacity 0.2s ease',\n });\n\n // Play/Pause button\n const playPauseBtn = document.createElement('button');\n playPauseBtn.className = 'accessify-tts-btn accessify-tts-play';\n playPauseBtn.setAttribute('aria-label', 'Play / Pause');\n playPauseBtn.title = 'Play / Pause';\n playPauseBtn.innerHTML = iconPlay();\n playPauseBtn.addEventListener('click', () => {\n if (isPlaying) {\n togglePause();\n }\n // If nothing is playing, the user should click an element to start\n });\n\n // Stop button\n const stopBtn = document.createElement('button');\n stopBtn.className = 'accessify-tts-btn accessify-tts-stop';\n stopBtn.setAttribute('aria-label', 'Stop');\n stopBtn.title = 'Stop';\n stopBtn.innerHTML = iconStop();\n stopBtn.addEventListener('click', () => {\n stopSpeech();\n });\n\n // Speed control\n const speedContainer = document.createElement('div');\n Object.assign(speedContainer.style, {\n display: 'flex',\n alignItems: 'center',\n gap: '4px',\n marginLeft: '4px',\n });\n\n const speedLabel = document.createElement('label');\n speedLabel.textContent = 'Speed';\n speedLabel.setAttribute('for', 'accessify-tts-speed');\n Object.assign(speedLabel.style, {\n fontSize: '11px',\n opacity: '0.8',\n whiteSpace: 'nowrap',\n });\n\n const speedSlider = document.createElement('input');\n speedSlider.type = 'range';\n speedSlider.id = 'accessify-tts-speed';\n speedSlider.min = '0.5';\n speedSlider.max = '2';\n speedSlider.step = '0.1';\n speedSlider.value = String(options.speed);\n speedSlider.setAttribute('aria-label', 'Speech speed');\n Object.assign(speedSlider.style, {\n width: '70px',\n accentColor: '#4ea8de',\n cursor: 'pointer',\n });\n\n const speedValue = document.createElement('span');\n speedValue.className = 'accessify-tts-speed-val';\n speedValue.textContent = `${options.speed.toFixed(1)}x`;\n Object.assign(speedValue.style, {\n fontSize: '11px',\n minWidth: '30px',\n textAlign: 'center',\n opacity: '0.8',\n });\n\n speedSlider.addEventListener('input', () => {\n const newSpeed = parseFloat(speedSlider.value);\n options.speed = newSpeed;\n speedValue.textContent = `${newSpeed.toFixed(1)}x`;\n savePreferences();\n\n // Update rate on any currently queued utterances\n for (const u of currentUtterances) {\n u.rate = newSpeed;\n }\n });\n\n speedContainer.appendChild(speedLabel);\n speedContainer.appendChild(speedSlider);\n speedContainer.appendChild(speedValue);\n\n // Separator\n const sep = document.createElement('div');\n Object.assign(sep.style, {\n width: '1px',\n height: '18px',\n background: 'rgba(255,255,255,0.2)',\n margin: '0 4px',\n });\n\n // Hint text\n const hint = document.createElement('span');\n hint.className = 'accessify-tts-hint';\n hint.textContent = 'Click any text to read it aloud';\n Object.assign(hint.style, {\n fontSize: '11px',\n opacity: '0.6',\n whiteSpace: 'nowrap',\n });\n\n // Screen reader note (visually hidden, available to AT)\n const srNote = document.createElement('span');\n Object.assign(srNote.style, {\n position: 'absolute',\n width: '1px',\n height: '1px',\n padding: '0',\n margin: '-1px',\n overflow: 'hidden',\n clip: 'rect(0,0,0,0)',\n whiteSpace: 'nowrap',\n border: '0',\n });\n srNote.textContent =\n 'For full screen reader support, use NVDA, JAWS, or VoiceOver';\n\n bar.appendChild(playPauseBtn);\n bar.appendChild(stopBtn);\n bar.appendChild(sep);\n bar.appendChild(speedContainer);\n bar.appendChild(sep.cloneNode(true));\n bar.appendChild(hint);\n bar.appendChild(srNote);\n\n return bar;\n }\n\n function updateControlBarState() {\n if (!controlBar) return;\n\n const playBtn = controlBar.querySelector<HTMLButtonElement>(\n '.accessify-tts-play'\n );\n const hintEl = controlBar.querySelector<HTMLSpanElement>(\n '.accessify-tts-hint'\n );\n\n if (playBtn) {\n if (isPlaying && !isPaused) {\n playBtn.innerHTML = iconPause();\n playBtn.setAttribute('aria-label', 'Pause');\n playBtn.title = 'Pause';\n } else if (isPlaying && isPaused) {\n playBtn.innerHTML = iconPlay();\n playBtn.setAttribute('aria-label', 'Resume');\n playBtn.title = 'Resume';\n } else {\n playBtn.innerHTML = iconPlay();\n playBtn.setAttribute('aria-label', 'Play / Pause');\n playBtn.title = 'Play / Pause';\n }\n }\n\n if (hintEl) {\n if (isPlaying && !isPaused) {\n hintEl.textContent = 'Speaking...';\n } else if (isPaused) {\n hintEl.textContent = 'Paused';\n } else {\n hintEl.textContent = 'Click any text to read it aloud';\n }\n }\n }\n\n // ---------------------------------------------------------------------------\n // Inline SVG icons for control buttons\n // ---------------------------------------------------------------------------\n\n function iconPlay(): string {\n return '<svg width=\"16\" height=\"16\" viewBox=\"0 0 24 24\" fill=\"currentColor\" stroke=\"none\" aria-hidden=\"true\"><polygon points=\"6,3 20,12 6,21\"/></svg>';\n }\n\n function iconPause(): string {\n return '<svg width=\"16\" height=\"16\" viewBox=\"0 0 24 24\" fill=\"currentColor\" stroke=\"none\" aria-hidden=\"true\"><rect x=\"5\" y=\"3\" width=\"4\" height=\"18\"/><rect x=\"15\" y=\"3\" width=\"4\" height=\"18\"/></svg>';\n }\n\n function iconStop(): string {\n return '<svg width=\"16\" height=\"16\" viewBox=\"0 0 24 24\" fill=\"currentColor\" stroke=\"none\" aria-hidden=\"true\"><rect x=\"4\" y=\"4\" width=\"16\" height=\"16\" rx=\"2\"/></svg>';\n }\n\n // ---------------------------------------------------------------------------\n // Styles (injected into <head>, not Shadow DOM)\n // ---------------------------------------------------------------------------\n\n function getStyles(): string {\n return `\n /* accessify TTS styles */\n .${HIGHLIGHT_CLASS} {\n background-color: rgba(78, 168, 222, 0.15) !important;\n outline: 2px solid rgba(78, 168, 222, 0.5) !important;\n outline-offset: 2px !important;\n border-radius: 3px !important;\n transition: background-color 0.2s ease, outline-color 0.2s ease !important;\n }\n\n #${CONTROL_BAR_ID} .accessify-tts-btn {\n display: inline-flex;\n align-items: center;\n justify-content: center;\n width: 28px;\n height: 28px;\n padding: 0;\n margin: 0;\n border: 1px solid rgba(255,255,255,0.15);\n border-radius: 6px;\n background: rgba(255,255,255,0.08);\n color: #f0f0f0;\n cursor: pointer;\n transition: background 0.15s ease, border-color 0.15s ease;\n line-height: 1;\n }\n\n #${CONTROL_BAR_ID} .accessify-tts-btn:hover {\n background: rgba(255,255,255,0.18);\n border-color: rgba(255,255,255,0.3);\n }\n\n #${CONTROL_BAR_ID} .accessify-tts-btn:focus-visible {\n outline: 2px solid #4ea8de;\n outline-offset: 1px;\n }\n\n #${CONTROL_BAR_ID} .accessify-tts-btn:active {\n background: rgba(255,255,255,0.25);\n }\n `;\n }\n\n function injectStyles() {\n let styleEl = document.getElementById(STYLE_ID);\n if (!styleEl) {\n styleEl = document.createElement('style');\n styleEl.id = STYLE_ID;\n document.head.appendChild(styleEl);\n }\n styleEl.textContent = getStyles();\n }\n\n function removeStyles() {\n const styleEl = document.getElementById(STYLE_ID);\n styleEl?.remove();\n }\n\n // ---------------------------------------------------------------------------\n // Preferences persistence\n // ---------------------------------------------------------------------------\n\n function loadPreferences() {\n const saved = localStorage.getItem(STORAGE_KEY);\n if (saved) {\n try {\n const parsed = JSON.parse(saved);\n options.speed = Math.min(2, Math.max(0.5, parsed.speed ?? DEFAULT_OPTIONS.speed));\n } catch {\n options = { ...DEFAULT_OPTIONS };\n }\n }\n }\n\n function savePreferences() {\n localStorage.setItem(STORAGE_KEY, JSON.stringify({ speed: options.speed }));\n }\n\n // ---------------------------------------------------------------------------\n // Module lifecycle\n // ---------------------------------------------------------------------------\n\n function activate() {\n if (enabled) return;\n enabled = true;\n\n loadPreferences();\n\n // Detect language\n detectedLang = detectPageLanguage();\n\n // Select voice (voices may load asynchronously)\n refreshVoice();\n if (!selectedVoice && speechSynthesis.onvoiceschanged !== undefined) {\n speechSynthesis.onvoiceschanged = () => {\n refreshVoice();\n };\n }\n\n // Inject styles and control bar\n injectStyles();\n controlBar = createControlBar();\n document.documentElement.appendChild(controlBar);\n\n // Enable click-to-read\n clickToReadActive = true;\n document.addEventListener('click', handleClick, true);\n }\n\n function deactivate() {\n enabled = false;\n\n // Stop all speech\n stopSpeech();\n\n // Remove click handler\n document.removeEventListener('click', handleClick, true);\n clickToReadActive = false;\n\n // Remove control bar\n if (controlBar) {\n controlBar.remove();\n controlBar = null;\n }\n\n // Remove highlights\n removeHighlight();\n\n // Remove styles\n removeStyles();\n\n // Clean up voices listener\n if (speechSynthesis.onvoiceschanged !== undefined) {\n speechSynthesis.onvoiceschanged = null;\n }\n }\n\n // ---------------------------------------------------------------------------\n // FeatureModule interface\n // ---------------------------------------------------------------------------\n\n return {\n id: 'tts',\n name: () => 'Text-to-Speech',\n description: 'Click any text to hear it read aloud with adjustable speed',\n icon: 'tts',\n category: 'cognitive',\n activate,\n deactivate,\n getState: (): FeatureState => ({\n id: 'tts',\n enabled,\n value: { speed: options.speed },\n }),\n setState: (newState: Partial<TTSOptions>) => {\n if (newState.speed !== undefined) {\n options.speed = Math.min(2, Math.max(0.5, newState.speed));\n savePreferences();\n\n // Update the slider in the control bar if present\n const slider = document.getElementById(\n 'accessify-tts-speed'\n ) as HTMLInputElement | null;\n if (slider) {\n slider.value = String(options.speed);\n }\n const valDisplay = controlBar?.querySelector<HTMLSpanElement>(\n '.accessify-tts-speed-val'\n );\n if (valDisplay) {\n valDisplay.textContent = `${options.speed.toFixed(1)}x`;\n }\n }\n },\n };\n}\n"],"names":[],"mappings":"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"}