import * as React from 'react'; import classNames from 'classnames'; import omit from 'rc-util/lib/omit'; import CloseOutlined from '@ant-design/icons/CloseOutlined'; import CheckableTag from './CheckableTag'; import { ConfigContext } from '../config-provider'; import type { PresetColorType, PresetStatusColorType } from '../_util/colors'; import { PresetColorTypes, PresetStatusColorTypes } from '../_util/colors'; import Wave from '../_util/wave'; import type { LiteralUnion } from '../_util/type'; import useStyle from './style'; export { CheckableTagProps } from './CheckableTag'; export interface TagProps extends React.HTMLAttributes { prefixCls?: string; className?: string; color?: LiteralUnion; closable?: boolean; closeIcon?: React.ReactNode; visible?: boolean; onClose?: (e: React.MouseEvent) => void; style?: React.CSSProperties; icon?: React.ReactNode; } const PresetColorRegex = new RegExp(`^(${PresetColorTypes.join('|')})(-inverse)?$`); const PresetStatusColorRegex = new RegExp(`^(${PresetStatusColorTypes.join('|')})$`); export interface TagType extends React.ForwardRefExoticComponent> { CheckableTag: typeof CheckableTag; } const InternalTag: React.ForwardRefRenderFunction = ( { prefixCls: customizePrefixCls, className, style, children, icon, color, onClose, closeIcon, closable = false, ...props }, ref, ) => { const { getPrefixCls, direction } = React.useContext(ConfigContext); const [visible, setVisible] = React.useState(true); React.useEffect(() => { if ('visible' in props) { setVisible(props.visible!); } }, [props.visible]); const isPresetColor = (): boolean => { if (!color) { return false; } return PresetColorRegex.test(color) || PresetStatusColorRegex.test(color); }; const tagStyle = { backgroundColor: color && !isPresetColor() ? color : undefined, ...style, }; const presetColor = isPresetColor(); const prefixCls = getPrefixCls('tag', customizePrefixCls); // Style const [wrapSSR, hashId] = useStyle(prefixCls); const tagClassName = classNames( prefixCls, { [`${prefixCls}-${color}`]: presetColor, [`${prefixCls}-has-color`]: color && !presetColor, [`${prefixCls}-hidden`]: !visible, [`${prefixCls}-rtl`]: direction === 'rtl', }, className, hashId, ); const handleCloseClick = (e: React.MouseEvent) => { e.stopPropagation(); onClose?.(e); if (e.defaultPrevented) { return; } if (!('visible' in props)) { setVisible(false); } }; const renderCloseIcon = () => { if (closable) { return closeIcon ? ( {closeIcon} ) : ( ); } return null; }; const isNeedWave = 'onClick' in props || (children && (children as React.ReactElement).type === 'a'); const tagProps = omit(props, ['visible']); const iconNode = icon || null; const kids = iconNode ? ( <> {iconNode} {children} ) : ( children ); const tagNode = ( {kids} {renderCloseIcon()} ); return wrapSSR(isNeedWave ? {tagNode} : tagNode); }; const Tag = React.forwardRef(InternalTag) as TagType; Tag.displayName = 'Tag'; Tag.CheckableTag = CheckableTag; export default Tag;