diff --git a/frontend/src/Components/CircularProgressBar.js b/frontend/src/Components/CircularProgressBar.js deleted file mode 100644 index 3af5665a9..000000000 --- a/frontend/src/Components/CircularProgressBar.js +++ /dev/null @@ -1,138 +0,0 @@ -import PropTypes from 'prop-types'; -import React, { Component } from 'react'; -import styles from './CircularProgressBar.css'; - -class CircularProgressBar extends Component { - - // - // Lifecycle - - constructor(props, context) { - super(props, context); - - this.state = { - progress: 0 - }; - } - - componentDidMount() { - this._progressStep(); - } - - componentDidUpdate(prevProps) { - const progress = this.props.progress; - - if (prevProps.progress !== progress) { - this._cancelProgressStep(); - this._progressStep(); - } - } - - componentWillUnmount() { - this._cancelProgressStep(); - } - - // - // Control - - _progressStep() { - this.requestAnimationFrame = window.requestAnimationFrame(() => { - this.setState({ - progress: this.state.progress + 1 - }, () => { - if (this.state.progress < this.props.progress) { - this._progressStep(); - } - }); - }); - } - - _cancelProgressStep() { - if (this.requestAnimationFrame) { - window.cancelAnimationFrame(this.requestAnimationFrame); - } - } - - // - // Render - - render() { - const { - className, - containerClassName, - size, - strokeWidth, - strokeColor, - showProgressText - } = this.props; - - const progress = this.state.progress; - - const center = size / 2; - const radius = center - strokeWidth; - const circumference = Math.PI * (radius * 2); - const sizeInPixels = `${size}px`; - const strokeDashoffset = ((100 - progress) / 100) * circumference; - const progressText = `${Math.round(progress)}%`; - - return ( -
- - - - - { - showProgressText && -
- {progressText} -
- } -
- ); - } -} - -CircularProgressBar.propTypes = { - className: PropTypes.string, - containerClassName: PropTypes.string, - size: PropTypes.number, - progress: PropTypes.number.isRequired, - strokeWidth: PropTypes.number, - strokeColor: PropTypes.string, - showProgressText: PropTypes.bool -}; - -CircularProgressBar.defaultProps = { - className: styles.circularProgressBar, - containerClassName: styles.circularProgressBarContainer, - size: 60, - strokeWidth: 5, - strokeColor: '#35c5f4', - showProgressText: false -}; - -export default CircularProgressBar; diff --git a/frontend/src/Components/CircularProgressBar.tsx b/frontend/src/Components/CircularProgressBar.tsx new file mode 100644 index 000000000..b14f5fc6a --- /dev/null +++ b/frontend/src/Components/CircularProgressBar.tsx @@ -0,0 +1,99 @@ +import React, { useCallback, useEffect, useState } from 'react'; +import styles from './CircularProgressBar.css'; + +interface CircularProgressBarProps { + className?: string; + containerClassName?: string; + size?: number; + progress: number; + strokeWidth?: number; + strokeColor?: string; + showProgressText?: boolean; +} + +function CircularProgressBar({ + className = styles.circularProgressBar, + containerClassName = styles.circularProgressBarContainer, + size = 60, + strokeWidth = 5, + strokeColor = '#35c5f4', + showProgressText = false, + progress, +}: CircularProgressBarProps) { + const [currentProgress, setCurrentProgress] = useState(0); + const raf = React.useRef(0); + const center = size / 2; + const radius = center - strokeWidth; + const circumference = Math.PI * (radius * 2); + const sizeInPixels = `${size}px`; + const strokeDashoffset = ((100 - currentProgress) / 100) * circumference; + const progressText = `${Math.round(currentProgress)}%`; + + const handleAnimation = useCallback( + (p: number) => { + setCurrentProgress((prevProgress) => { + if (prevProgress < p) { + return prevProgress + Math.min(1, p - prevProgress); + } + + return prevProgress; + }); + }, + [setCurrentProgress] + ); + + useEffect(() => { + if (progress > currentProgress) { + cancelAnimationFrame(raf.current); + + raf.current = requestAnimationFrame(() => handleAnimation(progress)); + } + }, [progress, currentProgress, handleAnimation]); + + useEffect( + () => { + return () => cancelAnimationFrame(raf.current); + }, + // We only want to run this effect once + // eslint-disable-next-line react-hooks/exhaustive-deps + [] + ); + + return ( +
+ + + + + {showProgressText && ( +
{progressText}
+ )} +
+ ); +} + +export default CircularProgressBar; diff --git a/frontend/src/Components/ProgressBar.js b/frontend/src/Components/ProgressBar.js deleted file mode 100644 index 171b4c0fa..000000000 --- a/frontend/src/Components/ProgressBar.js +++ /dev/null @@ -1,114 +0,0 @@ -import classNames from 'classnames'; -import PropTypes from 'prop-types'; -import React from 'react'; -import { ColorImpairedConsumer } from 'App/ColorImpairedContext'; -import { kinds, sizes } from 'Helpers/Props'; -import translate from 'Utilities/String/translate'; -import styles from './ProgressBar.css'; - -function ProgressBar(props) { - const { - className, - containerClassName, - title, - progress, - precision, - showText, - text, - kind, - size, - width - } = props; - - const progressPercent = `${progress.toFixed(precision)}%`; - const progressText = text || progressPercent; - const actualWidth = width ? `${width}px` : '100%'; - - return ( - - {(enableColorImpairedMode) => { - return ( -
- { - showText && width ? -
-
-
- {progressText} -
-
-
: - null - } - -
- - { - showText ? -
-
-
- {progressText} -
-
-
: - null - } -
- ); - }} - - ); -} - -ProgressBar.propTypes = { - className: PropTypes.string, - containerClassName: PropTypes.string, - title: PropTypes.string, - progress: PropTypes.number.isRequired, - precision: PropTypes.number.isRequired, - showText: PropTypes.bool.isRequired, - text: PropTypes.string, - kind: PropTypes.oneOf(kinds.all).isRequired, - size: PropTypes.oneOf(sizes.all).isRequired, - width: PropTypes.number -}; - -ProgressBar.defaultProps = { - className: styles.progressBar, - containerClassName: styles.container, - precision: 1, - showText: false, - kind: kinds.PRIMARY, - size: sizes.MEDIUM -}; - -export default ProgressBar; diff --git a/frontend/src/Components/ProgressBar.tsx b/frontend/src/Components/ProgressBar.tsx new file mode 100644 index 000000000..07b20d8a4 --- /dev/null +++ b/frontend/src/Components/ProgressBar.tsx @@ -0,0 +1,94 @@ +import classNames from 'classnames'; +import React from 'react'; +import { ColorImpairedConsumer } from 'App/ColorImpairedContext'; +import { Kind } from 'Helpers/Props/kinds'; +import { Size } from 'Helpers/Props/sizes'; +import translate from 'Utilities/String/translate'; +import styles from './ProgressBar.css'; + +interface ProgressBarProps { + className?: string; + containerClassName?: string; + title?: string; + progress: number; + precision?: number; + showText?: boolean; + text?: string; + kind?: Extract; + size?: Extract; + width?: number; +} + +function ProgressBar({ + className = styles.progressBar, + containerClassName = styles.container, + title, + progress, + precision = 1, + showText = false, + text, + kind = 'primary', + size = 'medium', + width, +}: ProgressBarProps) { + const progressPercent = `${progress.toFixed(precision)}%`; + const progressText = text || progressPercent; + const actualWidth = width ? `${width}px` : '100%'; + + return ( + + {(enableColorImpairedMode) => { + return ( +
+ {showText && width ? ( +
+
+
{progressText}
+
+
+ ) : null} + +
+ + {showText ? ( +
+
+
{progressText}
+
+
+ ) : null} +
+ ); + }} + + ); +} + +export default ProgressBar;