mirror of
https://github.com/ant-design/ant-design.git
synced 2025-01-01 05:38:03 +08:00
863f61d908
Co-authored-by: afc163 <afc163@gmail.com>
140 lines
3.9 KiB
TypeScript
140 lines
3.9 KiB
TypeScript
import * as React from 'react';
|
|
|
|
import usePatchElement from '../../_util/hooks/usePatchElement';
|
|
import type { ModalFunc, ModalStaticFunctions } from '../confirm';
|
|
import { withConfirm, withError, withInfo, withSuccess, withWarn } from '../confirm';
|
|
import destroyFns from '../destroyFns';
|
|
import type { ModalFuncProps } from '../interface';
|
|
import type { HookModalRef } from './HookModal';
|
|
import HookModal from './HookModal';
|
|
|
|
let uuid = 0;
|
|
|
|
interface ElementsHolderRef {
|
|
patchElement: ReturnType<typeof usePatchElement>[1];
|
|
}
|
|
|
|
// Add `then` field for `ModalFunc` return instance.
|
|
export type ModalFuncWithPromise = (...args: Parameters<ModalFunc>) => ReturnType<ModalFunc> & {
|
|
then<T>(resolve: (confirmed: boolean) => T, reject: VoidFunction): Promise<T>;
|
|
};
|
|
|
|
export type HookAPI = Omit<Record<keyof ModalStaticFunctions, ModalFuncWithPromise>, 'warn'>;
|
|
|
|
const ElementsHolder = React.memo(
|
|
React.forwardRef<ElementsHolderRef>((_props, ref) => {
|
|
const [elements, patchElement] = usePatchElement();
|
|
React.useImperativeHandle(
|
|
ref,
|
|
() => ({
|
|
patchElement,
|
|
}),
|
|
[],
|
|
);
|
|
return <>{elements}</>;
|
|
}),
|
|
);
|
|
|
|
function useModal(): readonly [instance: HookAPI, contextHolder: React.ReactElement] {
|
|
const holderRef = React.useRef<ElementsHolderRef>(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<HookModalRef>();
|
|
|
|
// Proxy to promise with `onClose`
|
|
let resolvePromise: (confirmed: boolean) => void;
|
|
const promise = new Promise<boolean>((resolve) => {
|
|
resolvePromise = resolve;
|
|
});
|
|
let silent = false;
|
|
|
|
let closeFunc: (() => void) | undefined;
|
|
const modal = (
|
|
<HookModal
|
|
key={`modal-${uuid}`}
|
|
config={withFunc(config)}
|
|
ref={modalRef}
|
|
afterClose={() => {
|
|
closeFunc?.();
|
|
}}
|
|
isSilent={() => silent}
|
|
onConfirm={(confirmed) => {
|
|
resolvePromise(confirmed);
|
|
}}
|
|
/>
|
|
);
|
|
|
|
closeFunc = holderRef.current?.patchElement(modal);
|
|
|
|
if (closeFunc) {
|
|
destroyFns.push(closeFunc);
|
|
}
|
|
|
|
const instance: ReturnType<ModalFuncWithPromise> = {
|
|
destroy: () => {
|
|
function destroyAction() {
|
|
modalRef.current?.destroy();
|
|
}
|
|
|
|
if (modalRef.current) {
|
|
destroyAction();
|
|
} else {
|
|
setActionQueue((prev) => [...prev, destroyAction]);
|
|
}
|
|
},
|
|
update: (newConfig) => {
|
|
function updateAction() {
|
|
modalRef.current?.update(newConfig as ModalFuncProps);
|
|
}
|
|
|
|
if (modalRef.current) {
|
|
updateAction();
|
|
} else {
|
|
setActionQueue((prev) => [...prev, updateAction]);
|
|
}
|
|
},
|
|
then: (resolve) => {
|
|
silent = true;
|
|
return promise.then(resolve);
|
|
},
|
|
};
|
|
|
|
return instance;
|
|
},
|
|
[],
|
|
);
|
|
|
|
const fns = React.useMemo<HookAPI>(
|
|
() => ({
|
|
info: getConfirmFunc(withInfo),
|
|
success: getConfirmFunc(withSuccess),
|
|
error: getConfirmFunc(withError),
|
|
warning: getConfirmFunc(withWarn),
|
|
confirm: getConfirmFunc(withConfirm),
|
|
}),
|
|
[],
|
|
);
|
|
return [fns, <ElementsHolder key="modal-holder" ref={holderRef} />] as const;
|
|
}
|
|
|
|
export default useModal;
|