2022-03-26 14:52:54 +00:00
|
|
|
import type { ExtractPropTypes, PropType } from 'vue';
|
2021-06-10 15:04:57 +00:00
|
|
|
import { defineComponent, onBeforeUnmount, onMounted, onUpdated, ref } from 'vue';
|
2022-03-26 14:52:54 +00:00
|
|
|
import omit from '../_util/omit';
|
2020-10-17 03:51:17 +00:00
|
|
|
import initDefaultProps from '../_util/props-util/initDefaultProps';
|
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';
|
2020-10-17 03:23:31 +00:00
|
|
|
|
|
|
|
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();
|
2020-10-17 03:23:31 +00:00
|
|
|
}
|
2022-03-26 14:52:54 +00:00
|
|
|
export const countdownProps = () => {
|
|
|
|
return {
|
|
|
|
...statisticProps(),
|
2022-05-21 02:57:08 +00:00
|
|
|
value: [Number, String, Object] as PropType<countdownValueType>,
|
2022-03-26 14:52:54 +00:00
|
|
|
format: String,
|
|
|
|
onFinish: Function as PropType<() => void>,
|
|
|
|
onChange: Function as PropType<(value?: countdownValueType) => void>,
|
|
|
|
};
|
|
|
|
};
|
2020-10-17 03:23:31 +00:00
|
|
|
|
2022-03-26 14:52:54 +00:00
|
|
|
export type CountdownProps = Partial<ExtractPropTypes<ReturnType<typeof countdownProps>>>;
|
2020-10-17 03:51:17 +00:00
|
|
|
export default defineComponent({
|
2022-09-26 13:33:41 +00:00
|
|
|
compatConfig: { MODE: 3 },
|
2020-10-17 03:23:31 +00:00
|
|
|
name: 'AStatisticCountdown',
|
2022-03-26 14:52:54 +00:00
|
|
|
props: initDefaultProps(countdownProps(), {
|
2020-10-17 03:51:17 +00:00
|
|
|
format: 'HH:mm:ss',
|
|
|
|
}),
|
2022-03-26 14:52:54 +00:00
|
|
|
// emits: ['finish', 'change'],
|
2021-12-08 02:39:36 +00:00
|
|
|
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);
|
2020-10-17 03:23:31 +00:00
|
|
|
if (timestamp >= Date.now()) {
|
2021-06-10 15:04:57 +00:00
|
|
|
startTimer();
|
2020-10-17 03:23:31 +00:00
|
|
|
} else {
|
2021-06-10 15:04:57 +00:00
|
|
|
stopTimer();
|
2020-10-17 03:23:31 +00:00
|
|
|
}
|
2021-06-10 15:04:57 +00:00
|
|
|
};
|
2020-10-17 03:23:31 +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());
|
|
|
|
}
|
2021-06-18 08:09:03 +00:00
|
|
|
syncTimer();
|
2020-10-17 03:23:31 +00:00
|
|
|
}, REFRESH_INTERVAL);
|
2021-06-10 15:04:57 +00:00
|
|
|
};
|
2020-10-17 03:23:31 +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:23:31 +00:00
|
|
|
|
2020-10-17 03:51:17 +00:00
|
|
|
const timestamp = getTime(value);
|
2020-10-17 03:23:31 +00:00
|
|
|
if (timestamp < Date.now()) {
|
2021-06-10 15:04:57 +00:00
|
|
|
emit('finish');
|
2020-10-17 03:23:31 +00:00
|
|
|
}
|
|
|
|
}
|
2021-06-10 15:04:57 +00:00
|
|
|
};
|
2020-10-17 03:23:31 +00:00
|
|
|
|
2021-06-10 15:04:57 +00:00
|
|
|
const formatCountdown = ({
|
|
|
|
value,
|
|
|
|
config,
|
|
|
|
}: {
|
|
|
|
value: countdownValueType;
|
|
|
|
config: FormatConfig;
|
|
|
|
}) => {
|
|
|
|
const { format } = props;
|
|
|
|
return formatCD(value, { ...config, format });
|
|
|
|
};
|
2020-10-17 03:23:31 +00:00
|
|
|
|
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}
|
|
|
|
{...{
|
2022-03-26 14:52:54 +00:00
|
|
|
...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,
|
|
|
|
}}
|
2021-12-08 02:39:36 +00:00
|
|
|
v-slots={slots}
|
2021-06-10 15:04:57 +00:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
2020-10-17 03:23:31 +00:00
|
|
|
},
|
|
|
|
});
|