accessify-widget 0.2.8 → 0.2.10

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/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "accessify-widget",
3
- "version": "0.2.8",
3
+ "version": "0.2.10",
4
4
  "description": "Accessify-Widget accessibility widget for any website",
5
5
  "license": "MIT",
6
6
  "homepage": "https://github.com/maddesv1-ctrl/accessify-widget",
@@ -1,533 +0,0 @@
1
- function createAnimationStopModule() {
2
- let enabled = false;
3
- const STYLE_ID = "accessify-animation-stop";
4
- const STORAGE_KEY = "accessify-animation-stop";
5
- let pausedVideos = [];
6
- let pausedAnimations = [];
7
- let originalAnimate = null;
8
- let originalRAF = null;
9
- let originalMatchMedia = null;
10
- let mutationObserver = null;
11
- let rafLoopTracker = /* @__PURE__ */ new Map();
12
- let blockedRAFs = /* @__PURE__ */ new Set();
13
- let gifOriginals = /* @__PURE__ */ new Map();
14
- let styleObserver = null;
15
- let delayedTimers = [];
16
- let originalIntersectionObserver = null;
17
- function getStyles() {
18
- return `
19
- /* accessify animation stop - WCAG 2.3.1 compliance */
20
- *, *::before, *::after {
21
- animation-duration: 0.001s !important;
22
- animation-delay: 0s !important;
23
- animation-iteration-count: 1 !important;
24
- animation-fill-mode: forwards !important;
25
- transition-duration: 0.001s !important;
26
- transition-delay: 0s !important;
27
- scroll-behavior: auto !important;
28
- }
29
-
30
- /* Framer appear elements — force visible only (do NOT touch transform — it breaks layout) */
31
- [data-framer-appear-id] {
32
- opacity: 1 !important;
33
- }
34
-
35
- marquee { -moz-binding: none; }
36
-
37
- /* Pause HTML5 video playback visually */
38
- video {
39
- animation-play-state: paused !important;
40
- }
41
- `;
42
- }
43
- function injectStyles() {
44
- let styleEl = document.getElementById(STYLE_ID);
45
- if (!styleEl) {
46
- styleEl = document.createElement("style");
47
- styleEl.id = STYLE_ID;
48
- document.head.appendChild(styleEl);
49
- }
50
- styleEl.textContent = getStyles();
51
- }
52
- function removeStyles() {
53
- document.getElementById(STYLE_ID)?.remove();
54
- }
55
- function finishAllWebAnimations() {
56
- try {
57
- const animations = document.getAnimations();
58
- for (const anim of animations) {
59
- if (anim.playState === "running" || anim.playState === "pending") {
60
- pausedAnimations.push(anim);
61
- try {
62
- anim.finish();
63
- } catch {
64
- try {
65
- anim.cancel();
66
- } catch {
67
- }
68
- }
69
- }
70
- }
71
- } catch {
72
- }
73
- }
74
- function overrideElementAnimate() {
75
- if (originalAnimate) return;
76
- originalAnimate = Element.prototype.animate;
77
- Element.prototype.animate = function(keyframes, options) {
78
- const anim = originalAnimate.call(this, keyframes, options);
79
- if (enabled) {
80
- pausedAnimations.push(anim);
81
- try {
82
- anim.finish();
83
- } catch {
84
- try {
85
- anim.cancel();
86
- } catch {
87
- }
88
- }
89
- }
90
- return anim;
91
- };
92
- }
93
- function restoreElementAnimate() {
94
- if (originalAnimate) {
95
- Element.prototype.animate = originalAnimate;
96
- originalAnimate = null;
97
- }
98
- }
99
- function resumePausedAnimations() {
100
- pausedAnimations = [];
101
- }
102
- function overrideRAF() {
103
- if (originalRAF) return;
104
- originalRAF = window.requestAnimationFrame;
105
- const origCancel = window.cancelAnimationFrame;
106
- window.requestAnimationFrame = function(callback) {
107
- if (!enabled) return originalRAF.call(window, callback);
108
- const key = callback.toString().slice(0, 200);
109
- const now = performance.now();
110
- const entry = rafLoopTracker.get(key);
111
- if (entry) {
112
- entry.count++;
113
- if (now - entry.lastTime < 100 && entry.count > 3) {
114
- const id = originalRAF.call(window, () => {
115
- });
116
- blockedRAFs.add(id);
117
- return id;
118
- }
119
- entry.lastTime = now;
120
- } else {
121
- rafLoopTracker.set(key, { count: 1, lastTime: now });
122
- }
123
- return originalRAF.call(window, callback);
124
- };
125
- window.cancelAnimationFrame = function(id) {
126
- blockedRAFs.delete(id);
127
- return origCancel.call(window, id);
128
- };
129
- }
130
- function restoreRAF() {
131
- if (originalRAF) {
132
- window.requestAnimationFrame = originalRAF;
133
- originalRAF = null;
134
- }
135
- rafLoopTracker.clear();
136
- blockedRAFs.clear();
137
- }
138
- function setupMutationObserver() {
139
- if (mutationObserver) return;
140
- mutationObserver = new MutationObserver((mutations) => {
141
- if (!enabled) return;
142
- for (const mutation of mutations) {
143
- for (const node of mutation.addedNodes) {
144
- if (!(node instanceof HTMLElement)) continue;
145
- try {
146
- const anims = node.getAnimations({ subtree: true });
147
- for (const anim of anims) {
148
- if (anim.playState === "running" || anim.playState === "pending") {
149
- pausedAnimations.push(anim);
150
- try {
151
- anim.finish();
152
- } catch {
153
- try {
154
- anim.cancel();
155
- } catch {
156
- }
157
- }
158
- }
159
- }
160
- } catch {
161
- }
162
- if (node.tagName === "VIDEO") {
163
- const video = node;
164
- if (!video.paused) {
165
- video.pause();
166
- pausedVideos.push(video);
167
- }
168
- }
169
- node.querySelectorAll?.("video")?.forEach((video) => {
170
- if (!video.paused) {
171
- video.pause();
172
- pausedVideos.push(video);
173
- }
174
- });
175
- if (node.tagName === "IMG") {
176
- freezeGif(node);
177
- }
178
- node.querySelectorAll?.("img")?.forEach((img) => freezeGif(img));
179
- forceElementVisibleSingle(node);
180
- node.querySelectorAll?.("[style]")?.forEach((el) => {
181
- forceElementVisibleSingle(el);
182
- });
183
- }
184
- }
185
- });
186
- mutationObserver.observe(document.body, {
187
- childList: true,
188
- subtree: true
189
- });
190
- }
191
- function disconnectMutationObserver() {
192
- mutationObserver?.disconnect();
193
- mutationObserver = null;
194
- }
195
- function emulateReducedMotion() {
196
- if (originalMatchMedia) return;
197
- originalMatchMedia = window.matchMedia;
198
- window.matchMedia = function(query) {
199
- const mql = originalMatchMedia.call(window, query);
200
- if (query.includes("prefers-reduced-motion")) {
201
- const proxy = Object.create(mql);
202
- Object.defineProperty(proxy, "matches", { get: () => enabled, configurable: true });
203
- Object.defineProperty(proxy, "media", { get: () => query, configurable: true });
204
- return proxy;
205
- }
206
- return mql;
207
- };
208
- try {
209
- const mql = originalMatchMedia.call(window, "(prefers-reduced-motion: reduce)");
210
- mql.dispatchEvent(new MediaQueryListEvent("change", { matches: true, media: mql.media }));
211
- } catch {
212
- }
213
- }
214
- function restoreMatchMedia() {
215
- if (originalMatchMedia) {
216
- window.matchMedia = originalMatchMedia;
217
- originalMatchMedia = null;
218
- try {
219
- const mql = window.matchMedia("(prefers-reduced-motion: reduce)");
220
- mql.dispatchEvent(new MediaQueryListEvent("change", { matches: mql.matches, media: mql.media }));
221
- } catch {
222
- }
223
- }
224
- }
225
- function freezeGif(img) {
226
- const src = img.src || img.getAttribute("src") || "";
227
- if (!src.toLowerCase().includes(".gif")) return;
228
- if (gifOriginals.has(img)) return;
229
- const freeze = () => {
230
- try {
231
- const canvas = document.createElement("canvas");
232
- canvas.width = img.naturalWidth || img.width;
233
- canvas.height = img.naturalHeight || img.height;
234
- if (canvas.width === 0 || canvas.height === 0) return;
235
- const ctx = canvas.getContext("2d");
236
- if (!ctx) return;
237
- ctx.drawImage(img, 0, 0);
238
- gifOriginals.set(img, src);
239
- img.src = canvas.toDataURL("image/png");
240
- } catch {
241
- }
242
- };
243
- if (img.complete && img.naturalWidth > 0) {
244
- freeze();
245
- } else {
246
- img.addEventListener("load", freeze, { once: true });
247
- }
248
- }
249
- function freezeAllGifs() {
250
- document.querySelectorAll("img").forEach(freezeGif);
251
- }
252
- function restoreGifs() {
253
- for (const [img, originalSrc] of gifOriginals) {
254
- try {
255
- img.src = originalSrc;
256
- } catch {
257
- }
258
- }
259
- gifOriginals.clear();
260
- }
261
- function overrideIntersectionObserver() {
262
- if (originalIntersectionObserver) return;
263
- originalIntersectionObserver = window.IntersectionObserver;
264
- const OrigIO = originalIntersectionObserver;
265
- window.IntersectionObserver = function(callback, options) {
266
- const wrappedCallback = (entries, observer2) => {
267
- if (!enabled) {
268
- callback(entries, observer2);
269
- return;
270
- }
271
- const faked = entries.map((entry) => {
272
- try {
273
- return new Proxy(entry, {
274
- get(target, prop) {
275
- if (prop === "isIntersecting") return true;
276
- if (prop === "intersectionRatio") return 1;
277
- const val = target[prop];
278
- return typeof val === "function" ? val.bind(target) : val;
279
- }
280
- });
281
- } catch {
282
- return entry;
283
- }
284
- });
285
- callback(faked, observer2);
286
- };
287
- const observer = new OrigIO(wrappedCallback, options);
288
- return observer;
289
- };
290
- window.IntersectionObserver.prototype = OrigIO.prototype;
291
- Object.defineProperty(window.IntersectionObserver, "name", { value: "IntersectionObserver" });
292
- }
293
- function restoreIntersectionObserver() {
294
- if (originalIntersectionObserver) {
295
- window.IntersectionObserver = originalIntersectionObserver;
296
- originalIntersectionObserver = null;
297
- }
298
- }
299
- const FIXED_ATTR = "data-accessify-anim-fixed";
300
- let fixedElements = [];
301
- function forceAllPageElementsVisible() {
302
- const allEls = document.querySelectorAll("[style]");
303
- for (const el of allEls) {
304
- if (el.hasAttribute(FIXED_ATTR)) continue;
305
- if (el.closest("#accessify-root")) continue;
306
- if (el.style.display === "none") continue;
307
- const inlineOpacity = el.style.opacity;
308
- if (inlineOpacity === "" || inlineOpacity === "1") continue;
309
- const val = parseFloat(inlineOpacity);
310
- if (!isNaN(val) && val < 0.9) {
311
- fixedElements.push({ el, origStyle: el.getAttribute("style") || "" });
312
- el.style.setProperty("opacity", "1", "important");
313
- el.setAttribute(FIXED_ATTR, "true");
314
- }
315
- }
316
- }
317
- function restoreFixedElements() {
318
- for (const { el, origStyle } of fixedElements) {
319
- try {
320
- el.setAttribute("style", origStyle);
321
- el.removeAttribute(FIXED_ATTR);
322
- } catch {
323
- }
324
- }
325
- fixedElements = [];
326
- }
327
- function forceElementVisibleSingle(el) {
328
- if (el.hasAttribute(FIXED_ATTR)) return;
329
- if (el.closest("#accessify-root")) return;
330
- const inlineOpacity = el.style.opacity;
331
- if (inlineOpacity === "" || inlineOpacity === "1") return;
332
- const val = parseFloat(inlineOpacity);
333
- if (!isNaN(val) && val < 0.9) {
334
- fixedElements.push({ el, origStyle: el.getAttribute("style") || "" });
335
- el.style.setProperty("opacity", "1", "important");
336
- el.setAttribute(FIXED_ATTR, "true");
337
- }
338
- }
339
- function setupStyleObserver() {
340
- if (styleObserver) return;
341
- let isOurChange = false;
342
- styleObserver = new MutationObserver((mutations) => {
343
- if (!enabled || isOurChange) return;
344
- isOurChange = true;
345
- for (const m of mutations) {
346
- if (m.type !== "attributes" || m.attributeName !== "style") continue;
347
- const el = m.target;
348
- if (el.closest("#accessify-root")) continue;
349
- const inlineOpacity = el.style.opacity;
350
- if (inlineOpacity !== "" && inlineOpacity !== "1") {
351
- const val = parseFloat(inlineOpacity);
352
- if (!isNaN(val) && val < 0.9) {
353
- if (!el.hasAttribute(FIXED_ATTR)) {
354
- fixedElements.push({ el, origStyle: m.oldValue || "" });
355
- }
356
- el.style.setProperty("opacity", "1", "important");
357
- el.setAttribute(FIXED_ATTR, "true");
358
- }
359
- }
360
- }
361
- isOurChange = false;
362
- });
363
- styleObserver.observe(document.body, {
364
- attributes: true,
365
- attributeFilter: ["style"],
366
- attributeOldValue: true,
367
- subtree: true
368
- });
369
- }
370
- function disconnectStyleObserver() {
371
- styleObserver?.disconnect();
372
- styleObserver = null;
373
- }
374
- function scheduleDelayedScans() {
375
- const delays = [300, 800, 1500, 3e3, 5e3];
376
- for (const delay of delays) {
377
- const timer = setTimeout(() => {
378
- if (!enabled) return;
379
- finishAllWebAnimations();
380
- forceAllPageElementsVisible();
381
- pauseAutoplayVideos();
382
- freezeAllGifs();
383
- pauseSVGAnimations();
384
- stopMarquees();
385
- }, delay);
386
- delayedTimers.push(timer);
387
- }
388
- }
389
- function clearDelayedScans() {
390
- for (const timer of delayedTimers) clearTimeout(timer);
391
- delayedTimers = [];
392
- }
393
- function stopMarquees() {
394
- document.querySelectorAll("marquee").forEach((el) => {
395
- try {
396
- el.stop();
397
- } catch {
398
- }
399
- });
400
- }
401
- function startMarquees() {
402
- document.querySelectorAll("marquee").forEach((el) => {
403
- try {
404
- el.start();
405
- } catch {
406
- }
407
- });
408
- }
409
- function pauseSVGAnimations() {
410
- document.querySelectorAll("svg").forEach((svg) => {
411
- try {
412
- if (typeof svg.pauseAnimations === "function") {
413
- svg.pauseAnimations();
414
- }
415
- } catch {
416
- }
417
- });
418
- }
419
- function resumeSVGAnimations() {
420
- document.querySelectorAll("svg").forEach((svg) => {
421
- try {
422
- if (typeof svg.unpauseAnimations === "function") {
423
- svg.unpauseAnimations();
424
- }
425
- } catch {
426
- }
427
- });
428
- }
429
- let originalVideoPlay = null;
430
- function pauseAutoplayVideos() {
431
- const videos = document.querySelectorAll("video");
432
- pausedVideos = [];
433
- videos.forEach((video) => {
434
- if (!video.paused) {
435
- video.pause();
436
- pausedVideos.push(video);
437
- }
438
- });
439
- videos.forEach((video) => {
440
- if (video.hasAttribute("autoplay")) {
441
- video.dataset.accessifyAutoplay = "true";
442
- video.removeAttribute("autoplay");
443
- }
444
- });
445
- if (!originalVideoPlay) {
446
- originalVideoPlay = HTMLVideoElement.prototype.play;
447
- HTMLVideoElement.prototype.play = function() {
448
- if (enabled) {
449
- return Promise.resolve();
450
- }
451
- return originalVideoPlay.call(this);
452
- };
453
- }
454
- }
455
- function resumePausedVideos() {
456
- if (originalVideoPlay) {
457
- HTMLVideoElement.prototype.play = originalVideoPlay;
458
- originalVideoPlay = null;
459
- }
460
- document.querySelectorAll("video[data-accessify-autoplay]").forEach((video) => {
461
- video.setAttribute("autoplay", "");
462
- delete video.dataset.accessifyAutoplay;
463
- });
464
- pausedVideos.forEach((video) => {
465
- try {
466
- video.play();
467
- } catch {
468
- }
469
- });
470
- pausedVideos = [];
471
- }
472
- function activate() {
473
- if (enabled) return;
474
- enabled = true;
475
- injectStyles();
476
- finishAllWebAnimations();
477
- overrideElementAnimate();
478
- overrideRAF();
479
- setupMutationObserver();
480
- emulateReducedMotion();
481
- freezeAllGifs();
482
- overrideIntersectionObserver();
483
- forceAllPageElementsVisible();
484
- stopMarquees();
485
- pauseSVGAnimations();
486
- setupStyleObserver();
487
- scheduleDelayedScans();
488
- pauseAutoplayVideos();
489
- localStorage.setItem(STORAGE_KEY, "true");
490
- }
491
- function deactivate() {
492
- enabled = false;
493
- clearDelayedScans();
494
- disconnectStyleObserver();
495
- restoreFixedElements();
496
- resumePausedVideos();
497
- resumeSVGAnimations();
498
- startMarquees();
499
- restoreIntersectionObserver();
500
- restoreGifs();
501
- restoreMatchMedia();
502
- disconnectMutationObserver();
503
- restoreRAF();
504
- restoreElementAnimate();
505
- resumePausedAnimations();
506
- removeStyles();
507
- localStorage.removeItem(STORAGE_KEY);
508
- }
509
- return {
510
- id: "animation-stop",
511
- name: () => "Stop Animations",
512
- description: "Pause all animations, transitions, and auto-playing videos (WCAG 2.3.1)",
513
- icon: "animation-stop",
514
- category: "visual",
515
- activate,
516
- deactivate,
517
- getState: () => ({
518
- id: "animation-stop",
519
- enabled
520
- }),
521
- setState: (state) => {
522
- if (state.enabled) {
523
- activate();
524
- } else {
525
- deactivate();
526
- }
527
- }
528
- };
529
- }
530
- export {
531
- createAnimationStopModule as default
532
- };
533
- //# sourceMappingURL=animation-stop-BhCnv5Nn.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"animation-stop-BhCnv5Nn.js","sources":["../src/features/animation-stop.ts"],"sourcesContent":["import type { FeatureModule, FeatureState } from '../types';\n\nexport default function createAnimationStopModule(): FeatureModule {\n let enabled = false;\n const STYLE_ID = 'accessify-animation-stop';\n const STORAGE_KEY = 'accessify-animation-stop';\n\n // --- State for clean restore ---\n let pausedVideos: HTMLVideoElement[] = [];\n let pausedAnimations: Animation[] = [];\n let originalAnimate: typeof Element.prototype.animate | null = null;\n let originalRAF: typeof window.requestAnimationFrame | null = null;\n let originalMatchMedia: typeof window.matchMedia | null = null;\n let mutationObserver: MutationObserver | null = null;\n let rafLoopTracker = new Map<string, { count: number; lastTime: number }>();\n let blockedRAFs = new Set<number>();\n let gifOriginals = new Map<HTMLImageElement, string>();\n let reducedMotionListeners: Array<{ mql: MediaQueryList; handler: () => void }> = [];\n let styleObserver: MutationObserver | null = null;\n let delayedTimers: ReturnType<typeof setTimeout>[] = [];\n // frozenInlineElements removed — replaced by fixedElements array in Layer 7\n let originalIntersectionObserver: typeof IntersectionObserver | null = null;\n\n // -------------------------------------------------------------------------\n // Layer 0: CSS overrides (existing — catches pure CSS animations)\n // -------------------------------------------------------------------------\n\n function getStyles(): string {\n return `\n /* accessify animation stop - WCAG 2.3.1 compliance */\n *, *::before, *::after {\n animation-duration: 0.001s !important;\n animation-delay: 0s !important;\n animation-iteration-count: 1 !important;\n animation-fill-mode: forwards !important;\n transition-duration: 0.001s !important;\n transition-delay: 0s !important;\n scroll-behavior: auto !important;\n }\n\n /* Framer appear elements — force visible only (do NOT touch transform — it breaks layout) */\n [data-framer-appear-id] {\n opacity: 1 !important;\n }\n\n marquee { -moz-binding: none; }\n\n /* Pause HTML5 video playback visually */\n video {\n animation-play-state: paused !important;\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 document.getElementById(STYLE_ID)?.remove();\n }\n\n // -------------------------------------------------------------------------\n // Layer 1: Web Animations API — FINISH all running animations instantly\n // Key insight: pause() freezes elements at START state (invisible).\n // finish() jumps to END state (visible). This is what we need.\n // -------------------------------------------------------------------------\n\n function finishAllWebAnimations() {\n try {\n const animations = document.getAnimations();\n for (const anim of animations) {\n if (anim.playState === 'running' || anim.playState === 'pending') {\n pausedAnimations.push(anim);\n try {\n // finish() jumps to end state — element becomes fully visible\n anim.finish();\n } catch {\n // finish() fails on infinite animations — cancel them instead\n // and the CSS !important rules will handle visibility\n try { anim.cancel(); } catch { /* ignore */ }\n }\n }\n }\n } catch { /* getAnimations not supported */ }\n }\n\n function overrideElementAnimate() {\n if (originalAnimate) return;\n originalAnimate = Element.prototype.animate;\n\n Element.prototype.animate = function (\n this: Element,\n keyframes: Keyframe[] | PropertyIndexedKeyframes | null,\n options?: number | KeyframeAnimationOptions,\n ): Animation {\n const anim = originalAnimate!.call(this, keyframes, options);\n if (enabled) {\n pausedAnimations.push(anim);\n try {\n anim.finish();\n } catch {\n try { anim.cancel(); } catch { /* ignore */ }\n }\n }\n return anim;\n };\n }\n\n function restoreElementAnimate() {\n if (originalAnimate) {\n Element.prototype.animate = originalAnimate;\n originalAnimate = null;\n }\n }\n\n function resumePausedAnimations() {\n // We finished/cancelled animations — nothing to resume.\n // Just clear the list.\n pausedAnimations = [];\n }\n\n // -------------------------------------------------------------------------\n // Layer 2: requestAnimationFrame loop detection\n // -------------------------------------------------------------------------\n\n function overrideRAF() {\n if (originalRAF) return;\n originalRAF = window.requestAnimationFrame;\n const origCancel = window.cancelAnimationFrame;\n\n window.requestAnimationFrame = function (callback: FrameRequestCallback): number {\n if (!enabled) return originalRAF!.call(window, callback);\n\n // Fingerprint callback to detect animation loops\n const key = callback.toString().slice(0, 200);\n const now = performance.now();\n const entry = rafLoopTracker.get(key);\n\n if (entry) {\n entry.count++;\n if (now - entry.lastTime < 100 && entry.count > 3) {\n // This callback is re-registering rapidly — animation loop detected\n const id = originalRAF!.call(window, () => {}); // no-op\n blockedRAFs.add(id);\n return id;\n }\n entry.lastTime = now;\n } else {\n rafLoopTracker.set(key, { count: 1, lastTime: now });\n }\n\n // Allow through but wrap to track re-registration\n return originalRAF!.call(window, callback);\n };\n\n window.cancelAnimationFrame = function (id: number) {\n blockedRAFs.delete(id);\n return origCancel.call(window, id);\n };\n }\n\n function restoreRAF() {\n if (originalRAF) {\n window.requestAnimationFrame = originalRAF;\n originalRAF = null;\n }\n rafLoopTracker.clear();\n blockedRAFs.clear();\n }\n\n // -------------------------------------------------------------------------\n // Layer 3: MutationObserver — catch dynamically added animated elements\n // -------------------------------------------------------------------------\n\n function setupMutationObserver() {\n if (mutationObserver) return;\n\n mutationObserver = new MutationObserver((mutations) => {\n if (!enabled) return;\n for (const mutation of mutations) {\n for (const node of mutation.addedNodes) {\n if (!(node instanceof HTMLElement)) continue;\n // Finish animations on newly added elements (jump to end state)\n try {\n const anims = node.getAnimations({ subtree: true });\n for (const anim of anims) {\n if (anim.playState === 'running' || anim.playState === 'pending') {\n pausedAnimations.push(anim);\n try { anim.finish(); } catch { try { anim.cancel(); } catch { /* ignore */ } }\n }\n }\n } catch { /* not supported */ }\n\n // Pause new videos\n if (node.tagName === 'VIDEO') {\n const video = node as HTMLVideoElement;\n if (!video.paused) {\n video.pause();\n pausedVideos.push(video);\n }\n }\n node.querySelectorAll?.('video')?.forEach((video) => {\n if (!video.paused) {\n video.pause();\n pausedVideos.push(video);\n }\n });\n\n // Freeze new GIFs\n if (node.tagName === 'IMG') {\n freezeGif(node as HTMLImageElement);\n }\n node.querySelectorAll?.('img')?.forEach((img) => freezeGif(img));\n\n // Force new elements visible (scroll-reveal, Framer appear, etc.)\n forceElementVisibleSingle(node);\n node.querySelectorAll?.('[style]')?.forEach((el) => {\n forceElementVisibleSingle(el as HTMLElement);\n });\n }\n }\n });\n\n mutationObserver.observe(document.body, {\n childList: true,\n subtree: true,\n });\n }\n\n function disconnectMutationObserver() {\n mutationObserver?.disconnect();\n mutationObserver = null;\n }\n\n // -------------------------------------------------------------------------\n // Layer 4: prefers-reduced-motion emulation\n // -------------------------------------------------------------------------\n\n function emulateReducedMotion() {\n if (originalMatchMedia) return;\n originalMatchMedia = window.matchMedia;\n\n window.matchMedia = function (query: string): MediaQueryList {\n const mql = originalMatchMedia!.call(window, query);\n\n if (query.includes('prefers-reduced-motion')) {\n // Return a proxy that always reports matches: true\n const proxy = Object.create(mql);\n Object.defineProperty(proxy, 'matches', { get: () => enabled, configurable: true });\n Object.defineProperty(proxy, 'media', { get: () => query, configurable: true });\n return proxy;\n }\n return mql;\n };\n\n // Trigger change event on existing reduced-motion media queries\n // so frameworks (Framer Motion, GSAP) react immediately\n try {\n const mql = originalMatchMedia.call(window, '(prefers-reduced-motion: reduce)');\n mql.dispatchEvent(new MediaQueryListEvent('change', { matches: true, media: mql.media }));\n } catch { /* older browsers */ }\n }\n\n function restoreMatchMedia() {\n if (originalMatchMedia) {\n window.matchMedia = originalMatchMedia;\n originalMatchMedia = null;\n\n // Dispatch change event to signal motion is allowed again\n try {\n const mql = window.matchMedia('(prefers-reduced-motion: reduce)');\n mql.dispatchEvent(new MediaQueryListEvent('change', { matches: mql.matches, media: mql.media }));\n } catch { /* older browsers */ }\n }\n }\n\n // -------------------------------------------------------------------------\n // Layer 5: GIF freeze — replace with static first frame\n // -------------------------------------------------------------------------\n\n function freezeGif(img: HTMLImageElement) {\n const src = img.src || img.getAttribute('src') || '';\n if (!src.toLowerCase().includes('.gif')) return;\n if (gifOriginals.has(img)) return;\n\n const freeze = () => {\n try {\n const canvas = document.createElement('canvas');\n canvas.width = img.naturalWidth || img.width;\n canvas.height = img.naturalHeight || img.height;\n if (canvas.width === 0 || canvas.height === 0) return;\n const ctx = canvas.getContext('2d');\n if (!ctx) return;\n ctx.drawImage(img, 0, 0);\n gifOriginals.set(img, src);\n img.src = canvas.toDataURL('image/png');\n } catch { /* cross-origin, skip */ }\n };\n\n if (img.complete && img.naturalWidth > 0) {\n freeze();\n } else {\n img.addEventListener('load', freeze, { once: true });\n }\n }\n\n function freezeAllGifs() {\n document.querySelectorAll<HTMLImageElement>('img').forEach(freezeGif);\n }\n\n function restoreGifs() {\n for (const [img, originalSrc] of gifOriginals) {\n try { img.src = originalSrc; } catch { /* element may be gone */ }\n }\n gifOriginals.clear();\n }\n\n // -------------------------------------------------------------------------\n // Layer 6: IntersectionObserver override\n // Scroll-triggered animations use IntersectionObserver to start only when\n // elements scroll into view. Override it so ALL elements are reported as\n // already visible — animations fire immediately and our finish() catches them.\n // -------------------------------------------------------------------------\n\n function overrideIntersectionObserver() {\n if (originalIntersectionObserver) return;\n originalIntersectionObserver = window.IntersectionObserver;\n const OrigIO = originalIntersectionObserver;\n\n window.IntersectionObserver = function (\n callback: IntersectionObserverCallback,\n options?: IntersectionObserverInit,\n ) {\n // Wrap callback: when enabled, force all entries to isIntersecting = true\n const wrappedCallback: IntersectionObserverCallback = (entries, observer) => {\n if (!enabled) {\n callback(entries, observer);\n return;\n }\n // Create proxied entries that always report as intersecting\n const faked = entries.map((entry) => {\n try {\n return new Proxy(entry, {\n get(target, prop) {\n if (prop === 'isIntersecting') return true;\n if (prop === 'intersectionRatio') return 1;\n const val = (target as any)[prop];\n return typeof val === 'function' ? val.bind(target) : val;\n },\n });\n } catch {\n return entry;\n }\n });\n callback(faked, observer);\n };\n\n const observer = new OrigIO(wrappedCallback, options);\n return observer;\n } as any;\n\n // Preserve prototype and static properties\n window.IntersectionObserver.prototype = OrigIO.prototype;\n Object.defineProperty(window.IntersectionObserver, 'name', { value: 'IntersectionObserver' });\n }\n\n function restoreIntersectionObserver() {\n if (originalIntersectionObserver) {\n window.IntersectionObserver = originalIntersectionObserver;\n originalIntersectionObserver = null;\n }\n }\n\n // -------------------------------------------------------------------------\n // Layer 7: Force ALL hidden elements on the page visible\n // Aggressive full-page sweep: any element with inline opacity < 1\n // that isn't our widget gets forced to opacity: 1.\n // This catches scroll-reveal, appear, and any other animation pattern.\n // -------------------------------------------------------------------------\n\n const FIXED_ATTR = 'data-accessify-anim-fixed';\n let fixedElements: Array<{ el: HTMLElement; origStyle: string }> = [];\n\n function forceAllPageElementsVisible() {\n // Walk ALL elements that have opacity in their inline style\n const allEls = document.querySelectorAll<HTMLElement>('[style]');\n for (const el of allEls) {\n if (el.hasAttribute(FIXED_ATTR)) continue;\n if (el.closest('#accessify-root')) continue;\n // Skip elements that are display:none — those are hidden for structural reasons\n if (el.style.display === 'none') continue;\n\n const inlineOpacity = el.style.opacity;\n if (inlineOpacity === '' || inlineOpacity === '1') continue;\n\n const val = parseFloat(inlineOpacity);\n if (!isNaN(val) && val < 0.9) {\n fixedElements.push({ el, origStyle: el.getAttribute('style') || '' });\n el.style.setProperty('opacity', '1', 'important');\n el.setAttribute(FIXED_ATTR, 'true');\n }\n }\n }\n\n function restoreFixedElements() {\n for (const { el, origStyle } of fixedElements) {\n try {\n el.setAttribute('style', origStyle);\n el.removeAttribute(FIXED_ATTR);\n } catch { /* element may be gone */ }\n }\n fixedElements = [];\n }\n\n // Also handle dynamically added or changed elements\n function forceElementVisibleSingle(el: HTMLElement) {\n if (el.hasAttribute(FIXED_ATTR)) return;\n if (el.closest('#accessify-root')) return;\n const inlineOpacity = el.style.opacity;\n if (inlineOpacity === '' || inlineOpacity === '1') return;\n const val = parseFloat(inlineOpacity);\n if (!isNaN(val) && val < 0.9) {\n fixedElements.push({ el, origStyle: el.getAttribute('style') || '' });\n el.style.setProperty('opacity', '1', 'important');\n el.setAttribute(FIXED_ATTR, 'true');\n }\n }\n\n // -------------------------------------------------------------------------\n // Layer 8: Style attribute mutation observer — counter inline style animations\n // -------------------------------------------------------------------------\n\n function setupStyleObserver() {\n if (styleObserver) return;\n let isOurChange = false;\n\n styleObserver = new MutationObserver((mutations) => {\n if (!enabled || isOurChange) return;\n isOurChange = true;\n\n for (const m of mutations) {\n if (m.type !== 'attributes' || m.attributeName !== 'style') continue;\n const el = m.target as HTMLElement;\n if (el.closest('#accessify-root')) continue;\n\n // Force any element whose opacity was just set to < 0.9 back to 1\n const inlineOpacity = el.style.opacity;\n if (inlineOpacity !== '' && inlineOpacity !== '1') {\n const val = parseFloat(inlineOpacity);\n if (!isNaN(val) && val < 0.9) {\n if (!el.hasAttribute(FIXED_ATTR)) {\n fixedElements.push({ el, origStyle: m.oldValue || '' });\n }\n el.style.setProperty('opacity', '1', 'important');\n el.setAttribute(FIXED_ATTR, 'true');\n }\n }\n }\n\n isOurChange = false;\n });\n\n styleObserver.observe(document.body, {\n attributes: true,\n attributeFilter: ['style'],\n attributeOldValue: true,\n subtree: true,\n });\n }\n\n function disconnectStyleObserver() {\n styleObserver?.disconnect();\n styleObserver = null;\n }\n\n // -------------------------------------------------------------------------\n // Layer 9: Delayed re-scans — Framer loads async, elements appear late\n // -------------------------------------------------------------------------\n\n function scheduleDelayedScans() {\n const delays = [300, 800, 1500, 3000, 5000];\n for (const delay of delays) {\n const timer = setTimeout(() => {\n if (!enabled) return;\n finishAllWebAnimations();\n forceAllPageElementsVisible();\n pauseAutoplayVideos();\n freezeAllGifs();\n pauseSVGAnimations();\n stopMarquees();\n }, delay);\n delayedTimers.push(timer);\n }\n }\n\n function clearDelayedScans() {\n for (const timer of delayedTimers) clearTimeout(timer);\n delayedTimers = [];\n }\n\n // -------------------------------------------------------------------------\n // Layer 7: Marquee & SVG SMIL animations\n // -------------------------------------------------------------------------\n\n function stopMarquees() {\n document.querySelectorAll('marquee').forEach((el) => {\n try { (el as any).stop(); } catch { /* not supported */ }\n });\n }\n\n function startMarquees() {\n document.querySelectorAll('marquee').forEach((el) => {\n try { (el as any).start(); } catch { /* not supported */ }\n });\n }\n\n function pauseSVGAnimations() {\n document.querySelectorAll('svg').forEach((svg) => {\n try {\n if (typeof (svg as any).pauseAnimations === 'function') {\n (svg as any).pauseAnimations();\n }\n } catch { /* not supported */ }\n });\n }\n\n function resumeSVGAnimations() {\n document.querySelectorAll('svg').forEach((svg) => {\n try {\n if (typeof (svg as any).unpauseAnimations === 'function') {\n (svg as any).unpauseAnimations();\n }\n } catch { /* not supported */ }\n });\n }\n\n // -------------------------------------------------------------------------\n // Video control — pause all videos + intercept play()\n // -------------------------------------------------------------------------\n\n let originalVideoPlay: typeof HTMLVideoElement.prototype.play | null = null;\n\n function pauseAutoplayVideos() {\n const videos = document.querySelectorAll<HTMLVideoElement>('video');\n pausedVideos = [];\n videos.forEach((video) => {\n if (!video.paused) {\n video.pause();\n pausedVideos.push(video);\n }\n });\n\n // Also remove autoplay attribute to prevent re-play\n videos.forEach((video) => {\n if (video.hasAttribute('autoplay')) {\n video.dataset.accessifyAutoplay = 'true';\n video.removeAttribute('autoplay');\n }\n });\n\n // Override play() to prevent JS from starting videos\n if (!originalVideoPlay) {\n originalVideoPlay = HTMLVideoElement.prototype.play;\n HTMLVideoElement.prototype.play = function () {\n if (enabled) {\n // Block autoplay, return resolved promise to avoid errors\n return Promise.resolve();\n }\n return originalVideoPlay!.call(this);\n };\n }\n }\n\n function resumePausedVideos() {\n // Restore play()\n if (originalVideoPlay) {\n HTMLVideoElement.prototype.play = originalVideoPlay;\n originalVideoPlay = null;\n }\n\n // Restore autoplay attributes\n document.querySelectorAll<HTMLVideoElement>('video[data-accessify-autoplay]').forEach((video) => {\n video.setAttribute('autoplay', '');\n delete video.dataset.accessifyAutoplay;\n });\n\n // Resume previously paused videos\n pausedVideos.forEach((video) => {\n try { video.play(); } catch { /* removed from DOM */ }\n });\n pausedVideos = [];\n }\n\n // -------------------------------------------------------------------------\n // Lifecycle\n // -------------------------------------------------------------------------\n\n function activate() {\n if (enabled) return;\n enabled = true;\n\n // Layer 0: CSS overrides (includes Framer-specific !important rules)\n injectStyles();\n\n // Layer 1: Web Animations API — finish all (jump to end state)\n finishAllWebAnimations();\n overrideElementAnimate();\n\n // Layer 2: RAF loop detection\n overrideRAF();\n\n // Layer 3: MutationObserver for dynamic content\n setupMutationObserver();\n\n // Layer 4: prefers-reduced-motion emulation\n emulateReducedMotion();\n\n // Layer 5: GIF freeze\n freezeAllGifs();\n\n // Layer 6: IntersectionObserver override — scroll-triggered anims fire immediately\n overrideIntersectionObserver();\n\n // Layer 7: Force ALL hidden elements visible (full-page sweep)\n forceAllPageElementsVisible();\n\n // Marquee & SVG SMIL\n stopMarquees();\n pauseSVGAnimations();\n\n // Layer 8: Style mutation observer (counters JS setting opacity back to low values)\n setupStyleObserver();\n\n // Layer 9: Delayed re-scans (async-loaded elements)\n scheduleDelayedScans();\n\n // Videos\n pauseAutoplayVideos();\n\n localStorage.setItem(STORAGE_KEY, 'true');\n }\n\n function deactivate() {\n enabled = false;\n\n // Reverse order for clean teardown\n clearDelayedScans();\n disconnectStyleObserver();\n restoreFixedElements();\n resumePausedVideos();\n resumeSVGAnimations();\n startMarquees();\n restoreIntersectionObserver();\n restoreGifs();\n restoreMatchMedia();\n disconnectMutationObserver();\n restoreRAF();\n restoreElementAnimate();\n resumePausedAnimations();\n removeStyles();\n\n localStorage.removeItem(STORAGE_KEY);\n }\n\n return {\n id: 'animation-stop',\n name: () => 'Stop Animations',\n description: 'Pause all animations, transitions, and auto-playing videos (WCAG 2.3.1)',\n icon: 'animation-stop',\n category: 'visual',\n activate,\n deactivate,\n getState: (): FeatureState => ({\n id: 'animation-stop',\n enabled,\n }),\n setState: (state: { enabled: boolean }) => {\n if (state.enabled) {\n activate();\n } else {\n deactivate();\n }\n },\n };\n}\n"],"names":["observer"],"mappings":"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"}