anvil/striker-ui/hooks/useFormUtils.ts

145 lines
3.6 KiB
TypeScript
Raw Normal View History

import { MutableRefObject, useCallback, useMemo, useState } from 'react';
2023-03-03 05:03:24 +00:00
import api from '../lib/api';
import buildObjectStateSetterCallback, {
buildRegExpObjectStateSetterCallback,
} from '../lib/buildObjectStateSetterCallback';
import handleAPIError from '../lib/handleAPIError';
import { Message } from '../components/MessageBox';
2023-03-03 05:03:24 +00:00
import { MessageGroupForwardedRefContent } from '../components/MessageGroup';
import useProtectedState from './useProtectedState';
2023-03-03 05:03:24 +00:00
const useFormUtils = <
U extends string,
I extends InputIds<U>,
M extends MapToInputId<U, I>,
>(
ids: I,
messageGroupRef: MutableRefObject<MessageGroupForwardedRefContent>,
): FormUtils<M> => {
const [formSubmitting, setFormSubmitting] = useProtectedState<boolean>(false);
2023-03-03 05:03:24 +00:00
const [formValidity, setFormValidity] = useState<FormValidity<M>>({});
const setMessage = useCallback(
(key: keyof M, message?: Message) => {
messageGroupRef?.current?.setMessage?.call(null, String(key), message);
},
[messageGroupRef],
);
const setMessageRe = useCallback(
(re: RegExp, message?: Message) => {
messageGroupRef?.current?.setMessageRe?.call(null, re, message);
},
[messageGroupRef],
);
2023-03-03 05:03:24 +00:00
const setValidity = useCallback((key: keyof M, value?: boolean) => {
setFormValidity(
buildObjectStateSetterCallback<FormValidity<M>>(key, value),
);
}, []);
const setValidityRe = useCallback((re: RegExp, value?: boolean) => {
setFormValidity(
buildRegExpObjectStateSetterCallback<FormValidity<M>>(re, value),
);
}, []);
const unsetKey = useCallback(
(key: keyof M) => {
setMessage(key);
setValidity(key);
},
[setMessage, setValidity],
);
const unsetKeyRe = useCallback(
(re: RegExp) => {
setMessageRe(re);
setValidityRe(re);
},
[setMessageRe, setValidityRe],
);
2023-03-03 05:03:24 +00:00
const buildFinishInputTestBatchFunction = useCallback(
(key: keyof M) => (result: boolean) => {
setValidity(key, result);
2023-03-03 05:03:24 +00:00
},
[setValidity],
2023-03-03 05:03:24 +00:00
);
const buildInputFirstRenderFunction = useCallback(
(key: keyof M) =>
({ isValid }: InputFirstRenderFunctionArgs) => {
setValidity(key, isValid);
2023-03-03 05:03:24 +00:00
},
[setValidity],
2023-03-03 05:03:24 +00:00
);
const buildInputUnmountFunction = useCallback(
(key: keyof M) => () => {
unsetKey(key);
},
[unsetKey],
);
const submitForm = useCallback<SubmitFormFunction>(
({
body,
getErrorMsg,
msgKey = 'api',
method,
setMsg = messageGroupRef?.current?.setMessage,
successMsg,
url,
}) => {
setFormSubmitting(true);
api
.request({ data: body, method, url })
.then(() => {
setMsg?.call(null, msgKey, {
children: successMsg,
type: 'info',
});
})
.catch((apiError) => {
const emsg = handleAPIError(apiError);
emsg.children = getErrorMsg(emsg.children);
setMsg?.call(null, msgKey, emsg);
})
.finally(() => {
setFormSubmitting(false);
});
},
[messageGroupRef, setFormSubmitting],
);
const formInvalid = useMemo(
2023-03-03 05:03:24 +00:00
() => Object.values(formValidity).some((isInputValid) => !isInputValid),
[formValidity],
);
return {
buildFinishInputTestBatchFunction,
buildInputFirstRenderFunction,
buildInputUnmountFunction,
2023-03-03 05:03:24 +00:00
formValidity,
isFormInvalid: formInvalid,
isFormSubmitting: formSubmitting,
2023-03-03 05:03:24 +00:00
setFormValidity,
setMessage,
setMessageRe,
setValidity,
setValidityRe,
submitForm,
unsetKey,
unsetKeyRe,
2023-03-03 05:03:24 +00:00
};
};
export default useFormUtils;