UNPKG

@mui/core

Version:

Unstyled React components with which to implement custom design systems.

372 lines (315 loc) 13.6 kB
"use strict"; 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 _objectWithoutPropertiesLoose2 = _interopRequireDefault(require("@babel/runtime/helpers/objectWithoutPropertiesLoose")); var _system = require("@mui/system"); var _utils = require("@mui/utils"); var _core = require("@popperjs/core"); var _propTypes = _interopRequireDefault(require("prop-types")); var React = _interopRequireWildcard(require("react")); var _Portal = _interopRequireDefault(require("../Portal")); var _jsxRuntime = require("react/jsx-runtime"); const _excluded = ["anchorEl", "children", "disablePortal", "modifiers", "open", "placement", "popperOptions", "popperRef", "TransitionProps"], _excluded2 = ["anchorEl", "children", "container", "disablePortal", "keepMounted", "modifiers", "open", "placement", "popperOptions", "popperRef", "style", "transition"]; function _getRequireWildcardCache(nodeInterop) { if (typeof WeakMap !== "function") return null; var cacheBabelInterop = new WeakMap(); var cacheNodeInterop = new WeakMap(); return (_getRequireWildcardCache = function (nodeInterop) { return nodeInterop ? cacheNodeInterop : cacheBabelInterop; })(nodeInterop); } function _interopRequireWildcard(obj, nodeInterop) { if (!nodeInterop && obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(nodeInterop); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (key !== "default" && Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; } function flipPlacement(placement, theme) { const direction = theme && theme.direction || 'ltr'; if (direction === 'ltr') { return placement; } switch (placement) { case 'bottom-end': return 'bottom-start'; case 'bottom-start': return 'bottom-end'; case 'top-end': return 'top-start'; case 'top-start': return 'top-end'; default: return placement; } } function resolveAnchorEl(anchorEl) { return typeof anchorEl === 'function' ? anchorEl() : anchorEl; } const defaultPopperOptions = {}; /* eslint-disable react/prop-types */ const PopperTooltip = /*#__PURE__*/React.forwardRef(function PopperTooltip(props, ref) { const { anchorEl, children, disablePortal, modifiers, open, placement: initialPlacement, popperOptions, popperRef: popperRefProp, TransitionProps } = props, other = (0, _objectWithoutPropertiesLoose2.default)(props, _excluded); const tooltipRef = React.useRef(null); const ownRef = (0, _utils.unstable_useForkRef)(tooltipRef, ref); const popperRef = React.useRef(null); const handlePopperRef = (0, _utils.unstable_useForkRef)(popperRef, popperRefProp); const handlePopperRefRef = React.useRef(handlePopperRef); (0, _utils.unstable_useEnhancedEffect)(() => { handlePopperRefRef.current = handlePopperRef; }, [handlePopperRef]); React.useImperativeHandle(popperRefProp, () => popperRef.current, []); const theme = (0, _system.useThemeWithoutDefault)(); const rtlPlacement = flipPlacement(initialPlacement, theme); /** * placement initialized from prop but can change during lifetime if modifiers.flip. * modifiers.flip is essentially a flip for controlled/uncontrolled behavior */ const [placement, setPlacement] = React.useState(rtlPlacement); React.useEffect(() => { if (popperRef.current) { popperRef.current.forceUpdate(); } }); (0, _utils.unstable_useEnhancedEffect)(() => { if (!anchorEl || !open) { return undefined; } const handlePopperUpdate = data => { setPlacement(data.placement); }; const resolvedAnchorEl = resolveAnchorEl(anchorEl); if (process.env.NODE_ENV !== 'production') { if (resolvedAnchorEl && resolvedAnchorEl.nodeType === 1) { const box = resolvedAnchorEl.getBoundingClientRect(); if (process.env.NODE_ENV !== 'test' && box.top === 0 && box.left === 0 && box.right === 0 && box.bottom === 0) { console.warn(['MUI: The `anchorEl` prop provided to the component is invalid.', 'The anchor element should be part of the document layout.', "Make sure the element is present in the document or that it's not display none."].join('\n')); } } } let popperModifiers = [{ name: 'preventOverflow', options: { altBoundary: disablePortal } }, { name: 'flip', options: { altBoundary: disablePortal } }, { name: 'onUpdate', enabled: true, phase: 'afterWrite', fn: ({ state }) => { handlePopperUpdate(state); } }]; if (modifiers != null) { popperModifiers = popperModifiers.concat(modifiers); } if (popperOptions && popperOptions.modifiers != null) { popperModifiers = popperModifiers.concat(popperOptions.modifiers); } const popper = (0, _core.createPopper)(resolveAnchorEl(anchorEl), tooltipRef.current, (0, _extends2.default)({ placement: rtlPlacement }, popperOptions, { modifiers: popperModifiers })); handlePopperRefRef.current(popper); return () => { popper.destroy(); handlePopperRefRef.current(null); }; }, [anchorEl, disablePortal, modifiers, open, popperOptions, rtlPlacement]); const childProps = { placement }; if (TransitionProps !== null) { childProps.TransitionProps = TransitionProps; } return /*#__PURE__*/(0, _jsxRuntime.jsx)("div", (0, _extends2.default)({ ref: ownRef, role: "tooltip" }, other, { children: typeof children === 'function' ? children(childProps) : children })); }); /* eslint-enable react/prop-types */ /** * Poppers rely on the 3rd party library [Popper.js](https://popper.js.org/docs/v2/) for positioning. */ const Popper = /*#__PURE__*/React.forwardRef(function Popper(props, ref) { const { anchorEl, children, container: containerProp, disablePortal = false, keepMounted = false, modifiers, open, placement = 'bottom', popperOptions = defaultPopperOptions, popperRef, style, transition = false } = props, other = (0, _objectWithoutPropertiesLoose2.default)(props, _excluded2); const [exited, setExited] = React.useState(true); const handleEnter = () => { setExited(false); }; const handleExited = () => { setExited(true); }; if (!keepMounted && !open && (!transition || exited)) { return null; } // If the container prop is provided, use that // If the anchorEl prop is provided, use its parent body element as the container // If neither are provided let the Modal take care of choosing the container const container = containerProp || (anchorEl ? (0, _utils.unstable_ownerDocument)(resolveAnchorEl(anchorEl)).body : undefined); return /*#__PURE__*/(0, _jsxRuntime.jsx)(_Portal.default, { disablePortal: disablePortal, container: container, children: /*#__PURE__*/(0, _jsxRuntime.jsx)(PopperTooltip, (0, _extends2.default)({ anchorEl: anchorEl, disablePortal: disablePortal, modifiers: modifiers, ref: ref, open: transition ? !exited : open, placement: placement, popperOptions: popperOptions, popperRef: popperRef }, other, { style: (0, _extends2.default)({ // Prevents scroll issue, waiting for Popper.js to add this style once initiated. position: 'fixed', // Fix Popper.js display issue top: 0, left: 0, display: !open && keepMounted && (!transition || exited) ? 'none' : null }, style), TransitionProps: transition ? { in: open, onEnter: handleEnter, onExited: handleExited } : null, children: children })) }); }); process.env.NODE_ENV !== "production" ? Popper.propTypes /* remove-proptypes */ = { // ----------------------------- Warning -------------------------------- // | These PropTypes are generated from the TypeScript type definitions | // | To update them edit the d.ts file and run "yarn proptypes" | // ---------------------------------------------------------------------- /** * An HTML element, [virtualElement](https://popper.js.org/docs/v2/virtual-elements/), * or a function that returns either. * It's used to set the position of the popper. * The return value will passed as the reference object of the Popper instance. */ anchorEl: (0, _utils.chainPropTypes)(_propTypes.default.oneOfType([_utils.HTMLElementType, _propTypes.default.object, _propTypes.default.func]), props => { if (props.open) { const resolvedAnchorEl = resolveAnchorEl(props.anchorEl); if (resolvedAnchorEl && resolvedAnchorEl.nodeType === 1) { const box = resolvedAnchorEl.getBoundingClientRect(); if (process.env.NODE_ENV !== 'test' && box.top === 0 && box.left === 0 && box.right === 0 && box.bottom === 0) { return new Error(['MUI: The `anchorEl` prop provided to the component is invalid.', 'The anchor element should be part of the document layout.', "Make sure the element is present in the document or that it's not display none."].join('\n')); } } else if (!resolvedAnchorEl || typeof resolvedAnchorEl.getBoundingClientRect !== 'function' || resolvedAnchorEl.contextElement != null && resolvedAnchorEl.contextElement.nodeType !== 1) { return new Error(['MUI: The `anchorEl` prop provided to the component is invalid.', 'It should be an HTML element instance or a virtualElement ', '(https://popper.js.org/docs/v2/virtual-elements/).'].join('\n')); } } return null; }), /** * Popper render function or node. */ children: _propTypes.default /* @typescript-to-proptypes-ignore */ .oneOfType([_propTypes.default.node, _propTypes.default.func]), /** * An HTML element or function that returns one. * The `container` will have the portal children appended to it. * * By default, it uses the body of the top-level document object, * so it's simply `document.body` most of the time. */ container: _propTypes.default /* @typescript-to-proptypes-ignore */ .oneOfType([_utils.HTMLElementType, _propTypes.default.func]), /** * The `children` will be under the DOM hierarchy of the parent component. * @default false */ disablePortal: _propTypes.default.bool, /** * Always keep the children in the DOM. * This prop can be useful in SEO situation or * when you want to maximize the responsiveness of the Popper. * @default false */ keepMounted: _propTypes.default.bool, /** * Popper.js is based on a "plugin-like" architecture, * most of its features are fully encapsulated "modifiers". * * A modifier is a function that is called each time Popper.js needs to * compute the position of the popper. * For this reason, modifiers should be very performant to avoid bottlenecks. * To learn how to create a modifier, [read the modifiers documentation](https://popper.js.org/docs/v2/modifiers/). */ modifiers: _propTypes.default.arrayOf(_propTypes.default.shape({ data: _propTypes.default.object, effect: _propTypes.default.func, enabled: _propTypes.default.bool, fn: _propTypes.default.func, name: _propTypes.default.any.isRequired, options: _propTypes.default.object, phase: _propTypes.default.oneOf(['afterMain', 'afterRead', 'afterWrite', 'beforeMain', 'beforeRead', 'beforeWrite', 'main', 'read', 'write']), requires: _propTypes.default.arrayOf(_propTypes.default.string), requiresIfExists: _propTypes.default.arrayOf(_propTypes.default.string) })), /** * If `true`, the component is shown. */ open: _propTypes.default.bool.isRequired, /** * Popper placement. * @default 'bottom' */ placement: _propTypes.default.oneOf(['auto-end', 'auto-start', 'auto', 'bottom-end', 'bottom-start', 'bottom', 'left-end', 'left-start', 'left', 'right-end', 'right-start', 'right', 'top-end', 'top-start', 'top']), /** * Options provided to the [`Popper.js`](https://popper.js.org/docs/v2/constructors/#options) instance. * @default {} */ popperOptions: _propTypes.default.shape({ modifiers: _propTypes.default.array, onFirstUpdate: _propTypes.default.func, placement: _propTypes.default.oneOf(['auto-end', 'auto-start', 'auto', 'bottom-end', 'bottom-start', 'bottom', 'left-end', 'left-start', 'left', 'right-end', 'right-start', 'right', 'top-end', 'top-start', 'top']), strategy: _propTypes.default.oneOf(['absolute', 'fixed']) }), /** * A ref that points to the used popper instance. */ popperRef: _utils.refType, /** * @ignore */ style: _propTypes.default.object, /** * Help supporting a react-transition-group/Transition component. * @default false */ transition: _propTypes.default.bool } : void 0; var _default = Popper; exports.default = _default;