ant-design/components/message/useMessage.tsx
EmilyyyLiu 62c9814ba7
feat: Replace 'rc-notification' with '@rc-component/notification' (#53319)
* feat:依赖新增@rc-component/notification

* feat: 对notification的closable改造,并将rc-notification使用修改为@rc-component/notification

* @rc-component/notification已经删除closeIcon,use Message删除closeIcon的配置

* reactivate PR

* feat: 修改notification mergedCloseIcon 的获取方式,并增加useClosable 非hook的方法

* 补充computeClosable方法的单测

* 补充computeClosable 方法单测

* test: 补充行覆盖率

* 提高useClosable代码复用

* feat: Add missing type definitions due to merging

* test: Supplement the row coverage of computeClassable method

* test: Supplement the row coverage of computeClassable method(2)

* test: Supplement the row coverage of computeClassable method(3)

---------

Co-authored-by: 刘欢 <lh01217311@antgroup.com>
2025-05-15 11:53:38 +08:00

285 lines
8.0 KiB
TypeScript

import * as React from 'react';
import {
NotificationProvider,
useNotification as useRcNotification,
} from '@rc-component/notification';
import type {
NotificationAPI,
NotificationConfig as RcNotificationConfig,
} from '@rc-component/notification';
import classNames from 'classnames';
import { devUseWarning } from '../_util/warning';
import { ConfigContext } from '../config-provider';
import { useComponentConfig } from '../config-provider/context';
import type { MessageConfig } from '../config-provider/context';
import useCSSVarCls from '../config-provider/hooks/useCSSVarCls';
import type {
ArgsProps,
ConfigOptions,
MessageInstance,
MessageType,
NoticeType,
TypeOpen,
} from './interface';
import { PureContent } from './PurePanel';
import useStyle from './style';
import { getMotion, wrapPromiseFn } from './util';
const DEFAULT_OFFSET = 8;
const DEFAULT_DURATION = 3;
// ==============================================================================
// == Holder ==
// ==============================================================================
type HolderProps = ConfigOptions & {
onAllRemoved?: VoidFunction;
};
interface HolderRef extends NotificationAPI {
prefixCls: string;
message?: MessageConfig;
}
const Wrapper: React.FC<React.PropsWithChildren<{ prefixCls: string }>> = ({
children,
prefixCls,
}) => {
const rootCls = useCSSVarCls(prefixCls);
const [hashId, cssVarCls] = useStyle(prefixCls, rootCls);
return (
<NotificationProvider classNames={{ list: classNames(hashId, cssVarCls, rootCls) }}>
{children}
</NotificationProvider>
);
};
const renderNotifications: RcNotificationConfig['renderNotifications'] = (
node,
{ prefixCls, key },
) => (
<Wrapper prefixCls={prefixCls} key={key}>
{node}
</Wrapper>
);
const Holder = React.forwardRef<HolderRef, HolderProps>((props, ref) => {
const {
top,
prefixCls: staticPrefixCls,
getContainer: staticGetContainer,
maxCount,
duration = DEFAULT_DURATION,
rtl,
transitionName,
onAllRemoved,
pauseOnHover = true,
} = props;
const { getPrefixCls, direction, getPopupContainer } = useComponentConfig('message');
const { message } = React.useContext(ConfigContext);
const prefixCls = staticPrefixCls || getPrefixCls('message');
// =============================== Style ===============================
const getStyle = (): React.CSSProperties => ({
left: '50%',
transform: 'translateX(-50%)',
top: top ?? DEFAULT_OFFSET,
});
const getClassName = () => classNames({ [`${prefixCls}-rtl`]: rtl ?? direction === 'rtl' });
// ============================== Motion ===============================
const getNotificationMotion = () => getMotion(prefixCls, transitionName);
// ============================== Origin ===============================
const [api, holder] = useRcNotification({
prefixCls,
style: getStyle,
className: getClassName,
motion: getNotificationMotion,
// closable=false requires-no closeIcon
closable: false,
duration,
getContainer: () => staticGetContainer?.() || getPopupContainer?.() || document.body,
maxCount,
onAllRemoved,
renderNotifications,
pauseOnHover,
});
// ================================ Ref ================================
React.useImperativeHandle(ref, () => ({
...api,
prefixCls,
message,
}));
return holder;
});
// ==============================================================================
// == Hook ==
// ==============================================================================
let keyIndex = 0;
export function useInternalMessage(
messageConfig?: HolderProps,
): readonly [MessageInstance, React.ReactElement] {
const holderRef = React.useRef<HolderRef>(null);
const warning = devUseWarning('Message');
// ================================ API ================================
const wrapAPI = React.useMemo<MessageInstance>(() => {
// Wrap with notification content
// >>> close
const close = (key: React.Key) => {
holderRef.current?.close(key);
};
// >>> Open
const open = (config: ArgsProps): MessageType => {
if (!holderRef.current) {
warning(
false,
'usage',
'You are calling notice in render which will break in React 18 concurrent mode. Please trigger in effect instead.',
);
const fakeResult: any = () => {};
fakeResult.then = () => {};
return fakeResult;
}
const { open: originOpen, prefixCls, message } = holderRef.current;
const contextClassName = message?.className || {};
const contextClassNames = message?.classNames || {};
const contextStyle = message?.style || {};
const contextStyles = message?.styles || {};
const noticePrefixCls = `${prefixCls}-notice`;
const {
content,
icon,
type,
key,
className,
style,
onClose,
classNames: configClassNames,
styles,
...restConfig
} = config;
let mergedKey: React.Key = key!;
if (mergedKey === undefined || mergedKey === null) {
keyIndex += 1;
mergedKey = `antd-message-${keyIndex}`;
}
return wrapPromiseFn((resolve) => {
originOpen({
...restConfig,
key: mergedKey,
content: (
<PureContent
prefixCls={prefixCls}
type={type}
icon={icon}
classNames={{
icon: classNames(configClassNames?.icon, contextClassNames.icon),
content: classNames(configClassNames?.content, contextClassNames.content),
}}
styles={{
icon: { ...contextStyles.icon, ...styles?.icon },
content: { ...contextStyles.content, ...styles?.content },
}}
>
{content}
</PureContent>
),
placement: 'top',
className: classNames(
type && `${noticePrefixCls}-${type}`,
className,
contextClassName,
contextClassNames.root,
configClassNames?.root,
),
style: { ...contextStyles.root, ...styles?.root, ...contextStyle, ...style },
onClose: () => {
onClose?.();
resolve();
},
});
// Return close function
return () => {
close(mergedKey);
};
});
};
// >>> destroy
const destroy = (key?: React.Key) => {
if (key !== undefined) {
close(key);
} else {
holderRef.current?.destroy();
}
};
const clone = {
open,
destroy,
} as MessageInstance;
const keys: NoticeType[] = ['info', 'success', 'warning', 'error', 'loading'];
keys.forEach((type) => {
const typeOpen: TypeOpen = (jointContent, duration, onClose) => {
let config: ArgsProps;
if (jointContent && typeof jointContent === 'object' && 'content' in jointContent) {
config = jointContent;
} else {
config = {
content: jointContent,
};
}
// Params
let mergedDuration: number | undefined;
let mergedOnClose: VoidFunction | undefined;
if (typeof duration === 'function') {
mergedOnClose = duration;
} else {
mergedDuration = duration;
mergedOnClose = onClose;
}
const mergedConfig = {
onClose: mergedOnClose,
duration: mergedDuration,
...config,
type,
};
return open(mergedConfig);
};
clone[type] = typeOpen;
});
return clone;
}, []);
// ============================== Return ===============================
return [wrapAPI, <Holder key="message-holder" {...messageConfig} ref={holderRef} />] as const;
}
export default function useMessage(messageConfig?: ConfigOptions) {
return useInternalMessage(messageConfig);
}