import upperFirst from 'lodash/upperFirst'; import React, { useCallback, useEffect, useRef } from 'react'; import PropTypes from 'prop-types'; import classNames from 'classnames'; import { useForceUpdate, usePrevious } from '../../lib/hooks'; import { formatTimer } from '../../utils/timer'; import styles from './Timer.module.scss'; const SIZES = { TINY: 'tiny', SMALL: 'small', MEDIUM: 'medium', }; const Timer = React.memo(({ startedAt, total, size, isDisabled, onClick }) => { const prevStartedAt = usePrevious(startedAt); const forceUpdate = useForceUpdate(); const interval = useRef(null); const start = useCallback(() => { interval.current = setInterval(() => { forceUpdate(); }, 1000); }, [forceUpdate]); const stop = useCallback(() => { clearInterval(interval.current); }, []); useEffect(() => { if (prevStartedAt) { if (!startedAt) { stop(); } } else if (startedAt) { start(); } }, [startedAt, prevStartedAt, start, stop]); useEffect( () => () => { stop(); }, [stop], ); const contentNode = ( {formatTimer({ startedAt, total })} ); return onClick ? ( ) : ( contentNode ); }); Timer.propTypes = { startedAt: PropTypes.instanceOf(Date), total: PropTypes.number.isRequired, // eslint-disable-line react/no-unused-prop-types size: PropTypes.oneOf(Object.values(SIZES)), isDisabled: PropTypes.bool, onClick: PropTypes.func, }; Timer.defaultProps = { startedAt: undefined, size: SIZES.MEDIUM, isDisabled: false, onClick: undefined, }; export default Timer;