diff options
author | dan <dan.abramov@gmail.com> | 2024-11-01 03:47:53 +0000 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-11-01 03:47:53 +0000 |
commit | 21b82fa19c60499636b8109dea6934e334107aa3 (patch) | |
tree | 91320bfa82bb3fe73d5a24078ff557959c07d67e /src/view/com/composer/Composer.tsx | |
parent | 7a08d61d889328ff5e3b8ba61faab71a5568df2f (diff) | |
download | voidsky-21b82fa19c60499636b8109dea6934e334107aa3.tar.zst |
Fixes for thread composer on Android (#6045)
* Extract function to read contentHeight later * Remove autoscroll to bottom We're going to implement this in the UI layer instead. * Remove worklet from non-worklets to avoid confusion * Rename and invert hasScrolled* variables Their naming was too ambiguous (they used to represent "has scrolled _away_ from X"). I inverted them and clarified the naming. No functional changes. * This should not be necessary It's already called not just from UI thread. And it only sets shared values, which can be done from either thread. * Make hasScrolledTo* derived values It wasn't always correct to derive them manually because reading from .value is stale on JS thread. We could fix that by using the local variables but it makes more conceptualy sense to treat these as derived anyway. * Reimplement autoscroll-to-bottom in UI layer Doing it here ensures we also do it when you add an image at the end of the thread. Otherwise it's very confusing where it went. * Use fancy ScrollView This seems to fix ScrollView getting stuck after inserting images at the thread end on Android. * More aggressive scroll-to-bottom * "Fix" cursor getting stuck on Android * Revert "Use fancy ScrollView" This reverts commit 04e34a54e3b75f8a77de5062bff5fe6e76420bbb.
Diffstat (limited to 'src/view/com/composer/Composer.tsx')
-rw-r--r-- | src/view/com/composer/Composer.tsx | 85 |
1 files changed, 52 insertions, 33 deletions
diff --git a/src/view/com/composer/Composer.tsx b/src/view/com/composer/Composer.tsx index 3c9808448..006e0c7dc 100644 --- a/src/view/com/composer/Composer.tsx +++ b/src/view/com/composer/Composer.tsx @@ -22,6 +22,7 @@ import { // @ts-expect-error no type definition import ProgressCircle from 'react-native-progress/Circle' import Animated, { + AnimatedRef, Easing, FadeIn, FadeOut, @@ -507,25 +508,24 @@ export const ComposePost = ({ useEffect(() => { if (composerState.mutableNeedsFocusActive) { composerState.mutableNeedsFocusActive = false - textInput.current?.focus() + // On Android, this risks getting the cursor stuck behind the keyboard. + // Not worth it. + if (!isAndroid) { + textInput.current?.focus() + } } }, [composerState]) const { - contentHeight, scrollHandler, onScrollViewContentSizeChange, onScrollViewLayout, topBarAnimatedStyle, bottomBarAnimatedStyle, - } = useAnimatedBorders() - - useEffect(() => { - if (composerState.mutableNeedsScrollToBottom) { - composerState.mutableNeedsScrollToBottom = false - runOnUI(scrollTo)(scrollViewRef, 0, contentHeight.value, true) - } - }, [composerState, scrollViewRef, contentHeight]) + } = useScrollTracker({ + scrollViewRef, + stickyBottom: true, + }) const keyboardVerticalOffset = useKeyboardVerticalOffset() @@ -1213,17 +1213,30 @@ export function useComposerCancelRef() { return useRef<CancelRef>(null) } -function useAnimatedBorders() { +function useScrollTracker({ + scrollViewRef, + stickyBottom, +}: { + scrollViewRef: AnimatedRef<Animated.ScrollView> + stickyBottom: boolean +}) { const t = useTheme() - const hasScrolledTop = useSharedValue(0) - const hasScrolledBottom = useSharedValue(0) const contentOffset = useSharedValue(0) const scrollViewHeight = useSharedValue(Infinity) const contentHeight = useSharedValue(0) - /** - * Make sure to run this on the UI thread! - */ + const hasScrolledToTop = useDerivedValue(() => + withTiming(contentOffset.value === 0 ? 1 : 0), + ) + + const hasScrolledToBottom = useDerivedValue(() => + withTiming( + contentHeight.value - contentOffset.value - 5 <= scrollViewHeight.value + ? 1 + : 0, + ), + ) + const showHideBottomBorder = useCallback( ({ newContentHeight, @@ -1235,27 +1248,19 @@ function useAnimatedBorders() { newScrollViewHeight?: number }) => { 'worklet' - if (typeof newContentHeight === 'number') contentHeight.value = Math.floor(newContentHeight) if (typeof newContentOffset === 'number') contentOffset.value = Math.floor(newContentOffset) if (typeof newScrollViewHeight === 'number') scrollViewHeight.value = Math.floor(newScrollViewHeight) - - hasScrolledBottom.value = withTiming( - contentHeight.value - contentOffset.value - 5 > scrollViewHeight.value - ? 1 - : 0, - ) }, - [contentHeight, contentOffset, scrollViewHeight, hasScrolledBottom], + [contentHeight, contentOffset, scrollViewHeight], ) const scrollHandler = useAnimatedScrollHandler({ onScroll: event => { 'worklet' - hasScrolledTop.value = withTiming(event.contentOffset.y > 0 ? 1 : 0) showHideBottomBorder({ newContentOffset: event.contentOffset.y, newContentHeight: event.contentSize.height, @@ -1266,17 +1271,32 @@ function useAnimatedBorders() { const onScrollViewContentSizeChange = useCallback( (_width: number, height: number) => { - 'worklet' + if (stickyBottom && height > contentHeight.value) { + const isFairlyCloseToBottom = + contentHeight.value - contentOffset.value - 100 <= + scrollViewHeight.value + if (isFairlyCloseToBottom) { + runOnUI(() => { + scrollTo(scrollViewRef, 0, contentHeight.value, true) + })() + } + } showHideBottomBorder({ newContentHeight: height, }) }, - [showHideBottomBorder], + [ + showHideBottomBorder, + scrollViewRef, + contentHeight, + stickyBottom, + contentOffset, + scrollViewHeight, + ], ) const onScrollViewLayout = useCallback( (evt: LayoutChangeEvent) => { - 'worklet' showHideBottomBorder({ newScrollViewHeight: evt.nativeEvent.layout.height, }) @@ -1288,9 +1308,9 @@ function useAnimatedBorders() { return { borderBottomWidth: StyleSheet.hairlineWidth, borderColor: interpolateColor( - hasScrolledTop.value, + hasScrolledToTop.value, [0, 1], - ['transparent', t.atoms.border_contrast_medium.borderColor], + [t.atoms.border_contrast_medium.borderColor, 'transparent'], ), } }) @@ -1298,15 +1318,14 @@ function useAnimatedBorders() { return { borderTopWidth: StyleSheet.hairlineWidth, borderColor: interpolateColor( - hasScrolledBottom.value, + hasScrolledToBottom.value, [0, 1], - ['transparent', t.atoms.border_contrast_medium.borderColor], + [t.atoms.border_contrast_medium.borderColor, 'transparent'], ), } }) return { - contentHeight, scrollHandler, onScrollViewContentSizeChange, onScrollViewLayout, |