MediaWiki:Gadget-bhc-accessibility.js: Difference between revisions
No edit summary |
No edit summary |
||
| (2 intermediate revisions by the same user not shown) | |||
| Line 11: | Line 11: | ||
// - cookie fallback only if localStorage unavailable | // - cookie fallback only if localStorage unavailable | ||
// ========================================================= | // ========================================================= | ||
const LS_KEY = " | const LS_KEY = "bhc_a11y_state_v2"; // bumped because state model changed | ||
const COOKIE_PREFIX = "a11y_"; | const COOKIE_PREFIX = "a11y_"; | ||
| Line 59: | Line 59: | ||
} | } | ||
// cookie fallback (per-key) | // cookie fallback (per-key) | ||
setCookie(" | setCookie("contrastMode", stateObj.contrastMode); // off|yellow|white | ||
setCookie("underline", stateObj.underline ? "1" : "0"); | setCookie("underline", stateObj.underline ? "1" : "0"); | ||
setCookie("spacing", stateObj.spacing); | setCookie("spacing", stateObj.spacing); | ||
setCookie("cursor", stateObj.cursor); | setCookie("cursor", stateObj.cursor); // off|white|yellow | ||
setCookie("ruler", stateObj.ruler ? "1" : "0"); | setCookie("ruler", stateObj.ruler ? "1" : "0"); | ||
setCookie("fontPct", String(stateObj.fontPct)); | setCookie("fontPct", String(stateObj.fontPct)); | ||
| Line 80: | Line 80: | ||
// cookie fallback (legacy / no-LS) | // cookie fallback (legacy / no-LS) | ||
return { | return { | ||
contrastMode: getCookie("contrastMode", "off"), | |||
underline: getCookie("underline", "0") === "1", | underline: getCookie("underline", "0") === "1", | ||
spacing: getCookie("spacing", "off"), | spacing: getCookie("spacing", "off"), | ||
| Line 94: | Line 94: | ||
return; | return; | ||
} | } | ||
[" | ["contrastMode","underline","spacing","cursor","ruler","fontPct"].forEach(delCookie); | ||
} | } | ||
| Line 101: | Line 101: | ||
// ========================================================= | // ========================================================= | ||
const STATE = { | const STATE = { | ||
contrastMode: "off", // off | yellow | white | |||
underline: false, | underline: false, | ||
spacing: "off", | spacing: "off", // off | plus | minus | ||
cursor: "off", | cursor: "off", // off | white | yellow | ||
ruler: false, | ruler: false, | ||
fontPct: 100 | fontPct: 100 // 90..175 | ||
}; | }; | ||
const FONT_STEPS = [90, 102, 114, 126, 138, 150, 162, 175]; | const FONT_STEPS = [90, 102, 114, 126, 138, 150, 162, 175]; | ||
let liveEl = null; | let liveEl = null; | ||
let statusEl = null; | let statusEl = null; | ||
let rulerEl = null; | let rulerEl = null; | ||
let rulerY = 0; | let rulerY = 0; | ||
| Line 132: | Line 131: | ||
} | } | ||
if (dir > 0) return FONT_STEPS[Math.min(FONT_STEPS.length - 1, idx + 1)]; | if (dir > 0) return FONT_STEPS[Math.min(FONT_STEPS.length - 1, idx + 1)]; | ||
if (FONT_STEPS[idx] === cur) return FONT_STEPS[Math.max(0, idx - 1)]; | if (FONT_STEPS[idx] === cur) return FONT_STEPS[Math.max(0, idx - 1)]; | ||
return FONT_STEPS[Math.max(0, idx - 1)]; | return FONT_STEPS[Math.max(0, idx - 1)]; | ||
| Line 144: | Line 141: | ||
if (!statusEl) return; | if (!statusEl) return; | ||
const contrast = STATE. | const contrast = | ||
STATE.contrastMode === "yellow" ? "Contrast: Yellow/Black" : | |||
STATE.contrastMode === "white" ? "Contrast: White/Black" : | |||
"Contrast: Off"; | |||
const text = `Text ${STATE.fontPct}%`; | const text = `Text ${STATE.fontPct}%`; | ||
const underline = STATE.underline ? "Links | const underline = STATE.underline ? "Links: Underlined" : "Links: Normal"; | ||
const spacing = | const spacing = | ||
STATE.spacing === "plus" ? "Spacing | STATE.spacing === "plus" ? "Spacing +" : | ||
STATE.spacing === "minus" ? "Spacing | STATE.spacing === "minus" ? "Spacing −" : | ||
"Spacing | "Spacing: Normal"; | ||
const cursor = | const cursor = | ||
STATE.cursor === " | STATE.cursor === "yellow" ? "Cursor: Yellow" : | ||
STATE.cursor === " | STATE.cursor === "white" ? "Cursor: White" : | ||
"Cursor | "Cursor: Off"; | ||
const ruler = STATE.ruler ? "Ruler: On" : "Ruler: Off"; | |||
statusEl.textContent = `Status: ${contrast} · ${text} · ${underline} · ${spacing} · ${cursor} · ${ruler}`; | |||
statusEl.textContent = | |||
} | } | ||
| Line 168: | Line 165: | ||
// ========================================================= | // ========================================================= | ||
function applyState() { | function applyState() { | ||
document.body.classList.toggle("a11y-contrast", | // Contrast scheme classes | ||
const contrastOn = STATE.contrastMode !== "off"; | |||
document.body.classList.toggle("a11y-contrast", contrastOn); | |||
document.body.classList.toggle("a11y-contrast-yellow", STATE.contrastMode === "yellow"); | |||
document.body.classList.toggle("a11y-contrast-white", STATE.contrastMode === "white"); | |||
// Other features | |||
document.body.classList.toggle("a11y-underline-links", !!STATE.underline); | document.body.classList.toggle("a11y-underline-links", !!STATE.underline); | ||
| Line 190: | Line 193: | ||
updateUiButtons(); | updateUiButtons(); | ||
updateStatusLine(); | updateStatusLine(); | ||
saveStateToStorage(STATE); | saveStateToStorage(STATE); | ||
} | } | ||
| Line 235: | Line 238: | ||
announce("Reading ruler off"); | announce("Reading ruler off"); | ||
} | } | ||
} | |||
// Named resize handler so we can remove it cleanly (prevents handler buildup) | |||
function onResize() { | |||
setRulerY(rulerY); | |||
} | } | ||
| Line 243: | Line 251: | ||
window.addEventListener("mousemove", onMouseMove, { passive: true }); | window.addEventListener("mousemove", onMouseMove, { passive: true }); | ||
window.addEventListener("keydown", onKeyDown); | window.addEventListener("keydown", onKeyDown); | ||
window.addEventListener("resize", | window.addEventListener("resize", onResize, { passive: true }); | ||
} | } | ||
function disableRulerListeners() { | function disableRulerListeners() { | ||
| Line 250: | Line 258: | ||
window.removeEventListener("mousemove", onMouseMove); | window.removeEventListener("mousemove", onMouseMove); | ||
window.removeEventListener("keydown", onKeyDown); | window.removeEventListener("keydown", onKeyDown); | ||
window.removeEventListener("resize", onResize); | |||
} | } | ||
| Line 268: | Line 277: | ||
return style.display !== "none" && style.visibility !== "hidden"; | return style.display !== "none" && style.visibility !== "hidden"; | ||
}); | }); | ||
} | } | ||
| Line 363: | Line 311: | ||
panel.hidden = true; | panel.hidden = true; | ||
// | // Layout per your request | ||
panel.innerHTML = ` | panel.innerHTML = ` | ||
<div class="a11y-head"> | <div class="a11y-head"> | ||
| Line 371: | Line 319: | ||
<div class="a11y-row" role="group" aria-label="Text size"> | <div class="a11y-row" role="group" aria-label="Text size"> | ||
<button type="button" class="a11y-btn" data-action="font-minus | <button type="button" class="a11y-btn" data-action="font-minus" aria-label="Decrease text size">A−</button> | ||
<button type="button" class="a11y-btn" data-action="font-plus | <button type="button" class="a11y-btn" data-action="font-plus" aria-label="Increase text size">A+</button> | ||
<button type="button" class="a11y-btn" data-action="font-reset | <button type="button" class="a11y-btn" data-action="font-reset" aria-label="Reset text size">Reset</button> | ||
</div> | </div> | ||
<div class="a11y-row" role="group" aria-label="Text spacing"> | <div class="a11y-row" role="group" aria-label="Text spacing"> | ||
<button type="button" class="a11y-btn" data-action="spacing-minus | <button type="button" class="a11y-btn" data-action="spacing-minus">Spacing−</button> | ||
<button type="button" class="a11y-btn" data-action="spacing-plus | <button type="button" class="a11y-btn" data-action="spacing-plus">Spacing+</button> | ||
<button type="button" class="a11y-btn" data-action="spacing-reset | <button type="button" class="a11y-btn" data-action="spacing-reset">Spacing reset</button> | ||
</div> | </div> | ||
<div class="a11y-row" role="group" aria-label=" | <div class="a11y-row" role="group" aria-label="High contrast"> | ||
<button type="button" class="a11y-btn" data-action="contrast" | <button type="button" class="a11y-btn" data-action="contrast-yellow">High contrast (yellow)</button> | ||
<button type="button" class="a11y-btn" data-action=" | <button type="button" class="a11y-btn" data-action="contrast-white">High contrast (white)</button> | ||
<button type="button" class="a11y-btn" data-action="contrast-reset">Contrast reset</button> | |||
</div> | </div> | ||
<div class="a11y-row" role="group" aria-label=" | <div class="a11y-row" role="group" aria-label="Large cursor"> | ||
<button type="button" class="a11y-btn" data-action="cursor" | <button type="button" class="a11y-btn" data-action="cursor-yellow">Cursor yellow</button> | ||
<button type="button" class="a11y-btn" data-action=" | <button type="button" class="a11y-btn" data-action="cursor-white">Cursor white</button> | ||
<button type="button" class="a11y-btn" data-action="cursor-reset">Cursor reset</button> | |||
</div> | |||
<div class="a11y-row" role="group" aria-label="Links and reading aid"> | |||
<button type="button" class="a11y-btn" data-action="underline">Underline links</button> | |||
<button type="button" class="a11y-btn" data-action="ruler">Reading ruler</button> | |||
</div> | </div> | ||
<div class="a11y-row" role="group" aria-label="Reset all"> | <div class="a11y-row" role="group" aria-label="Reset all"> | ||
<button type="button" class="a11y-btn" data-action="reset | <button type="button" class="a11y-btn" data-action="reset">Reset all</button> | ||
</div> | </div> | ||
| Line 399: | Line 354: | ||
<div class="a11y-hint"> | <div class="a11y-hint"> | ||
Settings are saved on this device. | Settings are saved on this device. When Reading ruler is on, use <strong>↑/↓</strong> to move it and <strong>Esc</strong> to turn it off. | ||
</div> | </div> | ||
| Line 408: | Line 362: | ||
liveEl = panel.querySelector("#a11y-live"); | liveEl = panel.querySelector("#a11y-live"); | ||
statusEl = panel.querySelector("#a11y-status"); | statusEl = panel.querySelector("#a11y-status"); | ||
updateStatusLine(); | updateStatusLine(); | ||
function openPanel() { | function openPanel() { | ||
| Line 417: | Line 371: | ||
if (firstBtn) firstBtn.focus(); | if (firstBtn) firstBtn.focus(); | ||
} | } | ||
function closePanel(opts = {}) { | function closePanel(opts = {}) { | ||
const { returnFocusToToggle = true } = opts; | const { returnFocusToToggle = true } = opts; | ||
panel.hidden = true; | panel.hidden = true; | ||
toggle.setAttribute("aria-expanded", "false"); | toggle.setAttribute("aria-expanded", "false"); | ||
if (returnFocusToToggle) toggle.focus(); | |||
if (returnFocusToToggle) | |||
} | } | ||
toggle.addEventListener("click", () => { | toggle.addEventListener("click", () => { | ||
if (panel.hidden) openPanel(); else closePanel({ returnFocusToToggle: false }); | if (panel.hidden) openPanel(); | ||
else closePanel({ returnFocusToToggle: false }); | |||
}); | }); | ||
| Line 457: | Line 394: | ||
if (e.key === "Escape") { | if (e.key === "Escape") { | ||
e.preventDefault(); | e.preventDefault(); | ||
closePanel({ returnFocusToToggle: false }); | |||
} | } | ||
}); | }); | ||
| Line 473: | Line 410: | ||
const active = document.activeElement; | const active = document.activeElement; | ||
if (e.shiftKey && active === first) { | if (e.shiftKey && active === first) { | ||
e.preventDefault(); | e.preventDefault(); | ||
closePanel({ returnFocusToToggle: true }); | closePanel({ returnFocusToToggle: true }); | ||
return; | return; | ||
} | } | ||
if (!e.shiftKey && active === last) { | if (!e.shiftKey && active === last) { | ||
e.preventDefault(); | e.preventDefault(); | ||
closePanel({ returnFocusToToggle: false }); | closePanel({ returnFocusToToggle: false }); | ||
const all = Array.from(document.querySelectorAll( | const all = Array.from(document.querySelectorAll( | ||
'a[href],button:not([disabled]),input:not([disabled]),select:not([disabled]),textarea:not([disabled]),[tabindex]:not([tabindex="-1"])' | 'a[href],button:not([disabled]),input:not([disabled]),select:not([disabled]),textarea:not([disabled]),[tabindex]:not([tabindex="-1"])' | ||
| Line 494: | Line 426: | ||
return style.display !== "none" && style.visibility !== "hidden"; | return style.display !== "none" && style.visibility !== "hidden"; | ||
}); | }); | ||
let next = null; | let next = null; | ||
for (const el of all) { | for (const el of all) { | ||
if (panel.contains(el) || toggle.contains(el)) continue; | if (panel.contains(el) || toggle.contains(el)) continue; | ||
if (panel.compareDocumentPosition(el) & Node.DOCUMENT_POSITION_FOLLOWING) { | |||
if ( | |||
next = el; | next = el; | ||
break; | break; | ||
} | } | ||
} | } | ||
if (next) next.focus(); | if (next) next.focus(); | ||
return; | return; | ||
| Line 516: | Line 440: | ||
}); | }); | ||
// | // Close if focus moves outside (screen reader / keyboard) | ||
document.addEventListener("focusin", (e) => { | document.addEventListener("focusin", (e) => { | ||
if (panel.hidden) return; | if (panel.hidden) return; | ||
const target = e.target; | const target = e.target; | ||
if (!target) return; | if (!target) return; | ||
if (panel.contains(target) || toggle.contains(target)) return; | if (panel.contains(target) || toggle.contains(target)) return; | ||
closePanel({ returnFocusToToggle: false }); | closePanel({ returnFocusToToggle: false }); | ||
}, true); | }, true); | ||
| Line 548: | Line 466: | ||
}; | }; | ||
setPressed(" | // Font | ||
setPressed(" | setPressed("font-reset", STATE.fontPct !== 100); | ||
setPressed(" | setPressed("font-plus", false); | ||
setPressed("font-minus", false); | |||
// Spacing | |||
setPressed("spacing-plus", STATE.spacing === "plus"); | setPressed("spacing-plus", STATE.spacing === "plus"); | ||
setPressed("spacing-minus", STATE.spacing === "minus"); | setPressed("spacing-minus", STATE.spacing === "minus"); | ||
setPressed("spacing-reset", | setPressed("spacing-reset", STATE.spacing === "off"); | ||
setPressed(" | // Contrast scheme | ||
setPressed(" | setPressed("contrast-yellow", STATE.contrastMode === "yellow"); | ||
setPressed(" | setPressed("contrast-white", STATE.contrastMode === "white"); | ||
setPressed("contrast-reset", STATE.contrastMode === "off"); | |||
// Cursor scheme | |||
setPressed("cursor-yellow", STATE.cursor === "yellow"); | |||
setPressed("cursor-white", STATE.cursor === "white"); | |||
setPressed("cursor-reset", STATE.cursor === "off"); | |||
// Toggles | |||
setPressed("underline", STATE.underline); | |||
setPressed("ruler", STATE.ruler); | |||
setPressed("reset", false); | setPressed("reset", false); | ||
} | } | ||
| Line 580: | Line 498: | ||
function handleAction(action) { | function handleAction(action) { | ||
switch (action) { | switch (action) { | ||
case "contrast": | // Contrast schemes | ||
STATE. | case "contrast-yellow": | ||
STATE.contrastMode = (STATE.contrastMode === "yellow") ? "off" : "yellow"; | |||
applyState(); | applyState(); | ||
announce( | announce(STATE.contrastMode === "yellow" ? "High contrast yellow on" : "High contrast off"); | ||
return; | return; | ||
case "contrast-white": | |||
STATE.contrastMode = (STATE.contrastMode === "white") ? "off" : "white"; | |||
applyState(); | |||
announce(STATE.contrastMode === "white" ? "High contrast white on" : "High contrast off"); | |||
return; | |||
case "contrast-reset": | |||
STATE.contrastMode = "off"; | |||
applyState(); | |||
announce("High contrast off"); | |||
return; | |||
// Underline + Ruler toggles | |||
case "underline": | case "underline": | ||
STATE.underline = !STATE.underline; | STATE.underline = !STATE.underline; | ||
| Line 592: | Line 524: | ||
return; | return; | ||
case "ruler": | |||
STATE.ruler = !STATE.ruler; | |||
if (STATE.ruler && !rulerY) rulerY = Math.floor(window.innerHeight / 2); | |||
applyState(); | |||
announce(`Reading ruler ${STATE.ruler ? "on" : "off"}`); | |||
return; | |||
// Spacing | |||
case "spacing-plus": | case "spacing-plus": | ||
STATE.spacing = (STATE.spacing === "plus") ? "off" : "plus"; | STATE.spacing = (STATE.spacing === "plus") ? "off" : "plus"; | ||
| Line 610: | Line 550: | ||
return; | return; | ||
case "cursor": | // Cursor schemes | ||
STATE.cursor = (STATE.cursor === " | case "cursor-yellow": | ||
STATE.cursor = (STATE.cursor === "yellow") ? "off" : "yellow"; | |||
applyState(); | applyState(); | ||
announce( | announce(STATE.cursor === "yellow" ? "Large cursor yellow" : "Large cursor off"); | ||
return; | return; | ||
case " | case "cursor-white": | ||
STATE. | STATE.cursor = (STATE.cursor === "white") ? "off" : "white"; | ||
applyState(); | |||
announce(STATE.cursor === "white" ? "Large cursor white" : "Large cursor off"); | |||
return; | |||
case "cursor-reset": | |||
STATE.cursor = "off"; | |||
applyState(); | applyState(); | ||
announce( | announce("Large cursor off"); | ||
return; | return; | ||
// Text size | |||
case "font-plus": | case "font-plus": | ||
STATE.fontPct = nextFontPct(STATE.fontPct, +1); | STATE.fontPct = nextFontPct(STATE.fontPct, +1); | ||
| Line 645: | Line 588: | ||
return; | return; | ||
// Reset all | |||
case "reset": | case "reset": | ||
clearStoredState(); | clearStoredState(); | ||
STATE. | STATE.contrastMode = "off"; | ||
STATE.underline = false; | STATE.underline = false; | ||
STATE.spacing = "off"; | STATE.spacing = "off"; | ||
| Line 667: | Line 611: | ||
if (!saved) return; | if (!saved) return; | ||
STATE. | const cm = saved.contrastMode; | ||
STATE.contrastMode = (cm === "yellow" || cm === "white") ? cm : "off"; | |||
STATE.underline = !!saved.underline; | STATE.underline = !!saved.underline; | ||
| Line 687: | Line 633: | ||
function init() { | function init() { | ||
ensureUi(); | ensureUi(); | ||
loadState(); | loadState(); | ||
applyState(); | applyState(); | ||
} | } | ||
if (document.readyState === "loading") { | if (document.readyState === "loading") { | ||
Latest revision as of 08:33, 29 January 2026
/*
⚠️ Do not reorder rules below this line — contrast overrides depend on cascade order.
*/
(() => {
"use strict";
// =========================================================
// Storage policy:
// - localStorage first
// - cookie fallback only if localStorage unavailable
// =========================================================
const LS_KEY = "bhc_a11y_state_v2"; // bumped because state model changed
const COOKIE_PREFIX = "a11y_";
function canUseLocalStorage() {
try {
const k = "__a11y_test__";
window.localStorage.setItem(k, "1");
window.localStorage.removeItem(k);
return true;
} catch (e) {
return false;
}
}
const HAS_LS = canUseLocalStorage();
// Cookie helpers (fallback)
function setCookie(name, value, days = 365) {
const exp = new Date();
exp.setTime(exp.getTime() + days * 24 * 60 * 60 * 1000);
document.cookie =
`${encodeURIComponent(COOKIE_PREFIX + name)}=${encodeURIComponent(String(value))}; ` +
`expires=${exp.toUTCString()}; path=/; SameSite=Lax`;
}
function getCookie(name, fallback = "") {
const key = encodeURIComponent(COOKIE_PREFIX + name) + "=";
const parts = document.cookie.split(";").map(s => s.trim());
for (const p of parts) {
if (p.startsWith(key)) return decodeURIComponent(p.substring(key.length));
}
return fallback;
}
function delCookie(name) {
document.cookie =
`${encodeURIComponent(COOKIE_PREFIX + name)}=; ` +
`expires=Thu, 01 Jan 1970 00:00:00 GMT; path=/; SameSite=Lax`;
}
// Unified read/write
function saveStateToStorage(stateObj) {
if (HAS_LS) {
try {
window.localStorage.setItem(LS_KEY, JSON.stringify(stateObj));
return;
} catch (e) {
// fall through to cookies
}
}
// cookie fallback (per-key)
setCookie("contrastMode", stateObj.contrastMode); // off|yellow|white
setCookie("underline", stateObj.underline ? "1" : "0");
setCookie("spacing", stateObj.spacing);
setCookie("cursor", stateObj.cursor); // off|white|yellow
setCookie("ruler", stateObj.ruler ? "1" : "0");
setCookie("fontPct", String(stateObj.fontPct));
}
function loadStateFromStorage() {
if (HAS_LS) {
try {
const raw = window.localStorage.getItem(LS_KEY);
if (!raw) return null;
const obj = JSON.parse(raw);
return (obj && typeof obj === "object") ? obj : null;
} catch (e) {
// fall through to cookies
}
}
// cookie fallback (legacy / no-LS)
return {
contrastMode: getCookie("contrastMode", "off"),
underline: getCookie("underline", "0") === "1",
spacing: getCookie("spacing", "off"),
cursor: getCookie("cursor", "off"),
ruler: getCookie("ruler", "0") === "1",
fontPct: parseInt(getCookie("fontPct", "100"), 10)
};
}
function clearStoredState() {
if (HAS_LS) {
try { window.localStorage.removeItem(LS_KEY); } catch (e) {}
return;
}
["contrastMode","underline","spacing","cursor","ruler","fontPct"].forEach(delCookie);
}
// =========================================================
// State + announcements
// =========================================================
const STATE = {
contrastMode: "off", // off | yellow | white
underline: false,
spacing: "off", // off | plus | minus
cursor: "off", // off | white | yellow
ruler: false,
fontPct: 100 // 90..175
};
const FONT_STEPS = [90, 102, 114, 126, 138, 150, 162, 175];
let liveEl = null;
let statusEl = null;
let rulerEl = null;
let rulerY = 0;
function announce(msg) {
if (!liveEl) return;
liveEl.textContent = "";
setTimeout(() => { liveEl.textContent = msg; }, 10);
}
function clamp(n, min, max) { return Math.max(min, Math.min(max, n)); }
function nextFontPct(current, dir) {
const cur = clamp(current, FONT_STEPS[0], FONT_STEPS[FONT_STEPS.length - 1]);
let idx = 0;
for (let i = 0; i < FONT_STEPS.length; i++) {
if (FONT_STEPS[i] >= cur) { idx = i; break; }
}
if (dir > 0) return FONT_STEPS[Math.min(FONT_STEPS.length - 1, idx + 1)];
if (FONT_STEPS[idx] === cur) return FONT_STEPS[Math.max(0, idx - 1)];
return FONT_STEPS[Math.max(0, idx - 1)];
}
// =========================================================
// Status line (display-only)
// =========================================================
function updateStatusLine() {
if (!statusEl) return;
const contrast =
STATE.contrastMode === "yellow" ? "Contrast: Yellow/Black" :
STATE.contrastMode === "white" ? "Contrast: White/Black" :
"Contrast: Off";
const text = `Text ${STATE.fontPct}%`;
const underline = STATE.underline ? "Links: Underlined" : "Links: Normal";
const spacing =
STATE.spacing === "plus" ? "Spacing +" :
STATE.spacing === "minus" ? "Spacing −" :
"Spacing: Normal";
const cursor =
STATE.cursor === "yellow" ? "Cursor: Yellow" :
STATE.cursor === "white" ? "Cursor: White" :
"Cursor: Off";
const ruler = STATE.ruler ? "Ruler: On" : "Ruler: Off";
statusEl.textContent = `Status: ${contrast} · ${text} · ${underline} · ${spacing} · ${cursor} · ${ruler}`;
}
// =========================================================
// Apply state to DOM
// =========================================================
function applyState() {
// Contrast scheme classes
const contrastOn = STATE.contrastMode !== "off";
document.body.classList.toggle("a11y-contrast", contrastOn);
document.body.classList.toggle("a11y-contrast-yellow", STATE.contrastMode === "yellow");
document.body.classList.toggle("a11y-contrast-white", STATE.contrastMode === "white");
// Other features
document.body.classList.toggle("a11y-underline-links", !!STATE.underline);
document.body.classList.toggle("a11y-spacing-plus", STATE.spacing === "plus");
document.body.classList.toggle("a11y-spacing-minus", STATE.spacing === "minus");
document.body.classList.toggle("a11y-cursor-white", STATE.cursor === "white");
document.body.classList.toggle("a11y-cursor-yellow", STATE.cursor === "yellow");
document.documentElement.style.fontSize = `${STATE.fontPct}%`;
ensureRuler();
if (STATE.ruler) {
rulerEl.hidden = false;
setRulerY(rulerY || Math.floor(window.innerHeight / 2));
enableRulerListeners();
} else {
if (rulerEl) rulerEl.hidden = true;
disableRulerListeners();
}
updateUiButtons();
updateStatusLine();
saveStateToStorage(STATE);
}
// =========================================================
// Ruler support
// =========================================================
function ensureRuler() {
if (rulerEl) return;
rulerEl = document.getElementById("a11y-ruler");
if (!rulerEl) {
rulerEl = document.createElement("div");
rulerEl.id = "a11y-ruler";
rulerEl.hidden = true;
document.body.appendChild(rulerEl);
}
}
function setRulerY(y) {
rulerY = clamp(y, 0, Math.max(0, window.innerHeight - 1));
if (rulerEl) rulerEl.style.transform = `translateY(${rulerY}px)`;
}
function onMouseMove(e) {
if (!STATE.ruler) return;
setRulerY(e.clientY - 11);
}
function onKeyDown(e) {
if (!STATE.ruler) return;
if (e.key === "ArrowDown") {
e.preventDefault();
setRulerY(rulerY + 10);
announce("Reading ruler moved down");
} else if (e.key === "ArrowUp") {
e.preventDefault();
setRulerY(rulerY - 10);
announce("Reading ruler moved up");
} else if (e.key === "Escape") {
e.preventDefault();
STATE.ruler = false;
applyState();
announce("Reading ruler off");
}
}
// Named resize handler so we can remove it cleanly (prevents handler buildup)
function onResize() {
setRulerY(rulerY);
}
let rulerListenersOn = false;
function enableRulerListeners() {
if (rulerListenersOn) return;
rulerListenersOn = true;
window.addEventListener("mousemove", onMouseMove, { passive: true });
window.addEventListener("keydown", onKeyDown);
window.addEventListener("resize", onResize, { passive: true });
}
function disableRulerListeners() {
if (!rulerListenersOn) return;
rulerListenersOn = false;
window.removeEventListener("mousemove", onMouseMove);
window.removeEventListener("keydown", onKeyDown);
window.removeEventListener("resize", onResize);
}
function getFocusableInside(container) {
if (!container) return [];
const selectors = [
'a[href]',
'button:not([disabled])',
'input:not([disabled])',
'select:not([disabled])',
'textarea:not([disabled])',
'[tabindex]:not([tabindex="-1"])'
].join(',');
return Array.from(container.querySelectorAll(selectors)).filter(el => {
if (el.hidden) return false;
const style = window.getComputedStyle(el);
return style.display !== "none" && style.visibility !== "hidden";
});
}
// =========================================================
// UI creation
// =========================================================
function ensureUi() {
if (document.getElementById("a11y-toggle")) return;
const helpHref = (window.mw && mw.util && mw.util.getUrl)
? mw.util.getUrl("Accessibility")
: "/index.php/Accessibility";
const toggle = document.createElement("button");
toggle.type = "button";
toggle.id = "a11y-toggle";
toggle.className = "a11y-toggle";
toggle.setAttribute("aria-label", "Accessibility options");
toggle.setAttribute("aria-haspopup", "dialog");
toggle.setAttribute("aria-expanded", "false");
toggle.innerHTML = `
<svg viewBox="0 0 24 24" aria-hidden="true" focusable="false">
<path d="M12 2a2 2 0 1 0 .001 4.001A2 2 0 0 0 12 2zm9 7h-6.5c-.7 0-1.3.4-1.6 1L12 12l-.9-2c-.3-.6-.9-1-1.6-1H3a1 1 0 0 0 0 2h5.3l1.3 2.9-1.7 6.2a1 1 0 0 0 1.9.6l1.2-4.3 1.2 4.3a1 1 0 0 0 1.9-.6l-1.7-6.2 1.3-2.9H21a1 1 0 0 0 0-2z"/>
</svg>
`;
document.body.appendChild(toggle);
const panel = document.createElement("div");
panel.id = "a11y-panel";
panel.className = "a11y-panel";
panel.setAttribute("role", "dialog");
panel.setAttribute("aria-label", "Accessibility options");
panel.hidden = true;
// Layout per your request
panel.innerHTML = `
<div class="a11y-head">
<h2>Accessibility</h2>
<a class="a11y-help" href="${helpHref}" aria-label="Accessibility help">?</a>
</div>
<div class="a11y-row" role="group" aria-label="Text size">
<button type="button" class="a11y-btn" data-action="font-minus" aria-label="Decrease text size">A−</button>
<button type="button" class="a11y-btn" data-action="font-plus" aria-label="Increase text size">A+</button>
<button type="button" class="a11y-btn" data-action="font-reset" aria-label="Reset text size">Reset</button>
</div>
<div class="a11y-row" role="group" aria-label="Text spacing">
<button type="button" class="a11y-btn" data-action="spacing-minus">Spacing−</button>
<button type="button" class="a11y-btn" data-action="spacing-plus">Spacing+</button>
<button type="button" class="a11y-btn" data-action="spacing-reset">Spacing reset</button>
</div>
<div class="a11y-row" role="group" aria-label="High contrast">
<button type="button" class="a11y-btn" data-action="contrast-yellow">High contrast (yellow)</button>
<button type="button" class="a11y-btn" data-action="contrast-white">High contrast (white)</button>
<button type="button" class="a11y-btn" data-action="contrast-reset">Contrast reset</button>
</div>
<div class="a11y-row" role="group" aria-label="Large cursor">
<button type="button" class="a11y-btn" data-action="cursor-yellow">Cursor yellow</button>
<button type="button" class="a11y-btn" data-action="cursor-white">Cursor white</button>
<button type="button" class="a11y-btn" data-action="cursor-reset">Cursor reset</button>
</div>
<div class="a11y-row" role="group" aria-label="Links and reading aid">
<button type="button" class="a11y-btn" data-action="underline">Underline links</button>
<button type="button" class="a11y-btn" data-action="ruler">Reading ruler</button>
</div>
<div class="a11y-row" role="group" aria-label="Reset all">
<button type="button" class="a11y-btn" data-action="reset">Reset all</button>
</div>
<div id="a11y-status" class="a11y-status" aria-live="off"></div>
<div class="a11y-hint">
Settings are saved on this device. When Reading ruler is on, use <strong>↑/↓</strong> to move it and <strong>Esc</strong> to turn it off.
</div>
<div id="a11y-live" class="a11y-sr" aria-live="polite" aria-atomic="true"></div>
`;
document.body.appendChild(panel);
liveEl = panel.querySelector("#a11y-live");
statusEl = panel.querySelector("#a11y-status");
updateStatusLine();
function openPanel() {
panel.hidden = false;
toggle.setAttribute("aria-expanded", "true");
const firstBtn = panel.querySelector("button.a11y-btn");
if (firstBtn) firstBtn.focus();
}
function closePanel(opts = {}) {
const { returnFocusToToggle = true } = opts;
panel.hidden = true;
toggle.setAttribute("aria-expanded", "false");
if (returnFocusToToggle) toggle.focus();
}
toggle.addEventListener("click", () => {
if (panel.hidden) openPanel();
else closePanel({ returnFocusToToggle: false });
});
document.addEventListener("click", (e) => {
if (panel.hidden) return;
if (panel.contains(e.target) || toggle.contains(e.target)) return;
closePanel({ returnFocusToToggle: false });
});
document.addEventListener("keydown", (e) => {
if (panel.hidden) return;
if (e.key === "Escape") {
e.preventDefault();
closePanel({ returnFocusToToggle: false });
}
});
// STRICT 1A (keyboard): close immediately when tabbing out of the panel
panel.addEventListener("keydown", (e) => {
if (panel.hidden) return;
if (e.key !== "Tab") return;
const focusables = getFocusableInside(panel);
if (!focusables.length) return;
const first = focusables[0];
const last = focusables[focusables.length - 1];
const active = document.activeElement;
if (e.shiftKey && active === first) {
e.preventDefault();
closePanel({ returnFocusToToggle: true });
return;
}
if (!e.shiftKey && active === last) {
e.preventDefault();
closePanel({ returnFocusToToggle: false });
const all = Array.from(document.querySelectorAll(
'a[href],button:not([disabled]),input:not([disabled]),select:not([disabled]),textarea:not([disabled]),[tabindex]:not([tabindex="-1"])'
)).filter(el => {
const style = window.getComputedStyle(el);
return style.display !== "none" && style.visibility !== "hidden";
});
let next = null;
for (const el of all) {
if (panel.contains(el) || toggle.contains(el)) continue;
if (panel.compareDocumentPosition(el) & Node.DOCUMENT_POSITION_FOLLOWING) {
next = el;
break;
}
}
if (next) next.focus();
return;
}
});
// Close if focus moves outside (screen reader / keyboard)
document.addEventListener("focusin", (e) => {
if (panel.hidden) return;
const target = e.target;
if (!target) return;
if (panel.contains(target) || toggle.contains(target)) return;
closePanel({ returnFocusToToggle: false });
}, true);
panel.addEventListener("click", (e) => {
const btn = e.target.closest("button[data-action]");
if (!btn) return;
const action = btn.getAttribute("data-action") || "";
handleAction(action);
});
}
function updateUiButtons() {
const panel = document.getElementById("a11y-panel");
if (!panel) return;
const setPressed = (action, pressed) => {
const b = panel.querySelector(`button[data-action="${action}"]`);
if (b) b.setAttribute("aria-pressed", pressed ? "true" : "false");
};
// Font
setPressed("font-reset", STATE.fontPct !== 100);
setPressed("font-plus", false);
setPressed("font-minus", false);
// Spacing
setPressed("spacing-plus", STATE.spacing === "plus");
setPressed("spacing-minus", STATE.spacing === "minus");
setPressed("spacing-reset", STATE.spacing === "off");
// Contrast scheme
setPressed("contrast-yellow", STATE.contrastMode === "yellow");
setPressed("contrast-white", STATE.contrastMode === "white");
setPressed("contrast-reset", STATE.contrastMode === "off");
// Cursor scheme
setPressed("cursor-yellow", STATE.cursor === "yellow");
setPressed("cursor-white", STATE.cursor === "white");
setPressed("cursor-reset", STATE.cursor === "off");
// Toggles
setPressed("underline", STATE.underline);
setPressed("ruler", STATE.ruler);
setPressed("reset", false);
}
// =========================================================
// Actions
// =========================================================
function handleAction(action) {
switch (action) {
// Contrast schemes
case "contrast-yellow":
STATE.contrastMode = (STATE.contrastMode === "yellow") ? "off" : "yellow";
applyState();
announce(STATE.contrastMode === "yellow" ? "High contrast yellow on" : "High contrast off");
return;
case "contrast-white":
STATE.contrastMode = (STATE.contrastMode === "white") ? "off" : "white";
applyState();
announce(STATE.contrastMode === "white" ? "High contrast white on" : "High contrast off");
return;
case "contrast-reset":
STATE.contrastMode = "off";
applyState();
announce("High contrast off");
return;
// Underline + Ruler toggles
case "underline":
STATE.underline = !STATE.underline;
applyState();
announce(`Underline links ${STATE.underline ? "on" : "off"}`);
return;
case "ruler":
STATE.ruler = !STATE.ruler;
if (STATE.ruler && !rulerY) rulerY = Math.floor(window.innerHeight / 2);
applyState();
announce(`Reading ruler ${STATE.ruler ? "on" : "off"}`);
return;
// Spacing
case "spacing-plus":
STATE.spacing = (STATE.spacing === "plus") ? "off" : "plus";
applyState();
announce(STATE.spacing === "plus" ? "Text spacing increased" : "Text spacing normal");
return;
case "spacing-minus":
STATE.spacing = (STATE.spacing === "minus") ? "off" : "minus";
applyState();
announce(STATE.spacing === "minus" ? "Text spacing decreased" : "Text spacing normal");
return;
case "spacing-reset":
STATE.spacing = "off";
applyState();
announce("Text spacing reset");
return;
// Cursor schemes
case "cursor-yellow":
STATE.cursor = (STATE.cursor === "yellow") ? "off" : "yellow";
applyState();
announce(STATE.cursor === "yellow" ? "Large cursor yellow" : "Large cursor off");
return;
case "cursor-white":
STATE.cursor = (STATE.cursor === "white") ? "off" : "white";
applyState();
announce(STATE.cursor === "white" ? "Large cursor white" : "Large cursor off");
return;
case "cursor-reset":
STATE.cursor = "off";
applyState();
announce("Large cursor off");
return;
// Text size
case "font-plus":
STATE.fontPct = nextFontPct(STATE.fontPct, +1);
applyState();
announce(`Text size ${STATE.fontPct}%`);
return;
case "font-minus":
STATE.fontPct = nextFontPct(STATE.fontPct, -1);
applyState();
announce(`Text size ${STATE.fontPct}%`);
return;
case "font-reset":
STATE.fontPct = 100;
applyState();
announce("Text size reset");
return;
// Reset all
case "reset":
clearStoredState();
STATE.contrastMode = "off";
STATE.underline = false;
STATE.spacing = "off";
STATE.cursor = "off";
STATE.ruler = false;
STATE.fontPct = 100;
rulerY = 0;
applyState();
announce("Accessibility settings reset");
return;
}
}
// =========================================================
// Load saved settings
// =========================================================
function loadState() {
const saved = loadStateFromStorage();
if (!saved) return;
const cm = saved.contrastMode;
STATE.contrastMode = (cm === "yellow" || cm === "white") ? cm : "off";
STATE.underline = !!saved.underline;
const spacing = saved.spacing;
STATE.spacing = (spacing === "plus" || spacing === "minus") ? spacing : "off";
const cursor = saved.cursor;
STATE.cursor = (cursor === "white" || cursor === "yellow") ? cursor : "off";
STATE.ruler = !!saved.ruler;
const fp = parseInt(saved.fontPct, 10);
STATE.fontPct = isNaN(fp) ? 100 : clamp(fp, FONT_STEPS[0], FONT_STEPS[FONT_STEPS.length - 1]);
}
// =========================================================
// Init
// =========================================================
function init() {
ensureUi();
loadState();
applyState();
}
if (document.readyState === "loading") {
document.addEventListener("DOMContentLoaded", init);
} else {
init();
}
})();
