@material-ui/core
Version:
React components that implement Google's Material Design.
43 lines (38 loc) • 1.77 kB
JavaScript
import keycode from 'keycode';
import warning from 'warning';
import ownerDocument from '../utils/ownerDocument';
const internal = {
focusKeyPressed: false,
keyUpEventTimeout: -1
};
export function detectFocusVisible(instance, element, callback, attempt = 1) {
process.env.NODE_ENV !== "production" ? warning(instance.focusVisibleCheckTime, 'Material-UI: missing instance.focusVisibleCheckTime.') : void 0;
process.env.NODE_ENV !== "production" ? warning(instance.focusVisibleMaxCheckTimes, 'Material-UI: missing instance.focusVisibleMaxCheckTimes.') : void 0;
instance.focusVisibleTimeout = setTimeout(() => {
const doc = ownerDocument(element);
if (internal.focusKeyPressed && (doc.activeElement === element || element.contains(doc.activeElement))) {
callback();
} else if (attempt < instance.focusVisibleMaxCheckTimes) {
detectFocusVisible(instance, element, callback, attempt + 1);
}
}, instance.focusVisibleCheckTime);
}
const FOCUS_KEYS = ['tab', 'enter', 'space', 'esc', 'up', 'down', 'left', 'right'];
function isFocusKey(event) {
return FOCUS_KEYS.indexOf(keycode(event)) > -1;
}
const handleKeyUpEvent = event => {
if (isFocusKey(event)) {
internal.focusKeyPressed = true; // Let's consider that the user is using a keyboard during a window frame of 1s.
clearTimeout(internal.keyUpEventTimeout);
internal.keyUpEventTimeout = setTimeout(() => {
internal.focusKeyPressed = false;
}, 1e3);
}
};
export function listenForFocusKeys(win) {
// The event listener will only be added once per window.
// Duplicate event listeners will be ignored by addEventListener.
// Also, this logic is client side only, we don't need a teardown.
win.addEventListener('keyup', handleKeyUpEvent);
}