reka-ui
Version:
Vue port for Radix UI Primitives.
70 lines (68 loc) • 3.09 kB
JavaScript
//#region src/shared/useArrowNavigation.ts
const ignoredElement = ["INPUT", "TEXTAREA"];
/**
* Allow arrow navigation for every html element with data-reka-collection-item tag
*
* @param e Keyboard event
* @param currentElement Event initiator element or any element that wants to handle the navigation
* @param parentElement Parent element where contains all the collection items, this will collect every item to be used when nav
* @param options further options
* @returns the navigated html element or null if none
*/
function useArrowNavigation(e, currentElement, parentElement, options = {}) {
if (!currentElement || options.enableIgnoredElement && ignoredElement.includes(currentElement.nodeName)) return null;
const { arrowKeyOptions = "both", attributeName = "[data-reka-collection-item]", itemsArray = [], loop = true, dir = "ltr", preventScroll = true, focus = false } = options;
const [right, left, up, down, home, end] = [
e.key === "ArrowRight",
e.key === "ArrowLeft",
e.key === "ArrowUp",
e.key === "ArrowDown",
e.key === "Home",
e.key === "End"
];
const goingVertical = up || down;
const goingHorizontal = right || left;
if (!home && !end && (!goingVertical && !goingHorizontal || arrowKeyOptions === "vertical" && goingHorizontal || arrowKeyOptions === "horizontal" && goingVertical)) return null;
const allCollectionItems = parentElement ? Array.from(parentElement.querySelectorAll(attributeName)) : itemsArray;
if (!allCollectionItems.length) return null;
if (preventScroll) e.preventDefault();
let item = null;
if (goingHorizontal || goingVertical) {
const goForward = goingVertical ? down : dir === "ltr" ? right : left;
item = findNextFocusableElement(allCollectionItems, currentElement, {
goForward,
loop
});
} else if (home) item = allCollectionItems.at(0) || null;
else if (end) item = allCollectionItems.at(-1) || null;
if (focus) item?.focus();
return item;
}
/**
* Recursive function to find the next focusable element to avoid disabled elements
*
* @param elements Elements to navigate
* @param currentElement Current active element
* @param options
* @returns next focusable element
*/
function findNextFocusableElement(elements, currentElement, options, iterations = elements.length) {
if (--iterations === 0) return null;
const index = elements.indexOf(currentElement);
const newIndex = options.goForward ? index + 1 : index - 1;
if (!options.loop && (newIndex < 0 || newIndex >= elements.length)) return null;
const adjustedNewIndex = (newIndex + elements.length) % elements.length;
const candidate = elements[adjustedNewIndex];
if (!candidate) return null;
const isDisabled = candidate.hasAttribute("disabled") && candidate.getAttribute("disabled") !== "false";
if (isDisabled) return findNextFocusableElement(elements, candidate, options, iterations);
return candidate;
}
//#endregion
Object.defineProperty(exports, 'useArrowNavigation', {
enumerable: true,
get: function () {
return useArrowNavigation;
}
});
//# sourceMappingURL=useArrowNavigation.cjs.map