123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108 |
- import _extends from "@babel/runtime/helpers/esm/extends";
- import _objectWithoutPropertiesLoose from "@babel/runtime/helpers/esm/objectWithoutPropertiesLoose";
- import qsa from 'dom-helpers/query/querySelectorAll';
- import React, { useContext, useEffect, useRef } from 'react';
- import useForceUpdate from '@restart/hooks/useForceUpdate';
- import useMergedRefs from '@restart/hooks/useMergedRefs';
- import NavContext from './NavContext';
- import SelectableContext, { makeEventKey } from './SelectableContext';
- import TabContext from './TabContext';
-
- var noop = function noop() {};
-
- var AbstractNav = React.forwardRef(function (_ref, ref) {
- var _ref$as = _ref.as,
- Component = _ref$as === void 0 ? 'ul' : _ref$as,
- onSelect = _ref.onSelect,
- activeKey = _ref.activeKey,
- role = _ref.role,
- onKeyDown = _ref.onKeyDown,
- props = _objectWithoutPropertiesLoose(_ref, ["as", "onSelect", "activeKey", "role", "onKeyDown"]);
-
- // A ref and forceUpdate for refocus, b/c we only want to trigger when needed
- // and don't want to reset the set in the effect
- var forceUpdate = useForceUpdate();
- var needsRefocusRef = useRef(false);
- var parentOnSelect = useContext(SelectableContext);
- var tabContext = useContext(TabContext);
- var getControlledId, getControllerId;
-
- if (tabContext) {
- role = role || 'tablist';
- activeKey = tabContext.activeKey;
- getControlledId = tabContext.getControlledId;
- getControllerId = tabContext.getControllerId;
- }
-
- var listNode = useRef(null);
-
- var getNextActiveChild = function getNextActiveChild(offset) {
- if (!listNode.current) return null;
- var items = qsa(listNode.current, '[data-rb-event-key]:not(.disabled)');
- var activeChild = listNode.current.querySelector('.active');
- var index = items.indexOf(activeChild);
- if (index === -1) return null;
- var nextIndex = index + offset;
- if (nextIndex >= items.length) nextIndex = 0;
- if (nextIndex < 0) nextIndex = items.length - 1;
- return items[nextIndex];
- };
-
- var handleSelect = function handleSelect(key, event) {
- if (key == null) return;
- if (onSelect) onSelect(key, event);
- if (parentOnSelect) parentOnSelect(key, event);
- };
-
- var handleKeyDown = function handleKeyDown(event) {
- if (onKeyDown) onKeyDown(event);
- var nextActiveChild;
-
- switch (event.key) {
- case 'ArrowLeft':
- case 'ArrowUp':
- nextActiveChild = getNextActiveChild(-1);
- break;
-
- case 'ArrowRight':
- case 'ArrowDown':
- nextActiveChild = getNextActiveChild(1);
- break;
-
- default:
- return;
- }
-
- if (!nextActiveChild) return;
- event.preventDefault();
- handleSelect(nextActiveChild.dataset.rbEventKey, event);
- needsRefocusRef.current = true;
- forceUpdate();
- };
-
- useEffect(function () {
- if (listNode.current && needsRefocusRef.current) {
- var activeChild = listNode.current.querySelector('[data-rb-event-key].active');
- if (activeChild) activeChild.focus();
- }
-
- needsRefocusRef.current = false;
- });
- var mergedRef = useMergedRefs(ref, listNode);
- return React.createElement(SelectableContext.Provider, {
- value: handleSelect
- }, React.createElement(NavContext.Provider, {
- value: {
- role: role,
- // used by NavLink to determine it's role
- activeKey: makeEventKey(activeKey),
- getControlledId: getControlledId || noop,
- getControllerId: getControllerId || noop
- }
- }, React.createElement(Component, _extends({}, props, {
- onKeyDown: handleKeyDown,
- ref: mergedRef,
- role: role
- }))));
- });
- export default AbstractNav;
|