Fix IntersectionObserver rootMargin in web List implementation, add onStartReached (#3866)
				
					
				
			* add `onStartReached` to web list * fix `rootMargin`
This commit is contained in:
		
							parent
							
								
									6b615f3720
								
							
						
					
					
						commit
						594b40c3ae
					
				
					 1 changed files with 30 additions and 8 deletions
				
			
		|  | @ -1,11 +1,12 @@ | |||
| import React, {isValidElement, memo, useRef, startTransition} from 'react' | ||||
| import React, {isValidElement, memo, startTransition, useRef} from 'react' | ||||
| import {FlatListProps, StyleSheet, View, ViewProps} from 'react-native' | ||||
| import {addStyle} from 'lib/styles' | ||||
| 
 | ||||
| import {batchedUpdates} from '#/lib/batchedUpdates' | ||||
| import {useNonReactiveCallback} from '#/lib/hooks/useNonReactiveCallback' | ||||
| import {useScrollHandlers} from '#/lib/ScrollContext' | ||||
| import {usePalette} from 'lib/hooks/usePalette' | ||||
| import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries' | ||||
| import {useScrollHandlers} from '#/lib/ScrollContext' | ||||
| import {useNonReactiveCallback} from '#/lib/hooks/useNonReactiveCallback' | ||||
| import {batchedUpdates} from '#/lib/batchedUpdates' | ||||
| import {addStyle} from 'lib/styles' | ||||
| 
 | ||||
| export type ListMethods = any // TODO: Better types.
 | ||||
| export type ListProps<ItemT> = Omit< | ||||
|  | @ -32,6 +33,8 @@ function ListImpl<ItemT>( | |||
|     headerOffset, | ||||
|     keyExtractor, | ||||
|     refreshing: _unsupportedRefreshing, | ||||
|     onStartReached, | ||||
|     onStartReachedThreshold = 0, | ||||
|     onEndReached, | ||||
|     onEndReachedThreshold = 0, | ||||
|     onRefresh: _unsupportedOnRefresh, | ||||
|  | @ -148,6 +151,17 @@ function ListImpl<ItemT>( | |||
|     } | ||||
|   } | ||||
| 
 | ||||
|   // --- onStartReached ---
 | ||||
|   const onHeadVisibilityChange = useNonReactiveCallback( | ||||
|     (isHeadVisible: boolean) => { | ||||
|       if (isHeadVisible) { | ||||
|         onStartReached?.({ | ||||
|           distanceFromStart: onStartReachedThreshold || 0, | ||||
|         }) | ||||
|       } | ||||
|     }, | ||||
|   ) | ||||
| 
 | ||||
|   // --- onEndReached ---
 | ||||
|   const onTailVisibilityChange = useNonReactiveCallback( | ||||
|     (isTailVisible: boolean) => { | ||||
|  | @ -181,6 +195,12 @@ function ListImpl<ItemT>( | |||
|           onVisibleChange={handleAboveTheFoldVisibleChange} | ||||
|           style={[styles.aboveTheFoldDetector, {height: headerOffset}]} | ||||
|         /> | ||||
|         {onStartReached && ( | ||||
|           <Visibility | ||||
|             onVisibleChange={onHeadVisibilityChange} | ||||
|             topMargin={(onStartReachedThreshold ?? 0) * 100 + '%'} | ||||
|           /> | ||||
|         )} | ||||
|         {header} | ||||
|         {(data as Array<ItemT>).map((item, index) => ( | ||||
|           <Row<ItemT> | ||||
|  | @ -193,8 +213,8 @@ function ListImpl<ItemT>( | |||
|         ))} | ||||
|         {onEndReached && ( | ||||
|           <Visibility | ||||
|             topMargin={(onEndReachedThreshold ?? 0) * 100 + '%'} | ||||
|             onVisibleChange={onTailVisibilityChange} | ||||
|             bottomMargin={(onEndReachedThreshold ?? 0) * 100 + '%'} | ||||
|           /> | ||||
|         )} | ||||
|         {footer} | ||||
|  | @ -256,10 +276,12 @@ Row = React.memo(Row) | |||
| 
 | ||||
| let Visibility = ({ | ||||
|   topMargin = '0px', | ||||
|   bottomMargin = '0px', | ||||
|   onVisibleChange, | ||||
|   style, | ||||
| }: { | ||||
|   topMargin?: string | ||||
|   bottomMargin?: string | ||||
|   onVisibleChange: (isVisible: boolean) => void | ||||
|   style?: ViewProps['style'] | ||||
| }): React.ReactNode => { | ||||
|  | @ -281,14 +303,14 @@ let Visibility = ({ | |||
| 
 | ||||
|   React.useEffect(() => { | ||||
|     const observer = new IntersectionObserver(handleIntersection, { | ||||
|       rootMargin: `${topMargin} 0px 0px 0px`, | ||||
|       rootMargin: `${topMargin} 0px ${bottomMargin} 0px`, | ||||
|     }) | ||||
|     const tail: Element | null = tailRef.current! | ||||
|     observer.observe(tail) | ||||
|     return () => { | ||||
|       observer.unobserve(tail) | ||||
|     } | ||||
|   }, [handleIntersection, topMargin]) | ||||
|   }, [bottomMargin, handleIntersection, topMargin]) | ||||
| 
 | ||||
|   return ( | ||||
|     <View ref={tailRef} style={addStyle(styles.visibilityDetector, style)} /> | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue