import React, { Component, cloneElement } from 'react'; import PropTypes from 'prop-types'; import classNames from 'classnames'; import omit from 'omit.js'; import Group from './Group'; import Search from './Search'; import TextArea from './TextArea'; function fixControlledValue(value) { if (typeof value === 'undefined' || value === null) { return ''; } return value; } export interface AbstractInputProps { prefixCls?: string; className?: string; defaultValue?: any; value?: any; style?: React.CSSProperties; } export interface InputProps extends AbstractInputProps { placeholder?: string; type?: string; id?: number | string; name?: string; size?: 'large' | 'default' | 'small'; disabled?: boolean; readOnly?: boolean; addonBefore?: React.ReactNode; addonAfter?: React.ReactNode; onPressEnter?: React.FormEventHandler; onKeyDown?: React.FormEventHandler; onChange?: React.ChangeEventHandler; onClick?: React.FormEventHandler; onFocus?: React.FormEventHandler; onBlur?: React.FormEventHandler; autoComplete?: string; prefix?: React.ReactNode; suffix?: React.ReactNode; spellCheck?: boolean; autoFocus?: boolean; } export default class Input extends Component { static Group: typeof Group; static Search: typeof Search; static TextArea: typeof TextArea; static defaultProps = { prefixCls: 'ant-input', type: 'text', disabled: false, }; static propTypes = { type: PropTypes.string, id: PropTypes.oneOfType([ PropTypes.string, PropTypes.number, ]), size: PropTypes.oneOf(['small', 'default', 'large']), disabled: PropTypes.bool, value: PropTypes.any, defaultValue: PropTypes.any, className: PropTypes.string, addonBefore: PropTypes.node, addonAfter: PropTypes.node, prefixCls: PropTypes.string, autosize: PropTypes.oneOfType([PropTypes.bool, PropTypes.object]), onPressEnter: PropTypes.func, onKeyDown: PropTypes.func, onFocus: PropTypes.func, onBlur: PropTypes.func, prefix: PropTypes.node, suffix: PropTypes.node, }; refs: { input: HTMLInputElement; }; handleKeyDown = (e) => { const { onPressEnter, onKeyDown } = this.props; if (e.keyCode === 13 && onPressEnter) { onPressEnter(e); } if (onKeyDown) { onKeyDown(e); } } focus() { this.refs.input.focus(); } blur() { this.refs.input.blur(); } getInputClassName() { const { prefixCls, size, disabled } = this.props; return classNames(prefixCls, { [`${prefixCls}-sm`]: size === 'small', [`${prefixCls}-lg`]: size === 'large', [`${prefixCls}-disabled`]: disabled, }); } renderLabeledInput(children) { const props = this.props; // Not wrap when there is not addons if ((!props.addonBefore && !props.addonAfter)) { return children; } const wrapperClassName = `${props.prefixCls}-group`; const addonClassName = `${wrapperClassName}-addon`; const addonBefore = props.addonBefore ? ( {props.addonBefore} ) : null; const addonAfter = props.addonAfter ? ( {props.addonAfter} ) : null; const className = classNames(`${props.prefixCls}-wrapper`, { [wrapperClassName]: (addonBefore || addonAfter), }); // Need another wrapper for changing display:table to display:inline-block // and put style prop in wrapper if (addonBefore || addonAfter) { return ( {addonBefore} {cloneElement(children, { style: null })} {addonAfter} ); } return ( {addonBefore} {children} {addonAfter} ); } renderLabeledIcon(children) { const { props } = this; if (!('prefix' in props || 'suffix' in props)) { return children; } const prefix = props.prefix ? ( {props.prefix} ) : null; const suffix = props.suffix ? ( {props.suffix} ) : null; return ( {prefix} {cloneElement(children, { style: null, className: this.getInputClassName() })} {suffix} ); } renderInput() { const { value, className } = this.props; // Fix https://fb.me/react-unknown-prop const otherProps = omit(this.props, [ 'prefixCls', 'onPressEnter', 'addonBefore', 'addonAfter', 'prefix', 'suffix', ]); if ('value' in this.props) { otherProps.value = fixControlledValue(value); // Input elements must be either controlled or uncontrolled, // specify either the value prop, or the defaultValue prop, but not both. delete otherProps.defaultValue; } return this.renderLabeledIcon( , ); } render() { if (this.props.type === 'textarea') { return