ant-design-vue/components/statistic/Statistic.tsx
2023-08-30 22:48:17 +08:00

90 lines
3.0 KiB
Vue

import type { CSSProperties, ExtractPropTypes, PropType } from 'vue';
import { defineComponent } from 'vue';
import initDefaultProps from '../_util/props-util/initDefaultProps';
import StatisticNumber from './Number';
import type { valueType, Formatter } from './utils';
import Skeleton from '../skeleton/Skeleton';
import useConfigInject from '../config-provider/hooks/useConfigInject';
// CSSINJS
import useStyle from './style';
import { anyType, booleanType, functionType, someType, vNodeType } from '../_util/type';
import type { CustomSlotsType, VueNode } from '../_util/type';
export const statisticProps = () => ({
prefixCls: String,
decimalSeparator: String,
groupSeparator: String,
format: String,
value: someType<valueType>([Number, String, Object]),
valueStyle: { type: Object as PropType<CSSProperties>, default: undefined as CSSProperties },
valueRender: functionType<(node: VueNode) => VueNode>(),
formatter: anyType<Formatter>(),
precision: Number,
prefix: vNodeType(),
suffix: vNodeType(),
title: vNodeType(),
loading: booleanType(),
});
export type StatisticProps = Partial<ExtractPropTypes<ReturnType<typeof statisticProps>>>;
export default defineComponent({
compatConfig: { MODE: 3 },
name: 'AStatistic',
inheritAttrs: false,
props: initDefaultProps(statisticProps(), {
decimalSeparator: '.',
groupSeparator: ',',
loading: false,
}),
slots: Object as CustomSlotsType<{
title?: any;
prefix?: any;
suffix?: any;
formatter?: any;
default?: any;
}>,
setup(props, { slots, attrs }) {
const { prefixCls, direction } = useConfigInject('statistic', props);
// Style
const [wrapSSR, hashId] = useStyle(prefixCls);
return () => {
const { value = 0, valueStyle, valueRender } = props;
const pre = prefixCls.value;
const title = props.title ?? slots.title?.();
const prefix = props.prefix ?? slots.prefix?.();
const suffix = props.suffix ?? slots.suffix?.();
const formatter = props.formatter ?? (slots.formatter as unknown as Formatter);
// data-for-update just for update component
// https://github.com/vueComponent/ant-design-vue/pull/3170
let valueNode: VueNode = (
<StatisticNumber
data-for-update={Date.now()}
{...{ ...props, prefixCls: pre, value, formatter }}
/>
);
if (valueRender) {
valueNode = valueRender(valueNode);
}
return wrapSSR(
<div
{...attrs}
class={[pre, { [`${pre}-rtl`]: direction.value === 'rtl' }, attrs.class, hashId.value]}
>
{title && <div class={`${pre}-title`}>{title}</div>}
<Skeleton paragraph={false} loading={props.loading}>
<div style={valueStyle} class={`${pre}-content`}>
{prefix && <span class={`${pre}-content-prefix`}>{prefix}</span>}
{valueNode}
{suffix && <span class={`${pre}-content-suffix`}>{suffix}</span>}
</div>
</Skeleton>
</div>,
);
};
},
});