Skip to content

Instantly share code, notes, and snippets.

@yzalvov
Last active December 22, 2021 18:48
Show Gist options
  • Save yzalvov/28133b3654ddb2724b201902937608cd to your computer and use it in GitHub Desktop.
Save yzalvov/28133b3654ddb2724b201902937608cd to your computer and use it in GitHub Desktop.
/**
* 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 }}
/>
)
}
)
@yzalvov
Copy link
Author

yzalvov commented Dec 22, 2021

@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.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment