forked from swiftlang/swift-docc-render
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathchangeElementVOVisibility.js
128 lines (111 loc) · 3.42 KB
/
changeElementVOVisibility.js
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
125
126
127
128
/**
* This source file is part of the Swift.org open source project
*
* Copyright (c) 2021-2022 Apple Inc. and the Swift project authors
* Licensed under Apache License v2.0 with Runtime Library Exception
*
* See https://swift.org/LICENSE.txt for license information
* See https://swift.org/CONTRIBUTORS.txt for Swift project authors
*/
import TabManager from 'docc-render/utils/TabManager';
const OG_PREFIX = 'data-original-';
const ARIA = 'aria-hidden';
const TABINDEX = 'tabindex';
function cacheOriginalAttribute(element, prop) {
const attr = OG_PREFIX + prop;
// make sure that prop isn't cached already
if (element.getAttribute(attr)) return;
const originalValue = element.getAttribute(prop) || '';
element.setAttribute(attr, originalValue);
}
function retrieveOriginalAttribute(element, prop) {
const attr = OG_PREFIX + prop;
// return if attribute doesn't exist
if (!element.hasAttribute(attr)) return;
// get the cached property
const originalValue = element.getAttribute(attr);
// remove the prefixed attribute
element.removeAttribute(attr);
// if there is a value, set it back.
if (originalValue.length) {
element.setAttribute(prop, originalValue);
} else {
// otherwise remove the attribute entirely.
element.removeAttribute(prop);
}
}
/* eslint-disable no-cond-assign */
function iterateOverSiblings(el, callback) {
const ancestor = document.body;
let previousNode = el;
let nextNode = el;
while ((previousNode = previousNode.previousElementSibling)) {
callback(previousNode);
}
while ((nextNode = nextNode.nextElementSibling)) {
callback(nextNode);
}
if (el.parentElement && el.parentElement !== ancestor) {
iterateOverSiblings(el.parentElement, callback);
}
}
/**
* Hides an element from VO
* @param {HTMLElement} element
*/
const hideElement = (element) => {
// set original value for prefixed properties and tabindex
// store the prop temporarily, to retrieve later.
cacheOriginalAttribute(element, ARIA);
cacheOriginalAttribute(element, TABINDEX);
// hide the component from VO
element.setAttribute(ARIA, 'true');
// hide the component from tabbing
element.setAttribute(TABINDEX, '-1');
// make sure element's tabbable children are hidden as well
const tabbables = TabManager.getTabbableElements(element);
let i = tabbables.length - 1;
while (i >= 0) {
cacheOriginalAttribute(tabbables[i], TABINDEX);
tabbables[i].setAttribute(TABINDEX, '-1');
i -= 1;
}
};
/**
* Show an element
* @param {HTMLElement} element
*/
const showElement = (element) => {
retrieveOriginalAttribute(element, ARIA);
retrieveOriginalAttribute(element, TABINDEX);
// make sure element's tabbable children are restored as well
const tabbables = element.querySelectorAll(`[${OG_PREFIX + TABINDEX}]`);
let i = tabbables.length - 1;
while (i >= 0) {
retrieveOriginalAttribute(tabbables[i], TABINDEX);
i -= 1;
}
};
/**
* Hides and reveals all siblings,
* up to the root element, of the passed element.
* Used for modals, to capture VO in a specific container.
*/
export default {
/**
* Hide all sibling elements,
* of the passed element to VO.
* @param {HTMLElement} element
*/
hide(element) {
iterateOverSiblings(element, hideElement);
},
/**
* Reveal all sibling elements,
* of the passed element to VO.
* @param {HTMLElement} element
*/
show(element) {
iterateOverSiblings(element, showElement);
},
};