forked from Expensify/App
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request Expensify#48615 from bernhardoj/fix/48036-animate-…
…pay-button Animate settlement button when pay and trigger a haptic feedback
- Loading branch information
Showing
8 changed files
with
249 additions
and
126 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
93 changes: 93 additions & 0 deletions
93
src/components/SettlementButton/AnimatedSettlementButton.tsx
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,93 @@ | ||
import React, {useCallback, useEffect} from 'react'; | ||
import Animated, {runOnJS, useAnimatedStyle, useSharedValue, withDelay, withTiming} from 'react-native-reanimated'; | ||
import Text from '@components/Text'; | ||
import useThemeStyles from '@hooks/useThemeStyles'; | ||
import variables from '@styles/variables'; | ||
import CONST from '@src/CONST'; | ||
import SettlementButton from '.'; | ||
import type SettlementButtonProps from './types'; | ||
|
||
type AnimatedSettlementButtonProps = SettlementButtonProps & { | ||
isPaidAnimationRunning: boolean; | ||
onAnimationFinish: () => void; | ||
}; | ||
|
||
function AnimatedSettlementButton({isPaidAnimationRunning, onAnimationFinish, isDisabled, ...settlementButtonProps}: AnimatedSettlementButtonProps) { | ||
const styles = useThemeStyles(); | ||
const buttonScale = useSharedValue(1); | ||
const buttonOpacity = useSharedValue(1); | ||
const paymentCompleteTextScale = useSharedValue(0); | ||
const paymentCompleteTextOpacity = useSharedValue(1); | ||
const height = useSharedValue<number>(variables.componentSizeNormal); | ||
const buttonStyles = useAnimatedStyle(() => ({ | ||
transform: [{scale: buttonScale.value}], | ||
opacity: buttonOpacity.value, | ||
})); | ||
const paymentCompleteTextStyles = useAnimatedStyle(() => ({ | ||
transform: [{scale: paymentCompleteTextScale.value}], | ||
opacity: paymentCompleteTextOpacity.value, | ||
position: 'absolute', | ||
alignSelf: 'center', | ||
})); | ||
const containerStyles = useAnimatedStyle(() => ({ | ||
height: height.value, | ||
justifyContent: 'center', | ||
overflow: 'hidden', | ||
})); | ||
const buttonDisabledStyle = isPaidAnimationRunning | ||
? { | ||
opacity: 1, | ||
...styles.cursorDefault, | ||
} | ||
: undefined; | ||
|
||
const resetAnimation = useCallback(() => { | ||
// eslint-disable-next-line react-compiler/react-compiler | ||
buttonScale.value = 1; | ||
buttonOpacity.value = 1; | ||
paymentCompleteTextScale.value = 0; | ||
paymentCompleteTextOpacity.value = 1; | ||
height.value = variables.componentSizeNormal; | ||
}, [buttonScale, buttonOpacity, paymentCompleteTextScale, paymentCompleteTextOpacity, height]); | ||
|
||
useEffect(() => { | ||
if (!isPaidAnimationRunning) { | ||
resetAnimation(); | ||
return; | ||
} | ||
// eslint-disable-next-line react-compiler/react-compiler | ||
buttonScale.value = withTiming(0, {duration: CONST.ANIMATION_PAY_BUTTON_DURATION}); | ||
buttonOpacity.value = withTiming(0, {duration: CONST.ANIMATION_PAY_BUTTON_DURATION}); | ||
paymentCompleteTextScale.value = withTiming(1, {duration: CONST.ANIMATION_PAY_BUTTON_DURATION}); | ||
|
||
// Wait for the above animation + 1s delay before hiding the component | ||
const totalDelay = CONST.ANIMATION_PAY_BUTTON_DURATION + CONST.ANIMATION_PAY_BUTTON_HIDE_DELAY; | ||
height.value = withDelay( | ||
totalDelay, | ||
withTiming(0, {duration: CONST.ANIMATION_PAY_BUTTON_DURATION}, () => runOnJS(onAnimationFinish)()), | ||
); | ||
paymentCompleteTextOpacity.value = withDelay(totalDelay, withTiming(0, {duration: CONST.ANIMATION_PAY_BUTTON_DURATION})); | ||
}, [isPaidAnimationRunning, onAnimationFinish, buttonOpacity, buttonScale, height, paymentCompleteTextOpacity, paymentCompleteTextScale, resetAnimation]); | ||
|
||
return ( | ||
<Animated.View style={containerStyles}> | ||
{isPaidAnimationRunning && ( | ||
<Animated.View style={paymentCompleteTextStyles}> | ||
<Text style={[styles.buttonMediumText]}>Payment complete</Text> | ||
</Animated.View> | ||
)} | ||
<Animated.View style={buttonStyles}> | ||
<SettlementButton | ||
// eslint-disable-next-line react/jsx-props-no-spreading | ||
{...settlementButtonProps} | ||
isDisabled={isPaidAnimationRunning || isDisabled} | ||
disabledStyle={buttonDisabledStyle} | ||
/> | ||
</Animated.View> | ||
</Animated.View> | ||
); | ||
} | ||
|
||
AnimatedSettlementButton.displayName = 'AnimatedSettlementButton'; | ||
|
||
export default AnimatedSettlementButton; |
Oops, something went wrong.