71 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			TypeScript
		
	
	
			
		
		
	
	
			71 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			TypeScript
		
	
	
import type { VNodeTypes } from 'vue';
 | 
						|
import padStart from 'lodash-es/padStart';
 | 
						|
 | 
						|
export type valueType = number | string;
 | 
						|
export type countdownValueType = valueType | string;
 | 
						|
 | 
						|
export type Formatter =
 | 
						|
  | false
 | 
						|
  | 'number'
 | 
						|
  | 'countdown'
 | 
						|
  | (({ value, config }: { value: valueType; config?: FormatConfig }) => VNodeTypes);
 | 
						|
 | 
						|
export interface FormatConfig {
 | 
						|
  formatter?: Formatter;
 | 
						|
  decimalSeparator?: string;
 | 
						|
  groupSeparator?: string;
 | 
						|
  precision?: number;
 | 
						|
  prefixCls?: string;
 | 
						|
}
 | 
						|
 | 
						|
export interface CountdownFormatConfig extends FormatConfig {
 | 
						|
  format?: string;
 | 
						|
}
 | 
						|
 | 
						|
// Countdown
 | 
						|
const timeUnits: [string, number][] = [
 | 
						|
  ['Y', 1000 * 60 * 60 * 24 * 365], // years
 | 
						|
  ['M', 1000 * 60 * 60 * 24 * 30], // months
 | 
						|
  ['D', 1000 * 60 * 60 * 24], // days
 | 
						|
  ['H', 1000 * 60 * 60], // hours
 | 
						|
  ['m', 1000 * 60], // minutes
 | 
						|
  ['s', 1000], // seconds
 | 
						|
  ['S', 1], // million seconds
 | 
						|
];
 | 
						|
 | 
						|
export function formatTimeStr(duration: number, format: string) {
 | 
						|
  let leftDuration: number = duration;
 | 
						|
 | 
						|
  const escapeRegex = /\[[^\]]*]/g;
 | 
						|
  const keepList: string[] = (format.match(escapeRegex) || []).map(str => str.slice(1, -1));
 | 
						|
  const templateText = format.replace(escapeRegex, '[]');
 | 
						|
 | 
						|
  const replacedText = timeUnits.reduce((current, [name, unit]) => {
 | 
						|
    if (current.indexOf(name) !== -1) {
 | 
						|
      const value = Math.floor(leftDuration / unit);
 | 
						|
      leftDuration -= value * unit;
 | 
						|
      return current.replace(new RegExp(`${name}+`, 'g'), (match: string) => {
 | 
						|
        const len = match.length;
 | 
						|
        return padStart(value.toString(), len, '0');
 | 
						|
      });
 | 
						|
    }
 | 
						|
    return current;
 | 
						|
  }, templateText);
 | 
						|
 | 
						|
  let index = 0;
 | 
						|
  return replacedText.replace(escapeRegex, () => {
 | 
						|
    const match = keepList[index];
 | 
						|
    index += 1;
 | 
						|
    return match;
 | 
						|
  });
 | 
						|
}
 | 
						|
 | 
						|
export function formatCountdown(value: countdownValueType, config: CountdownFormatConfig) {
 | 
						|
  const { format = '' } = config;
 | 
						|
  const target = new Date(value).getTime();
 | 
						|
  const current = Date.now();
 | 
						|
  const diff = Math.max(target - current, 0);
 | 
						|
 | 
						|
  return formatTimeStr(diff, format);
 | 
						|
}
 |