2018-03-01 12:53:29 -05:00
|
|
|
import React, { HTMLProps } from 'react';
|
2018-05-13 15:24:50 -04:00
|
|
|
import classnames from 'classnames';
|
2018-03-01 12:53:29 -05:00
|
|
|
import './Input.scss';
|
|
|
|
|
2018-05-21 18:10:51 -05:00
|
|
|
interface OwnProps extends HTMLProps<HTMLInputElement> {
|
|
|
|
showInvalidBeforeBlur?: boolean;
|
|
|
|
setInnerRef?(ref: HTMLInputElement | null): void;
|
|
|
|
}
|
|
|
|
|
2018-03-01 12:53:29 -05:00
|
|
|
interface State {
|
|
|
|
hasBlurred: boolean;
|
2018-05-13 15:24:50 -04:00
|
|
|
/**
|
|
|
|
* @description when the input has not had any values inputted yet
|
|
|
|
* e.g. "Pristine" condition
|
|
|
|
*/
|
|
|
|
isStateless: boolean;
|
2018-03-01 12:53:29 -05:00
|
|
|
}
|
|
|
|
|
2018-05-13 15:24:50 -04:00
|
|
|
interface OwnProps extends HTMLProps<HTMLInputElement> {
|
|
|
|
isValid: boolean;
|
|
|
|
showValidAsPlain?: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
class Input extends React.Component<OwnProps, State> {
|
2018-03-01 12:53:29 -05:00
|
|
|
public state: State = {
|
2018-05-13 15:24:50 -04:00
|
|
|
hasBlurred: false,
|
|
|
|
isStateless: true
|
2018-03-01 12:53:29 -05:00
|
|
|
};
|
2018-03-03 14:11:05 -05:00
|
|
|
|
2018-03-01 12:53:29 -05:00
|
|
|
public render() {
|
2018-05-21 18:10:51 -05:00
|
|
|
const {
|
|
|
|
setInnerRef,
|
|
|
|
showInvalidBeforeBlur,
|
|
|
|
showValidAsPlain,
|
|
|
|
isValid,
|
|
|
|
...htmlProps
|
|
|
|
} = this.props;
|
2018-05-13 15:24:50 -04:00
|
|
|
const hasValue = !!this.props.value && this.props.value.toString().length > 0;
|
|
|
|
const classname = classnames(
|
|
|
|
this.props.className,
|
|
|
|
'input-group-input',
|
2018-05-13 21:39:54 -04:00
|
|
|
this.state.isStateless ? '' : isValid ? (showValidAsPlain ? '' : '') : `invalid`,
|
2018-05-21 18:10:51 -05:00
|
|
|
(showInvalidBeforeBlur || this.state.hasBlurred) && 'has-blurred',
|
2018-05-13 15:24:50 -04:00
|
|
|
hasValue && 'has-value'
|
|
|
|
);
|
|
|
|
|
2018-03-01 12:53:29 -05:00
|
|
|
return (
|
|
|
|
<input
|
2018-05-13 15:24:50 -04:00
|
|
|
{...htmlProps}
|
2018-05-21 18:10:51 -05:00
|
|
|
ref={node => setInnerRef && setInnerRef(node)}
|
2018-03-01 12:53:29 -05:00
|
|
|
onBlur={e => {
|
|
|
|
this.setState({ hasBlurred: true });
|
|
|
|
if (this.props && this.props.onBlur) {
|
|
|
|
this.props.onBlur(e);
|
|
|
|
}
|
|
|
|
}}
|
2018-05-13 15:24:50 -04:00
|
|
|
onChange={this.handleOnChange}
|
2018-03-03 14:11:05 -05:00
|
|
|
onWheel={this.props.type === 'number' ? this.preventNumberScroll : undefined}
|
2018-05-13 15:24:50 -04:00
|
|
|
className={classname}
|
2018-03-01 12:53:29 -05:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
2018-03-03 14:11:05 -05:00
|
|
|
|
2018-05-13 15:24:50 -04:00
|
|
|
private handleOnChange = (args: React.FormEvent<HTMLInputElement>) => {
|
|
|
|
if (this.state.isStateless) {
|
|
|
|
this.setState({ isStateless: false });
|
|
|
|
}
|
|
|
|
if (this.props.onChange) {
|
|
|
|
this.props.onChange(args);
|
|
|
|
}
|
|
|
|
};
|
2018-03-03 14:11:05 -05:00
|
|
|
// When number inputs are scrolled on while in focus, the number changes. So we blur
|
|
|
|
// it if it's focused to prevent that behavior, without preventing the scroll.
|
|
|
|
private preventNumberScroll(ev: React.WheelEvent<HTMLInputElement>) {
|
|
|
|
if (document.activeElement === ev.currentTarget) {
|
|
|
|
ev.currentTarget.blur();
|
|
|
|
}
|
|
|
|
}
|
2018-03-01 12:53:29 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
export default Input;
|