Push useAnimatedScrollHandler down everywhere to work around bugs (#1866)
* Move useOnMainScroll handlers to leaves * Force Feed to always take handlers * Pass handlers from the pagerzio/stable
parent
e0e5bc8fd8
commit
65def37165
|
@ -1 +1,15 @@
|
||||||
|
// Be warned. This Hook is very buggy unless used in a very constrained way.
|
||||||
|
// To use it safely:
|
||||||
|
//
|
||||||
|
// - DO NOT pass its return value as a prop to any user-defined component.
|
||||||
|
// - DO NOT pass its return value to more than a single component.
|
||||||
|
//
|
||||||
|
// In other words, the only safe way to use it is next to the leaf Reanimated View.
|
||||||
|
//
|
||||||
|
// Relevant bug reports:
|
||||||
|
// - https://github.com/software-mansion/react-native-reanimated/issues/5345
|
||||||
|
// - https://github.com/software-mansion/react-native-reanimated/issues/5360
|
||||||
|
// - https://github.com/software-mansion/react-native-reanimated/issues/5364
|
||||||
|
//
|
||||||
|
// It's great when it works though.
|
||||||
export {useAnimatedScrollHandler} from 'react-native-reanimated'
|
export {useAnimatedScrollHandler} from 'react-native-reanimated'
|
||||||
|
|
|
@ -1,11 +1,15 @@
|
||||||
import {useState, useCallback} from 'react'
|
import {useState, useCallback, useMemo} from 'react'
|
||||||
import {NativeSyntheticEvent, NativeScrollEvent} from 'react-native'
|
import {NativeSyntheticEvent, NativeScrollEvent} from 'react-native'
|
||||||
import {useSetMinimalShellMode, useMinimalShellMode} from '#/state/shell'
|
import {useSetMinimalShellMode, useMinimalShellMode} from '#/state/shell'
|
||||||
import {useShellLayout} from '#/state/shell/shell-layout'
|
import {useShellLayout} from '#/state/shell/shell-layout'
|
||||||
import {s} from 'lib/styles'
|
import {s} from 'lib/styles'
|
||||||
import {isWeb} from 'platform/detection'
|
import {isWeb} from 'platform/detection'
|
||||||
import {useSharedValue, interpolate, runOnJS} from 'react-native-reanimated'
|
import {
|
||||||
import {useAnimatedScrollHandler} from './useAnimatedScrollHandler_FIXED'
|
useSharedValue,
|
||||||
|
interpolate,
|
||||||
|
runOnJS,
|
||||||
|
ScrollHandlers,
|
||||||
|
} from 'react-native-reanimated'
|
||||||
|
|
||||||
function clamp(num: number, min: number, max: number) {
|
function clamp(num: number, min: number, max: number) {
|
||||||
'worklet'
|
'worklet'
|
||||||
|
@ -15,9 +19,10 @@ function clamp(num: number, min: number, max: number) {
|
||||||
export type OnScrollCb = (
|
export type OnScrollCb = (
|
||||||
event: NativeSyntheticEvent<NativeScrollEvent>,
|
event: NativeSyntheticEvent<NativeScrollEvent>,
|
||||||
) => void
|
) => void
|
||||||
|
export type OnScrollHandler = ScrollHandlers<any>
|
||||||
export type ResetCb = () => void
|
export type ResetCb = () => void
|
||||||
|
|
||||||
export function useOnMainScroll(): [OnScrollCb, boolean, ResetCb] {
|
export function useOnMainScroll(): [OnScrollHandler, boolean, ResetCb] {
|
||||||
const {headerHeight} = useShellLayout()
|
const {headerHeight} = useShellLayout()
|
||||||
const [isScrolledDown, setIsScrolledDown] = useState(false)
|
const [isScrolledDown, setIsScrolledDown] = useState(false)
|
||||||
const mode = useMinimalShellMode()
|
const mode = useMinimalShellMode()
|
||||||
|
@ -25,12 +30,18 @@ export function useOnMainScroll(): [OnScrollCb, boolean, ResetCb] {
|
||||||
const startDragOffset = useSharedValue<number | null>(null)
|
const startDragOffset = useSharedValue<number | null>(null)
|
||||||
const startMode = useSharedValue<number | null>(null)
|
const startMode = useSharedValue<number | null>(null)
|
||||||
|
|
||||||
const scrollHandler = useAnimatedScrollHandler({
|
const onBeginDrag = useCallback(
|
||||||
onBeginDrag(e) {
|
(e: NativeScrollEvent) => {
|
||||||
|
'worklet'
|
||||||
startDragOffset.value = e.contentOffset.y
|
startDragOffset.value = e.contentOffset.y
|
||||||
startMode.value = mode.value
|
startMode.value = mode.value
|
||||||
},
|
},
|
||||||
onEndDrag(e) {
|
[mode, startDragOffset, startMode],
|
||||||
|
)
|
||||||
|
|
||||||
|
const onEndDrag = useCallback(
|
||||||
|
(e: NativeScrollEvent) => {
|
||||||
|
'worklet'
|
||||||
startDragOffset.value = null
|
startDragOffset.value = null
|
||||||
startMode.value = null
|
startMode.value = null
|
||||||
if (e.contentOffset.y < headerHeight.value / 2) {
|
if (e.contentOffset.y < headerHeight.value / 2) {
|
||||||
|
@ -41,7 +52,12 @@ export function useOnMainScroll(): [OnScrollCb, boolean, ResetCb] {
|
||||||
setMode(Math.round(mode.value) === 1)
|
setMode(Math.round(mode.value) === 1)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
onScroll(e) {
|
[startDragOffset, startMode, setMode, mode, headerHeight],
|
||||||
|
)
|
||||||
|
|
||||||
|
const onScroll = useCallback(
|
||||||
|
(e: NativeScrollEvent) => {
|
||||||
|
'worklet'
|
||||||
// Keep track of whether we want to show "scroll to top".
|
// Keep track of whether we want to show "scroll to top".
|
||||||
if (!isScrolledDown && e.contentOffset.y > s.window.height) {
|
if (!isScrolledDown && e.contentOffset.y > s.window.height) {
|
||||||
runOnJS(setIsScrolledDown)(true)
|
runOnJS(setIsScrolledDown)(true)
|
||||||
|
@ -86,7 +102,17 @@ export function useOnMainScroll(): [OnScrollCb, boolean, ResetCb] {
|
||||||
startMode.value = mode.value
|
startMode.value = mode.value
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
})
|
[headerHeight, mode, setMode, isScrolledDown, startDragOffset, startMode],
|
||||||
|
)
|
||||||
|
|
||||||
|
const scrollHandler: ScrollHandlers<any> = useMemo(
|
||||||
|
() => ({
|
||||||
|
onBeginDrag,
|
||||||
|
onEndDrag,
|
||||||
|
onScroll,
|
||||||
|
}),
|
||||||
|
[onBeginDrag, onEndDrag, onScroll],
|
||||||
|
)
|
||||||
|
|
||||||
return [
|
return [
|
||||||
scrollHandler,
|
scrollHandler,
|
||||||
|
|
|
@ -19,9 +19,10 @@ import {useAnalytics} from 'lib/analytics/analytics'
|
||||||
import {usePalette} from 'lib/hooks/usePalette'
|
import {usePalette} from 'lib/hooks/usePalette'
|
||||||
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
||||||
import {s} from 'lib/styles'
|
import {s} from 'lib/styles'
|
||||||
import {OnScrollCb} from 'lib/hooks/useOnMainScroll'
|
import {OnScrollHandler} from 'lib/hooks/useOnMainScroll'
|
||||||
import {logger} from '#/logger'
|
import {logger} from '#/logger'
|
||||||
import {useModalControls} from '#/state/modals'
|
import {useModalControls} from '#/state/modals'
|
||||||
|
import {useAnimatedScrollHandler} from '#/lib/hooks/useAnimatedScrollHandler_FIXED'
|
||||||
|
|
||||||
const LOADING_ITEM = {_reactKey: '__loading__'}
|
const LOADING_ITEM = {_reactKey: '__loading__'}
|
||||||
const EMPTY_ITEM = {_reactKey: '__empty__'}
|
const EMPTY_ITEM = {_reactKey: '__empty__'}
|
||||||
|
@ -44,7 +45,7 @@ export const ListItems = observer(function ListItemsImpl({
|
||||||
list: ListModel
|
list: ListModel
|
||||||
style?: StyleProp<ViewStyle>
|
style?: StyleProp<ViewStyle>
|
||||||
scrollElRef?: MutableRefObject<FlatList<any> | null>
|
scrollElRef?: MutableRefObject<FlatList<any> | null>
|
||||||
onScroll?: OnScrollCb
|
onScroll: OnScrollHandler
|
||||||
onPressTryAgain?: () => void
|
onPressTryAgain?: () => void
|
||||||
renderHeader: () => JSX.Element
|
renderHeader: () => JSX.Element
|
||||||
renderEmptyState: () => JSX.Element
|
renderEmptyState: () => JSX.Element
|
||||||
|
@ -205,6 +206,7 @@ export const ListItems = observer(function ListItemsImpl({
|
||||||
[list.isLoading],
|
[list.isLoading],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const scrollHandler = useAnimatedScrollHandler(onScroll)
|
||||||
return (
|
return (
|
||||||
<View testID={testID} style={style}>
|
<View testID={testID} style={style}>
|
||||||
<FlatList
|
<FlatList
|
||||||
|
@ -226,7 +228,7 @@ export const ListItems = observer(function ListItemsImpl({
|
||||||
}
|
}
|
||||||
contentContainerStyle={s.contentContainer}
|
contentContainerStyle={s.contentContainer}
|
||||||
style={{paddingTop: headerOffset}}
|
style={{paddingTop: headerOffset}}
|
||||||
onScroll={onScroll}
|
onScroll={scrollHandler}
|
||||||
onEndReached={onEndReached}
|
onEndReached={onEndReached}
|
||||||
onEndReachedThreshold={0.6}
|
onEndReachedThreshold={0.6}
|
||||||
scrollEventThrottle={scrollEventThrottle}
|
scrollEventThrottle={scrollEventThrottle}
|
||||||
|
|
|
@ -8,7 +8,8 @@ import {NotificationFeedLoadingPlaceholder} from '../util/LoadingPlaceholder'
|
||||||
import {ErrorMessage} from '../util/error/ErrorMessage'
|
import {ErrorMessage} from '../util/error/ErrorMessage'
|
||||||
import {LoadMoreRetryBtn} from '../util/LoadMoreRetryBtn'
|
import {LoadMoreRetryBtn} from '../util/LoadMoreRetryBtn'
|
||||||
import {EmptyState} from '../util/EmptyState'
|
import {EmptyState} from '../util/EmptyState'
|
||||||
import {OnScrollCb} from 'lib/hooks/useOnMainScroll'
|
import {OnScrollHandler} from 'lib/hooks/useOnMainScroll'
|
||||||
|
import {useAnimatedScrollHandler} from '#/lib/hooks/useAnimatedScrollHandler_FIXED'
|
||||||
import {s} from 'lib/styles'
|
import {s} from 'lib/styles'
|
||||||
import {usePalette} from 'lib/hooks/usePalette'
|
import {usePalette} from 'lib/hooks/usePalette'
|
||||||
import {logger} from '#/logger'
|
import {logger} from '#/logger'
|
||||||
|
@ -27,7 +28,7 @@ export const Feed = observer(function Feed({
|
||||||
view: NotificationsFeedModel
|
view: NotificationsFeedModel
|
||||||
scrollElRef?: MutableRefObject<FlatList<any> | null>
|
scrollElRef?: MutableRefObject<FlatList<any> | null>
|
||||||
onPressTryAgain?: () => void
|
onPressTryAgain?: () => void
|
||||||
onScroll?: OnScrollCb
|
onScroll?: OnScrollHandler
|
||||||
ListHeaderComponent?: () => JSX.Element
|
ListHeaderComponent?: () => JSX.Element
|
||||||
}) {
|
}) {
|
||||||
const pal = usePalette('default')
|
const pal = usePalette('default')
|
||||||
|
@ -129,6 +130,7 @@ export const Feed = observer(function Feed({
|
||||||
[view],
|
[view],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const scrollHandler = useAnimatedScrollHandler(onScroll || {})
|
||||||
return (
|
return (
|
||||||
<View style={s.hContentRegion}>
|
<View style={s.hContentRegion}>
|
||||||
<CenteredView>
|
<CenteredView>
|
||||||
|
@ -161,7 +163,7 @@ export const Feed = observer(function Feed({
|
||||||
}
|
}
|
||||||
onEndReached={onEndReached}
|
onEndReached={onEndReached}
|
||||||
onEndReachedThreshold={0.6}
|
onEndReachedThreshold={0.6}
|
||||||
onScroll={onScroll}
|
onScroll={scrollHandler}
|
||||||
scrollEventThrottle={1}
|
scrollEventThrottle={1}
|
||||||
contentContainerStyle={s.contentContainer}
|
contentContainerStyle={s.contentContainer}
|
||||||
// @ts-ignore our .web version only -prf
|
// @ts-ignore our .web version only -prf
|
||||||
|
|
|
@ -1,5 +1,10 @@
|
||||||
import * as React from 'react'
|
import * as React from 'react'
|
||||||
import {LayoutChangeEvent, StyleSheet, View} from 'react-native'
|
import {
|
||||||
|
LayoutChangeEvent,
|
||||||
|
NativeScrollEvent,
|
||||||
|
StyleSheet,
|
||||||
|
View,
|
||||||
|
} from 'react-native'
|
||||||
import Animated, {
|
import Animated, {
|
||||||
Easing,
|
Easing,
|
||||||
useAnimatedReaction,
|
useAnimatedReaction,
|
||||||
|
@ -11,14 +16,13 @@ import Animated, {
|
||||||
import {Pager, PagerRef, RenderTabBarFnProps} from 'view/com/pager/Pager'
|
import {Pager, PagerRef, RenderTabBarFnProps} from 'view/com/pager/Pager'
|
||||||
import {TabBar} from './TabBar'
|
import {TabBar} from './TabBar'
|
||||||
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
||||||
import {OnScrollCb} from 'lib/hooks/useOnMainScroll'
|
import {OnScrollHandler} from 'lib/hooks/useOnMainScroll'
|
||||||
import {useAnimatedScrollHandler} from 'lib/hooks/useAnimatedScrollHandler_FIXED'
|
|
||||||
|
|
||||||
const SCROLLED_DOWN_LIMIT = 200
|
const SCROLLED_DOWN_LIMIT = 200
|
||||||
|
|
||||||
interface PagerWithHeaderChildParams {
|
interface PagerWithHeaderChildParams {
|
||||||
headerHeight: number
|
headerHeight: number
|
||||||
onScroll: OnScrollCb
|
onScroll: OnScrollHandler
|
||||||
isScrolledDown: boolean
|
isScrolledDown: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -141,11 +145,10 @@ export const PagerWithHeader = React.forwardRef<PagerRef, PagerWithHeaderProps>(
|
||||||
)
|
)
|
||||||
|
|
||||||
// props to pass into children render functions
|
// props to pass into children render functions
|
||||||
const onScroll = useAnimatedScrollHandler({
|
function onScrollWorklet(e: NativeScrollEvent) {
|
||||||
onScroll(e) {
|
'worklet'
|
||||||
scrollY.value = e.contentOffset.y
|
scrollY.value = e.contentOffset.y
|
||||||
},
|
}
|
||||||
})
|
|
||||||
|
|
||||||
const onPageSelectedInner = React.useCallback(
|
const onPageSelectedInner = React.useCallback(
|
||||||
(index: number) => {
|
(index: number) => {
|
||||||
|
@ -192,7 +195,9 @@ export const PagerWithHeader = React.forwardRef<PagerRef, PagerWithHeaderProps>(
|
||||||
output = child({
|
output = child({
|
||||||
headerHeight,
|
headerHeight,
|
||||||
isScrolledDown,
|
isScrolledDown,
|
||||||
onScroll: i === currentPage ? onScroll : noop,
|
onScroll: {
|
||||||
|
onScroll: i === currentPage ? onScrollWorklet : noop,
|
||||||
|
},
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
// Pager children must be noncollapsible plain <View>s.
|
// Pager children must be noncollapsible plain <View>s.
|
||||||
|
@ -225,7 +230,9 @@ const styles = StyleSheet.create({
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
function noop() {}
|
function noop() {
|
||||||
|
'worklet'
|
||||||
|
}
|
||||||
|
|
||||||
function toArray<T>(v: T | T[]): T[] {
|
function toArray<T>(v: T | T[]): T[] {
|
||||||
if (Array.isArray(v)) {
|
if (Array.isArray(v)) {
|
||||||
|
|
|
@ -14,10 +14,11 @@ import {FeedErrorMessage} from './FeedErrorMessage'
|
||||||
import {PostsFeedModel} from 'state/models/feeds/posts'
|
import {PostsFeedModel} from 'state/models/feeds/posts'
|
||||||
import {FeedSlice} from './FeedSlice'
|
import {FeedSlice} from './FeedSlice'
|
||||||
import {LoadMoreRetryBtn} from '../util/LoadMoreRetryBtn'
|
import {LoadMoreRetryBtn} from '../util/LoadMoreRetryBtn'
|
||||||
import {OnScrollCb} from 'lib/hooks/useOnMainScroll'
|
import {OnScrollHandler} from 'lib/hooks/useOnMainScroll'
|
||||||
import {s} from 'lib/styles'
|
import {s} from 'lib/styles'
|
||||||
import {useAnalytics} from 'lib/analytics/analytics'
|
import {useAnalytics} from 'lib/analytics/analytics'
|
||||||
import {usePalette} from 'lib/hooks/usePalette'
|
import {usePalette} from 'lib/hooks/usePalette'
|
||||||
|
import {useAnimatedScrollHandler} from '#/lib/hooks/useAnimatedScrollHandler_FIXED'
|
||||||
import {useTheme} from 'lib/ThemeContext'
|
import {useTheme} from 'lib/ThemeContext'
|
||||||
import {logger} from '#/logger'
|
import {logger} from '#/logger'
|
||||||
|
|
||||||
|
@ -43,7 +44,7 @@ export const Feed = observer(function Feed({
|
||||||
feed: PostsFeedModel
|
feed: PostsFeedModel
|
||||||
style?: StyleProp<ViewStyle>
|
style?: StyleProp<ViewStyle>
|
||||||
scrollElRef?: MutableRefObject<FlatList<any> | null>
|
scrollElRef?: MutableRefObject<FlatList<any> | null>
|
||||||
onScroll?: OnScrollCb
|
onScroll?: OnScrollHandler
|
||||||
scrollEventThrottle?: number
|
scrollEventThrottle?: number
|
||||||
renderEmptyState: () => JSX.Element
|
renderEmptyState: () => JSX.Element
|
||||||
renderEndOfFeed?: () => JSX.Element
|
renderEndOfFeed?: () => JSX.Element
|
||||||
|
@ -157,6 +158,7 @@ export const Feed = observer(function Feed({
|
||||||
[feed.isLoadingMore, feed.hasMore, feed.isEmpty, renderEndOfFeed],
|
[feed.isLoadingMore, feed.hasMore, feed.isEmpty, renderEndOfFeed],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const scrollHandler = useAnimatedScrollHandler(onScroll || {})
|
||||||
return (
|
return (
|
||||||
<View testID={testID} style={style}>
|
<View testID={testID} style={style}>
|
||||||
<FlatList
|
<FlatList
|
||||||
|
@ -178,7 +180,7 @@ export const Feed = observer(function Feed({
|
||||||
}
|
}
|
||||||
contentContainerStyle={s.contentContainer}
|
contentContainerStyle={s.contentContainer}
|
||||||
style={{paddingTop: headerOffset}}
|
style={{paddingTop: headerOffset}}
|
||||||
onScroll={onScroll}
|
onScroll={onScroll != null ? scrollHandler : undefined}
|
||||||
scrollEventThrottle={scrollEventThrottle}
|
scrollEventThrottle={scrollEventThrottle}
|
||||||
indicatorStyle={theme.colorScheme === 'dark' ? 'white' : 'black'}
|
indicatorStyle={theme.colorScheme === 'dark' ? 'white' : 'black'}
|
||||||
onEndReached={onEndReached}
|
onEndReached={onEndReached}
|
||||||
|
|
|
@ -26,7 +26,7 @@ import {EmptyState} from 'view/com/util/EmptyState'
|
||||||
import * as Toast from 'view/com/util/Toast'
|
import * as Toast from 'view/com/util/Toast'
|
||||||
import {useSetTitle} from 'lib/hooks/useSetTitle'
|
import {useSetTitle} from 'lib/hooks/useSetTitle'
|
||||||
import {useCustomFeed} from 'lib/hooks/useCustomFeed'
|
import {useCustomFeed} from 'lib/hooks/useCustomFeed'
|
||||||
import {OnScrollCb} from 'lib/hooks/useOnMainScroll'
|
import {OnScrollHandler} from 'lib/hooks/useOnMainScroll'
|
||||||
import {shareUrl} from 'lib/sharing'
|
import {shareUrl} from 'lib/sharing'
|
||||||
import {toShareUrl} from 'lib/strings/url-helpers'
|
import {toShareUrl} from 'lib/strings/url-helpers'
|
||||||
import {Haptics} from 'lib/haptics'
|
import {Haptics} from 'lib/haptics'
|
||||||
|
@ -44,6 +44,7 @@ import {logger} from '#/logger'
|
||||||
import {Trans, msg} from '@lingui/macro'
|
import {Trans, msg} from '@lingui/macro'
|
||||||
import {useLingui} from '@lingui/react'
|
import {useLingui} from '@lingui/react'
|
||||||
import {useModalControls} from '#/state/modals'
|
import {useModalControls} from '#/state/modals'
|
||||||
|
import {useAnimatedScrollHandler} from '#/lib/hooks/useAnimatedScrollHandler_FIXED'
|
||||||
|
|
||||||
const SECTION_TITLES = ['Posts', 'About']
|
const SECTION_TITLES = ['Posts', 'About']
|
||||||
|
|
||||||
|
@ -383,7 +384,7 @@ export const ProfileFeedScreenInner = observer(
|
||||||
|
|
||||||
interface FeedSectionProps {
|
interface FeedSectionProps {
|
||||||
feed: PostsFeedModel
|
feed: PostsFeedModel
|
||||||
onScroll: OnScrollCb
|
onScroll: OnScrollHandler
|
||||||
headerHeight: number
|
headerHeight: number
|
||||||
isScrolledDown: boolean
|
isScrolledDown: boolean
|
||||||
}
|
}
|
||||||
|
@ -443,10 +444,11 @@ const AboutSection = observer(function AboutPageImpl({
|
||||||
feedInfo: FeedSourceModel | undefined
|
feedInfo: FeedSourceModel | undefined
|
||||||
headerHeight: number
|
headerHeight: number
|
||||||
onToggleLiked: () => void
|
onToggleLiked: () => void
|
||||||
onScroll: OnScrollCb
|
onScroll: OnScrollHandler
|
||||||
}) {
|
}) {
|
||||||
const pal = usePalette('default')
|
const pal = usePalette('default')
|
||||||
const {_} = useLingui()
|
const {_} = useLingui()
|
||||||
|
const scrollHandler = useAnimatedScrollHandler(onScroll)
|
||||||
|
|
||||||
if (!feedInfo) {
|
if (!feedInfo) {
|
||||||
return <View />
|
return <View />
|
||||||
|
@ -456,7 +458,7 @@ const AboutSection = observer(function AboutPageImpl({
|
||||||
<ScrollView
|
<ScrollView
|
||||||
scrollEventThrottle={1}
|
scrollEventThrottle={1}
|
||||||
contentContainerStyle={{paddingTop: headerHeight}}
|
contentContainerStyle={{paddingTop: headerHeight}}
|
||||||
onScroll={onScroll}>
|
onScroll={scrollHandler}>
|
||||||
<View
|
<View
|
||||||
style={[
|
style={[
|
||||||
{
|
{
|
||||||
|
|
|
@ -33,7 +33,7 @@ import {useStores} from 'state/index'
|
||||||
import {usePalette} from 'lib/hooks/usePalette'
|
import {usePalette} from 'lib/hooks/usePalette'
|
||||||
import {useSetTitle} from 'lib/hooks/useSetTitle'
|
import {useSetTitle} from 'lib/hooks/useSetTitle'
|
||||||
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
||||||
import {OnScrollCb} from 'lib/hooks/useOnMainScroll'
|
import {OnScrollHandler} from 'lib/hooks/useOnMainScroll'
|
||||||
import {NavigationProp} from 'lib/routes/types'
|
import {NavigationProp} from 'lib/routes/types'
|
||||||
import {toShareUrl} from 'lib/strings/url-helpers'
|
import {toShareUrl} from 'lib/strings/url-helpers'
|
||||||
import {shareUrl} from 'lib/sharing'
|
import {shareUrl} from 'lib/sharing'
|
||||||
|
@ -554,7 +554,7 @@ const Header = observer(function HeaderImpl({
|
||||||
|
|
||||||
interface FeedSectionProps {
|
interface FeedSectionProps {
|
||||||
feed: PostsFeedModel
|
feed: PostsFeedModel
|
||||||
onScroll: OnScrollCb
|
onScroll: OnScrollHandler
|
||||||
headerHeight: number
|
headerHeight: number
|
||||||
isScrolledDown: boolean
|
isScrolledDown: boolean
|
||||||
}
|
}
|
||||||
|
@ -608,7 +608,7 @@ interface AboutSectionProps {
|
||||||
isCurateList: boolean | undefined
|
isCurateList: boolean | undefined
|
||||||
isOwner: boolean | undefined
|
isOwner: boolean | undefined
|
||||||
onPressAddUser: () => void
|
onPressAddUser: () => void
|
||||||
onScroll: OnScrollCb
|
onScroll: OnScrollHandler
|
||||||
headerHeight: number
|
headerHeight: number
|
||||||
isScrolledDown: boolean
|
isScrolledDown: boolean
|
||||||
}
|
}
|
||||||
|
|
|
@ -14,6 +14,7 @@ import {
|
||||||
} from 'lib/routes/types'
|
} from 'lib/routes/types'
|
||||||
import {observer} from 'mobx-react-lite'
|
import {observer} from 'mobx-react-lite'
|
||||||
import {Text} from 'view/com/util/text/Text'
|
import {Text} from 'view/com/util/text/Text'
|
||||||
|
import {useAnimatedScrollHandler} from '#/lib/hooks/useAnimatedScrollHandler_FIXED'
|
||||||
import {useStores} from 'state/index'
|
import {useStores} from 'state/index'
|
||||||
import {UserAutocompleteModel} from 'state/models/discovery/user-autocomplete'
|
import {UserAutocompleteModel} from 'state/models/discovery/user-autocomplete'
|
||||||
import {SearchUIModel} from 'state/models/ui/search'
|
import {SearchUIModel} from 'state/models/ui/search'
|
||||||
|
@ -131,6 +132,7 @@ export const SearchScreen = withAuthRequired(
|
||||||
}
|
}
|
||||||
}, [])
|
}, [])
|
||||||
|
|
||||||
|
const scrollHandler = useAnimatedScrollHandler(onMainScroll)
|
||||||
return (
|
return (
|
||||||
<TouchableWithoutFeedback onPress={onPress} accessible={false}>
|
<TouchableWithoutFeedback onPress={onPress} accessible={false}>
|
||||||
<View style={[pal.view, styles.container]}>
|
<View style={[pal.view, styles.container]}>
|
||||||
|
@ -156,8 +158,8 @@ export const SearchScreen = withAuthRequired(
|
||||||
ref={scrollViewRef}
|
ref={scrollViewRef}
|
||||||
testID="searchScrollView"
|
testID="searchScrollView"
|
||||||
style={pal.view}
|
style={pal.view}
|
||||||
onScroll={onMainScroll}
|
onScroll={scrollHandler}
|
||||||
scrollEventThrottle={100}>
|
scrollEventThrottle={1}>
|
||||||
{query && autocompleteView.suggestions.length ? (
|
{query && autocompleteView.suggestions.length ? (
|
||||||
<>
|
<>
|
||||||
{autocompleteView.suggestions.map((suggestion, index) => (
|
{autocompleteView.suggestions.map((suggestion, index) => (
|
||||||
|
|
Loading…
Reference in New Issue