import * as React from 'react'; import SearchOutlined from '@ant-design/icons/SearchOutlined'; import classNames from 'classnames'; import { composeRef } from 'rc-util/lib/ref'; import { cloneElement } from '../_util/reactNode'; import Button from '../button'; import { ConfigContext } from '../config-provider'; import useSize from '../config-provider/hooks/useSize'; import { useCompactItemContext } from '../space/Compact'; import type { InputProps, InputRef } from './Input'; import Input from './Input'; export interface SearchProps extends InputProps { inputPrefixCls?: string; onSearch?: ( value: string, event?: | React.ChangeEvent | React.MouseEvent | React.KeyboardEvent, info?: { source?: 'clear' | 'input'; }, ) => void; enterButton?: React.ReactNode; loading?: boolean; } const Search = React.forwardRef((props, ref) => { const { prefixCls: customizePrefixCls, inputPrefixCls: customizeInputPrefixCls, className, size: customizeSize, suffix, enterButton = false, addonAfter, loading, disabled, onSearch: customOnSearch, onChange: customOnChange, onCompositionStart, onCompositionEnd, ...restProps } = props; const { getPrefixCls, direction } = React.useContext(ConfigContext); const composedRef = React.useRef(false); const prefixCls = getPrefixCls('input-search', customizePrefixCls); const inputPrefixCls = getPrefixCls('input', customizeInputPrefixCls); const { compactSize } = useCompactItemContext(prefixCls, direction); const size = useSize((ctx) => customizeSize ?? compactSize ?? ctx); const inputRef = React.useRef(null); const onChange = (e: React.ChangeEvent) => { if (e?.target && e.type === 'click' && customOnSearch) { customOnSearch((e as React.ChangeEvent).target.value, e, { source: 'clear', }); } customOnChange?.(e); }; const onMouseDown: React.MouseEventHandler = (e) => { if (document.activeElement === inputRef.current?.input) { e.preventDefault(); } }; const onSearch = (e: React.MouseEvent | React.KeyboardEvent) => { if (customOnSearch) { customOnSearch(inputRef.current?.input?.value!, e, { source: 'input', }); } }; const onPressEnter = (e: React.KeyboardEvent) => { if (composedRef.current || loading) { return; } onSearch(e); }; const searchIcon = typeof enterButton === 'boolean' ? : null; const btnClassName = `${prefixCls}-button`; let button: React.ReactNode; const enterButtonAsElement = (enterButton || {}) as React.ReactElement; const isAntdButton = enterButtonAsElement.type && (enterButtonAsElement.type as typeof Button).__ANT_BUTTON === true; if (isAntdButton || enterButtonAsElement.type === 'button') { button = cloneElement(enterButtonAsElement, { onMouseDown, onClick: (e: React.MouseEvent) => { ( enterButtonAsElement as React.ReactElement<{ onClick?: React.MouseEventHandler; }> )?.props?.onClick?.(e); onSearch(e); }, key: 'enterButton', ...(isAntdButton ? { className: btnClassName, size, } : {}), }); } else { button = ( ); } if (addonAfter) { button = [ button, cloneElement(addonAfter, { key: 'addonAfter', }), ]; } const cls = classNames( prefixCls, { [`${prefixCls}-rtl`]: direction === 'rtl', [`${prefixCls}-${size}`]: !!size, [`${prefixCls}-with-button`]: !!enterButton, }, className, ); const newProps: InputProps = { ...restProps, className: cls, prefixCls: inputPrefixCls, type: 'search', }; const handleOnCompositionStart: React.CompositionEventHandler = (e) => { composedRef.current = true; onCompositionStart?.(e); }; const handleOnCompositionEnd: React.CompositionEventHandler = (e) => { composedRef.current = false; onCompositionEnd?.(e); }; return ( (inputRef, ref)} onPressEnter={onPressEnter} {...newProps} size={size} onCompositionStart={handleOnCompositionStart} onCompositionEnd={handleOnCompositionEnd} addonAfter={button} suffix={suffix} onChange={onChange} disabled={disabled} /> ); }); if (process.env.NODE_ENV !== 'production') { Search.displayName = 'Search'; } export default Search;