import * as React from 'react'; import * as 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'; import { ConfigConsumer, ConfigConsumerProps } from '../config-provider'; import Password from './Password'; import { Omit } from '../_util/type'; function fixControlledValue(value: T) { if (typeof value === 'undefined' || value === null) { return ''; } return value; } export interface InputProps extends Omit, 'size' | 'prefix'> { prefixCls?: string; size?: 'large' | 'default' | 'small'; onPressEnter?: React.KeyboardEventHandler; addonBefore?: React.ReactNode; addonAfter?: React.ReactNode; prefix?: React.ReactNode; suffix?: React.ReactNode; } export default class Input extends React.Component { static Group: typeof Group; static Search: typeof Search; static TextArea: typeof TextArea; static Password: typeof Password; static defaultProps = { type: 'text', disabled: false, }; static propTypes = { type: PropTypes.string, id: PropTypes.oneOfType([ PropTypes.string, PropTypes.number, ]), size: PropTypes.oneOf(['small', 'default', 'large']), maxLength: PropTypes.oneOfType([ PropTypes.string, PropTypes.number, ]), disabled: PropTypes.bool, value: PropTypes.any, defaultValue: PropTypes.any, className: PropTypes.string, addonBefore: PropTypes.node, addonAfter: PropTypes.node, prefixCls: PropTypes.string, onPressEnter: PropTypes.func, onKeyDown: PropTypes.func, onKeyUp: PropTypes.func, onFocus: PropTypes.func, onBlur: PropTypes.func, prefix: PropTypes.node, suffix: PropTypes.node, }; input: HTMLInputElement; handleKeyDown = (e: React.KeyboardEvent) => { const { onPressEnter, onKeyDown } = this.props; if (e.keyCode === 13 && onPressEnter) { onPressEnter(e); } if (onKeyDown) { onKeyDown(e); } } focus() { this.input.focus(); } blur() { this.input.blur(); } select() { this.input.select(); } getInputClassName(prefixCls: string) { const { size, disabled } = this.props; return classNames(prefixCls, { [`${prefixCls}-sm`]: size === 'small', [`${prefixCls}-lg`]: size === 'large', [`${prefixCls}-disabled`]: disabled, }); } saveInput = (node: HTMLInputElement) => { this.input = node; } renderLabeledInput(prefixCls: string, children: React.ReactElement) { const props = this.props; // Not wrap when there is not addons if ((!props.addonBefore && !props.addonAfter)) { return children; } const wrapperClassName = `${prefixCls}-group`; const addonClassName = `${wrapperClassName}-addon`; const addonBefore = props.addonBefore ? ( {props.addonBefore} ) : null; const addonAfter = props.addonAfter ? ( {props.addonAfter} ) : null; const className = classNames(`${prefixCls}-wrapper`, { [wrapperClassName]: (addonBefore || addonAfter), }); const groupClassName = classNames(`${prefixCls}-group-wrapper`, { [`${prefixCls}-group-wrapper-sm`]: props.size === 'small', [`${prefixCls}-group-wrapper-lg`]: props.size === 'large', }); // Need another wrapper for changing display:table to display:inline-block // and put style prop in wrapper return ( {addonBefore} {React.cloneElement(children, { style: null })} {addonAfter} ); } renderLabeledIcon(prefixCls: string, children: React.ReactElement) { 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; const affixWrapperCls = classNames(props.className, `${prefixCls}-affix-wrapper`, { [`${prefixCls}-affix-wrapper-sm`]: props.size === 'small', [`${prefixCls}-affix-wrapper-lg`]: props.size === 'large', }); return ( {prefix} {React.cloneElement(children, { style: null, className: this.getInputClassName(prefixCls) })} {suffix} ); } renderInput(prefixCls: string) { 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( prefixCls, , ); } renderComponent = ({ getPrefixCls }: ConfigConsumerProps) => { const { prefixCls: customizePrefixCls } = this.props; const prefixCls = getPrefixCls('input', customizePrefixCls); return this.renderLabeledInput(prefixCls, this.renderInput(prefixCls)); }; render() { return ( {this.renderComponent} ); } }