devextreme
Version:
HTML5 JavaScript Component Suite for Responsive Web Development
181 lines (174 loc) • 7.6 kB
JavaScript
/**
* DevExtreme (esm/ui/shared/accessibility.js)
* Version: 21.1.4
* Build date: Mon Jun 21 2021
*
* Copyright (c) 2012 - 2021 Developer Express Inc. ALL RIGHTS RESERVED
* Read about DevExtreme licensing here: https://js.devexpress.com/Licensing/
*/
import $ from "../../core/renderer";
import eventsEngine from "../../events/core/events_engine";
import {
normalizeKeyName
} from "../../events/utils/index";
import {
extend
} from "../../core/utils/extend";
import domAdapter from "../../core/dom_adapter";
import {
noop
} from "../../core/utils/common";
var FOCUS_STATE_CLASS = "dx-state-focused";
var FOCUS_DISABLED_CLASS = "dx-cell-focus-disabled";
var FOCUSED_ROW_SELECTOR = ".dx-row-focused";
var GRID_ROW_SELECTOR = ".dx-datagrid-rowsview .dx-row";
var GRID_CELL_SELECTOR = "".concat(GRID_ROW_SELECTOR, " > td");
var TREELIST_ROW_SELECTOR = ".dx-treelist-rowsview .dx-row";
var TREELIST_CELL_SELECTOR = "".concat(TREELIST_ROW_SELECTOR, " > td");
var viewItemSelectorMap = {
groupPanel: [".dx-datagrid-group-panel .dx-group-panel-item[tabindex]"],
columnHeaders: [".dx-datagrid-headers .dx-header-row > td.dx-datagrid-action", ".dx-treelist-headers .dx-header-row > td.dx-treelist-action"],
filterRow: [".dx-datagrid-headers .dx-datagrid-filter-row .dx-editor-cell .dx-texteditor-input", ".dx-treelist-headers .dx-treelist-filter-row .dx-editor-cell .dx-texteditor-input"],
rowsView: ["".concat(FOCUSED_ROW_SELECTOR), "".concat(GRID_ROW_SELECTOR, "[tabindex]"), "".concat(GRID_CELL_SELECTOR, "[tabindex]"), "".concat(GRID_CELL_SELECTOR), "".concat(TREELIST_ROW_SELECTOR, "[tabindex]"), "".concat(TREELIST_CELL_SELECTOR, "[tabindex]"), "".concat(TREELIST_CELL_SELECTOR)],
footer: [".dx-datagrid-total-footer .dx-datagrid-summary-item", ".dx-treelist-total-footer .dx-treelist-summary-item"],
filterPanel: [".dx-datagrid-filter-panel .dx-icon-filter", ".dx-treelist-filter-panel .dx-icon-filter"],
pager: [".dx-datagrid-pager [tabindex]", ".dx-treelist-pager [tabindex]"]
};
var isMouseDown = false;
var isHiddenFocusing = false;
var focusedElementInfo = null;
function processKeyDown(viewName, instance, event, action, $mainElement, executeKeyDown) {
var isHandled = fireKeyDownEvent(instance, event.originalEvent, executeKeyDown);
if (isHandled) {
return
}
var keyName = normalizeKeyName(event);
if ("enter" === keyName || "space" === keyName) {
saveFocusedElementInfo(event.target, instance);
action && action({
event: event
})
} else if ("tab" === keyName) {
$mainElement.addClass(FOCUS_STATE_CLASS)
} else {
selectView(viewName, instance, event)
}
}
function saveFocusedElementInfo(target, instance) {
var $target = $(target);
var ariaLabel = $target.attr("aria-label");
var $activeElements = getActiveAccessibleElements(ariaLabel, instance.element());
var targetIndex = $activeElements.index($target);
focusedElementInfo = extend({}, {
ariaLabel: ariaLabel,
index: targetIndex
}, {
viewInstance: instance
})
}
function getActiveAccessibleElements(ariaLabel, viewElement) {
var $viewElement = $(viewElement);
var $activeElements;
if (ariaLabel) {
$activeElements = $viewElement.find('[aria-label="'.concat(ariaLabel, '"][tabindex]'))
} else {
$activeElements = $viewElement.find("[tabindex]")
}
return $activeElements
}
function findFocusedViewElement(viewSelectors) {
for (var index in viewSelectors) {
var selector = viewSelectors[index];
var $focusViewElement = $(selector).first();
if ($focusViewElement.length) {
return $focusViewElement
}
}
}
function fireKeyDownEvent(instance, event, executeAction) {
var args = {
event: event,
handled: false
};
if (executeAction) {
executeAction(args)
} else {
instance._createActionByOption("onKeyDown")(args)
}
return args.handled
}
function onDocumentVisibilityChange() {
isHiddenFocusing = "visible" === domAdapter.getDocument().visibilityState
}
export function subscribeVisibilityChange() {
eventsEngine.on(domAdapter.getDocument(), "visibilitychange", onDocumentVisibilityChange)
}
export function unsubscribeVisibilityChange() {
eventsEngine.off(domAdapter.getDocument(), "visibilitychange", onDocumentVisibilityChange)
}
export function hiddenFocus(element) {
isHiddenFocusing = true;
element.focus();
isHiddenFocusing = false
}
export function registerKeyboardAction(viewName, instance, $element, selector, action, executeKeyDown) {
if (instance.option("useLegacyKeyboardNavigation")) {
return noop
}
var getMainElement = () => $(instance.element());
var keyDownHandler = e => processKeyDown(viewName, instance, e, action, getMainElement(), executeKeyDown);
var mouseDownHandler = () => {
isMouseDown = true;
getMainElement().removeClass(FOCUS_STATE_CLASS)
};
var focusinHandler = () => {
var needShowOverlay = !isMouseDown && !isHiddenFocusing;
if (needShowOverlay) {
getMainElement().addClass(FOCUS_STATE_CLASS)
}
isMouseDown = false
};
eventsEngine.on($element, "keydown", selector, keyDownHandler);
eventsEngine.on($element, "mousedown", selector, mouseDownHandler);
eventsEngine.on($element, "focusin", selector, focusinHandler);
return () => {
eventsEngine.off($element, "keydown", selector, keyDownHandler);
eventsEngine.off($element, "mousedown", selector, mouseDownHandler);
eventsEngine.off($element, "focusin", selector, focusinHandler)
}
}
export function restoreFocus(instance) {
if (!instance.option("useLegacyKeyboardNavigation") && focusedElementInfo) {
var viewInstance = focusedElementInfo.viewInstance;
if (viewInstance) {
var $activeElements = getActiveAccessibleElements(focusedElementInfo.ariaLabel, viewInstance.element());
var $targetElement = $activeElements.eq(focusedElementInfo.index);
focusedElementInfo = null;
eventsEngine.trigger($targetElement, "focus")
}
}
}
export function selectView(viewName, instance, event) {
var keyName = normalizeKeyName(event);
if (event.ctrlKey && ("upArrow" === keyName || "downArrow" === keyName)) {
var viewNames = Object.keys(viewItemSelectorMap);
var viewItemIndex = viewNames.indexOf(viewName);
while (viewItemIndex >= 0 && viewItemIndex < viewNames.length) {
viewItemIndex = "upArrow" === keyName ? --viewItemIndex : ++viewItemIndex;
var _viewName = viewNames[viewItemIndex];
var viewSelectors = viewItemSelectorMap[_viewName];
var $focusViewElement = findFocusedViewElement(viewSelectors);
if ($focusViewElement && $focusViewElement.length) {
$focusViewElement.attr("tabindex", instance.option("tabindex") || 0);
eventsEngine.trigger($focusViewElement, "focus");
$focusViewElement.removeClass(FOCUS_DISABLED_CLASS);
break
}
}
}
}
export function setTabIndex(instance, $element) {
if (!instance.option("useLegacyKeyboardnavigation")) {
$element.attr("tabindex", instance.option("tabindex") || 0)
}
}