Last active
December 22, 2021 18:48
-
-
Save yzalvov/28133b3654ddb2724b201902937608cd to your computer and use it in GitHub Desktop.
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 is a patch wrapper for the React Native `reanimated-bottom-sheet`. | |
* | |
* Problems solved: | |
* - onCloseEnd was firing as soon as the sheet's mounted, even before its opening. https://github.com/osdnk/react-native-reanimated-bottom-sheet/issues/136 | |
* - onCloseEnd was not firing on repeated closings. https://github.com/osdnk/react-native-reanimated-bottom-sheet/issues/183 | |
* - Other open/close events were silent sometimes. | |
* | |
* How to use: | |
* 1. Make sure you have the imported below modules installed. | |
* 2. Put this file to your project folder. | |
* 3. `import BottomSheet from './ReanimatedBottomSheet_patched'` - import this patched version to where you need the `BottomSheet`. | |
* 4. Use the `BottomSheet` as you would normally do. | |
*/ | |
import React, { useRef } from 'react' // "react": "16.9.0" | |
import Animated from 'react-native-reanimated' // "react-native-reanimated": "~1.7.0" | |
import BottomSheet from 'reanimated-bottom-sheet' // "reanimated-bottom-sheet": "^1.0.0-alpha.20" | |
const { Value, useCode, cond, lessOrEq, or, eq, call, set, neq } = Animated | |
export default React.forwardRef( | |
( | |
{ | |
highest = 0, // 0 it the highest snap point and | |
lowest = 1, // 1 is the lowest (from `reanimated-bottom-sheet` docs). | |
callbackThreshold = 0.01, | |
callbackNode = useRef(new Value(1)).current, | |
onOpenStart, | |
onOpenEnd, | |
onCloseStart, | |
onCloseEnd, | |
...rest | |
}, | |
ref | |
) => { | |
const bottomTippingPoint = lowest - callbackThreshold | |
const topTippingPoint = highest + callbackThreshold | |
const animDirection = useRef(new Value(0)).current | |
const animLastFired = useRef(new Value(0)).current | |
// Prevent the onCloseEnd firing onMount #fix | |
const isCloseEndAllowed = useRef(false) | |
function handleOpenStart() { | |
isCloseEndAllowed.current = true | |
if (onOpenStart) onOpenStart() | |
} | |
function handleCloseEnd() { | |
if (!isCloseEndAllowed.current) return | |
if (onCloseEnd) onCloseEnd() | |
} | |
useCode( | |
() => | |
cond( | |
lessOrEq(callbackNode, bottomTippingPoint), | |
cond( | |
lessOrEq(callbackNode, topTippingPoint), | |
// onOpenEnd | |
cond( | |
or(eq(animDirection, 0), eq(animLastFired, topTippingPoint)), | |
call( | |
[set(animDirection, 1), set(animLastFired, 0)], | |
() => onOpenEnd && onOpenEnd() | |
) | |
), | |
cond( | |
eq(animDirection, 1), | |
// onCloseStart | |
cond( | |
neq(animLastFired, topTippingPoint), | |
call( | |
[set(animLastFired, topTippingPoint)], | |
() => onCloseStart && onCloseStart() | |
) | |
), | |
// onOpenStart | |
cond( | |
neq(animLastFired, bottomTippingPoint), | |
call([set(animLastFired, bottomTippingPoint)], handleOpenStart) | |
) | |
) | |
), | |
// onCloseEnd | |
cond( | |
or(eq(animDirection, 1), eq(animLastFired, bottomTippingPoint)), | |
call([set(animDirection, 0), set(animLastFired, 1)], handleCloseEnd) | |
) | |
), | |
[callbackNode] | |
) | |
return ( | |
<BottomSheet | |
ref={ref} | |
{...{ callbackNode, callbackThreshold, ...rest }} | |
/> | |
) | |
} | |
) |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
@shivainvenio sorry, mate. The patch is still working fine for my current config
react 16.13.1
react-native 0.63.4
react-native-reanimated 2.2.2
I may update once the react version I use is up. Not sure when to be honest.