Files
ant-design/components/empty/index.tsx
lijianan 5c52fea0bf refactor(types): derive SemanticName from semantic maps (#56389)
* refactor(types): derive SemanticName from semantic maps

* update type
2025-12-27 18:12:44 +08:00

174 lines
4.5 KiB
TypeScript

import * as React from 'react';
import { clsx } from 'clsx';
import { useMergeSemantic } from '../_util/hooks';
import type { SemanticClassNamesType, SemanticStylesType } from '../_util/hooks';
import { devUseWarning } from '../_util/warning';
import { useComponentConfig } from '../config-provider/context';
import { useLocale } from '../locale';
import DefaultEmptyImg from './empty';
import SimpleEmptyImg from './simple';
import useStyle from './style';
const defaultEmptyImg = <DefaultEmptyImg />;
const simpleEmptyImg = <SimpleEmptyImg />;
export interface TransferLocale {
description: string;
}
export type EmptySemanticName = keyof EmptySemanticClassNames & keyof EmptySemanticStyles;
export type EmptySemanticClassNames = {
root?: string;
image?: string;
description?: string;
footer?: string;
};
export type EmptySemanticStyles = {
root?: React.CSSProperties;
image?: React.CSSProperties;
description?: React.CSSProperties;
footer?: React.CSSProperties;
};
export type EmptyClassNamesType = SemanticClassNamesType<EmptyProps, EmptySemanticClassNames>;
export type EmptyStylesType = SemanticStylesType<EmptyProps, EmptySemanticStyles>;
// For backward compatibility
export type SemanticName = EmptySemanticName;
export interface EmptyProps {
prefixCls?: string;
className?: string;
rootClassName?: string;
style?: React.CSSProperties;
/** @deprecated Please use `styles.image` instead */
imageStyle?: React.CSSProperties;
image?: React.ReactNode;
description?: React.ReactNode;
children?: React.ReactNode;
classNames?: EmptyClassNamesType;
styles?: EmptyStylesType;
}
type CompoundedComponent = React.FC<EmptyProps> & {
PRESENTED_IMAGE_DEFAULT: React.ReactNode;
PRESENTED_IMAGE_SIMPLE: React.ReactNode;
};
const Empty: CompoundedComponent = (props) => {
const {
className,
rootClassName,
prefixCls: customizePrefixCls,
image,
description,
children,
imageStyle,
style,
classNames,
styles,
...restProps
} = props;
const {
getPrefixCls,
direction,
className: contextClassName,
style: contextStyle,
classNames: contextClassNames,
styles: contextStyles,
image: contextImage,
} = useComponentConfig('empty');
const prefixCls = getPrefixCls('empty', customizePrefixCls);
const [hashId, cssVarCls] = useStyle(prefixCls);
const [mergedClassNames, mergedStyles] = useMergeSemantic<
EmptyClassNamesType,
EmptyStylesType,
EmptyProps
>([contextClassNames, classNames], [contextStyles, styles], {
props,
});
const [locale] = useLocale('Empty');
const des = typeof description !== 'undefined' ? description : locale?.description;
const alt = typeof des === 'string' ? des : 'empty';
const mergedImage = image ?? contextImage ?? defaultEmptyImg;
let imageNode: React.ReactNode = null;
if (typeof mergedImage === 'string') {
imageNode = <img draggable={false} alt={alt} src={mergedImage} />;
} else {
imageNode = mergedImage;
}
// ============================= Warning ==============================
if (process.env.NODE_ENV !== 'production') {
const warning = devUseWarning('Empty');
[['imageStyle', 'styles.image']].forEach(([deprecatedName, newName]) => {
warning.deprecated(!(deprecatedName in props), deprecatedName, newName);
});
}
return (
<div
className={clsx(
hashId,
cssVarCls,
prefixCls,
contextClassName,
{
[`${prefixCls}-normal`]: mergedImage === simpleEmptyImg,
[`${prefixCls}-rtl`]: direction === 'rtl',
},
className,
rootClassName,
mergedClassNames.root,
)}
style={{ ...mergedStyles.root, ...contextStyle, ...style }}
{...restProps}
>
<div
className={clsx(`${prefixCls}-image`, mergedClassNames.image)}
style={{ ...imageStyle, ...mergedStyles.image }}
>
{imageNode}
</div>
{des && (
<div
className={clsx(`${prefixCls}-description`, mergedClassNames.description)}
style={mergedStyles.description}
>
{des}
</div>
)}
{children && (
<div
className={clsx(`${prefixCls}-footer`, mergedClassNames.footer)}
style={mergedStyles.footer}
>
{children}
</div>
)}
</div>
);
};
Empty.PRESENTED_IMAGE_DEFAULT = defaultEmptyImg;
Empty.PRESENTED_IMAGE_SIMPLE = simpleEmptyImg;
if (process.env.NODE_ENV !== 'production') {
Empty.displayName = 'Empty';
}
export default Empty;