Implement popular & following views on home screen
This commit is contained in:
		
							parent
							
								
									ad9da82612
								
							
						
					
					
						commit
						71209bb3ac
					
				
					 6 changed files with 273 additions and 163 deletions
				
			
		|  | @ -257,7 +257,7 @@ export class FeedModel { | ||||||
| 
 | 
 | ||||||
|   constructor( |   constructor( | ||||||
|     public rootStore: RootStoreModel, |     public rootStore: RootStoreModel, | ||||||
|     public feedType: 'home' | 'author' | 'suggested', |     public feedType: 'home' | 'author' | 'suggested' | 'goodstuff', | ||||||
|     params: GetTimeline.QueryParams | GetAuthorFeed.QueryParams, |     params: GetTimeline.QueryParams | GetAuthorFeed.QueryParams, | ||||||
|   ) { |   ) { | ||||||
|     makeAutoObservable( |     makeAutoObservable( | ||||||
|  | @ -634,6 +634,16 @@ export class FeedModel { | ||||||
|       return this.rootStore.api.app.bsky.feed.getTimeline( |       return this.rootStore.api.app.bsky.feed.getTimeline( | ||||||
|         params as GetTimeline.QueryParams, |         params as GetTimeline.QueryParams, | ||||||
|       ) |       ) | ||||||
|  |     } else if (this.feedType === 'goodstuff') { | ||||||
|  |       const res = await this.rootStore.api.app.bsky.feed.getAuthorFeed({ | ||||||
|  |         ...params, | ||||||
|  |         author: 'jay.bsky.social', | ||||||
|  |       } as GetAuthorFeed.QueryParams) | ||||||
|  |       res.data.feed = mergePosts([res], {repostsOnly: true}) | ||||||
|  |       res.data.feed.forEach(item => { | ||||||
|  |         delete item.reason | ||||||
|  |       }) | ||||||
|  |       return res | ||||||
|     } else { |     } else { | ||||||
|       return this.rootStore.api.app.bsky.feed.getAuthorFeed( |       return this.rootStore.api.app.bsky.feed.getAuthorFeed( | ||||||
|         params as GetAuthorFeed.QueryParams, |         params as GetAuthorFeed.QueryParams, | ||||||
|  |  | ||||||
|  | @ -42,7 +42,7 @@ const FollowButton = observer( | ||||||
| 
 | 
 | ||||||
|     return ( |     return ( | ||||||
|       <Button |       <Button | ||||||
|         type={isFollowing ? 'default' : 'primary'} |         type={isFollowing ? 'default' : 'inverted'} | ||||||
|         onPress={onToggleFollowInner} |         onPress={onToggleFollowInner} | ||||||
|         label={isFollowing ? 'Unfollow' : 'Follow'} |         label={isFollowing ? 'Unfollow' : 'Follow'} | ||||||
|       /> |       /> | ||||||
|  |  | ||||||
|  | @ -2,16 +2,25 @@ import React from 'react' | ||||||
| import {Animated, StyleSheet, View} from 'react-native' | import {Animated, StyleSheet, View} from 'react-native' | ||||||
| import PagerView, {PagerViewOnPageSelectedEvent} from 'react-native-pager-view' | import PagerView, {PagerViewOnPageSelectedEvent} from 'react-native-pager-view' | ||||||
| import {useAnimatedValue} from 'lib/hooks/useAnimatedValue' | import {useAnimatedValue} from 'lib/hooks/useAnimatedValue' | ||||||
| import {TabBar} from './TabBar' | import {s} from 'lib/styles' | ||||||
| 
 | 
 | ||||||
| export type PageSelectedEvent = PagerViewOnPageSelectedEvent | export type PageSelectedEvent = PagerViewOnPageSelectedEvent | ||||||
| const AnimatedPagerView = Animated.createAnimatedComponent(PagerView) | const AnimatedPagerView = Animated.createAnimatedComponent(PagerView) | ||||||
| 
 | 
 | ||||||
|  | export interface TabBarProps { | ||||||
|  |   selectedPage: number | ||||||
|  |   position: Animated.Value | ||||||
|  |   offset: Animated.Value | ||||||
|  |   onSelect?: (index: number) => void | ||||||
|  | } | ||||||
|  | 
 | ||||||
| interface Props { | interface Props { | ||||||
|  |   renderTabBar: (props: TabBarProps) => JSX.Element | ||||||
|   onPageSelected?: (e: PageSelectedEvent) => void |   onPageSelected?: (e: PageSelectedEvent) => void | ||||||
| } | } | ||||||
| export const Pager = ({ | export const Pager = ({ | ||||||
|   children, |   children, | ||||||
|  |   renderTabBar, | ||||||
|   onPageSelected, |   onPageSelected, | ||||||
| }: React.PropsWithChildren<Props>) => { | }: React.PropsWithChildren<Props>) => { | ||||||
|   const [selectedPage, setSelectedPage] = React.useState(0) |   const [selectedPage, setSelectedPage] = React.useState(0) | ||||||
|  | @ -36,16 +45,10 @@ export const Pager = ({ | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <View> |     <View> | ||||||
|       <TabBar |       {renderTabBar({selectedPage, position, offset, onSelect: onTabBarSelect})} | ||||||
|         position={position} |  | ||||||
|         offset={offset} |  | ||||||
|         items={['One', 'Two', 'Three']} |  | ||||||
|         selectedPage={selectedPage} |  | ||||||
|         onSelect={onTabBarSelect} |  | ||||||
|       /> |  | ||||||
|       <AnimatedPagerView |       <AnimatedPagerView | ||||||
|         ref={pagerView} |         ref={pagerView} | ||||||
|         style={{height: '100%'}} |         style={s.h100pct} | ||||||
|         initialPage={0} |         initialPage={0} | ||||||
|         onPageSelected={onPageSelectedInner} |         onPageSelected={onPageSelectedInner} | ||||||
|         onPageScroll={Animated.event( |         onPageScroll={Animated.event( | ||||||
|  | @ -64,9 +67,3 @@ export const Pager = ({ | ||||||
|     </View> |     </View> | ||||||
|   ) |   ) | ||||||
| } | } | ||||||
| 
 |  | ||||||
| const styles = StyleSheet.create({ |  | ||||||
|   tabBar: { |  | ||||||
|     flexDirection: 'row', |  | ||||||
|   }, |  | ||||||
| }) |  | ||||||
|  |  | ||||||
|  | @ -44,7 +44,7 @@ export const PostMeta = observer(function (opts: PostMetaOpts) { | ||||||
|     // two-liner with follow button
 |     // two-liner with follow button
 | ||||||
|     return ( |     return ( | ||||||
|       <View style={styles.metaTwoLine}> |       <View style={styles.metaTwoLine}> | ||||||
|         <View> |         <View style={styles.metaTwoLineLeft}> | ||||||
|           <View style={styles.metaTwoLineTop}> |           <View style={styles.metaTwoLineTop}> | ||||||
|             <DesktopWebTextLink |             <DesktopWebTextLink | ||||||
|               type="lg-bold" |               type="lg-bold" | ||||||
|  | @ -69,6 +69,7 @@ export const PostMeta = observer(function (opts: PostMetaOpts) { | ||||||
|             type="md" |             type="md" | ||||||
|             style={[styles.metaItem, pal.textLight]} |             style={[styles.metaItem, pal.textLight]} | ||||||
|             lineHeight={1.2} |             lineHeight={1.2} | ||||||
|  |             numberOfLines={1} | ||||||
|             text={`@${handle}`} |             text={`@${handle}`} | ||||||
|             href={`/profile/${opts.authorHandle}`} |             href={`/profile/${opts.authorHandle}`} | ||||||
|           /> |           /> | ||||||
|  | @ -134,8 +135,13 @@ const styles = StyleSheet.create({ | ||||||
|     flexDirection: 'row', |     flexDirection: 'row', | ||||||
|     alignItems: 'center', |     alignItems: 'center', | ||||||
|     justifyContent: 'space-between', |     justifyContent: 'space-between', | ||||||
|  |     width: '100%', | ||||||
|     paddingBottom: 2, |     paddingBottom: 2, | ||||||
|   }, |   }, | ||||||
|  |   metaTwoLineLeft: { | ||||||
|  |     flex: 1, | ||||||
|  |     paddingRight: 40, | ||||||
|  |   }, | ||||||
|   metaTwoLineTop: { |   metaTwoLineTop: { | ||||||
|     flexDirection: 'row', |     flexDirection: 'row', | ||||||
|     alignItems: 'baseline', |     alignItems: 'baseline', | ||||||
|  |  | ||||||
|  | @ -37,7 +37,7 @@ export function TabBar({ | ||||||
|   const panX = Animated.add(position, offset) |   const panX = Animated.add(position, offset) | ||||||
| 
 | 
 | ||||||
|   const underlineStyle = { |   const underlineStyle = { | ||||||
|     backgroundColor: pal.colors.text, |     backgroundColor: pal.colors.link, | ||||||
|     left: panX.interpolate({ |     left: panX.interpolate({ | ||||||
|       inputRange: items.map((_item, i) => i), |       inputRange: items.map((_item, i) => i), | ||||||
|       outputRange: itemLayouts.map(l => l.x), |       outputRange: itemLayouts.map(l => l.x), | ||||||
|  | @ -79,11 +79,8 @@ export function TabBar({ | ||||||
|           <TouchableWithoutFeedback key={i} onPress={() => onPressItem(i)}> |           <TouchableWithoutFeedback key={i} onPress={() => onPressItem(i)}> | ||||||
|             <View style={styles.item} ref={itemRefs[i]}> |             <View style={styles.item} ref={itemRefs[i]}> | ||||||
|               <Text |               <Text | ||||||
|                 style={ |                 type="xl-medium" | ||||||
|                   selected |                 style={selected ? pal.text : pal.textLight}> | ||||||
|                     ? [styles.labelSelected, pal.text] |  | ||||||
|                     : [styles.label, pal.textLight] |  | ||||||
|                 }> |  | ||||||
|                 {item} |                 {item} | ||||||
|               </Text> |               </Text> | ||||||
|             </View> |             </View> | ||||||
|  | @ -100,20 +97,14 @@ const styles = StyleSheet.create({ | ||||||
|     paddingHorizontal: 14, |     paddingHorizontal: 14, | ||||||
|   }, |   }, | ||||||
|   item: { |   item: { | ||||||
|     paddingTop: 8, |     paddingTop: 6, | ||||||
|     paddingBottom: 12, |     paddingBottom: 14, | ||||||
|     marginRight: 14, |     marginRight: 24, | ||||||
|     paddingHorizontal: 10, |  | ||||||
|   }, |  | ||||||
|   label: { |  | ||||||
|     fontWeight: '600', |  | ||||||
|   }, |  | ||||||
|   labelSelected: { |  | ||||||
|     fontWeight: '600', |  | ||||||
|   }, |   }, | ||||||
|   underline: { |   underline: { | ||||||
|     position: 'absolute', |     position: 'absolute', | ||||||
|     height: 4, |     height: 3, | ||||||
|     bottom: 0, |     bottom: 0, | ||||||
|  |     borderRadius: 4, | ||||||
|   }, |   }, | ||||||
| }) | }) | ||||||
|  |  | ||||||
|  | @ -1,15 +1,23 @@ | ||||||
| import React from 'react' | import React from 'react' | ||||||
| import {FlatList, StyleSheet, View, useWindowDimensions} from 'react-native' | import { | ||||||
|  |   FlatList, | ||||||
|  |   StyleSheet, | ||||||
|  |   TouchableOpacity, | ||||||
|  |   View, | ||||||
|  |   useWindowDimensions, | ||||||
|  | } from 'react-native' | ||||||
| import {useFocusEffect, useIsFocused} from '@react-navigation/native' | import {useFocusEffect, useIsFocused} from '@react-navigation/native' | ||||||
| import {observer} from 'mobx-react-lite' | import {observer} from 'mobx-react-lite' | ||||||
| import useAppState from 'react-native-appstate-hook' | import useAppState from 'react-native-appstate-hook' | ||||||
| import {NativeStackScreenProps, HomeTabNavigatorParams} from 'lib/routes/types' | import {NativeStackScreenProps, HomeTabNavigatorParams} from 'lib/routes/types' | ||||||
|  | import {FeedModel} from 'state/models/feed-view' | ||||||
| import {withAuthRequired} from 'view/com/auth/withAuthRequired' | import {withAuthRequired} from 'view/com/auth/withAuthRequired' | ||||||
| import {ViewHeader} from '../com/util/ViewHeader' |  | ||||||
| import {Feed} from '../com/posts/Feed' | import {Feed} from '../com/posts/Feed' | ||||||
| import {LoadLatestBtn} from '../com/util/LoadLatestBtn' | import {LoadLatestBtn} from '../com/util/LoadLatestBtn' | ||||||
| import {WelcomeBanner} from '../com/util/WelcomeBanner' | import {WelcomeBanner} from '../com/util/WelcomeBanner' | ||||||
| import {UserAvatar} from 'view/com/util/UserAvatar' | import {UserAvatar} from 'view/com/util/UserAvatar' | ||||||
|  | import {TabBar} from 'view/com/util/TabBar' | ||||||
|  | import {Pager, PageSelectedEvent, TabBarProps} from 'view/com/util/Pager' | ||||||
| import {FAB} from '../com/util/FAB' | import {FAB} from '../com/util/FAB' | ||||||
| import {useStores} from 'state/index' | import {useStores} from 'state/index' | ||||||
| import {usePalette} from 'lib/hooks/usePalette' | import {usePalette} from 'lib/hooks/usePalette' | ||||||
|  | @ -18,67 +26,60 @@ import {useOnMainScroll} from 'lib/hooks/useOnMainScroll' | ||||||
| import {useAnalytics} from 'lib/analytics' | import {useAnalytics} from 'lib/analytics' | ||||||
| import {ComposeIcon2} from 'lib/icons' | import {ComposeIcon2} from 'lib/icons' | ||||||
| 
 | 
 | ||||||
| import {Pager, PageSelectedEvent} from 'view/com/util/Pager' | const TAB_BAR_HEIGHT = 82 | ||||||
| import {Text} from 'view/com/util/text/Text' |  | ||||||
| 
 |  | ||||||
| const HEADER_HEIGHT = 42 |  | ||||||
| 
 | 
 | ||||||
| type Props = NativeStackScreenProps<HomeTabNavigatorParams, 'Home'> | type Props = NativeStackScreenProps<HomeTabNavigatorParams, 'Home'> | ||||||
| export const HomeScreen = withAuthRequired((_opts: Props) => { | export const HomeScreen = withAuthRequired((_opts: Props) => { | ||||||
|   const store = useStores() |   const store = useStores() | ||||||
|  |   const pal = usePalette('default') | ||||||
|  |   const [selectedPage, setSelectedPage] = React.useState(0) | ||||||
|  | 
 | ||||||
|  |   useFocusEffect( | ||||||
|  |     React.useCallback(() => { | ||||||
|  |       store.shell.setIsDrawerSwipeDisabled(selectedPage > 0) | ||||||
|  |       return () => { | ||||||
|  |         store.shell.setIsDrawerSwipeDisabled(false) | ||||||
|  |       } | ||||||
|  |     }, [store, selectedPage]), | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|   const onPageSelected = React.useCallback( |   const onPageSelected = React.useCallback( | ||||||
|     (e: PageSelectedEvent) => { |     (e: PageSelectedEvent) => { | ||||||
|  |       setSelectedPage(e.nativeEvent.position) | ||||||
|       store.shell.setIsDrawerSwipeDisabled(e.nativeEvent.position > 0) |       store.shell.setIsDrawerSwipeDisabled(e.nativeEvent.position > 0) | ||||||
|     }, |     }, | ||||||
|     [store], |     [store], | ||||||
|   ) |   ) | ||||||
| 
 | 
 | ||||||
|   useFocusEffect( |   const onPressAvi = React.useCallback(() => { | ||||||
|     React.useCallback(() => { |     store.shell.openDrawer() | ||||||
|       return () => { |   }, [store]) | ||||||
|         store.shell.setIsDrawerSwipeDisabled(false) | 
 | ||||||
|       } |   const renderTabBar = React.useCallback( | ||||||
|     }, [store]), |     (props: TabBarProps) => { | ||||||
|  |       return ( | ||||||
|  |         <View style={[pal.view, styles.tabBar]}> | ||||||
|  |           <TouchableOpacity style={styles.tabBarAvi} onPress={onPressAvi}> | ||||||
|  |             <UserAvatar avatar={store.me.avatar} size={32} /> | ||||||
|  |           </TouchableOpacity> | ||||||
|  |           <TabBar items={['Popular', 'Following']} {...props} /> | ||||||
|  |         </View> | ||||||
|  |       ) | ||||||
|  |     }, | ||||||
|  |     [store.me.avatar, pal, onPressAvi], | ||||||
|   ) |   ) | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|     <Pager onPageSelected={onPageSelected}> |     <Pager onPageSelected={onPageSelected} renderTabBar={renderTabBar}> | ||||||
|       <View key="1"> |       <AlgoView key="1" /> | ||||||
|         <MyPage>First page</MyPage> |  | ||||||
|       </View> |  | ||||||
|       <View key="2"> |       <View key="2"> | ||||||
|         <MyPage>Second page</MyPage> |         <FollowingView /> | ||||||
|       </View> |  | ||||||
|       <View key="3"> |  | ||||||
|         <MyPage>Third page</MyPage> |  | ||||||
|       </View> |       </View> | ||||||
|     </Pager> |     </Pager> | ||||||
|   ) |   ) | ||||||
| }) | }) | ||||||
| function MyPage({children}) { |  | ||||||
|   return ( |  | ||||||
|     <View |  | ||||||
|       style={{ |  | ||||||
|         flex: 1, |  | ||||||
|         justifyContent: 'center', |  | ||||||
|         alignItems: 'center', |  | ||||||
|         borderWidth: 1, |  | ||||||
|         backgroundColor: 'white', |  | ||||||
|       }}> |  | ||||||
|       <Text>{children}</Text> |  | ||||||
|     </View> |  | ||||||
|   ) |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
| const styles = StyleSheet.create({ | const AlgoView = observer(() => { | ||||||
|   tabBar: { |  | ||||||
|     flexDirection: 'row', |  | ||||||
|   }, |  | ||||||
| }) |  | ||||||
| /* |  | ||||||
| type Props = NativeStackScreenProps<HomeTabNavigatorParams, 'Home'> |  | ||||||
| export const HomeScreen = withAuthRequired( |  | ||||||
|   observer(function Home(_opts: Props) { |  | ||||||
|   const store = useStores() |   const store = useStores() | ||||||
|   const onMainScroll = useOnMainScroll(store) |   const onMainScroll = useOnMainScroll(store) | ||||||
|   const {screen, track} = useAnalytics() |   const {screen, track} = useAnalytics() | ||||||
|  | @ -87,6 +88,108 @@ export const HomeScreen = withAuthRequired( | ||||||
|     onForeground: () => doPoll(true), |     onForeground: () => doPoll(true), | ||||||
|   }) |   }) | ||||||
|   const isFocused = useIsFocused() |   const isFocused = useIsFocused() | ||||||
|  |   const winDim = useWindowDimensions() | ||||||
|  |   const containerStyle = React.useMemo( | ||||||
|  |     () => ({height: winDim.height - TAB_BAR_HEIGHT}), | ||||||
|  |     [winDim], | ||||||
|  |   ) | ||||||
|  |   const algoFeed = React.useMemo(() => { | ||||||
|  |     const feed = new FeedModel(store, 'goodstuff', {}) | ||||||
|  |     feed.setup() | ||||||
|  |     return feed | ||||||
|  |   }, [store]) | ||||||
|  | 
 | ||||||
|  |   const doPoll = React.useCallback( | ||||||
|  |     (knownActive = false) => { | ||||||
|  |       if ((!knownActive && appState !== 'active') || !isFocused) { | ||||||
|  |         return | ||||||
|  |       } | ||||||
|  |       if (algoFeed.isLoading) { | ||||||
|  |         return | ||||||
|  |       } | ||||||
|  |       store.log.debug('HomeScreen: Polling for new posts') | ||||||
|  |       algoFeed.checkForLatest() | ||||||
|  |     }, | ||||||
|  |     [appState, isFocused, store, algoFeed], | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   const scrollToTop = React.useCallback(() => { | ||||||
|  |     scrollElRef.current?.scrollToOffset({offset: 0}) | ||||||
|  |   }, [scrollElRef]) | ||||||
|  | 
 | ||||||
|  |   useFocusEffect( | ||||||
|  |     React.useCallback(() => { | ||||||
|  |       const softResetSub = store.onScreenSoftReset(scrollToTop) | ||||||
|  |       const feedCleanup = algoFeed.registerListeners() | ||||||
|  |       const pollInterval = setInterval(doPoll, 15e3) | ||||||
|  | 
 | ||||||
|  |       screen('Feed') | ||||||
|  |       store.log.debug('HomeScreen: Updating feed') | ||||||
|  |       if (algoFeed.hasContent) { | ||||||
|  |         algoFeed.update() | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       return () => { | ||||||
|  |         clearInterval(pollInterval) | ||||||
|  |         softResetSub.remove() | ||||||
|  |         feedCleanup() | ||||||
|  |       } | ||||||
|  |     }, [store, doPoll, scrollToTop, screen, algoFeed]), | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  |   const onPressCompose = React.useCallback(() => { | ||||||
|  |     track('HomeScreen:PressCompose') | ||||||
|  |     store.shell.openComposer({}) | ||||||
|  |   }, [store, track]) | ||||||
|  | 
 | ||||||
|  |   const onPressTryAgain = React.useCallback(() => { | ||||||
|  |     algoFeed.refresh() | ||||||
|  |   }, [algoFeed]) | ||||||
|  | 
 | ||||||
|  |   const onPressLoadLatest = React.useCallback(() => { | ||||||
|  |     algoFeed.refresh() | ||||||
|  |     scrollToTop() | ||||||
|  |   }, [algoFeed, scrollToTop]) | ||||||
|  | 
 | ||||||
|  |   return ( | ||||||
|  |     <View style={containerStyle}> | ||||||
|  |       {store.shell.isOnboarding && <WelcomeBanner />} | ||||||
|  |       <Feed | ||||||
|  |         testID="homeFeed" | ||||||
|  |         key="default" | ||||||
|  |         feed={algoFeed} | ||||||
|  |         scrollElRef={scrollElRef} | ||||||
|  |         style={s.hContentRegion} | ||||||
|  |         showPostFollowBtn | ||||||
|  |         onPressTryAgain={onPressTryAgain} | ||||||
|  |         onScroll={onMainScroll} | ||||||
|  |       /> | ||||||
|  |       {algoFeed.hasNewLatest && !algoFeed.isRefreshing && ( | ||||||
|  |         <LoadLatestBtn onPress={onPressLoadLatest} /> | ||||||
|  |       )} | ||||||
|  |       <FAB | ||||||
|  |         testID="composeFAB" | ||||||
|  |         onPress={onPressCompose} | ||||||
|  |         icon={<ComposeIcon2 strokeWidth={1.5} size={29} style={s.white} />} | ||||||
|  |       /> | ||||||
|  |     </View> | ||||||
|  |   ) | ||||||
|  | }) | ||||||
|  | 
 | ||||||
|  | const FollowingView = observer(() => { | ||||||
|  |   const store = useStores() | ||||||
|  |   const onMainScroll = useOnMainScroll(store) | ||||||
|  |   const {screen, track} = useAnalytics() | ||||||
|  |   const scrollElRef = React.useRef<FlatList>(null) | ||||||
|  |   const {appState} = useAppState({ | ||||||
|  |     onForeground: () => doPoll(true), | ||||||
|  |   }) | ||||||
|  |   const isFocused = useIsFocused() | ||||||
|  |   const winDim = useWindowDimensions() | ||||||
|  |   const containerStyle = React.useMemo( | ||||||
|  |     () => ({height: winDim.height - TAB_BAR_HEIGHT}), | ||||||
|  |     [winDim], | ||||||
|  |   ) | ||||||
| 
 | 
 | ||||||
|   const doPoll = React.useCallback( |   const doPoll = React.useCallback( | ||||||
|     (knownActive = false) => { |     (knownActive = false) => { | ||||||
|  | @ -103,9 +206,7 @@ export const HomeScreen = withAuthRequired( | ||||||
|   ) |   ) | ||||||
| 
 | 
 | ||||||
|   const scrollToTop = React.useCallback(() => { |   const scrollToTop = React.useCallback(() => { | ||||||
|       // NOTE: the feed is offset by the height of the collapsing header,
 |     scrollElRef.current?.scrollToOffset({offset: 0}) | ||||||
|       //       so we scroll to the negative of that height -prf
 |  | ||||||
|       scrollElRef.current?.scrollToOffset({offset: -HEADER_HEIGHT}) |  | ||||||
|   }, [scrollElRef]) |   }, [scrollElRef]) | ||||||
| 
 | 
 | ||||||
|   useFocusEffect( |   useFocusEffect( | ||||||
|  | @ -143,7 +244,7 @@ export const HomeScreen = withAuthRequired( | ||||||
|   }, [store, scrollToTop]) |   }, [store, scrollToTop]) | ||||||
| 
 | 
 | ||||||
|   return ( |   return ( | ||||||
|       <View style={s.hContentRegion}> |     <View style={containerStyle}> | ||||||
|       {store.shell.isOnboarding && <WelcomeBanner />} |       {store.shell.isOnboarding && <WelcomeBanner />} | ||||||
|       <Feed |       <Feed | ||||||
|         testID="homeFeed" |         testID="homeFeed" | ||||||
|  | @ -154,11 +255,7 @@ export const HomeScreen = withAuthRequired( | ||||||
|         showPostFollowBtn |         showPostFollowBtn | ||||||
|         onPressTryAgain={onPressTryAgain} |         onPressTryAgain={onPressTryAgain} | ||||||
|         onScroll={onMainScroll} |         onScroll={onMainScroll} | ||||||
|           headerOffset={store.shell.isOnboarding ? 0 : HEADER_HEIGHT} |  | ||||||
|       /> |       /> | ||||||
|         {!store.shell.isOnboarding && ( |  | ||||||
|           <ViewHeader title="Bluesky" canGoBack={false} hideOnScroll /> |  | ||||||
|         )} |  | ||||||
|       {store.me.mainFeed.hasNewLatest && !store.me.mainFeed.isRefreshing && ( |       {store.me.mainFeed.hasNewLatest && !store.me.mainFeed.isRefreshing && ( | ||||||
|         <LoadLatestBtn onPress={onPressLoadLatest} /> |         <LoadLatestBtn onPress={onPressLoadLatest} /> | ||||||
|       )} |       )} | ||||||
|  | @ -169,6 +266,15 @@ export const HomeScreen = withAuthRequired( | ||||||
|       /> |       /> | ||||||
|     </View> |     </View> | ||||||
|   ) |   ) | ||||||
|   }), | }) | ||||||
| ) | 
 | ||||||
| */ | const styles = StyleSheet.create({ | ||||||
|  |   tabBar: { | ||||||
|  |     flexDirection: 'row', | ||||||
|  |     alignItems: 'center', | ||||||
|  |     paddingHorizontal: 18, | ||||||
|  |   }, | ||||||
|  |   tabBarAvi: { | ||||||
|  |     marginRight: 16, | ||||||
|  |   }, | ||||||
|  | }) | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue