import {useState, useCallback} from 'react' import {NativeSyntheticEvent, NativeScrollEvent} from 'react-native' import {useSetMinimalShellMode, useMinimalShellMode} from '#/state/shell' import {useShellLayout} from '#/state/shell/shell-layout' import {s} from 'lib/styles' import {isWeb} from 'platform/detection' import {useSharedValue, interpolate, runOnJS} from 'react-native-reanimated' import {useAnimatedScrollHandler} from './useAnimatedScrollHandler_FIXED' function clamp(num: number, min: number, max: number) { 'worklet' return Math.min(Math.max(num, min), max) } export type OnScrollCb = ( event: NativeSyntheticEvent, ) => void export type ResetCb = () => void export function useOnMainScroll(): [OnScrollCb, boolean, ResetCb] { const {headerHeight} = useShellLayout() const [isScrolledDown, setIsScrolledDown] = useState(false) const mode = useMinimalShellMode() const setMode = useSetMinimalShellMode() const startDragOffset = useSharedValue(null) const startMode = useSharedValue(null) const scrollHandler = useAnimatedScrollHandler({ onBeginDrag(e) { startDragOffset.value = e.contentOffset.y startMode.value = mode.value }, onEndDrag(e) { startDragOffset.value = null startMode.value = null if (e.contentOffset.y < headerHeight.value / 2) { // If we're close to the top, show the shell. setMode(false) } else { // Snap to whichever state is the closest. setMode(Math.round(mode.value) === 1) } }, onScroll(e) { // Keep track of whether we want to show "scroll to top". if (!isScrolledDown && e.contentOffset.y > s.window.height) { runOnJS(setIsScrolledDown)(true) } else if (isScrolledDown && e.contentOffset.y < s.window.height) { runOnJS(setIsScrolledDown)(false) } if (startDragOffset.value === null || startMode.value === null) { if (mode.value !== 0 && e.contentOffset.y < headerHeight.value) { // If we're close enough to the top, always show the shell. // Even if we're not dragging. setMode(false) return } if (isWeb) { // On the web, there is no concept of "starting" the drag. // When we get the first scroll event, we consider that the start. startDragOffset.value = e.contentOffset.y startMode.value = mode.value } return } // The "mode" value is always between 0 and 1. // Figure out how much to move it based on the current dragged distance. const dy = e.contentOffset.y - startDragOffset.value const dProgress = interpolate( dy, [-headerHeight.value, headerHeight.value], [-1, 1], ) const newValue = clamp(startMode.value + dProgress, 0, 1) if (newValue !== mode.value) { // Manually adjust the value. This won't be (and shouldn't be) animated. mode.value = newValue } if (isWeb) { // On the web, there is no concept of "starting" the drag, // so we don't have any specific anchor point to calculate the distance. // Instead, update it continuosly along the way and diff with the last event. startDragOffset.value = e.contentOffset.y startMode.value = mode.value } }, }) return [ scrollHandler, isScrolledDown, useCallback(() => { setIsScrolledDown(false) setMode(false) }, [setMode]), ] }