-
-
Notifications
You must be signed in to change notification settings - Fork 452
Expand file tree
/
Copy pathNestableScrollContainer.tsx
More file actions
84 lines (73 loc) · 2.21 KB
/
NestableScrollContainer.tsx
File metadata and controls
84 lines (73 loc) · 2.21 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
import React from "react";
import { LayoutChangeEvent, ScrollViewProps } from "react-native";
import { ScrollView } from "react-native-gesture-handler";
import Animated, {
runOnJS,
useAnimatedScrollHandler,
} from "react-native-reanimated";
import {
NestableScrollContainerProvider,
useSafeNestableScrollContainerContext,
} from "../context/nestableScrollContainerContext";
import { useStableCallback } from "../hooks/useStableCallback";
const AnimatedScrollView = Animated.createAnimatedComponent(ScrollView);
type NestableScrollContainerInnerProps = {
onScrollOffsetChange?: (scrollOffset: number) => void;
} & Omit<ScrollViewProps, "onScroll">;
function NestableScrollContainerInner(
props: NestableScrollContainerInnerProps
) {
const {
outerScrollOffset,
containerSize,
scrollViewSize,
scrollableRef,
outerScrollEnabled,
} = useSafeNestableScrollContainerContext();
const onScroll = useStableCallback((scrollOffset: number) => {
props.onScrollOffsetChange?.(scrollOffset);
});
const scrollHandler = useAnimatedScrollHandler({
onScroll: (event) => {
outerScrollOffset.value = event.contentOffset.y;
runOnJS(onScroll)(event.contentOffset.y);
},
});
const onLayout = useStableCallback((event: LayoutChangeEvent) => {
const {
nativeEvent: { layout },
} = event;
containerSize.value = layout.height;
});
const onContentSizeChange = useStableCallback((w: number, h: number) => {
scrollViewSize.value = h;
props.onContentSizeChange?.(w, h);
});
return (
<AnimatedScrollView
{...props}
onLayout={onLayout}
onContentSizeChange={onContentSizeChange}
scrollEnabled={outerScrollEnabled}
ref={scrollableRef}
scrollEventThrottle={1}
onScroll={scrollHandler}
/>
);
}
export const NestableScrollContainer = React.forwardRef(
(
props: NestableScrollContainerInnerProps,
forwardedRef?: React.ForwardedRef<ScrollView>
) => {
return (
<NestableScrollContainerProvider
forwardedRef={
(forwardedRef as React.MutableRefObject<ScrollView>) || undefined
}
>
<NestableScrollContainerInner {...props} />
</NestableScrollContainerProvider>
);
}
);