@lskjs/navbar
Version:
LSK ux subrepo: navbar
154 lines (121 loc) • 15.8 kB
JavaScript
"use strict";
var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard");
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports["default"] = void 0;
var _extends2 = _interopRequireDefault(require("@babel/runtime/helpers/extends"));
var _objectWithoutProperties2 = _interopRequireDefault(require("@babel/runtime/helpers/objectWithoutProperties"));
var _querySelectorAll = _interopRequireDefault(require("dom-helpers/querySelectorAll"));
var _propTypes = _interopRequireDefault(require("prop-types"));
var _react = _interopRequireWildcard(require("react"));
var _useForceUpdate = _interopRequireDefault(require("@restart/hooks/useForceUpdate"));
var _useMergedRefs = _interopRequireDefault(require("@restart/hooks/useMergedRefs"));
var _NavContext = _interopRequireDefault(require("./NavContext"));
var _SelectableContext = _interopRequireWildcard(require("./SelectableContext"));
var _TabContext = _interopRequireDefault(require("./TabContext"));
var noop = function noop() {};
var propTypes = {
onSelect: _propTypes["default"].func.isRequired,
as: _propTypes["default"].elementType,
role: _propTypes["default"].string,
/** @private */
onKeyDown: _propTypes["default"].func,
/** @private */
parentOnSelect: _propTypes["default"].func,
/** @private */
getControlledId: _propTypes["default"].func,
/** @private */
getControllerId: _propTypes["default"].func,
/** @private */
activeKey: _propTypes["default"].any
};
var AbstractNav = _react["default"].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 = (0, _objectWithoutProperties2["default"])(_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 = (0, _useForceUpdate["default"])();
var needsRefocusRef = (0, _react.useRef)(false);
var parentOnSelect = (0, _react.useContext)(_SelectableContext["default"]);
var tabContext = (0, _react.useContext)(_TabContext["default"]);
var getControlledId, getControllerId;
if (tabContext) {
role = role || 'tablist';
activeKey = tabContext.activeKey;
getControlledId = tabContext.getControlledId;
getControllerId = tabContext.getControllerId;
}
var listNode = (0, _react.useRef)(null);
var getNextActiveChild = function getNextActiveChild(offset) {
if (!listNode.current) return null;
var items = (0, _querySelectorAll["default"])(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();
};
(0, _react.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 = (0, _useMergedRefs["default"])(ref, listNode);
return /*#__PURE__*/_react["default"].createElement(_SelectableContext["default"].Provider, {
value: handleSelect
}, /*#__PURE__*/_react["default"].createElement(_NavContext["default"].Provider, {
value: {
role: role,
// used by NavLink to determine it's role
activeKey: (0, _SelectableContext.makeEventKey)(activeKey),
getControlledId: getControlledId || noop,
getControllerId: getControllerId || noop
}
}, /*#__PURE__*/_react["default"].createElement(Component, (0, _extends2["default"])({}, props, {
onKeyDown: handleKeyDown,
ref: mergedRef,
role: role
}))));
});
AbstractNav.propTypes = propTypes;
var _default = AbstractNav;
exports["default"] = _default;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/AbstractNav.js"],"names":["noop","propTypes","onSelect","PropTypes","func","isRequired","as","elementType","role","string","onKeyDown","parentOnSelect","getControlledId","getControllerId","activeKey","any","AbstractNav","React","forwardRef","ref","Component","props","forceUpdate","needsRefocusRef","SelectableContext","tabContext","TabContext","listNode","getNextActiveChild","offset","current","items","activeChild","querySelector","index","indexOf","nextIndex","length","handleSelect","key","event","handleKeyDown","nextActiveChild","preventDefault","dataset","rbEventKey","focus","mergedRef"],"mappings":"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","sourcesContent":["import qsa from 'dom-helpers/querySelectorAll';\nimport PropTypes from 'prop-types';\nimport React, { useContext, useEffect, useRef } from 'react';\nimport useForceUpdate from '@restart/hooks/useForceUpdate';\nimport useMergedRefs from '@restart/hooks/useMergedRefs';\nimport NavContext from './NavContext';\nimport SelectableContext, { makeEventKey } from './SelectableContext';\nimport TabContext from './TabContext';\n\nconst noop = () => {};\n\nconst propTypes = {\n  onSelect: PropTypes.func.isRequired,\n\n  as: PropTypes.elementType,\n\n  role: PropTypes.string,\n\n  /** @private */\n  onKeyDown: PropTypes.func,\n  /** @private */\n  parentOnSelect: PropTypes.func,\n  /** @private */\n  getControlledId: PropTypes.func,\n  /** @private */\n  getControllerId: PropTypes.func,\n  /** @private */\n  activeKey: PropTypes.any,\n};\n\nconst AbstractNav = React.forwardRef(\n  (\n    {\n      // Need to define the default \"as\" during prop destructuring to be compatible with styled-components github.com/react-bootstrap/react-bootstrap/issues/3595\n      as: Component = 'ul',\n      onSelect,\n      activeKey,\n      role,\n      onKeyDown,\n      ...props\n    },\n    ref,\n  ) => {\n    // A ref and forceUpdate for refocus, b/c we only want to trigger when needed\n    // and don't want to reset the set in the effect\n    const forceUpdate = useForceUpdate();\n    const needsRefocusRef = useRef(false);\n\n    const parentOnSelect = useContext(SelectableContext);\n    const tabContext = useContext(TabContext);\n\n    let getControlledId, getControllerId;\n\n    if (tabContext) {\n      role = role || 'tablist';\n      activeKey = tabContext.activeKey;\n      getControlledId = tabContext.getControlledId;\n      getControllerId = tabContext.getControllerId;\n    }\n\n    const listNode = useRef(null);\n\n    const getNextActiveChild = offset => {\n      if (!listNode.current) return null;\n\n      let items = qsa(listNode.current, '[data-rb-event-key]:not(.disabled)');\n      let activeChild = listNode.current.querySelector('.active');\n\n      let index = items.indexOf(activeChild);\n      if (index === -1) return null;\n\n      let nextIndex = index + offset;\n      if (nextIndex >= items.length) nextIndex = 0;\n      if (nextIndex < 0) nextIndex = items.length - 1;\n      return items[nextIndex];\n    };\n\n    const handleSelect = (key, event) => {\n      if (key == null) return;\n      if (onSelect) onSelect(key, event);\n      if (parentOnSelect) parentOnSelect(key, event);\n    };\n\n    const handleKeyDown = event => {\n      if (onKeyDown) onKeyDown(event);\n\n      let nextActiveChild;\n      switch (event.key) {\n        case 'ArrowLeft':\n        case 'ArrowUp':\n          nextActiveChild = getNextActiveChild(-1);\n          break;\n        case 'ArrowRight':\n        case 'ArrowDown':\n          nextActiveChild = getNextActiveChild(1);\n          break;\n        default:\n          return;\n      }\n      if (!nextActiveChild) return;\n\n      event.preventDefault();\n      handleSelect(nextActiveChild.dataset.rbEventKey, event);\n      needsRefocusRef.current = true;\n      forceUpdate();\n    };\n\n    useEffect(() => {\n      if (listNode.current && needsRefocusRef.current) {\n        let activeChild = listNode.current.querySelector(\n          '[data-rb-event-key].active',\n        );\n\n        if (activeChild) activeChild.focus();\n      }\n\n      needsRefocusRef.current = false;\n    });\n\n    const mergedRef = useMergedRefs(ref, listNode);\n\n    return (\n      <SelectableContext.Provider value={handleSelect}>\n        <NavContext.Provider\n          value={{\n            role, // used by NavLink to determine it's role\n            activeKey: makeEventKey(activeKey),\n            getControlledId: getControlledId || noop,\n            getControllerId: getControllerId || noop,\n          }}\n        >\n          <Component\n            {...props}\n            onKeyDown={handleKeyDown}\n            ref={mergedRef}\n            role={role}\n          />\n        </NavContext.Provider>\n      </SelectableContext.Provider>\n    );\n  },\n);\n\nAbstractNav.propTypes = propTypes;\n\nexport default AbstractNav;\n"]}
//# sourceMappingURL=AbstractNav.js.map