import * as React from 'react'; import usePatchElement from '../../_util/hooks/usePatchElement'; import type { ModalStaticFunctions } from '../confirm'; import { withConfirm, withError, withInfo, withSuccess, withWarn } from '../confirm'; import type { ModalFuncProps } from '../Modal'; import type { HookModalRef } from './HookModal'; import HookModal from './HookModal'; import destroyFns from '../destroyFns'; let uuid = 0; interface ElementsHolderRef { patchElement: ReturnType[1]; } const ElementsHolder = React.memo( React.forwardRef((_props, ref) => { const [elements, patchElement] = usePatchElement(); React.useImperativeHandle( ref, () => ({ patchElement, }), [], ); // eslint-disable-next-line react/jsx-no-useless-fragment return <>{elements}; }), ); function useModal(): readonly [ instance: Omit, contextHolder: React.ReactElement, ] { const holderRef = React.useRef(null); // ========================== Effect ========================== const [actionQueue, setActionQueue] = React.useState<(() => void)[]>([]); React.useEffect(() => { if (actionQueue.length) { const cloneQueue = [...actionQueue]; cloneQueue.forEach((action) => { action(); }); setActionQueue([]); } }, [actionQueue]); // =========================== Hook =========================== const getConfirmFunc = React.useCallback( (withFunc: (config: ModalFuncProps) => ModalFuncProps) => function hookConfirm(config: ModalFuncProps) { uuid += 1; const modalRef = React.createRef(); let closeFunc: Function | undefined; const modal = ( { closeFunc?.(); }} /> ); closeFunc = holderRef.current?.patchElement(modal); if (closeFunc) { destroyFns.push(closeFunc); } return { destroy: () => { function destroyAction() { modalRef.current?.destroy(); } if (modalRef.current) { destroyAction(); } else { setActionQueue((prev) => [...prev, destroyAction]); } }, update: (newConfig: ModalFuncProps) => { function updateAction() { modalRef.current?.update(newConfig); } if (modalRef.current) { updateAction(); } else { setActionQueue((prev) => [...prev, updateAction]); } }, }; }, [], ); const fns = React.useMemo>( () => ({ info: getConfirmFunc(withInfo), success: getConfirmFunc(withSuccess), error: getConfirmFunc(withError), warning: getConfirmFunc(withWarn), confirm: getConfirmFunc(withConfirm), }), [], ); return [fns, ] as const; } export default useModal;