vuecssuiant-designantdreactantantd-vueenterprisefrontendui-designvue-antdvue-antd-uivue3vuecomponent
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
164 lines
4.9 KiB
164 lines
4.9 KiB
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 getSpinProps = () => ({ |
|
prefixCls: PropTypes.string, |
|
spinning: PropTypes.looseBool, |
|
size: SpinSize, |
|
wrapperClassName: PropTypes.string, |
|
tip: PropTypes.string, |
|
delay: PropTypes.number, |
|
indicator: PropTypes.any, |
|
}); |
|
|
|
export type SpinProps = Partial<ExtractPropTypes<ReturnType<typeof getSpinProps>>>; |
|
|
|
// 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 : () => <Indicator />; |
|
} |
|
|
|
export default defineComponent({ |
|
name: 'ASpin', |
|
mixins: [BaseMixin], |
|
inheritAttrs: false, |
|
props: initDefaultProps(getSpinProps(), { |
|
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 ( |
|
<span class={`${dotClassName} ${prefixCls}-dot-spin`}> |
|
<i class={`${prefixCls}-dot-item`} /> |
|
<i class={`${prefixCls}-dot-item`} /> |
|
<i class={`${prefixCls}-dot-item`} /> |
|
<i class={`${prefixCls}-dot-item`} /> |
|
</span> |
|
); |
|
}, |
|
}, |
|
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 = ( |
|
<div {...divProps} style={style} class={spinClassName}> |
|
{this.renderIndicator(prefixCls)} |
|
{tip ? <div class={`${prefixCls}-text`}>{tip}</div> : null} |
|
</div> |
|
); |
|
const children = getSlot(this); |
|
if (children && children.length) { |
|
const containerClassName = { |
|
[`${prefixCls}-container`]: true, |
|
[`${prefixCls}-blur`]: sSpinning, |
|
}; |
|
|
|
return ( |
|
<div class={[`${prefixCls}-nested-loading`, wrapperClassName]}> |
|
{sSpinning && <div key="loading">{spinElement}</div>} |
|
<div class={containerClassName} key="container"> |
|
{children} |
|
</div> |
|
</div> |
|
); |
|
} |
|
return spinElement; |
|
}, |
|
});
|
|
|