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, FloatButtonProps, FloatButtonShape, } from './interface'; import useStyle from './style'; export const floatButtonPrefixCls = 'float-btn'; const FloatButton: React.ForwardRefRenderFunction< HTMLAnchorElement | HTMLButtonElement, FloatButtonProps > = (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 [wrapSSR, hashId] = useStyle(prefixCls); const mergeShape = groupShape || shape; const classString = classNames( hashId, 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 wrapSSR( props.href ? ( } {...restProps} className={classString}> {buttonNode} ) : ( ), ); }; const ForwardFloatButton = React.forwardRef< HTMLAnchorElement | HTMLButtonElement, FloatButtonProps >(FloatButton) as CompoundedComponent; if (process.env.NODE_ENV !== 'production') { ForwardFloatButton.displayName = 'FloatButton'; } export default ForwardFloatButton;