UNPKG

zarm

Version:

基于 React 的移动端UI库

209 lines (187 loc) 7.76 kB
import _extends from "@babel/runtime/helpers/extends"; import _defineProperty from "@babel/runtime/helpers/defineProperty"; import _slicedToArray from "@babel/runtime/helpers/slicedToArray"; function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; } function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; } import React, { useEffect, useRef, useState, useMemo, useImperativeHandle, cloneElement, forwardRef } from 'react'; import { createBEM } from '@zarm-design/bem'; import { useClick, useFloating, flip, shift, useInteractions, offset, autoUpdate, arrow, useHover, useFocus, useDismiss } from '@floating-ui/react-dom-interactions'; import Transition from '../transition'; import { canUseDOM, getElementSize, renderToContainer } from '../utils/dom'; import { ConfigContext } from '../config-provider'; import { getTransitionName, getTransformOrigin } from './utils'; import mergeRefs from '../utils/mergeRefs'; var directionMap = { top: 'top', 'top-left': 'top-start', 'top-right': 'top-end', right: 'right', 'right-top': 'right-start', 'right-bottom': 'right-end', bottom: 'bottom', 'bottom-left': 'bottom-start', 'bottom-right': 'bottom-end', left: 'left', 'left-top': 'left-start', 'left-bottom': 'left-end' }; var Popper = /*#__PURE__*/forwardRef(function (props, ref) { var _React$useContext = React.useContext(ConfigContext), prefixCls = _React$useContext.prefixCls, globalMountContainer = _React$useContext.mountContainer; var visible = props.visible, mountContainer = props.mountContainer, direction = props.direction, destroy = props.destroy, trigger = props.trigger, animationDuration = props.animationDuration, hasArrow = props.hasArrow, animationType = props.animationType, content = props.content, children = props.children, arrowPointAtCenter = props.arrowPointAtCenter, onVisibleChange = props.onVisibleChange; var bem = createBEM('popper', { prefixCls: prefixCls }); var isVisible = trigger === 'manual' && visible; var _useState = useState(isVisible), _useState2 = _slicedToArray(_useState, 2), open = _useState2[0], setOpen = _useState2[1]; var arrowRef = useRef(null); var middleware = [offset(function (_ref) { var placement = _ref.placement; var _getElementSize = getElementSize(arrowRef.current), width = _getElementSize.width, height = _getElementSize.height; var side = placement.split('-')[0]; var value = side === 'bottom' || side === 'top' ? height : width; return value; }), flip(), shift()]; if (hasArrow) { middleware.push(arrow({ element: arrowRef })); } var _useFloating = useFloating({ open: open, onOpenChange: function onOpenChange(state) { setOpen(state); onVisibleChange === null || onVisibleChange === void 0 ? void 0 : onVisibleChange(state); }, middleware: middleware, placement: directionMap[direction], whileElementsMounted: autoUpdate }), x = _useFloating.x, y = _useFloating.y, reference = _useFloating.reference, floating = _useFloating.floating, strategy = _useFloating.strategy, context = _useFloating.context, _update = _useFloating.update; useImperativeHandle(ref, function () { return { update: function update() { _update(); } }; }); function computeArrowStyle(ctx) { var _ctx$middlewareData; var _ref2 = (ctx === null || ctx === void 0 ? void 0 : (_ctx$middlewareData = ctx.middlewareData) === null || _ctx$middlewareData === void 0 ? void 0 : _ctx$middlewareData.arrow) || {}, arrowX = _ref2.x, arrowY = _ref2.y; if (arrowRef.current) { var postion = ctx.placement.split('-'); if (!postion[1] || arrowPointAtCenter) { return { left: x != null ? "".concat(arrowX, "px") : '', top: y != null ? "".concat(arrowY, "px") : '' }; } return {}; } } var _useInteractions = useInteractions([useHover(context, { restMs: 100, enabled: trigger === 'hover', delay: { open: props.mouseEnterDelay, close: props.mouseLeaveDelay } }), useClick(context, { enabled: trigger === 'click' }), useFocus(context, { enabled: trigger === 'focus' }), useDismiss(context, { enabled: trigger !== 'manual' })]), getReferenceProps = _useInteractions.getReferenceProps, getFloatingProps = _useInteractions.getFloatingProps; var transitionName = getTransitionName(prefixCls, directionMap[direction], animationType); useEffect(function () { setOpen(isVisible); }, [isVisible]); var hidden = function hidden() { setOpen(false); }; var toolTip = /*#__PURE__*/React.createElement(Transition, { nodeRef: floating, duration: animationDuration, visible: open, onLeaveEnd: hidden, tranisitionName: transitionName, destroy: destroy }, function (_ref3, setNodeRef) { var style = _ref3.style, className = _ref3.className; return renderToContainer(mountContainer !== null && mountContainer !== void 0 ? mountContainer : globalMountContainer, /*#__PURE__*/React.createElement("div", _extends({}, getFloatingProps({ ref: setNodeRef, className: bem([_defineProperty({}, "".concat(direction), true), props.className, className]), style: _objectSpread(_objectSpread(_objectSpread({}, props.style), style), {}, { position: strategy, top: y !== null && y !== void 0 ? y : 0, left: x !== null && x !== void 0 ? x : 0, // animationDuration: `${animationDuration}ms`, transformOrigin: getTransformOrigin(context.placement), WebkitTransformOrigin: getTransformOrigin(context.placement) }) }), { "data-popper-placement": context.placement }), /*#__PURE__*/React.createElement("div", { className: bem('content') }, content, hasArrow && /*#__PURE__*/React.createElement("span", { className: bem('arrow'), ref: arrowRef, style: computeArrowStyle(context) })))); }); var newRef = useMemo(function () { return mergeRefs([reference, children.ref]); }, [reference, children]); var child = /*#__PURE__*/React.isValidElement(children) ? children : /*#__PURE__*/React.createElement("span", null, children); var childrenProps = _objectSpread(_objectSpread({}, children && children.props), trigger === 'contextMenu' && { onContextMenu: function onContextMenu(e) { e.preventDefault(); setOpen(true); } }); return /*#__PURE__*/React.createElement(React.Fragment, null, toolTip, /*#__PURE__*/cloneElement(child, getReferenceProps(_objectSpread({ ref: newRef }, childrenProps)))); }); Popper.defaultProps = { hasArrow: false, destroy: false, arrowPointAtCenter: false, trigger: (canUseDOM && /(iPhone|iPad|iPod|iOS|Android)/i.test(navigator.userAgent) ? 'click' : 'hover') || 'click', direction: 'top', mouseEnterDelay: 150, mouseLeaveDelay: 100, visible: false, animationType: 'zoom-fade', onVisibleChange: function onVisibleChange() {} }; export default Popper;