import type { VNode, ExtractPropTypes } from 'vue'; import { inject, cloneVNode, isVNode, defineComponent, nextTick } from 'vue'; import debounce from 'lodash-es/debounce'; import { tuple } from '../_util/type'; import PropTypes from '../_util/vue-types'; import BaseMixin from '../_util/BaseMixin'; import { getComponent, getSlot } from '../_util/props-util'; import initDefaultProps from '../_util/props-util/initDefaultProps'; import { defaultConfigProvider } from '../config-provider'; export const SpinSize = PropTypes.oneOf(tuple('small', 'default', 'large')); export const spinProps = () => ({ prefixCls: PropTypes.string, spinning: PropTypes.looseBool, size: SpinSize, wrapperClassName: PropTypes.string, tip: PropTypes.string, delay: PropTypes.number, indicator: PropTypes.any, }); export type SpinProps = Partial>>; // Render indicator let defaultIndicator: () => VNode = null; function shouldDelay(spinning?: boolean, delay?: number): boolean { return !!spinning && !!delay && !isNaN(Number(delay)); } export function setDefaultIndicator(Content: any) { const Indicator = Content.indicator; defaultIndicator = typeof Indicator === 'function' ? Indicator : () => ; } export default defineComponent({ name: 'ASpin', mixins: [BaseMixin], inheritAttrs: false, props: initDefaultProps(spinProps(), { size: 'default', spinning: true, wrapperClassName: '', }), setup() { return { originalUpdateSpinning: null, configProvider: inject('configProvider', defaultConfigProvider), }; }, data() { const { spinning, delay } = this; const shouldBeDelayed = shouldDelay(spinning, delay); return { sSpinning: spinning && !shouldBeDelayed, }; }, created() { this.originalUpdateSpinning = this.updateSpinning; this.debouncifyUpdateSpinning(this.$props); }, mounted() { this.updateSpinning(); }, updated() { nextTick(() => { this.debouncifyUpdateSpinning(); this.updateSpinning(); }); }, beforeUnmount() { this.cancelExistingSpin(); }, methods: { debouncifyUpdateSpinning(props?: any) { const { delay } = props || this.$props; if (delay) { this.cancelExistingSpin(); this.updateSpinning = debounce(this.originalUpdateSpinning, delay); } }, updateSpinning() { const { spinning, sSpinning } = this; if (sSpinning !== spinning) { this.setState({ sSpinning: spinning }); } }, cancelExistingSpin() { const { updateSpinning } = this; if (updateSpinning && (updateSpinning as any).cancel) { (updateSpinning as any).cancel(); } }, renderIndicator(prefixCls: string) { const dotClassName = `${prefixCls}-dot`; let indicator = getComponent(this, 'indicator'); // should not be render default indicator when indicator value is null if (indicator === null) { return null; } if (Array.isArray(indicator)) { indicator = indicator.length === 1 ? indicator[0] : indicator; } if (isVNode(indicator)) { return cloneVNode(indicator, { class: dotClassName }); } if (defaultIndicator && isVNode(defaultIndicator())) { return cloneVNode(defaultIndicator(), { class: dotClassName }); } return ( ); }, }, render() { const { size, prefixCls: customizePrefixCls, tip, wrapperClassName } = this.$props; const { class: cls, style, ...divProps } = this.$attrs; const { getPrefixCls, direction } = this.configProvider; const prefixCls = getPrefixCls('spin', customizePrefixCls); const { sSpinning } = this; const spinClassName = { [prefixCls]: true, [`${prefixCls}-sm`]: size === 'small', [`${prefixCls}-lg`]: size === 'large', [`${prefixCls}-spinning`]: sSpinning, [`${prefixCls}-show-text`]: !!tip, [`${prefixCls}-rtl`]: direction === 'rtl', [cls as string]: !!cls, }; const spinElement = (
{this.renderIndicator(prefixCls)} {tip ?
{tip}
: null}
); const children = getSlot(this); if (children && children.length) { const containerClassName = { [`${prefixCls}-container`]: true, [`${prefixCls}-blur`]: sSpinning, }; return (
{sSpinning &&
{spinElement}
}
{children}
); } return spinElement; }, });