import * as React from 'react'; import toArray from '@rc-component/util/lib/Children/toArray'; import omit from '@rc-component/util/lib/omit'; import classNames from 'classnames'; import type { BaseSelectRef } from 'rc-select'; import { useZIndex } from '../_util/hooks/useZIndex'; import genPurePanel from '../_util/PurePanel'; import type { InputStatus } from '../_util/statusUtils'; import { devUseWarning } from '../_util/warning'; import type { ConfigConsumerProps } from '../config-provider'; import { ConfigContext } from '../config-provider'; import type { BaseOptionType, DefaultOptionType, InternalSelectProps, RefSelectProps, SelectProps, } from '../select'; import Select from '../select'; const { Option } = Select; export interface DataSourceItemObject { value: string; text: string; } export type DataSourceItemType = DataSourceItemObject | React.ReactNode; export interface AutoCompleteProps< ValueType = any, OptionType extends BaseOptionType | DefaultOptionType = DefaultOptionType, > extends Omit< InternalSelectProps, 'loading' | 'mode' | 'optionLabelProp' | 'labelInValue' > { /** @deprecated Please use `options` instead */ dataSource?: DataSourceItemType[]; status?: InputStatus; popupClassName?: string; /** @deprecated Please use `popupMatchSelectWidth` instead */ dropdownMatchSelectWidth?: boolean | number; popupMatchSelectWidth?: boolean | number; } function isSelectOptionOrSelectOptGroup(child: any): boolean { return child?.type && (child.type.isSelectOption || child.type.isSelectOptGroup); } const AutoComplete: React.ForwardRefRenderFunction = ( props, ref, ) => { const { prefixCls: customizePrefixCls, className, popupClassName, children, dataSource } = props; const childNodes: React.ReactElement[] = toArray(children); // ============================= Input ============================= let customizeInput: React.ReactElement | undefined; if ( childNodes.length === 1 && React.isValidElement(childNodes[0]) && !isSelectOptionOrSelectOptGroup(childNodes[0]) ) { [customizeInput] = childNodes; } const getInputElement = customizeInput ? (): React.ReactElement => customizeInput! : undefined; // ============================ Options ============================ let optionChildren: React.ReactNode; // [Legacy] convert `children` or `dataSource` into option children if (childNodes.length && isSelectOptionOrSelectOptGroup(childNodes[0])) { optionChildren = children; } else { optionChildren = dataSource ? dataSource.map((item) => { if (React.isValidElement(item)) { return item; } switch (typeof item) { case 'string': return ( ); case 'object': { const { value: optionValue } = item as DataSourceItemObject; return ( ); } default: return undefined; } }) : []; } if (process.env.NODE_ENV !== 'production') { const warning = devUseWarning('AutoComplete'); warning.deprecated(!('dataSource' in props), 'dataSource', 'options'); warning( !customizeInput || !('size' in props), 'usage', 'You need to control style self instead of setting `size` when using customize input.', ); } const { getPrefixCls } = React.useContext(ConfigContext); const prefixCls = getPrefixCls('select', customizePrefixCls); // ============================ zIndex ============================ const [zIndex] = useZIndex('SelectLike', props.dropdownStyle?.zIndex as number); return ( ); }; const RefAutoComplete = React.forwardRef( AutoComplete, ) as unknown as (< ValueType = any, OptionType extends BaseOptionType | DefaultOptionType = DefaultOptionType, >( props: React.PropsWithChildren> & React.RefAttributes, ) => React.ReactElement) & { displayName?: string; Option: typeof Option; _InternalPanelDoNotUseOrYouWillBeFired: typeof PurePanel; }; // We don't care debug panel /* istanbul ignore next */ const PurePanel = genPurePanel(RefAutoComplete, 'dropdownAlign', (props: any) => omit(props, ['visible']), ); RefAutoComplete.Option = Option; RefAutoComplete._InternalPanelDoNotUseOrYouWillBeFired = PurePanel; if (process.env.NODE_ENV !== 'production') { RefAutoComplete.displayName = 'AutoComplete'; } export default RefAutoComplete;