import * as React from 'react'; import classNames from 'classnames'; import SearchOutlined from '@ant-design/icons/SearchOutlined'; import LoadingOutlined from '@ant-design/icons/LoadingOutlined'; import Input, { InputProps } from './Input'; import Button from '../button'; import SizeContext, { SizeType } from '../config-provider/SizeContext'; import { ConfigConsumer, ConfigConsumerProps } from '../config-provider'; import { cloneElement, replaceElement } 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 inputRef = (ref as any) || React.createRef(); const onChange = (e: React.ChangeEvent) => { const { onChange: customOnChange, onSearch: customOnSearch } = props; 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) => { const { onSearch: customOnSearch, loading, disabled } = props; if (loading || disabled) { return; } if (customOnSearch) { customOnSearch(inputRef.current.input.value, e); } }; const renderLoading = (prefixCls: string) => { const { enterButton, size: customizeSize } = props; if (enterButton) { return ( {size => ( )} ); } return ; }; const renderSuffix = (prefixCls: string) => { const { suffix, enterButton, loading } = props; if (loading && !enterButton) { return [suffix, renderLoading(prefixCls)]; } if (enterButton) return suffix; const icon = ( ); if (suffix) { return [ replaceElement(suffix, null, { key: 'suffix', }), icon, ]; } return icon; }; const renderAddonAfter = (prefixCls: string, size: SizeType) => { const { enterButton, disabled, addonAfter, loading } = props; const btnClassName = `${prefixCls}-button`; if (loading && enterButton) { return [renderLoading(prefixCls), addonAfter]; } if (!enterButton) return addonAfter; 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) { return [ button, replaceElement(addonAfter, null, { key: 'addonAfter', }), ]; } return button; }; const renderSearch = ({ getPrefixCls, direction }: ConfigConsumerProps) => { const { prefixCls: customizePrefixCls, inputPrefixCls: customizeInputPrefixCls, enterButton, className, size: customizeSize, ...restProps } = props; delete (restProps as any).onSearch; delete (restProps as any).loading; const prefixCls = getPrefixCls('input-search', customizePrefixCls); const inputPrefixCls = getPrefixCls('input', customizeInputPrefixCls); const getClassName = (size: SizeType) => { let inputClassName; if (enterButton) { inputClassName = classNames(prefixCls, className, { [`${prefixCls}-rtl`]: direction === 'rtl', [`${prefixCls}-enter-button`]: !!enterButton, [`${prefixCls}-${size}`]: !!size, }); } else { inputClassName = classNames(prefixCls, className, { [`${prefixCls}-rtl`]: direction === 'rtl', }); } return inputClassName; }; return ( {size => ( )} ); }; return {renderSearch}; }); Search.defaultProps = { enterButton: false, }; export default Search;