import * as React from 'react'; import classNames from 'classnames'; import omit from 'omit.js'; import Group from './Group'; import Search from './Search'; import TextArea from './TextArea'; import Password from './Password'; import { Omit, LiteralUnion } from '../_util/type'; import ClearableLabeledInput, { hasPrefixSuffix } from './ClearableLabeledInput'; import { ConfigConsumer, ConfigConsumerProps } from '../config-provider'; import SizeContext, { SizeType } from '../config-provider/SizeContext'; import devWarning from '../_util/devWarning'; export interface InputProps extends Omit, 'size' | 'prefix' | 'type'> { prefixCls?: string; size?: SizeType; // ref: https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input#%3Cinput%3E_types type?: LiteralUnion< | 'button' | 'checkbox' | 'color' | 'date' | 'datetime-local' | 'email' | 'file' | 'hidden' | 'image' | 'month' | 'number' | 'password' | 'radio' | 'range' | 'reset' | 'search' | 'submit' | 'tel' | 'text' | 'time' | 'url' | 'week', string >; onPressEnter?: React.KeyboardEventHandler; addonBefore?: React.ReactNode; addonAfter?: React.ReactNode; prefix?: React.ReactNode; suffix?: React.ReactNode; allowClear?: boolean; bordered?: boolean; } export function fixControlledValue(value: T) { if (typeof value === 'undefined' || value === null) { return ''; } return value; } export function resolveOnChange( target: HTMLInputElement | HTMLTextAreaElement, e: | React.ChangeEvent | React.MouseEvent, onChange?: (event: React.ChangeEvent) => void, ) { if (onChange) { let event = e; if (e.type === 'click') { // click clear icon event = Object.create(e); event.target = target; event.currentTarget = target; const originalInputValue = target.value; // change target ref value cause e.target.value should be '' when clear input target.value = ''; onChange(event as React.ChangeEvent); // reset target ref value target.value = originalInputValue; return; } onChange(event as React.ChangeEvent); } } export function getInputClassName( prefixCls: string, bordered: boolean, size?: SizeType, disabled?: boolean, direction?: any, ) { return classNames(prefixCls, { [`${prefixCls}-sm`]: size === 'small', [`${prefixCls}-lg`]: size === 'large', [`${prefixCls}-disabled`]: disabled, [`${prefixCls}-rtl`]: direction === 'rtl', [`${prefixCls}-borderless`]: !bordered, }); } export interface InputState { value: any; focused: boolean; /** `value` from prev props */ prevValue: any; } 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', }; input: HTMLInputElement; clearableInput: ClearableLabeledInput; removePasswordTimeout: number; direction: any = 'ltr'; constructor(props: InputProps) { super(props); const value = typeof props.value === 'undefined' ? props.defaultValue : props.value; this.state = { value, focused: false, // eslint-disable-next-line react/no-unused-state prevValue: props.value, }; } static getDerivedStateFromProps(nextProps: InputProps, { prevValue }: InputState) { const newState: Partial = { prevValue: nextProps.value }; if (nextProps.value !== undefined || prevValue !== nextProps.value) { newState.value = nextProps.value; } return newState; } componentDidMount() { this.clearPasswordValueAttribute(); } // Since polyfill `getSnapshotBeforeUpdate` need work with `componentDidUpdate`. // We keep an empty function here. componentDidUpdate() {} getSnapshotBeforeUpdate(prevProps: InputProps) { if (hasPrefixSuffix(prevProps) !== hasPrefixSuffix(this.props)) { devWarning( this.input !== document.activeElement, 'Input', `When Input is focused, dynamic add or remove prefix / suffix will make it lose focus caused by dom structure change. Read more: https://ant.design/components/input/#FAQ`, ); } return null; } componentWillUnmount() { if (this.removePasswordTimeout) { clearTimeout(this.removePasswordTimeout); } } focus = () => { this.input.focus(); }; blur() { this.input.blur(); } select() { this.input.select(); } saveClearableInput = (input: ClearableLabeledInput) => { this.clearableInput = input; }; saveInput = (input: HTMLInputElement) => { this.input = input; }; onFocus: React.FocusEventHandler = e => { const { onFocus } = this.props; this.setState({ focused: true }, this.clearPasswordValueAttribute); if (onFocus) { onFocus(e); } }; onBlur: React.FocusEventHandler = e => { const { onBlur } = this.props; this.setState({ focused: false }, this.clearPasswordValueAttribute); if (onBlur) { onBlur(e); } }; setValue(value: string, callback?: () => void) { if (this.props.value === undefined) { this.setState({ value }, callback); } } handleReset = (e: React.MouseEvent) => { this.setValue('', () => { this.focus(); }); resolveOnChange(this.input, e, this.props.onChange); }; renderInput = ( prefixCls: string, size: SizeType | undefined, bordered: boolean, input: ConfigConsumerProps['input'] = {}, ) => { const { className, addonBefore, addonAfter, size: customizeSize, disabled } = this.props; // Fix https://fb.me/react-unknown-prop const otherProps = omit(this.props, [ 'prefixCls', 'onPressEnter', 'addonBefore', 'addonAfter', 'prefix', 'suffix', 'allowClear', // Input elements must be either controlled or uncontrolled, // specify either the value prop, or the defaultValue prop, but not both. 'defaultValue', 'size', 'inputType', 'bordered', ]); return ( ); }; clearPasswordValueAttribute = () => { // https://github.com/ant-design/ant-design/issues/20541 this.removePasswordTimeout = setTimeout(() => { if ( this.input && this.input.getAttribute('type') === 'password' && this.input.hasAttribute('value') ) { this.input.removeAttribute('value'); } }); }; handleChange = (e: React.ChangeEvent) => { this.setValue(e.target.value, this.clearPasswordValueAttribute); resolveOnChange(this.input, e, this.props.onChange); }; handleKeyDown = (e: React.KeyboardEvent) => { const { onPressEnter, onKeyDown } = this.props; if (e.keyCode === 13 && onPressEnter) { onPressEnter(e); } if (onKeyDown) { onKeyDown(e); } }; renderComponent = ({ getPrefixCls, direction, input }: ConfigConsumerProps) => { const { value, focused } = this.state; const { prefixCls: customizePrefixCls, bordered = true } = this.props; const prefixCls = getPrefixCls('input', customizePrefixCls); this.direction = direction; return ( {size => ( )} ); }; render() { return {this.renderComponent}; } } export default Input;