import CloseOutlined from '@ant-design/icons/CloseOutlined'; import classNames from 'classnames'; import Dialog from 'rc-dialog'; import * as React from 'react'; import Button from '../button'; import type { ButtonProps, LegacyButtonType } from '../button/button'; import { convertLegacyProps } from '../button/button'; import type { DirectionType } from '../config-provider'; import { ConfigContext } from '../config-provider'; import { NoFormStyle } from '../form/context'; import LocaleReceiver from '../locale-provider/LocaleReceiver'; import { getTransitionName } from '../_util/motion'; import { canUseDocElement } from '../_util/styleChecker'; import warning from '../_util/warning'; import { getConfirmLocale } from './locale'; type MousePosition = { x: number; y: number } | null; let mousePosition: MousePosition; // ref: https://github.com/ant-design/ant-design/issues/15795 const getClickPosition = (e: MouseEvent) => { mousePosition = { x: e.pageX, y: e.pageY, }; // 100ms 内发生过点击事件,则从点击位置动画展示 // 否则直接 zoom 展示 // 这样可以兼容非点击方式展开 setTimeout(() => { mousePosition = null; }, 100); }; // 只有点击事件支持从鼠标位置动画展开 if (canUseDocElement()) { document.documentElement.addEventListener('click', getClickPosition, true); } export interface ModalProps { /** 对话框是否可见 */ /** * @deprecated `visible` is deprecated which will be removed in next major version. Please use * `open` instead. */ visible?: boolean; open?: boolean; /** 确定按钮 loading */ confirmLoading?: boolean; /** 标题 */ title?: React.ReactNode; /** 是否显示右上角的关闭按钮 */ closable?: boolean; /** 点击确定回调 */ onOk?: (e: React.MouseEvent) => void; /** 点击模态框右上角叉、取消按钮、Props.maskClosable 值为 true 时的遮罩层或键盘按下 Esc 时的回调 */ onCancel?: (e: React.MouseEvent) => void; afterClose?: () => void; /** 垂直居中 */ centered?: boolean; /** 宽度 */ width?: string | number; /** 底部内容 */ footer?: React.ReactNode; /** 确认按钮文字 */ okText?: React.ReactNode; /** 确认按钮类型 */ okType?: LegacyButtonType; /** 取消按钮文字 */ cancelText?: React.ReactNode; /** 点击蒙层是否允许关闭 */ maskClosable?: boolean; /** 强制渲染 Modal */ forceRender?: boolean; okButtonProps?: ButtonProps; cancelButtonProps?: ButtonProps; destroyOnClose?: boolean; style?: React.CSSProperties; wrapClassName?: string; maskTransitionName?: string; transitionName?: string; className?: string; getContainer?: string | HTMLElement | getContainerFunc | false; zIndex?: number; bodyStyle?: React.CSSProperties; maskStyle?: React.CSSProperties; mask?: boolean; keyboard?: boolean; wrapProps?: any; prefixCls?: string; closeIcon?: React.ReactNode; modalRender?: (node: React.ReactNode) => React.ReactNode; focusTriggerAfterClose?: boolean; children?: React.ReactNode; mousePosition?: MousePosition; } type getContainerFunc = () => HTMLElement; export interface ModalFuncProps { prefixCls?: string; className?: string; /** * @deprecated `visible` is deprecated which will be removed in next major version. Please use * `open` instead. */ visible?: boolean; open?: boolean; title?: React.ReactNode; closable?: boolean; content?: React.ReactNode; // TODO: find out exact types onOk?: (...args: any[]) => any; onCancel?: (...args: any[]) => any; afterClose?: () => void; okButtonProps?: ButtonProps; cancelButtonProps?: ButtonProps; centered?: boolean; width?: string | number; okText?: React.ReactNode; okType?: LegacyButtonType; cancelText?: React.ReactNode; icon?: React.ReactNode; mask?: boolean; maskClosable?: boolean; zIndex?: number; okCancel?: boolean; style?: React.CSSProperties; wrapClassName?: string; maskStyle?: React.CSSProperties; type?: 'info' | 'success' | 'error' | 'warn' | 'warning' | 'confirm'; keyboard?: boolean; getContainer?: string | HTMLElement | getContainerFunc | false; autoFocusButton?: null | 'ok' | 'cancel'; transitionName?: string; maskTransitionName?: string; direction?: DirectionType; bodyStyle?: React.CSSProperties; closeIcon?: React.ReactNode; modalRender?: (node: React.ReactNode) => React.ReactNode; focusTriggerAfterClose?: boolean; } export interface ModalLocale { okText: string; cancelText: string; justOkText: string; } const Modal: React.FC = props => { const { getPopupContainer: getContextPopupContainer, getPrefixCls, direction, } = React.useContext(ConfigContext); const handleCancel = (e: React.MouseEvent) => { const { onCancel } = props; onCancel?.(e); }; const handleOk = (e: React.MouseEvent) => { const { onOk } = props; onOk?.(e); }; warning( !('visible' in props), 'Modal', `\`visible\` will be removed in next major version, please use \`open\` instead.`, ); const { prefixCls: customizePrefixCls, footer, visible, open = false, wrapClassName, centered, getContainer, closeIcon, focusTriggerAfterClose = true, width = 520, ...restProps } = props; const prefixCls = getPrefixCls('modal', customizePrefixCls); const rootPrefixCls = getPrefixCls(); const defaultFooter = ( {contextLocale => { const { okText, okType = 'primary', cancelText, confirmLoading = false } = props; return ( <> ); }} ); const closeIconToRender = ( {closeIcon || } ); const wrapClassNameExtended = classNames(wrapClassName, { [`${prefixCls}-centered`]: !!centered, [`${prefixCls}-wrap-rtl`]: direction === 'rtl', }); return ( ); }; export default Modal;