import * as React from 'react'; import classNames from 'classnames'; import { ConfigContext } from '../config-provider'; import { cloneElement } from '../_util/reactNode'; function getNumberArray(num: string | number | undefined | null) { return num ? num .toString() .split('') .reverse() .map(i => { const current = Number(i); return isNaN(current) ? i : current; }) : []; } function renderNumberList(position: number, className: string) { const childrenToReturn: React.ReactElement[] = []; for (let i = 0; i < 30; i++) { childrenToReturn.push(

{i % 10}

, ); } return childrenToReturn; } export interface ScrollNumberProps { prefixCls?: string; className?: string; count?: string | number | null; children?: React.ReactElement; component?: string; onAnimated?: Function; style?: React.CSSProperties; title?: string | number | null; show: boolean; } export interface ScrollNumberState { animateStarted?: boolean; count?: string | number | null; } const ScrollNumber: React.FC = ({ prefixCls: customizePrefixCls, count: customizeCount, className, style, title, show, component = 'sup', children, onAnimated = () => {}, ...restProps }) => { const [animateStarted, setAnimateStarted] = React.useState(true); const [count, setCount] = React.useState(customizeCount); const [prevCount, setPrevCount] = React.useState(customizeCount); const [lastCount, setLastCount] = React.useState(customizeCount); const { getPrefixCls } = React.useContext(ConfigContext); const prefixCls = getPrefixCls('scroll-number', customizePrefixCls); if (prevCount !== customizeCount) { setAnimateStarted(true); setPrevCount(customizeCount); } React.useEffect(() => { setLastCount(count); let timeout: number; if (animateStarted) { // Let browser has time to reset the scroller before actually // performing the transition. timeout = setTimeout(() => { setAnimateStarted(false); setCount(customizeCount); onAnimated(); }); } return () => { if (timeout) { clearTimeout(timeout); } }; }, [animateStarted, customizeCount, onAnimated]); // =========================== Function =========================== const getPositionByNum = (num: number, i: number) => { const currentCount = Math.abs(Number(count)); const lstCount = Math.abs(Number(lastCount)); const currentDigit = Math.abs(getNumberArray(count)[i] as number); const lastDigit = Math.abs(getNumberArray(lstCount)[i] as number); if (animateStarted) { return 10 + num; } // 同方向则在同一侧切换数字 if (currentCount > lstCount) { if (currentDigit >= lastDigit) { return 10 + num; } return 20 + num; } if (currentDigit <= lastDigit) { return 10 + num; } return num; }; // ============================ Render ============================ const newProps = { ...restProps, 'data-show': show, style, className: classNames(prefixCls, className), title: title as string, }; const renderCurrentNumber = (num: number | string, i: number) => { if (typeof num === 'number') { const position = getPositionByNum(num, i); const removeTransition = animateStarted || getNumberArray(lastCount)[i] === undefined; return React.createElement( 'span', { className: `${prefixCls}-only`, style: { transition: removeTransition ? 'none' : undefined, msTransform: `translateY(${-position * 100}%)`, WebkitTransform: `translateY(${-position * 100}%)`, transform: `translateY(${-position * 100}%)`, }, key: i, }, renderNumberList(position, `${prefixCls}-only-unit`), ); } return ( {num} ); }; const numberNodeRef = React.useRef(null); if (show) { numberNodeRef.current = count && Number(count) % 1 === 0 ? getNumberArray(count) .map((num, i) => renderCurrentNumber(num, i)) .reverse() : count; } // allow specify the border // mock border-color by box-shadow for compatible with old usage: // if (style && style.borderColor) { newProps.style = { ...style, boxShadow: `0 0 0 1px ${style.borderColor} inset`, }; } if (children) { return cloneElement(children, oriProps => ({ className: classNames(`${prefixCls}-custom-component`, oriProps?.className), })); } return React.createElement(component as any, newProps, numberNodeRef.current); }; export default ScrollNumber;