2024-12-14 13:16:29 +01:00
|
|
|
import React, { ReactNode } from 'react';
|
2019-12-07 21:43:08 +02:00
|
|
|
import clsx from 'clsx';
|
2020-01-17 23:37:52 +03:00
|
|
|
|
2019-12-07 21:02:00 +02:00
|
|
|
import logger from 'app/services/logger';
|
2016-05-02 12:20:50 +03:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
import FormModel from './FormModel';
|
2019-06-30 16:32:50 +03:00
|
|
|
import styles from './form.scss';
|
2017-08-20 18:45:21 +03:00
|
|
|
|
2020-01-17 23:37:52 +03:00
|
|
|
interface BaseProps {
|
2020-05-24 02:08:24 +03:00
|
|
|
id: string;
|
|
|
|
isLoading: boolean;
|
|
|
|
onInvalid: (errors: Record<string, string>) => void;
|
2024-12-14 13:16:29 +01:00
|
|
|
className?: string;
|
|
|
|
children?: ReactNode;
|
2019-12-07 13:28:52 +02:00
|
|
|
}
|
2020-01-17 23:37:52 +03:00
|
|
|
|
|
|
|
interface PropsWithoutForm extends BaseProps {
|
2020-07-06 19:29:56 +03:00
|
|
|
onSubmit?: (form: FormData) => Promise<void> | void;
|
2020-01-17 23:37:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
interface PropsWithForm extends BaseProps {
|
2020-05-24 02:08:24 +03:00
|
|
|
form: FormModel;
|
2020-07-06 19:29:56 +03:00
|
|
|
onSubmit?: (form: FormModel) => Promise<void> | void;
|
2020-01-17 23:37:52 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
type Props = PropsWithoutForm | PropsWithForm;
|
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
interface State {
|
2020-05-24 02:08:24 +03:00
|
|
|
id: string; // just to track value for derived updates
|
|
|
|
isTouched: boolean;
|
|
|
|
isLoading: boolean;
|
2019-12-07 13:28:52 +02:00
|
|
|
}
|
2017-09-09 18:04:26 +03:00
|
|
|
type InputElement = HTMLInputElement | HTMLTextAreaElement;
|
2017-08-20 18:45:21 +03:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
export default class Form extends React.Component<Props, State> {
|
2020-05-24 02:08:24 +03:00
|
|
|
static defaultProps = {
|
|
|
|
id: 'default',
|
|
|
|
isLoading: false,
|
|
|
|
onSubmit() {},
|
|
|
|
onInvalid() {},
|
|
|
|
};
|
|
|
|
|
|
|
|
state: State = {
|
|
|
|
id: this.props.id,
|
|
|
|
isTouched: false,
|
|
|
|
isLoading: this.props.isLoading || false,
|
|
|
|
};
|
|
|
|
|
|
|
|
formEl: HTMLFormElement | null;
|
|
|
|
|
|
|
|
mounted = false;
|
|
|
|
|
|
|
|
componentDidMount() {
|
2024-12-14 13:16:29 +01:00
|
|
|
(this.props as PropsWithForm).form?.addLoadingListener(this.onLoading);
|
2020-05-24 02:08:24 +03:00
|
|
|
this.mounted = true;
|
2016-05-27 23:04:17 +03:00
|
|
|
}
|
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
static getDerivedStateFromProps(props: Props, state: State) {
|
|
|
|
const patch: Partial<State> = {};
|
2016-05-27 23:04:17 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
if (typeof props.isLoading !== 'undefined' && props.isLoading !== state.isLoading) {
|
|
|
|
patch.isLoading = props.isLoading;
|
|
|
|
}
|
2016-05-27 23:04:17 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
if (props.id !== state.id) {
|
|
|
|
patch.id = props.id;
|
|
|
|
patch.isTouched = true;
|
|
|
|
}
|
2016-05-27 23:04:17 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
return patch;
|
2019-12-10 09:47:32 +02:00
|
|
|
}
|
2018-05-07 22:23:26 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
componentDidUpdate(prevProps: Props) {
|
2024-12-14 13:16:29 +01:00
|
|
|
const nextForm = (this.props as PropsWithForm).form;
|
|
|
|
const prevForm = (prevProps as PropsWithForm).form;
|
2019-12-10 09:47:32 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
if (nextForm !== prevForm) {
|
|
|
|
if (prevForm) {
|
|
|
|
prevForm.removeLoadingListener(this.onLoading);
|
|
|
|
}
|
2019-12-10 09:47:32 +02:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
if (nextForm) {
|
|
|
|
nextForm.addLoadingListener(this.onLoading);
|
|
|
|
}
|
|
|
|
}
|
2016-05-02 10:15:42 +03:00
|
|
|
}
|
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
componentWillUnmount() {
|
2024-12-14 13:16:29 +01:00
|
|
|
(this.props as PropsWithForm).form?.removeLoadingListener(this.onLoading);
|
2020-05-24 02:08:24 +03:00
|
|
|
this.mounted = false;
|
2016-05-02 10:15:42 +03:00
|
|
|
}
|
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
render() {
|
|
|
|
const { isLoading } = this.state;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<form
|
2024-12-14 13:16:29 +01:00
|
|
|
className={clsx(styles.form, this.props.className, {
|
2020-05-24 02:08:24 +03:00
|
|
|
[styles.isFormLoading]: isLoading,
|
|
|
|
[styles.formTouched]: this.state.isTouched,
|
|
|
|
})}
|
|
|
|
onSubmit={this.onFormSubmit}
|
|
|
|
ref={(el: HTMLFormElement | null) => (this.formEl = el)}
|
|
|
|
noValidate
|
|
|
|
>
|
|
|
|
{this.props.children}
|
|
|
|
</form>
|
|
|
|
);
|
2019-11-27 11:03:32 +02:00
|
|
|
}
|
2016-05-02 10:15:42 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
submit() {
|
|
|
|
if (!this.state.isTouched) {
|
|
|
|
this.setState({
|
|
|
|
isTouched: true,
|
|
|
|
});
|
|
|
|
}
|
2017-10-28 17:03:38 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
const form = this.formEl;
|
2017-10-28 17:03:38 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
if (!form) {
|
|
|
|
return;
|
|
|
|
}
|
2020-01-17 23:37:52 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
if (form.checkValidity()) {
|
2020-08-06 00:27:22 +04:00
|
|
|
this.clearErrors();
|
2020-05-24 02:08:24 +03:00
|
|
|
let result: Promise<void> | void;
|
2016-05-02 12:20:50 +03:00
|
|
|
|
2024-12-14 13:16:29 +01:00
|
|
|
if ((this.props as PropsWithForm).form) {
|
|
|
|
result = (this.props as PropsWithForm).onSubmit!((this.props as PropsWithForm).form);
|
2020-05-24 02:08:24 +03:00
|
|
|
} else {
|
2024-12-14 13:16:29 +01:00
|
|
|
result = (this.props as PropsWithoutForm).onSubmit!(new FormData(form));
|
2020-05-24 02:08:24 +03:00
|
|
|
}
|
2017-06-12 22:32:59 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
if (result && result.then) {
|
|
|
|
this.setState({ isLoading: true });
|
2016-05-02 12:20:50 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
result
|
|
|
|
.catch((errors: Record<string, string>) => {
|
|
|
|
this.setErrors(errors);
|
|
|
|
})
|
|
|
|
.finally(() => this.mounted && this.setState({ isLoading: false }));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
const invalidEls: NodeListOf<InputElement> = form.querySelectorAll(':invalid');
|
|
|
|
const errors: Record<string, string> = {};
|
|
|
|
invalidEls[0].focus(); // focus on first error
|
2017-09-09 17:22:19 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
Array.from(invalidEls).reduce((acc, el) => {
|
|
|
|
if (!el.name) {
|
|
|
|
logger.warn('Found an element without name', { el });
|
2016-05-02 12:20:50 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
return acc;
|
|
|
|
}
|
2016-05-02 10:15:42 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
let errorMessage = el.validationMessage;
|
2016-05-02 10:15:42 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
if (el.validity.valueMissing) {
|
|
|
|
errorMessage = `error.${el.name}_required`;
|
|
|
|
} else if (el.validity.typeMismatch) {
|
|
|
|
errorMessage = `error.${el.name}_invalid`;
|
|
|
|
}
|
|
|
|
|
|
|
|
acc[el.name] = errorMessage;
|
|
|
|
|
|
|
|
return acc;
|
|
|
|
}, errors);
|
2017-08-20 18:45:21 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
this.setErrors(errors);
|
|
|
|
}
|
2020-01-17 23:37:52 +03:00
|
|
|
}
|
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
setErrors(errors: { [key: string]: string }) {
|
2024-12-14 13:16:29 +01:00
|
|
|
(this.props as PropsWithForm).form?.setErrors(errors);
|
2020-05-24 02:08:24 +03:00
|
|
|
this.props.onInvalid(errors);
|
|
|
|
}
|
2017-08-20 18:45:21 +03:00
|
|
|
|
2024-12-14 13:16:29 +01:00
|
|
|
clearErrors = () => (this.props as PropsWithForm).form?.clearErrors();
|
2020-08-06 00:27:22 +04:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
onFormSubmit = (event: React.FormEvent<HTMLFormElement>) => {
|
|
|
|
event.preventDefault();
|
2017-08-20 18:45:21 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
this.submit();
|
|
|
|
};
|
2016-05-27 23:04:17 +03:00
|
|
|
|
2020-05-24 02:08:24 +03:00
|
|
|
onLoading = (isLoading: boolean) => this.setState({ isLoading });
|
2016-05-02 10:15:42 +03:00
|
|
|
}
|