ant-design-vue/components/statistic/Countdown.tsx

103 lines
2.9 KiB
Vue
Raw Normal View History

import type { ExtractPropTypes, PropType } from 'vue';
2021-06-10 15:04:57 +00:00
import { defineComponent, onBeforeUnmount, onMounted, onUpdated, ref } from 'vue';
import omit from '../_util/omit';
2020-10-17 03:51:17 +00:00
import initDefaultProps from '../_util/props-util/initDefaultProps';
2023-02-08 07:14:43 +00:00
import { someType } from '../_util/type';
2021-06-23 13:47:53 +00:00
import Statistic, { statisticProps } from './Statistic';
2022-05-21 07:06:02 +00:00
import type { countdownValueType, FormatConfig, valueType } from './utils';
2021-06-26 01:35:40 +00:00
import { formatCountdown as formatCD } from './utils';
const REFRESH_INTERVAL = 1000 / 30;
2020-10-17 03:51:17 +00:00
function getTime(value?: countdownValueType) {
2021-08-05 06:36:50 +00:00
return new Date(value as any).getTime();
}
export const countdownProps = () => {
return {
...statisticProps(),
2023-02-08 07:14:43 +00:00
value: someType<countdownValueType>([Number, String, Object]),
format: String,
onFinish: Function as PropType<() => void>,
onChange: Function as PropType<(value?: countdownValueType) => void>,
};
};
export type CountdownProps = Partial<ExtractPropTypes<ReturnType<typeof countdownProps>>>;
2020-10-17 03:51:17 +00:00
export default defineComponent({
compatConfig: { MODE: 3 },
name: 'AStatisticCountdown',
props: initDefaultProps(countdownProps(), {
2020-10-17 03:51:17 +00:00
format: 'HH:mm:ss',
}),
// emits: ['finish', 'change'],
setup(props, { emit, slots }) {
2021-11-29 09:53:09 +00:00
const countdownId = ref<any>();
2021-06-10 15:04:57 +00:00
const statistic = ref();
const syncTimer = () => {
const { value } = props;
2020-10-17 03:51:17 +00:00
const timestamp = getTime(value);
if (timestamp >= Date.now()) {
2021-06-10 15:04:57 +00:00
startTimer();
} else {
2021-06-10 15:04:57 +00:00
stopTimer();
}
2021-06-10 15:04:57 +00:00
};
2021-06-10 15:04:57 +00:00
const startTimer = () => {
if (countdownId.value) return;
const timestamp = getTime(props.value);
2021-11-29 09:53:09 +00:00
countdownId.value = setInterval(() => {
2021-06-10 15:04:57 +00:00
statistic.value.$forceUpdate();
if (timestamp > Date.now()) {
emit('change', timestamp - Date.now());
}
syncTimer();
}, REFRESH_INTERVAL);
2021-06-10 15:04:57 +00:00
};
2021-06-10 15:04:57 +00:00
const stopTimer = () => {
const { value } = props;
2021-06-11 14:06:25 +00:00
if (countdownId.value) {
2021-06-10 15:04:57 +00:00
clearInterval(countdownId.value);
countdownId.value = undefined;
2020-10-17 03:51:17 +00:00
const timestamp = getTime(value);
if (timestamp < Date.now()) {
2021-06-10 15:04:57 +00:00
emit('finish');
}
}
2021-06-10 15:04:57 +00:00
};
2023-02-08 07:14:43 +00:00
const formatCountdown = ({ value, config }: { value: valueType; config: FormatConfig }) => {
2021-06-10 15:04:57 +00:00
const { format } = props;
return formatCD(value, { ...config, format });
};
2021-06-10 15:04:57 +00:00
const valueRenderHtml = (node: any) => node;
onMounted(() => {
syncTimer();
});
onUpdated(() => {
syncTimer();
});
onBeforeUnmount(() => {
stopTimer();
});
return () => {
2022-05-21 07:06:02 +00:00
const value = props.value as valueType;
2021-06-10 15:04:57 +00:00
return (
<Statistic
ref={statistic}
{...{
...omit(props, ['onFinish', 'onChange']),
2022-05-21 07:06:02 +00:00
value,
2021-06-10 15:04:57 +00:00
valueRender: valueRenderHtml,
formatter: formatCountdown,
}}
v-slots={slots}
2021-06-10 15:04:57 +00:00
/>
);
};
},
});