import React, { memo, useCallback, useContext, useEffect } 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 useMergedState from 'rc-util/lib/hooks/useMergedState'; import { useZIndex } from '../_util/hooks/useZIndex'; 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, FloatButtonRef } 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, ...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, }); // ============================ zIndex ============================ const [zIndex] = useZIndex('FloatButton', style?.zIndex as number); const mergedStyle: React.CSSProperties = { ...style, zIndex }; const wrapperCls = classNames(hashId, `${groupPrefixCls}-wrap`); const [open, setOpen] = useMergedState(false, { value: customOpen }); const floatButtonGroupRef = React.useRef(null); const floatButtonRef = React.useRef(null); const hoverAction = React.useMemo>(() => { const hoverTypeAction = { onMouseEnter() { setOpen(true); onOpenChange?.(true); }, onMouseLeave() { setOpen(false); onOpenChange?.(false); }, }; return trigger === 'hover' ? hoverTypeAction : {}; }, [trigger]); const handleOpenChange = () => { setOpen((prevState) => { onOpenChange?.(!prevState); return !prevState; }); }; const onClick = useCallback( (e: MouseEvent) => { if (floatButtonGroupRef.current?.contains(e.target as Node)) { if (floatButtonRef.current?.contains(e.target as Node)) { handleOpenChange(); } return; } setOpen(false); onOpenChange?.(false); }, [trigger], ); useEffect(() => { if (trigger === 'click') { document.addEventListener('click', onClick); return () => { document.removeEventListener('click', onClick); }; } }, [trigger]); // =================== 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`', ); } return wrapCSSVar(
{trigger && ['click', 'hover'].includes(trigger) ? ( <> {({ className: motionClassName }) => (
{children}
)}
) : ( children )}
, ); }; export default memo(FloatButtonGroup);