import * as React from 'react'; import type { ModalFuncProps } from '../Modal'; import usePatchElement from '../../_util/hooks/usePatchElement'; import type { HookModalRef } from './HookModal'; import HookModal from './HookModal'; import type { ModalStaticFunctions } from '../confirm'; import { withConfirm, withInfo, withSuccess, withError, withWarn } from '../confirm'; 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}; }), ); export default function useModal(): [Omit, React.ReactElement] { const holderRef = React.useRef(null as any); // ========================== 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; const modal = ( { closeFunc(); }} /> ); closeFunc = holderRef.current?.patchElement(modal); 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), }), [], ); // eslint-disable-next-line react/jsx-key return [fns, ]; }