From 03b487488b6b1bb3f0757d74b9a542c6f8a9b863 Mon Sep 17 00:00:00 2001 From: Khac Hung Le Date: Tue, 17 Sep 2024 14:26:40 +0200 Subject: [PATCH 1/9] feat: add circular progress bar --- src/components/CircularProgressBar.tsx | 209 +++++++++++++++++++++++++ 1 file changed, 209 insertions(+) create mode 100644 src/components/CircularProgressBar.tsx diff --git a/src/components/CircularProgressBar.tsx b/src/components/CircularProgressBar.tsx new file mode 100644 index 0000000000..fab1158d90 --- /dev/null +++ b/src/components/CircularProgressBar.tsx @@ -0,0 +1,209 @@ +import * as React from 'react'; +import { Animated, StyleProp, StyleSheet, View, ViewStyle } from 'react-native'; + +import setColor from 'color'; + +import { useInternalTheme } from '../core/theming'; +import type { ThemeProp } from '../types'; + +export type Props = React.ComponentPropsWithRef & { + /** + * Progress value (between 0 and 1). + */ + progress?: number; + /** + * Whether to show the indicator or hide it. + */ + animating?: boolean; + /** + * The color of the spinner. + */ + color?: string; + /** + * Size of the indicator. + */ + size?: 'small' | 'large' | number; + style?: StyleProp; + /** + * @optional + */ + theme?: ThemeProp; +}; + +/** + * Circular progress bar is an indicator used to present progress of some activity in the app. + */ +const CircularProgressBar = ({ + progress = 0, + animating = true, + color: indicatorColor, + size: indicatorSize = 'small', + style, + theme: themeOverrides, + ...rest +}: Props) => { + const theme = useInternalTheme(themeOverrides); + + // Progress must be between 0 and 1 + if (progress < 0) progress = 0; + if (progress > 1) progress = 1; + + const { current: timer } = React.useRef( + new Animated.Value(0) + ); + + const { scale } = theme.animation; + + Animated.timing(timer, { + duration: 200 * scale, + toValue: 1, + useNativeDriver: true, + isInteraction: false, + }).start(); + + const color = indicatorColor || theme.colors?.primary; + const tintColor = theme.isV3 + ? theme.colors.surfaceVariant + : setColor(color).alpha(0.38).rgb().string(); + + const size = + typeof indicatorSize === 'string' + ? indicatorSize === 'small' + ? 24 + : 48 + : indicatorSize + ? indicatorSize + : 24; + + const layerStyle = { + width: size, + height: size, + }; + + const backgroundStyle = { + borderColor: tintColor, + borderWidth: size / 10, + borderRadius: size / 2, + }; + + const progressInDegrees = progress * 360; + // Rotation of the left half of the circle + const leftRotation = progressInDegrees > 180 ? 180 : progressInDegrees; + // Rotation the right half of the circle + const rightRotation = progressInDegrees > 180 ? progressInDegrees - 180 : 0; + + return ( + + + + {[0, 1].map((index) => { + const fixWidth = + progress > 0.5 ? (index ? size / 2 : size / 2 + 1) : size / 2; + + const containerStyle = { + width: fixWidth, + height: size, + overflow: 'hidden' as const, + }; + + const offsetStyle = index + ? { + left: size / 2, + transform: [ + { + rotate: `180deg`, + }, + ], + } + : null; + + const middle = Math.floor(180 / progressInDegrees); + + const rotationStyle = animating + ? { + transform: [ + { + rotate: timer.interpolate({ + inputRange: + progressInDegrees > 180 + ? index + ? [0, middle, middle] + : [middle, middle, 1] + : index + ? [0, 1, 1] + : [0, 0, 1], + outputRange: index + ? [ + `180deg`, + `${leftRotation + 180}deg`, + `${leftRotation + 180}deg`, + ] + : [`180deg`, `180deg`, `${rightRotation + 180}deg`], + }), + }, + ], + } + : { + transform: [ + { + rotate: index + ? `${leftRotation - 180}deg` + : `${rightRotation - 180}deg`, + }, + ], + }; + + const lineStyle = { + width: size, + height: size, + borderColor: color, + borderWidth: progress === 0 ? 0 : size / 10, // Prevents clipping when progress is 0 + borderRadius: size / 2, + }; + + return ( + + + + + + + + + + + + ); + })} + + + ); +}; + +const styles = StyleSheet.create({ + container: { + justifyContent: 'center', + alignItems: 'center', + }, + + layer: { + ...StyleSheet.absoluteFillObject, + + justifyContent: 'center', + alignItems: 'center', + }, +}); + +export default CircularProgressBar; From 6b31f24d6e6c1f79290d3330ebbefbac2da36b09 Mon Sep 17 00:00:00 2001 From: Khac Hung Le Date: Thu, 19 Sep 2024 11:04:53 +0200 Subject: [PATCH 2/9] fix: add more animations for the circular progress bar --- src/components/CircularProgressBar.tsx | 109 ++++++++++++++++++------- 1 file changed, 79 insertions(+), 30 deletions(-) diff --git a/src/components/CircularProgressBar.tsx b/src/components/CircularProgressBar.tsx index fab1158d90..81bb3af71e 100644 --- a/src/components/CircularProgressBar.tsx +++ b/src/components/CircularProgressBar.tsx @@ -12,15 +12,15 @@ export type Props = React.ComponentPropsWithRef & { */ progress?: number; /** - * Whether to show the indicator or hide it. + * Whether to animate the circular progress bar or not. */ animating?: boolean; /** - * The color of the spinner. + * The color of the circular progress bar. */ color?: string; /** - * Size of the indicator. + * Size of the circular progress bar. */ size?: 'small' | 'large' | number; style?: StyleProp; @@ -32,6 +32,18 @@ export type Props = React.ComponentPropsWithRef & { /** * Circular progress bar is an indicator used to present progress of some activity in the app. + * + * ## Usage + * ```js + * import * as React from 'react'; + * import { CircularProgressBar, MD2Colors } from 'react-native-paper'; + * + * const MyComponent = () => ( + * + * ); + * + * export default MyComponent; + * ``` */ const CircularProgressBar = ({ progress = 0, @@ -52,6 +64,8 @@ const CircularProgressBar = ({ new Animated.Value(0) ); + const prevProgressValue = React.useRef(0); + const { scale } = theme.animation; Animated.timing(timer, { @@ -61,6 +75,20 @@ const CircularProgressBar = ({ isInteraction: false, }).start(); + React.useEffect(() => { + timer.setValue(0); + Animated.timing(timer, { + duration: 200 * scale, + toValue: 1, + useNativeDriver: true, + isInteraction: false, + }).start(); + }, [progress, scale, timer]); + + React.useEffect(() => { + prevProgressValue.current = progress; + }, [progress]); + const color = indicatorColor || theme.colors?.primary; const tintColor = theme.isV3 ? theme.colors.surfaceVariant @@ -86,12 +114,16 @@ const CircularProgressBar = ({ borderRadius: size / 2, }; - const progressInDegrees = progress * 360; - // Rotation of the left half of the circle + const progressInDegrees = Math.ceil(progress * 360); const leftRotation = progressInDegrees > 180 ? 180 : progressInDegrees; - // Rotation the right half of the circle const rightRotation = progressInDegrees > 180 ? progressInDegrees - 180 : 0; + const prevProgressInDegrees = Math.ceil(prevProgressValue.current * 360); + const prevLeftRotation = + prevProgressInDegrees > 180 ? 180 : prevProgressInDegrees; + const prevRightRotation = + prevProgressInDegrees > 180 ? prevProgressInDegrees - 180 : 0; + return ( {[0, 1].map((index) => { - const fixWidth = - progress > 0.5 ? (index ? size / 2 : size / 2 + 1) : size / 2; + const addProgress = progressInDegrees > prevProgressInDegrees; + + const middle = addProgress + ? progressInDegrees <= 180 + ? 1 + : prevProgressInDegrees >= 180 + ? 0 + : (180 - prevProgressInDegrees) / + (progressInDegrees - prevProgressInDegrees) + : progressInDegrees <= 180 + ? prevProgressInDegrees >= 180 + ? (prevProgressInDegrees - 180) / + (prevProgressInDegrees - progressInDegrees) + : 0 + : 1; const containerStyle = { - width: fixWidth, + width: size / 2, height: size, overflow: 'hidden' as const, }; const offsetStyle = index ? { - left: size / 2, transform: [ { rotate: `180deg`, @@ -126,28 +170,36 @@ const CircularProgressBar = ({ } : null; - const middle = Math.floor(180 / progressInDegrees); - const rotationStyle = animating ? { transform: [ { rotate: timer.interpolate({ - inputRange: - progressInDegrees > 180 - ? index - ? [0, middle, middle] - : [middle, middle, 1] - : index - ? [0, 1, 1] - : [0, 0, 1], + inputRange: addProgress + ? index + ? [0, middle, middle] + : [middle, middle, 1] + : index + ? [middle, middle, 1] + : [0, middle, middle], outputRange: index ? [ - `180deg`, - `${leftRotation + 180}deg`, + `${prevLeftRotation + 180}deg`, + `${ + (addProgress ? leftRotation : prevLeftRotation) + + 180 + }deg`, `${leftRotation + 180}deg`, ] - : [`180deg`, `180deg`, `${rightRotation + 180}deg`], + : [ + `${prevRightRotation + 180}deg`, + `${ + (addProgress + ? prevRightRotation + : rightRotation) + 180 + }deg`, + `${rightRotation + 180}deg`, + ], }), }, ], @@ -166,17 +218,14 @@ const CircularProgressBar = ({ width: size, height: size, borderColor: color, - borderWidth: progress === 0 ? 0 : size / 10, // Prevents clipping when progress is 0 + borderWidth: size / 10, // Prevents clipping when progress is 0 borderRadius: size / 2, }; return ( - - - + + + From 960b9b3fb4136c2b67ccd8a28b90e0150f078486 Mon Sep 17 00:00:00 2001 From: Khac Hung Le Date: Thu, 19 Sep 2024 11:42:44 +0200 Subject: [PATCH 3/9] feat(circularprogressbar): add example --- example/src/ExampleList.tsx | 2 + .../Examples/CircularProgressBarExample.tsx | 68 +++++++++++++++++++ src/components/CircularProgressBar.tsx | 10 ++- 3 files changed, 78 insertions(+), 2 deletions(-) create mode 100644 example/src/Examples/CircularProgressBarExample.tsx diff --git a/example/src/ExampleList.tsx b/example/src/ExampleList.tsx index 3fc4483c63..971bed5928 100644 --- a/example/src/ExampleList.tsx +++ b/example/src/ExampleList.tsx @@ -18,6 +18,7 @@ import CardExample from './Examples/CardExample'; import CheckboxExample from './Examples/CheckboxExample'; import CheckboxItemExample from './Examples/CheckboxItemExample'; import ChipExample from './Examples/ChipExample'; +import CircularProgressBarExample from './Examples/CircularProgressBarExample'; import DataTableExample from './Examples/DataTableExample'; import DialogExample from './Examples/DialogExample'; import DividerExample from './Examples/DividerExample'; @@ -70,6 +71,7 @@ export const mainExamples: Record< checkbox: CheckboxExample, checkboxItem: CheckboxItemExample, chip: ChipExample, + circularProgressBar: CircularProgressBarExample, dataTable: DataTableExample, dialog: DialogExample, divider: DividerExample, diff --git a/example/src/Examples/CircularProgressBarExample.tsx b/example/src/Examples/CircularProgressBarExample.tsx new file mode 100644 index 0000000000..cb75518d8a --- /dev/null +++ b/example/src/Examples/CircularProgressBarExample.tsx @@ -0,0 +1,68 @@ +import * as React from 'react'; +import { StyleSheet, View } from 'react-native'; + +import { FAB, List, MD2Colors, MD3Colors, TextInput } from 'react-native-paper'; + +import { useExampleTheme } from '..'; +import CircularProgressBar from '../../../src/components/CircularProgressBar'; +import ScreenWrapper from '../ScreenWrapper'; + +const CircularProgressBarExample = () => { + const [progress, setProgress] = React.useState(1); + const [text, setText] = React.useState(''); + const { isV3 } = useExampleTheme(); + + return ( + + setText(text)} + /> + + { + const x = Number(text); + !isNaN(x) ? setProgress(x) : null; + }} + /> + + + + + + + + + + + + + + + + + + + ); +}; + +CircularProgressBarExample.title = 'Circular Progress Bar'; + +const styles = StyleSheet.create({ + container: { + padding: 4, + }, + row: { + justifyContent: 'center', + alignItems: 'center', + margin: 10, + }, +}); + +export default CircularProgressBarExample; diff --git a/src/components/CircularProgressBar.tsx b/src/components/CircularProgressBar.tsx index 81bb3af71e..0efd217232 100644 --- a/src/components/CircularProgressBar.tsx +++ b/src/components/CircularProgressBar.tsx @@ -145,15 +145,21 @@ const CircularProgressBar = ({ ? 1 : prevProgressInDegrees >= 180 ? 0 + : progressInDegrees - prevProgressInDegrees === 0 + ? 0 : (180 - prevProgressInDegrees) / (progressInDegrees - prevProgressInDegrees) : progressInDegrees <= 180 ? prevProgressInDegrees >= 180 - ? (prevProgressInDegrees - 180) / - (prevProgressInDegrees - progressInDegrees) + ? progressInDegrees - prevProgressInDegrees === 0 + ? 0 + : (prevProgressInDegrees - 180) / + (prevProgressInDegrees - progressInDegrees) : 0 : 1; + console.log(middle); + const containerStyle = { width: size / 2, height: size, From 5218dedfadca40b052d24992175538fcb1100546 Mon Sep 17 00:00:00 2001 From: Khac Hung Le Date: Thu, 19 Sep 2024 11:45:58 +0200 Subject: [PATCH 4/9] docs: add circular progress bar --- docs/docusaurus.config.js | 1 + 1 file changed, 1 insertion(+) diff --git a/docs/docusaurus.config.js b/docs/docusaurus.config.js index 16cd59f009..dca532d528 100644 --- a/docs/docusaurus.config.js +++ b/docs/docusaurus.config.js @@ -95,6 +95,7 @@ const config = { CheckboxIOS: 'Checkbox/CheckboxIOS', CheckboxItem: 'Checkbox/CheckboxItem', }, + CircularProgressBar: 'CircularProgressBar', Chip: { Chip: 'Chip/Chip', }, From c57e927c1849d4edb042d52fa3ee76f5215dfb00 Mon Sep 17 00:00:00 2001 From: Khac Hung Le Date: Thu, 19 Sep 2024 18:03:31 +0200 Subject: [PATCH 5/9] fix(circularprogressbar): remove clipping when progress > 50% --- src/components/CircularProgressBar.tsx | 87 ++++++++++++++------------ 1 file changed, 48 insertions(+), 39 deletions(-) diff --git a/src/components/CircularProgressBar.tsx b/src/components/CircularProgressBar.tsx index 0efd217232..6c6438f9f4 100644 --- a/src/components/CircularProgressBar.tsx +++ b/src/components/CircularProgressBar.tsx @@ -1,5 +1,12 @@ import * as React from 'react'; -import { Animated, StyleProp, StyleSheet, View, ViewStyle } from 'react-native'; +import { + Animated, + StyleProp, + StyleSheet, + View, + ViewStyle, + PixelRatio, +} from 'react-native'; import setColor from 'color'; @@ -39,7 +46,7 @@ export type Props = React.ComponentPropsWithRef & { * import { CircularProgressBar, MD2Colors } from 'react-native-paper'; * * const MyComponent = () => ( - * + * * ); * * export default MyComponent; @@ -68,13 +75,6 @@ const CircularProgressBar = ({ const { scale } = theme.animation; - Animated.timing(timer, { - duration: 200 * scale, - toValue: 1, - useNativeDriver: true, - isInteraction: false, - }).start(); - React.useEffect(() => { timer.setValue(0); Animated.timing(timer, { @@ -94,7 +94,7 @@ const CircularProgressBar = ({ ? theme.colors.surfaceVariant : setColor(color).alpha(0.38).rgb().string(); - const size = + let size = typeof indicatorSize === 'string' ? indicatorSize === 'small' ? 24 @@ -102,12 +102,21 @@ const CircularProgressBar = ({ : indicatorSize ? indicatorSize : 24; + // Calculate the actual size of the circular progress bar to prevent a bug with clipping containers + const halfSize = PixelRatio.roundToNearestPixel(size / 2); + size = halfSize * 2; const layerStyle = { width: size, height: size, }; + const containerStyle = { + width: halfSize, + height: size, + overflow: 'hidden' as const, + }; + const backgroundStyle = { borderColor: tintColor, borderWidth: size / 10, @@ -124,6 +133,34 @@ const CircularProgressBar = ({ const prevRightRotation = prevProgressInDegrees > 180 ? prevProgressInDegrees - 180 : 0; + const addProgress = progressInDegrees > prevProgressInDegrees; + const noProgress = progressInDegrees - prevProgressInDegrees === 0; + const progressLteFiftyPercent = progressInDegrees <= 180; + const prevProgressGteFiftyPercent = prevProgressInDegrees >= 180; + + let middle = 0; + + if ( + noProgress || + (addProgress && prevProgressGteFiftyPercent) || + (!addProgress && !prevProgressGteFiftyPercent) + ) { + middle = 0; + } else if ( + (addProgress && progressLteFiftyPercent) || + (!addProgress && !progressLteFiftyPercent) + ) { + middle = 1; + } else if (addProgress) { + middle = + (180 - prevProgressInDegrees) / + (progressInDegrees - prevProgressInDegrees); + } else { + middle = + (prevProgressInDegrees - 180) / + (prevProgressInDegrees - progressInDegrees); + } + return ( {[0, 1].map((index) => { - const addProgress = progressInDegrees > prevProgressInDegrees; - - const middle = addProgress - ? progressInDegrees <= 180 - ? 1 - : prevProgressInDegrees >= 180 - ? 0 - : progressInDegrees - prevProgressInDegrees === 0 - ? 0 - : (180 - prevProgressInDegrees) / - (progressInDegrees - prevProgressInDegrees) - : progressInDegrees <= 180 - ? prevProgressInDegrees >= 180 - ? progressInDegrees - prevProgressInDegrees === 0 - ? 0 - : (prevProgressInDegrees - 180) / - (prevProgressInDegrees - progressInDegrees) - : 0 - : 1; - - console.log(middle); - - const containerStyle = { - width: size / 2, - height: size, - overflow: 'hidden' as const, - }; - const offsetStyle = index ? { transform: [ @@ -224,7 +233,7 @@ const CircularProgressBar = ({ width: size, height: size, borderColor: color, - borderWidth: size / 10, // Prevents clipping when progress is 0 + borderWidth: size / 10, borderRadius: size / 2, }; From ba1aef436a4820f6546b18eb220125f3e9aff6f9 Mon Sep 17 00:00:00 2001 From: Khac Hung Le Date: Thu, 19 Sep 2024 18:05:40 +0200 Subject: [PATCH 6/9] feat: add example for circular progress bar --- .../Examples/CircularProgressBarExample.tsx | 20 ++++++++++++------- 1 file changed, 13 insertions(+), 7 deletions(-) diff --git a/example/src/Examples/CircularProgressBarExample.tsx b/example/src/Examples/CircularProgressBarExample.tsx index cb75518d8a..b926a64122 100644 --- a/example/src/Examples/CircularProgressBarExample.tsx +++ b/example/src/Examples/CircularProgressBarExample.tsx @@ -8,17 +8,19 @@ import CircularProgressBar from '../../../src/components/CircularProgressBar'; import ScreenWrapper from '../ScreenWrapper'; const CircularProgressBarExample = () => { - const [progress, setProgress] = React.useState(1); - const [text, setText] = React.useState(''); + const [progress, setProgress] = React.useState(0.6); + const [text, setText] = React.useState('0.6'); const { isV3 } = useExampleTheme(); return ( - setText(text)} - /> + + setText(text)} + /> + { color={isV3 ? MD3Colors.error20 : MD2Colors.red500} /> + + + + ); }; From b27b9ca2433de4d4ee2b043cb42ea84f4ad92242 Mon Sep 17 00:00:00 2001 From: Khac Hung Le Date: Thu, 19 Sep 2024 18:06:34 +0200 Subject: [PATCH 7/9] test: create tests and snapshot for circular progress bar --- .../__tests__/CircularProgressBar.test.tsx | 85 + .../CircularProgressBar.test.tsx.snap | 1971 +++++++++++++++++ 2 files changed, 2056 insertions(+) create mode 100644 src/components/__tests__/CircularProgressBar.test.tsx create mode 100644 src/components/__tests__/__snapshots__/CircularProgressBar.test.tsx.snap diff --git a/src/components/__tests__/CircularProgressBar.test.tsx b/src/components/__tests__/CircularProgressBar.test.tsx new file mode 100644 index 0000000000..a2f17ada66 --- /dev/null +++ b/src/components/__tests__/CircularProgressBar.test.tsx @@ -0,0 +1,85 @@ +import * as React from 'react'; + +import { render } from '@testing-library/react-native'; + +import CircularProgressBar from '../CircularProgressBar'; + +it('renders animated circular progress bar with 100% progress', () => { + const tree = render( + + ).toJSON(); + + expect(tree).toMatchSnapshot(); +}); + +it('renders circular progress bar with 100% progress', () => { + const tree = render( + + ).toJSON(); + + expect(tree).toMatchSnapshot(); +}); + +it('renders animated circular progress bar and 70% progress', () => { + const tree = render( + + ).toJSON(); + + expect(tree).toMatchSnapshot(); +}); + +it('renders circular progress bar with 70% progress', () => { + const tree = render( + + ).toJSON(); + + expect(tree).toMatchSnapshot(); +}); + +it('renders animated circular progress bar with 40% progress', () => { + const tree = render( + + ).toJSON(); + + expect(tree).toMatchSnapshot(); +}); + +it('renders circular progress bar with 40% progress', () => { + const tree = render( + + ).toJSON(); + + expect(tree).toMatchSnapshot(); +}); + +it('renders animated circular progress bar with 0% progress', () => { + const tree = render( + + ).toJSON(); + + expect(tree).toMatchSnapshot(); +}); + +it('renders circular progress bar with 0% progress', () => { + const tree = render( + + ).toJSON(); + + expect(tree).toMatchSnapshot(); +}); + +it('renders large animated circular progress bar', () => { + const tree = render( + + ).toJSON(); + + expect(tree).toMatchSnapshot(); +}); + +it('renders colored animated circular progress bar', () => { + const tree = render( + + ).toJSON(); + + expect(tree).toMatchSnapshot(); +}); diff --git a/src/components/__tests__/__snapshots__/CircularProgressBar.test.tsx.snap b/src/components/__tests__/__snapshots__/CircularProgressBar.test.tsx.snap new file mode 100644 index 0000000000..23279ffe24 --- /dev/null +++ b/src/components/__tests__/__snapshots__/CircularProgressBar.test.tsx.snap @@ -0,0 +1,1971 @@ +// Jest Snapshot v1, https://goo.gl/fbAQLP + +exports[`renders animated circular progress bar and 70% progress 1`] = ` + + + + + + + + + + + + + + + + + + + + + + + + + + + +`; + +exports[`renders animated circular progress bar with 0% progress 1`] = ` + + + + + + + + + + + + + + + + + + + + + + + + + + + +`; + +exports[`renders animated circular progress bar with 40% progress 1`] = ` + + + + + + + + + + + + + + + + + + + + + + + + + + + +`; + +exports[`renders animated circular progress bar with 100% progress 1`] = ` + + + + + + + + + + + + + + + + + + + + + + + + + + + +`; + +exports[`renders circular progress bar with 0% progress 1`] = ` + + + + + + + + + + + + + + + + + + + + + + + + + + + +`; + +exports[`renders circular progress bar with 40% progress 1`] = ` + + + + + + + + + + + + + + + + + + + + + + + + + + + +`; + +exports[`renders circular progress bar with 70% progress 1`] = ` + + + + + + + + + + + + + + + + + + + + + + + + + + + +`; + +exports[`renders circular progress bar with 100% progress 1`] = ` + + + + + + + + + + + + + + + + + + + + + + + + + + + +`; + +exports[`renders colored animated circular progress bar 1`] = ` + + + + + + + + + + + + + + + + + + + + + + + + + + + +`; + +exports[`renders large animated circular progress bar 1`] = ` + + + + + + + + + + + + + + + + + + + + + + + + + + + +`; From 80832500124d28c5ca2130e25c4f472d94008ed9 Mon Sep 17 00:00:00 2001 From: Khac Hung Le Date: Mon, 23 Sep 2024 11:14:21 +0200 Subject: [PATCH 8/9] refactor: simplified logic for interpolation and added comments --- src/components/CircularProgressBar.tsx | 26 ++++++++++++++++---------- 1 file changed, 16 insertions(+), 10 deletions(-) diff --git a/src/components/CircularProgressBar.tsx b/src/components/CircularProgressBar.tsx index 6c6438f9f4..a549d33e87 100644 --- a/src/components/CircularProgressBar.tsx +++ b/src/components/CircularProgressBar.tsx @@ -138,24 +138,35 @@ const CircularProgressBar = ({ const progressLteFiftyPercent = progressInDegrees <= 180; const prevProgressGteFiftyPercent = prevProgressInDegrees >= 180; + /** + * The animation uses a timer which counts from 0 to 1 for each change in progress. + * Since we have 2 half circles rotating, we need to calculate the timing when the other half circle has to start rotating. + * This value is used for the interpolation in the rotation style. + */ let middle = 0; if ( + // There is no progress or the progress does not intersect the 50% mark noProgress || (addProgress && prevProgressGteFiftyPercent) || (!addProgress && !prevProgressGteFiftyPercent) ) { middle = 0; } else if ( + // The progress does not intersect the 50% mark (addProgress && progressLteFiftyPercent) || (!addProgress && !progressLteFiftyPercent) ) { middle = 1; - } else if (addProgress) { + } else if ( + // The progress intersects the 50% mark and both half circles need to rotate + addProgress + ) { middle = (180 - prevProgressInDegrees) / (progressInDegrees - prevProgressInDegrees); } else { + // The progress intersects the 50% mark and both half circles need to rotate middle = (prevProgressInDegrees - 180) / (prevProgressInDegrees - progressInDegrees); @@ -185,18 +196,13 @@ const CircularProgressBar = ({ } : null; + // The rotation both half circles need to do const rotationStyle = animating ? { transform: [ { rotate: timer.interpolate({ - inputRange: addProgress - ? index - ? [0, middle, middle] - : [middle, middle, 1] - : index - ? [middle, middle, 1] - : [0, middle, middle], + inputRange: [0, middle, 1], outputRange: index ? [ `${prevLeftRotation + 180}deg`, @@ -239,8 +245,8 @@ const CircularProgressBar = ({ return ( - - + + From 3dd43f9da8567edb9d0e32ddf710571767d03b8f Mon Sep 17 00:00:00 2001 From: Khac Hung Le Date: Tue, 24 Sep 2024 11:55:43 +0200 Subject: [PATCH 9/9] refactor: merged the two useEffects into one --- src/components/CircularProgressBar.tsx | 5 +---- 1 file changed, 1 insertion(+), 4 deletions(-) diff --git a/src/components/CircularProgressBar.tsx b/src/components/CircularProgressBar.tsx index a549d33e87..18f22c36a1 100644 --- a/src/components/CircularProgressBar.tsx +++ b/src/components/CircularProgressBar.tsx @@ -76,6 +76,7 @@ const CircularProgressBar = ({ const { scale } = theme.animation; React.useEffect(() => { + prevProgressValue.current = progress; timer.setValue(0); Animated.timing(timer, { duration: 200 * scale, @@ -85,10 +86,6 @@ const CircularProgressBar = ({ }).start(); }, [progress, scale, timer]); - React.useEffect(() => { - prevProgressValue.current = progress; - }, [progress]); - const color = indicatorColor || theme.colors?.primary; const tintColor = theme.isV3 ? theme.colors.surfaceVariant