accounts-frontend/packages/app/components/ui/form/Form.tsx

168 lines
3.8 KiB
TypeScript
Raw Normal View History

2019-12-07 16:58:52 +05:30
import React from 'react';
2016-05-02 12:45:42 +05:30
import classNames from 'classnames';
import logger from 'app/services/logger';
2019-12-07 16:58:52 +05:30
import FormModel from './FormModel';
import styles from './form.scss';
2019-12-07 16:58:52 +05:30
interface Props {
id: string;
isLoading: boolean;
form?: FormModel;
onSubmit: (form: FormModel | FormData) => void | Promise<void>;
onInvalid: (errors: { [errorKey: string]: string }) => void;
children: React.ReactNode;
}
interface State {
isTouched: boolean;
isLoading: boolean;
}
type InputElement = HTMLInputElement | HTMLTextAreaElement;
2019-12-07 16:58:52 +05:30
export default class Form extends React.Component<Props, State> {
static defaultProps = {
id: 'default',
isLoading: false,
onSubmit() {},
onInvalid() {},
};
2016-05-02 12:45:42 +05:30
state = {
isTouched: false,
isLoading: this.props.isLoading || false,
};
2019-12-07 16:58:52 +05:30
formEl: HTMLFormElement | null;
2018-05-08 00:53:26 +05:30
mounted = false;
2018-05-08 00:53:26 +05:30
componentDidMount() {
if (this.props.form) {
this.props.form.addLoadingListener(this.onLoading);
2016-05-28 01:34:17 +05:30
}
this.mounted = true;
}
2016-05-28 01:34:17 +05:30
componentWillReceiveProps(nextProps: Props) {
if (nextProps.id !== this.props.id) {
this.setState({
isTouched: false,
});
}
2016-05-28 01:34:17 +05:30
if (
typeof nextProps.isLoading !== 'undefined' &&
nextProps.isLoading !== this.state.isLoading
) {
this.setState({
isLoading: nextProps.isLoading,
});
2016-05-28 01:34:17 +05:30
}
const nextForm = nextProps.form;
2018-05-08 00:53:26 +05:30
if (nextForm && this.props.form && nextForm !== this.props.form) {
this.props.form.removeLoadingListener(this.onLoading);
nextForm.addLoadingListener(this.onLoading);
2016-05-02 12:45:42 +05:30
}
}
2016-05-02 12:45:42 +05:30
componentWillUnmount() {
if (this.props.form) {
this.props.form.removeLoadingListener(this.onLoading);
2016-05-02 12:45:42 +05:30
}
this.mounted = false;
}
render() {
const { isLoading } = this.state;
return (
<form
className={classNames(styles.form, {
[styles.isFormLoading]: isLoading,
[styles.formTouched]: this.state.isTouched,
})}
onSubmit={this.onFormSubmit}
2019-12-07 16:58:52 +05:30
ref={(el: HTMLFormElement | null) => (this.formEl = el)}
noValidate
>
{this.props.children}
</form>
);
}
submit() {
if (!this.state.isTouched) {
this.setState({
isTouched: true,
});
}
2016-05-02 12:45:42 +05:30
const form = this.formEl;
if (!form) {
return;
}
if (form.checkValidity()) {
const result = this.props.onSubmit(
this.props.form ? this.props.form : new FormData(form),
);
if (result && result.then) {
this.setState({ isLoading: true });
2017-06-13 01:02:59 +05:30
result
.catch((errors: { [key: string]: string }) => {
this.setErrors(errors);
})
.finally(() => this.mounted && this.setState({ isLoading: false }));
}
} else {
2019-12-07 16:58:52 +05:30
const invalidEls: NodeListOf<InputElement> = form.querySelectorAll(
':invalid',
2019-12-07 16:58:52 +05:30
);
const errors = {};
invalidEls[0].focus(); // focus on first error
2019-12-07 16:58:52 +05:30
Array.from(invalidEls).reduce((acc, el: InputElement) => {
if (!el.name) {
logger.warn('Found an element without name', { el });
2019-12-07 16:58:52 +05:30
return acc;
}
let errorMessage = el.validationMessage;
2017-09-09 19:52:19 +05:30
if (el.validity.valueMissing) {
errorMessage = `error.${el.name}_required`;
} else if (el.validity.typeMismatch) {
errorMessage = `error.${el.name}_invalid`;
}
2019-12-07 16:58:52 +05:30
acc[el.name] = errorMessage;
2016-05-02 12:45:42 +05:30
2019-12-07 16:58:52 +05:30
return acc;
}, errors);
2016-05-02 12:45:42 +05:30
this.setErrors(errors);
}
}
setErrors(errors: { [key: string]: string }) {
this.props.form && this.props.form.setErrors(errors);
this.props.onInvalid(errors);
}
2019-12-07 16:58:52 +05:30
onFormSubmit = (event: React.FormEvent<HTMLFormElement>) => {
event.preventDefault();
this.submit();
};
2016-05-28 01:34:17 +05:30
onLoading = (isLoading: boolean) => this.setState({ isLoading });
2016-05-02 12:45:42 +05:30
}