forked from solidjs-community/solid-aria
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathariaHideOutside.ts
124 lines (106 loc) · 4.41 KB
/
ariaHideOutside.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
/*
* Copyright 2022 Solid Aria Working Group.
* MIT License
*
* Portions of this file are based on code from react-spectrum.
* Copyright 2020 Adobe. All rights reserved.
*
* This file is licensed to you under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. You may obtain a copy
* of the License at http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under
* the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
* OF ANY KIND, either express or implied. See the License for the specific language
* governing permissions and limitations under the License.
*/
// Keeps a ref count of all hidden elements. Added to when hiding an element, and
// subtracted from when showing it again. When it reaches zero, aria-hidden is removed.
const refCountMap = new WeakMap<Element, number>();
/**
* Hides all elements in the DOM outside the given targets from screen readers using aria-hidden,
* and returns a function to revert these changes. In addition, changes to the DOM are watched
* and new elements outside the targets are automatically hidden.
* @param targets - The elements that should remain visible.
* @param root - Nothing will be hidden above this element.
* @returns - A function to restore all hidden elements.
*/
export function ariaHideOutside(targets: HTMLElement[], root = document.body) {
const visibleNodes = new Set<Element>(targets);
const hiddenNodes = new Set<Element>();
const walker = document.createTreeWalker(root, NodeFilter.SHOW_ELEMENT, {
acceptNode(node) {
// If this node is a live announcer, add it to the set of nodes to keep visible.
if (node instanceof HTMLElement && node.dataset.liveAnnouncer === "true") {
visibleNodes.add(node);
}
// Skip this node and its children if it is one of the target nodes, or a live announcer.
// Also skip children of already hidden nodes, as aria-hidden is recursive.
if (visibleNodes.has(node as Element) || hiddenNodes.has(node.parentElement as Element)) {
return NodeFilter.FILTER_REJECT;
}
// VoiceOver on iOS has issues hiding elements with role="row". Hide the cells inside instead.
// https://bugs.webkit.org/show_bug.cgi?id=222623
if (node instanceof HTMLElement && node.getAttribute("role") === "row") {
return NodeFilter.FILTER_SKIP;
}
// Skip this node but continue to children if one of the targets is inside the node.
if (targets.some(target => node.contains(target))) {
return NodeFilter.FILTER_SKIP;
}
return NodeFilter.FILTER_ACCEPT;
}
});
const hide = (node: Element) => {
const refCount = refCountMap.get(node) ?? 0;
// If already aria-hidden, and the ref count is zero, then this element
// was already hidden and there's nothing for us to do.
if (node.getAttribute("aria-hidden") === "true" && refCount === 0) {
return;
}
if (refCount === 0) {
node.setAttribute("aria-hidden", "true");
}
hiddenNodes.add(node);
refCountMap.set(node, refCount + 1);
};
let node = walker.nextNode() as Element;
while (node != null) {
hide(node);
node = walker.nextNode() as Element;
}
const observer = new MutationObserver(changes => {
for (const change of changes) {
if (change.type !== "childList" || change.addedNodes.length === 0) {
continue;
}
// If the parent element of the added nodes is not within one of the targets,
// and not already inside a hidden node, hide all of the new children.
if (![...visibleNodes, ...hiddenNodes].some(node => node.contains(change.target))) {
for (const node of change.addedNodes) {
if (node instanceof HTMLElement && node.dataset.liveAnnouncer === "true") {
visibleNodes.add(node);
} else if (node instanceof Element) {
hide(node);
}
}
}
}
});
observer.observe(root, { childList: true, subtree: true });
return () => {
observer.disconnect();
for (const node of hiddenNodes) {
const count = refCountMap.get(node);
if (count == null) {
continue;
}
if (count === 1) {
node.removeAttribute("aria-hidden");
refCountMap.delete(node);
} else {
refCountMap.set(node, count - 1);
}
}
};
}