ant-design/components/spin/index.tsx
叶枫 31efd7f4ea
Spin add styles.wrapper (#53448)
* feat: Spin add styles.wrap

* feat: wrapper
2025-04-09 12:30:51 +08:00

222 lines
6.0 KiB
TypeScript

import * as React from 'react';
import classNames from 'classnames';
import { debounce } from 'throttle-debounce';
import { devUseWarning } from '../_util/warning';
import { useComponentConfig } from '../config-provider/context';
import Indicator from './Indicator';
import useStyle from './style/index';
import usePercent from './usePercent';
const _SpinSizes = ['small', 'default', 'large'] as const;
export type SpinSize = (typeof _SpinSizes)[number];
export type SpinIndicator = React.ReactElement<HTMLElement>;
type SemanticName = 'root' | 'wrapper' | 'mask' | 'indicator';
export interface SpinProps {
/** Customize prefix class name */
prefixCls?: string;
/** Additional class name of Spin */
className?: string;
/** Additional root class name of Spin */
rootClassName?: string;
/** Whether Spin is spinning */
spinning?: boolean;
/** Style of Spin */
style?: React.CSSProperties;
/** Size of Spin, options: `small`, `default` and `large` */
size?: SpinSize;
/** Customize description content when Spin has children */
tip?: React.ReactNode;
/** Specifies a delay in milliseconds for loading state (prevent flush) */
delay?: number;
/** The className of wrapper when Spin has children */
wrapperClassName?: string;
/** React node of the spinning indicator */
indicator?: SpinIndicator;
/** Children of Spin */
children?: React.ReactNode;
/** Display a backdrop with the `Spin` component */
fullscreen?: boolean;
percent?: number | 'auto';
classNames?: Partial<Record<SemanticName, string>>;
styles?: Partial<Record<SemanticName, React.CSSProperties>>;
}
export type SpinType = React.FC<SpinProps> & {
setDefaultIndicator: (indicator: React.ReactNode) => void;
};
// Render indicator
let defaultIndicator: React.ReactNode | undefined;
function shouldDelay(spinning?: boolean, delay?: number): boolean {
return !!spinning && !!delay && !Number.isNaN(Number(delay));
}
const Spin: SpinType = (props) => {
const {
prefixCls: customizePrefixCls,
spinning: customSpinning = true,
delay = 0,
className,
rootClassName,
size = 'default',
tip,
wrapperClassName,
style,
children,
fullscreen = false,
indicator,
percent,
classNames: spinClassNames,
styles,
...restProps
} = props;
const {
getPrefixCls,
direction,
indicator: contextIndicator,
className: contextClassName,
style: contextStyle,
classNames: contextClassNames,
styles: contextStyles,
} = useComponentConfig('spin');
const prefixCls = getPrefixCls('spin', customizePrefixCls);
const [hashId, cssVarCls] = useStyle(prefixCls);
const [spinning, setSpinning] = React.useState<boolean>(
() => customSpinning && !shouldDelay(customSpinning, delay),
);
const mergedPercent = usePercent(spinning, percent);
React.useEffect(() => {
if (customSpinning) {
const showSpinning = debounce(delay, () => {
setSpinning(true);
});
showSpinning();
return () => {
showSpinning?.cancel?.();
};
}
setSpinning(false);
}, [delay, customSpinning]);
const isNestedPattern = React.useMemo<boolean>(
() => typeof children !== 'undefined' && !fullscreen,
[children, fullscreen],
);
if (process.env.NODE_ENV !== 'production') {
const warning = devUseWarning('Spin');
warning(
!tip || isNestedPattern || fullscreen,
'usage',
'`tip` only work in nest or fullscreen pattern.',
);
}
const spinClassName = classNames(
prefixCls,
contextClassName,
{
[`${prefixCls}-sm`]: size === 'small',
[`${prefixCls}-lg`]: size === 'large',
[`${prefixCls}-spinning`]: spinning,
[`${prefixCls}-show-text`]: !!tip,
[`${prefixCls}-rtl`]: direction === 'rtl',
},
className,
!fullscreen && rootClassName,
!fullscreen && spinClassNames?.root,
!fullscreen && contextClassNames.root,
hashId,
cssVarCls,
);
const containerClassName = classNames(`${prefixCls}-container`, {
[`${prefixCls}-blur`]: spinning,
});
const mergedIndicator = indicator ?? contextIndicator ?? defaultIndicator;
const rootStyle: React.CSSProperties = { ...contextStyles.root, ...styles?.root };
const wrapStyle: React.CSSProperties = { ...contextStyles.wrapper, ...styles?.wrapper };
const mergedStyle: React.CSSProperties = { ...contextStyle, ...style };
const spinElement: React.ReactNode = (
<div
{...restProps}
style={fullscreen ? mergedStyle : { ...rootStyle, ...mergedStyle }}
className={spinClassName}
aria-live="polite"
aria-busy={spinning}
>
<Indicator
className={classNames(spinClassNames?.indicator, contextClassNames.indicator)}
style={{ ...contextStyles.indicator, ...styles?.indicator }}
prefixCls={prefixCls}
indicator={mergedIndicator}
percent={mergedPercent}
/>
{tip && (isNestedPattern || fullscreen) ? (
<div className={`${prefixCls}-text`}>{tip}</div>
) : null}
</div>
);
if (isNestedPattern) {
return (
<div
{...restProps}
className={classNames(`${prefixCls}-nested-loading`, wrapperClassName, hashId, cssVarCls)}
style={wrapStyle}
>
{spinning && <div key="loading">{spinElement}</div>}
<div className={containerClassName} key="container">
{children}
</div>
</div>
);
}
if (fullscreen) {
return (
<div
className={classNames(
`${prefixCls}-fullscreen`,
{
[`${prefixCls}-fullscreen-show`]: spinning,
},
rootClassName,
hashId,
cssVarCls,
spinClassNames?.mask,
contextClassNames.mask,
)}
style={{ ...contextStyles.mask, ...styles?.mask }}
>
{spinElement}
</div>
);
}
return spinElement;
};
Spin.setDefaultIndicator = (indicator: React.ReactNode) => {
defaultIndicator = indicator;
};
if (process.env.NODE_ENV !== 'production') {
Spin.displayName = 'Spin';
}
export default Spin;