ant-design-vue/components/modal/useModal/index.tsx

152 lines
4.3 KiB
Vue

import type { Ref } from 'vue';
import { isRef, unref, computed, defineComponent, shallowRef, watch } from 'vue';
import type { MaybeRef, VueNode } from '../../_util/type';
import type { ModalFuncProps } from '../Modal';
import type { HookModalRef } from './HookModal';
import type { ModalStaticFunctions } from '../confirm';
import { withConfirm, withError, withInfo, withSuccess, withWarn } from '../confirm';
import HookModal from './HookModal';
import destroyFns from '../destroyFns';
let uuid = 0;
interface ElementsHolderRef {
addModal: (modal: () => VueNode) => () => void;
}
const ElementsHolder = defineComponent({
name: 'ElementsHolder',
inheritAttrs: false,
setup(_, { expose }) {
const modals = shallowRef<(() => VueNode)[]>([]);
const addModal = (modal: () => VueNode) => {
modals.value.push(modal);
modals.value = modals.value.slice();
return () => {
modals.value = modals.value.filter(currentModal => currentModal !== modal);
};
};
expose({ addModal });
return () => {
return modals.value.map(modal => modal());
};
},
});
export type ModalFuncWithRef = (props: MaybeRef<ModalFuncProps>) => {
destroy: () => void;
update: (configUpdate: ModalFuncProps) => void;
};
function useModal(): readonly [
Omit<ModalStaticFunctions<ModalFuncWithRef>, 'warn'>,
() => VueNode,
] {
const holderRef = shallowRef<ElementsHolderRef>(null);
// ========================== Effect ==========================
const actionQueue = shallowRef([]);
watch(
actionQueue,
() => {
if (actionQueue.value.length) {
const cloneQueue = [...actionQueue.value];
cloneQueue.forEach(action => {
action();
});
actionQueue.value = [];
}
},
{
immediate: true,
},
);
// =========================== Hook ===========================
const getConfirmFunc = (withFunc: (config: ModalFuncProps) => ModalFuncProps) =>
function hookConfirm(config: Ref<ModalFuncProps> | ModalFuncProps) {
uuid += 1;
const open = shallowRef(true);
const modalRef = shallowRef<HookModalRef>(null);
const configRef = shallowRef(unref(config));
const updateConfig = shallowRef({});
watch(
() => config,
val => {
updateAction({
...(isRef(val) ? val.value : val),
...updateConfig.value,
});
},
);
const destroyAction = (...args: any[]) => {
open.value = false;
const triggerCancel = args.some(param => param && param.triggerCancel);
if (configRef.value.onCancel && triggerCancel) {
configRef.value.onCancel(() => {}, ...args.slice(1));
}
};
// eslint-disable-next-line prefer-const
let closeFunc: Function | undefined;
const modal = () => (
<HookModal
key={`modal-${uuid}`}
config={withFunc(configRef.value)}
ref={modalRef}
open={open.value}
destroyAction={destroyAction}
afterClose={() => {
closeFunc?.();
}}
/>
);
closeFunc = holderRef.value?.addModal(modal);
if (closeFunc) {
destroyFns.push(closeFunc);
}
const updateAction = (newConfig: ModalFuncProps) => {
configRef.value = {
...configRef.value,
...newConfig,
};
};
const destroy = () => {
if (modalRef.value) {
destroyAction();
} else {
actionQueue.value = [...actionQueue.value, destroyAction];
}
};
const update = (newConfig: ModalFuncProps) => {
updateConfig.value = newConfig;
if (modalRef.value) {
updateAction(newConfig);
} else {
actionQueue.value = [...actionQueue.value, () => updateAction(newConfig)];
}
};
return {
destroy,
update,
};
};
const fns = computed(() => ({
info: getConfirmFunc(withInfo),
success: getConfirmFunc(withSuccess),
error: getConfirmFunc(withError),
warning: getConfirmFunc(withWarn),
confirm: getConfirmFunc(withConfirm),
}));
const holderKey = Symbol('modalHolderKey');
return [fns.value, () => <ElementsHolder key={holderKey} ref={holderRef} />] as const;
}
export default useModal;