import React, { memo, useContext } from 'react'; import CloseOutlined from '@ant-design/icons/CloseOutlined'; import FileTextOutlined from '@ant-design/icons/FileTextOutlined'; import classNames from 'classnames'; import CSSMotion from 'rc-motion'; import { useEvent } from 'rc-util'; import useMergedState from 'rc-util/lib/hooks/useMergedState'; import { devUseWarning } from '../_util/warning'; import type { ConfigConsumerProps } from '../config-provider'; import { ConfigContext } from '../config-provider'; import useCSSVarCls from '../config-provider/hooks/useCSSVarCls'; import { FloatButtonGroupProvider } from './context'; import FloatButton, { floatButtonPrefixCls } from './FloatButton'; import type { FloatButtonGroupProps } from './interface'; import useStyle from './style'; const FloatButtonGroup: React.FC = (props) => { const { prefixCls: customizePrefixCls, className, style, shape = 'circle', type = 'default', icon = , closeIcon, description, trigger, children, onOpenChange, open: customOpen, onClick: onTriggerButtonClick, ...floatButtonProps } = props; const { direction, getPrefixCls, floatButtonGroup } = useContext(ConfigContext); const mergedCloseIcon = closeIcon ?? floatButtonGroup?.closeIcon ?? ; const prefixCls = getPrefixCls(floatButtonPrefixCls, customizePrefixCls); const rootCls = useCSSVarCls(prefixCls); const [wrapCSSVar, hashId, cssVarCls] = useStyle(prefixCls, rootCls); const groupPrefixCls = `${prefixCls}-group`; const groupCls = classNames(groupPrefixCls, hashId, cssVarCls, rootCls, className, { [`${groupPrefixCls}-rtl`]: direction === 'rtl', [`${groupPrefixCls}-${shape}`]: shape, [`${groupPrefixCls}-${shape}-shadow`]: !trigger, }); const wrapperCls = classNames(hashId, `${groupPrefixCls}-wrap`); const [open, setOpen] = useMergedState(false, { value: customOpen }); const floatButtonGroupRef = React.useRef(null); // ========================== Open ========================== const hoverTrigger = trigger === 'hover'; const clickTrigger = trigger === 'click'; const triggerOpen = useEvent((nextOpen: boolean) => { if (open !== nextOpen) { setOpen(nextOpen); onOpenChange?.(nextOpen); } }); // ===================== Trigger: Hover ===================== const onMouseEnter = () => { if (hoverTrigger) { triggerOpen(true); } }; const onMouseLeave = () => { if (hoverTrigger) { triggerOpen(false); } }; // ===================== Trigger: Click ===================== const onInternalTriggerButtonClick: FloatButtonGroupProps['onClick'] = (e) => { if (clickTrigger) { triggerOpen(!open); } onTriggerButtonClick?.(e); }; React.useEffect(() => { if (clickTrigger) { const onDocClick = (e: MouseEvent) => { // Skip if click on the group if (floatButtonGroupRef.current?.contains(e.target as Node)) { return; } triggerOpen(false); }; document.addEventListener('click', onDocClick, { capture: true, }); return () => { document.removeEventListener('click', onDocClick, { capture: true, }); }; } }, [clickTrigger]); // ======================== Warning ========================= if (process.env.NODE_ENV !== 'production') { const warning = devUseWarning('FloatButton.Group'); warning( !('open' in props) || !!trigger, 'usage', '`open` need to be used together with `trigger`', ); } // ========================= Render ========================= return wrapCSSVar(
{trigger && ['click', 'hover'].includes(trigger) ? ( <> {({ className: motionClassName }) => (
{children}
)}
) : ( children )}
, ); }; export default memo(FloatButtonGroup);