ant-design/components/auto-complete/index.tsx

161 lines
4.5 KiB
TypeScript
Raw Normal View History

import * as React from 'react';
import { Option, OptGroup } from 'rc-select';
2016-07-25 17:46:45 +08:00
import classNames from 'classnames';
import InputElement from './InputElement';
2018-12-13 22:03:12 +08:00
import Input, { InputProps } from '../input';
import Select, { AbstractSelectProps, SelectValue, OptionProps, OptGroupProps } from '../select';
import { ConfigConsumer, ConfigConsumerProps } from '../config-provider';
import { Omit } from '../_util/type';
2018-12-07 20:02:01 +08:00
export interface DataSourceItemObject {
value: string;
text: string;
}
export type DataSourceItemType =
2018-12-07 20:02:01 +08:00
| string
| DataSourceItemObject
| React.ReactElement<OptionProps>
| React.ReactElement<OptGroupProps>;
2017-12-14 10:49:44 +08:00
export interface AutoCompleteInputProps {
2019-06-24 11:29:58 +08:00
onChange?: React.FormEventHandler<any>;
value: any;
}
export type ValidInputElement =
2018-12-07 20:02:01 +08:00
| HTMLInputElement
| HTMLTextAreaElement
| React.ReactElement<AutoCompleteInputProps>;
export interface AutoCompleteProps extends Omit<AbstractSelectProps, 'loading'> {
value?: SelectValue;
defaultValue?: SelectValue;
dataSource?: DataSourceItemType[];
dropdownMenuStyle?: React.CSSProperties;
autoFocus?: boolean;
backfill?: boolean;
optionLabelProp?: string;
onChange?: (value: SelectValue) => void;
2019-06-24 11:29:58 +08:00
onSelect?: (value: SelectValue, option: Object) => any;
onBlur?: (value: SelectValue) => void;
onFocus?: () => void;
2018-12-07 20:02:01 +08:00
children?:
| ValidInputElement
2018-12-13 22:03:12 +08:00
| React.ReactElement<InputProps>
2018-12-07 20:02:01 +08:00
| React.ReactElement<OptionProps>
| Array<React.ReactElement<OptionProps>>;
}
function isSelectOptionOrSelectOptGroup(child: any): Boolean {
return child && child.type && (child.type.isSelectOption || child.type.isSelectOptGroup);
}
export default class AutoComplete extends React.Component<AutoCompleteProps, {}> {
static Option = Option as React.ClassicComponentClass<OptionProps>;
2019-08-05 18:38:10 +08:00
static OptGroup = OptGroup as React.ClassicComponentClass<OptGroupProps>;
2016-07-25 17:46:45 +08:00
static defaultProps = {
transitionName: 'slide-up',
optionLabelProp: 'children',
choiceTransitionName: 'zoom',
showSearch: false,
2017-05-03 15:33:40 +08:00
filterOption: false,
2016-07-25 17:46:45 +08:00
};
private select: any;
2019-08-05 18:38:10 +08:00
saveSelect = (node: any) => {
this.select = node;
};
getInputElement = () => {
const { children } = this.props;
2018-12-07 20:02:01 +08:00
const element =
children && React.isValidElement(children) && children.type !== Option ? (
React.Children.only(this.props.children)
) : (
<Input />
);
2019-01-28 11:10:56 +08:00
const elementProps = { ...(element as React.ReactElement<any>).props };
// https://github.com/ant-design/ant-design/pull/7742
delete elementProps.children;
2018-12-07 20:02:01 +08:00
return <InputElement {...elementProps}>{element}</InputElement>;
};
2017-01-20 18:28:09 +08:00
focus() {
this.select.focus();
}
blur() {
this.select.blur();
}
renderAutoComplete = ({ getPrefixCls }: ConfigConsumerProps) => {
2018-11-08 18:02:31 +08:00
const {
prefixCls: customizePrefixCls,
2018-12-07 20:02:01 +08:00
size,
className = '',
notFoundContent,
optionLabelProp,
dataSource,
children,
2016-07-25 17:46:45 +08:00
} = this.props;
const prefixCls = getPrefixCls('select', customizePrefixCls);
2016-07-25 17:46:45 +08:00
const cls = classNames({
[`${prefixCls}-lg`]: size === 'large',
[`${prefixCls}-sm`]: size === 'small',
[className]: !!className,
[`${prefixCls}-show-search`]: true,
[`${prefixCls}-auto-complete`]: true,
2016-07-25 17:46:45 +08:00
});
let options;
const childArray = React.Children.toArray(children);
2018-12-07 20:02:01 +08:00
if (childArray.length && isSelectOptionOrSelectOptGroup(childArray[0])) {
options = children;
} else {
2018-12-07 20:02:01 +08:00
options = dataSource
? dataSource.map(item => {
if (React.isValidElement(item)) {
return item;
}
switch (typeof item) {
case 'string':
return <Option key={item}>{item}</Option>;
case 'object':
return (
<Option key={(item as DataSourceItemObject).value}>
{(item as DataSourceItemObject).text}
</Option>
);
default:
throw new Error(
'AutoComplete[dataSource] only supports type `string[] | Object[]`.',
);
}
})
: [];
}
2016-07-25 17:46:45 +08:00
return (
<Select
{...this.props}
2016-07-25 17:46:45 +08:00
className={cls}
2018-06-25 11:58:00 +08:00
mode={Select.SECRET_COMBOBOX_MODE_DO_NOT_USE}
2016-07-25 17:46:45 +08:00
optionLabelProp={optionLabelProp}
getInputElement={this.getInputElement}
notFoundContent={notFoundContent}
ref={this.saveSelect}
>
2016-07-25 17:46:45 +08:00
{options}
</Select>
);
2018-12-07 20:02:01 +08:00
};
render() {
2018-12-07 20:02:01 +08:00
return <ConfigConsumer>{this.renderAutoComplete}</ConfigConsumer>;
}
2016-07-25 17:46:45 +08:00
}