import * as React from 'react'; import classNames from 'classnames'; import { composeRef } from 'rc-util/lib/ref'; import SearchOutlined from '@ant-design/icons/SearchOutlined'; import Input, { InputProps } from './Input'; import Button from '../button'; import SizeContext from '../config-provider/SizeContext'; import { ConfigContext } from '../config-provider'; import { cloneElement } from '../_util/reactNode'; export interface SearchProps extends InputProps { inputPrefixCls?: string; onSearch?: ( value: string, event?: | React.ChangeEvent | React.MouseEvent | React.KeyboardEvent, ) => 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, ...restProps } = props; const { getPrefixCls, direction } = React.useContext(ConfigContext); const contextSize = React.useContext(SizeContext); const size = customizeSize || contextSize; const inputRef = React.useRef(null); const onChange = (e: React.ChangeEvent) => { if (e && e.target && e.type === 'click' && customOnSearch) { customOnSearch((e as React.ChangeEvent).target.value, e); } if (customOnChange) { 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); } }; const prefixCls = getPrefixCls('input-search', customizePrefixCls); const inputPrefixCls = getPrefixCls('input', customizeInputPrefixCls); const searchIcon = typeof enterButton === 'boolean' || typeof enterButton === 'undefined' ? ( ) : 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: onSearch, 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, ); return ( (inputRef, ref)} onPressEnter={onSearch} {...restProps} size={size} prefixCls={inputPrefixCls} addonAfter={button} suffix={suffix} onChange={onChange} className={cls} disabled={disabled} /> ); }); Search.displayName = 'Search'; export default Search;