import React, { useContext, useMemo } from 'react'; import classNames from 'classnames'; import omit from 'rc-util/lib/omit'; import { devUseWarning } from '../_util/warning'; import Badge from '../badge'; import type { ConfigConsumerProps } from '../config-provider'; import { ConfigContext } from '../config-provider'; import Tooltip from '../tooltip'; import FloatButtonGroupContext from './context'; import Content from './FloatButtonContent'; import type { CompoundedComponent, FloatButtonBadgeProps, FloatButtonContentProps, FloatButtonElement, FloatButtonProps, FloatButtonShape, } from './interface'; import useStyle from './style'; import useCSSVarCls from '../config-provider/hooks/useCSSVarCls'; export const floatButtonPrefixCls = 'float-btn'; const FloatButton = React.forwardRef((props, ref) => { const { prefixCls: customizePrefixCls, className, rootClassName, type = 'default', shape = 'circle', icon, description, tooltip, badge = {}, ...restProps } = props; const { getPrefixCls, direction } = useContext(ConfigContext); const groupShape = useContext(FloatButtonGroupContext); const prefixCls = getPrefixCls(floatButtonPrefixCls, customizePrefixCls); const rootCls = useCSSVarCls(prefixCls); const [wrapCSSVar, hashId, cssVarCls] = useStyle(prefixCls, rootCls); const mergeShape = groupShape || shape; const classString = classNames( hashId, cssVarCls, rootCls, prefixCls, className, rootClassName, `${prefixCls}-${type}`, `${prefixCls}-${mergeShape}`, { [`${prefixCls}-rtl`]: direction === 'rtl', }, ); // 虽然在 ts 中已经 omit 过了,但是为了防止多余的属性被透传进来,这里再 omit 一遍,以防万一 const badgeProps = useMemo( () => omit(badge, ['title', 'children', 'status', 'text'] as any[]), [badge], ); const contentProps = useMemo( () => ({ prefixCls, description, icon, type }), [prefixCls, description, icon, type], ); let buttonNode = (
); if ('badge' in props) { buttonNode = {buttonNode}; } if ('tooltip' in props) { buttonNode = ( {buttonNode} ); } if (process.env.NODE_ENV !== 'production') { const warning = devUseWarning('FloatButton'); warning( !(shape === 'circle' && description), 'usage', 'supported only when `shape` is `square`. Due to narrow space for text, short sentence is recommended.', ); } return wrapCSSVar( props.href ? ( {buttonNode} ) : ( ), ); }) as CompoundedComponent; if (process.env.NODE_ENV !== 'production') { FloatButton.displayName = 'FloatButton'; } export default FloatButton;