demarches-normaliennes/app/javascript/components/shared/hooks.ts

88 lines
2.2 KiB
TypeScript
Raw Normal View History

import { useRef, useCallback, useMemo, useState } from 'react';
import { fire } from '@utils';
2021-06-09 17:13:26 +02:00
export function useDeferredSubmit(input?: HTMLInputElement): {
(callback: () => void): void;
done: () => void;
} {
2021-06-09 17:13:26 +02:00
const calledRef = useRef(false);
const awaitFormSubmit = useCallback(
(callback: () => void) => {
2021-07-06 15:06:38 +02:00
const form = input?.form;
2021-06-09 17:13:26 +02:00
if (!form) {
return;
}
const interceptFormSubmit = (event: Event) => {
2021-06-09 17:13:26 +02:00
event.preventDefault();
runCallback();
if (
!Array.from(form.elements).some((e) =>
e.hasAttribute('data-direct-upload-url')
)
) {
form.submit();
}
// else: form will be submitted by diret upload once file have been uploaded
2021-06-09 17:13:26 +02:00
};
calledRef.current = false;
form.addEventListener('submit', interceptFormSubmit);
const runCallback = () => {
form.removeEventListener('submit', interceptFormSubmit);
clearTimeout(timer);
if (!calledRef.current) {
callback();
}
};
const timer = setTimeout(runCallback, 400);
},
[input]
);
const done = () => {
2021-06-09 17:13:26 +02:00
calledRef.current = true;
};
return Object.assign(awaitFormSubmit, { done });
2021-06-09 17:13:26 +02:00
}
export function groupId(id: string) {
return `#champ-${id.replace(/-input$/, '')}`;
}
export function useHiddenField(
group?: string,
name = 'value'
): [
value: string | undefined,
setValue: (value: string) => void,
input: HTMLInputElement | undefined
] {
const hiddenField = useMemo(
() => selectInputInGroup(group, name),
[group, name]
);
const [value, setValue] = useState(() => hiddenField?.value);
return [
value,
(value) => {
if (hiddenField) {
hiddenField.setAttribute('value', value);
setValue(value);
fire(hiddenField, 'change');
}
},
hiddenField ?? undefined
];
}
function selectInputInGroup(
group: string | undefined,
name: string
): HTMLInputElement | undefined | null {
if (group) {
return document.querySelector<HTMLInputElement>(
`${group} input[type="hidden"][name$="[${name}]"], ${group} input[type="hidden"][name="${name}"]`
);
}
}