Top/Latest for hashtags (#3625)

* Split HashtagScreen into two components

* Hashtag tabs

* Visual fixes
zio/stable
dan 2024-04-19 23:37:11 +01:00 committed by GitHub
parent c0ca891501
commit d3c0b48da3
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 184 additions and 79 deletions

View File

@ -1,11 +1,12 @@
import React from 'react' import React from 'react'
import {ListRenderItemInfo, Pressable} from 'react-native' import {ListRenderItemInfo, Pressable, StyleSheet, View} from 'react-native'
import {PostView} from '@atproto/api/dist/client/types/app/bsky/feed/defs' import {PostView} from '@atproto/api/dist/client/types/app/bsky/feed/defs'
import {msg} from '@lingui/macro' import {msg} from '@lingui/macro'
import {useLingui} from '@lingui/react' import {useLingui} from '@lingui/react'
import {useFocusEffect} from '@react-navigation/native' import {useFocusEffect} from '@react-navigation/native'
import {NativeStackScreenProps} from '@react-navigation/native-stack' import {NativeStackScreenProps} from '@react-navigation/native-stack'
import {usePalette} from '#/lib/hooks/usePalette'
import {HITSLOP_10} from 'lib/constants' import {HITSLOP_10} from 'lib/constants'
import {useInitialNumToRender} from 'lib/hooks/useInitialNumToRender' import {useInitialNumToRender} from 'lib/hooks/useInitialNumToRender'
import {CommonNavigatorParams} from 'lib/routes/types' import {CommonNavigatorParams} from 'lib/routes/types'
@ -13,18 +14,17 @@ import {shareUrl} from 'lib/sharing'
import {cleanError} from 'lib/strings/errors' import {cleanError} from 'lib/strings/errors'
import {sanitizeHandle} from 'lib/strings/handles' import {sanitizeHandle} from 'lib/strings/handles'
import {enforceLen} from 'lib/strings/helpers' import {enforceLen} from 'lib/strings/helpers'
import {isNative} from 'platform/detection' import {isNative, isWeb} from 'platform/detection'
import {useSearchPostsQuery} from 'state/queries/search-posts' import {useSearchPostsQuery} from 'state/queries/search-posts'
import {useSetMinimalShellMode} from 'state/shell' import {useSetDrawerSwipeDisabled, useSetMinimalShellMode} from 'state/shell'
import {Pager} from '#/view/com/pager/Pager'
import {TabBar} from '#/view/com/pager/TabBar'
import {CenteredView} from '#/view/com/util/Views'
import {Post} from 'view/com/post/Post' import {Post} from 'view/com/post/Post'
import {List} from 'view/com/util/List' import {List} from 'view/com/util/List'
import {ViewHeader} from 'view/com/util/ViewHeader' import {ViewHeader} from 'view/com/util/ViewHeader'
import {ArrowOutOfBox_Stroke2_Corner0_Rounded} from '#/components/icons/ArrowOutOfBox' import {ArrowOutOfBox_Stroke2_Corner0_Rounded} from '#/components/icons/ArrowOutOfBox'
import { import {ListFooter, ListMaybePlaceholder} from '#/components/Lists'
ListFooter,
ListHeaderDesktop,
ListMaybePlaceholder,
} from '#/components/Lists'
const renderItem = ({item}: ListRenderItemInfo<PostView>) => { const renderItem = ({item}: ListRenderItemInfo<PostView>) => {
return <Post post={item} /> return <Post post={item} />
@ -38,20 +38,13 @@ export default function HashtagScreen({
route, route,
}: NativeStackScreenProps<CommonNavigatorParams, 'Hashtag'>) { }: NativeStackScreenProps<CommonNavigatorParams, 'Hashtag'>) {
const {tag, author} = route.params const {tag, author} = route.params
const setMinimalShellMode = useSetMinimalShellMode()
const {_} = useLingui() const {_} = useLingui()
const initialNumToRender = useInitialNumToRender() const pal = usePalette('default')
const [isPTR, setIsPTR] = React.useState(false)
const fullTag = React.useMemo(() => { const fullTag = React.useMemo(() => {
return `#${decodeURIComponent(tag)}` return `#${decodeURIComponent(tag)}`
}, [tag]) }, [tag])
const queryParam = React.useMemo(() => {
if (!author) return fullTag
return `${fullTag} from:${sanitizeHandle(author)}`
}, [fullTag, author])
const headerTitle = React.useMemo(() => { const headerTitle = React.useMemo(() => {
return enforceLen(fullTag.toLowerCase(), 24, true, 'middle') return enforceLen(fullTag.toLowerCase(), 24, true, 'middle')
}, [fullTag]) }, [fullTag])
@ -61,27 +54,6 @@ export default function HashtagScreen({
return sanitizeHandle(author) return sanitizeHandle(author)
}, [author]) }, [author])
const {
data,
isFetchingNextPage,
isLoading,
isError,
error,
refetch,
fetchNextPage,
hasNextPage,
} = useSearchPostsQuery({query: queryParam})
const posts = React.useMemo(() => {
return data?.pages.flatMap(page => page.posts) || []
}, [data])
useFocusEffect(
React.useCallback(() => {
setMinimalShellMode(false)
}, [setMinimalShellMode]),
)
const onShare = React.useCallback(() => { const onShare = React.useCallback(() => {
const url = new URL('https://bsky.app') const url = new URL('https://bsky.app')
url.pathname = `/hashtag/${decodeURIComponent(tag)}` url.pathname = `/hashtag/${decodeURIComponent(tag)}`
@ -91,20 +63,57 @@ export default function HashtagScreen({
shareUrl(url.toString()) shareUrl(url.toString())
}, [tag, author]) }, [tag, author])
const onRefresh = React.useCallback(async () => { const [activeTab, setActiveTab] = React.useState(0)
setIsPTR(true) const setMinimalShellMode = useSetMinimalShellMode()
await refetch() const setDrawerSwipeDisabled = useSetDrawerSwipeDisabled()
setIsPTR(false)
}, [refetch])
const onEndReached = React.useCallback(() => { useFocusEffect(
if (isFetchingNextPage || !hasNextPage || error) return React.useCallback(() => {
fetchNextPage() setMinimalShellMode(false)
}, [isFetchingNextPage, hasNextPage, error, fetchNextPage]) }, [setMinimalShellMode]),
)
const onPageSelected = React.useCallback(
(index: number) => {
setMinimalShellMode(false)
setDrawerSwipeDisabled(index > 0)
setActiveTab(index)
},
[setDrawerSwipeDisabled, setMinimalShellMode],
)
const sections = React.useMemo(() => {
return [
{
title: _(msg`Top`),
component: (
<HashtagScreenTab
fullTag={fullTag}
author={author}
sort="top"
active={activeTab === 0}
/>
),
},
{
title: _(msg`Latest`),
component: (
<HashtagScreenTab
fullTag={fullTag}
author={author}
sort="latest"
active={activeTab === 1}
/>
),
},
]
}, [_, fullTag, author, activeTab])
return ( return (
<> <>
<CenteredView sideBorders style={[pal.border, pal.view]}>
<ViewHeader <ViewHeader
showOnDesktop
title={headerTitle} title={headerTitle}
subtitle={author ? _(msg`From @${sanitizedAuthor}`) : undefined} subtitle={author ? _(msg`From @${sanitizedAuthor}`) : undefined}
canGoBack canGoBack
@ -124,9 +133,77 @@ export default function HashtagScreen({
: undefined : undefined
} }
/> />
</CenteredView>
<Pager
onPageSelected={onPageSelected}
renderTabBar={props => (
<CenteredView
sideBorders
style={[pal.border, pal.view, styles.tabBarContainer]}>
<TabBar items={sections.map(section => section.title)} {...props} />
</CenteredView>
)}
initialPage={0}>
{sections.map((section, i) => (
<View key={i}>{section.component}</View>
))}
</Pager>
</>
)
}
function HashtagScreenTab({
fullTag,
author,
sort,
active,
}: {
fullTag: string
author: string | undefined
sort: 'top' | 'latest'
active: boolean
}) {
const {_} = useLingui()
const initialNumToRender = useInitialNumToRender()
const [isPTR, setIsPTR] = React.useState(false)
const queryParam = React.useMemo(() => {
if (!author) return fullTag
return `${fullTag} from:${sanitizeHandle(author)}`
}, [fullTag, author])
const {
data,
isFetched,
isFetchingNextPage,
isLoading,
isError,
error,
refetch,
fetchNextPage,
hasNextPage,
} = useSearchPostsQuery({query: queryParam, sort, enabled: active})
const posts = React.useMemo(() => {
return data?.pages.flatMap(page => page.posts) || []
}, [data])
const onRefresh = React.useCallback(async () => {
setIsPTR(true)
await refetch()
setIsPTR(false)
}, [refetch])
const onEndReached = React.useCallback(() => {
if (isFetchingNextPage || !hasNextPage || error) return
fetchNextPage()
}, [isFetchingNextPage, hasNextPage, error, fetchNextPage])
return (
<>
{posts.length < 1 ? ( {posts.length < 1 ? (
<ListMaybePlaceholder <ListMaybePlaceholder
isLoading={isLoading} isLoading={isLoading || !isFetched}
isError={isError} isError={isError}
onRetry={refetch} onRetry={refetch}
emptyType="results" emptyType="results"
@ -143,12 +220,6 @@ export default function HashtagScreen({
onEndReachedThreshold={4} onEndReachedThreshold={4}
// @ts-ignore web only -prf // @ts-ignore web only -prf
desktopFixedHeight desktopFixedHeight
ListHeaderComponent={
<ListHeaderDesktop
title={headerTitle}
subtitle={author ? _(msg`From @${sanitizedAuthor}`) : undefined}
/>
}
ListFooterComponent={ ListFooterComponent={
<ListFooter <ListFooter
isFetchingNextPage={isFetchingNextPage} isFetchingNextPage={isFetchingNextPage}
@ -163,3 +234,12 @@ export default function HashtagScreen({
</> </>
) )
} }
const styles = StyleSheet.create({
tabBarContainer: {
// @ts-ignore web only
position: isWeb ? 'sticky' : '',
top: 0,
zIndex: 1,
},
})

View File

@ -1,19 +1,20 @@
import React from 'react' import React from 'react'
import {StyleSheet, TouchableOpacity, View} from 'react-native' import {StyleSheet, TouchableOpacity, View} from 'react-native'
import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {useNavigation} from '@react-navigation/native'
import {CenteredView} from './Views'
import {Text} from './text/Text'
import {usePalette} from 'lib/hooks/usePalette'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {useAnalytics} from 'lib/analytics/analytics'
import {NavigationProp} from 'lib/routes/types'
import {useMinimalShellMode} from 'lib/hooks/useMinimalShellMode'
import Animated from 'react-native-reanimated' import Animated from 'react-native-reanimated'
import {useSetDrawerOpen} from '#/state/shell' import {FontAwesomeIcon} from '@fortawesome/react-native-fontawesome'
import {msg} from '@lingui/macro' import {msg} from '@lingui/macro'
import {useLingui} from '@lingui/react' import {useLingui} from '@lingui/react'
import {useNavigation} from '@react-navigation/native'
import {useSetDrawerOpen} from '#/state/shell'
import {useAnalytics} from 'lib/analytics/analytics'
import {useMinimalShellMode} from 'lib/hooks/useMinimalShellMode'
import {usePalette} from 'lib/hooks/usePalette'
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
import {NavigationProp} from 'lib/routes/types'
import {useTheme} from '#/alf' import {useTheme} from '#/alf'
import {Text} from './text/Text'
import {CenteredView} from './Views'
const BACK_HITSLOP = {left: 20, top: 20, right: 50, bottom: 20} const BACK_HITSLOP = {left: 20, top: 20, right: 50, bottom: 20}
@ -62,6 +63,7 @@ export function ViewHeader({
return ( return (
<DesktopWebHeader <DesktopWebHeader
title={title} title={title}
subtitle={subtitle}
renderButton={renderButton} renderButton={renderButton}
showBorder={showBorder} showBorder={showBorder}
/> />
@ -136,14 +138,17 @@ export function ViewHeader({
function DesktopWebHeader({ function DesktopWebHeader({
title, title,
subtitle,
renderButton, renderButton,
showBorder = true, showBorder = true,
}: { }: {
title: string title: string
subtitle?: string
renderButton?: () => JSX.Element renderButton?: () => JSX.Element
showBorder?: boolean showBorder?: boolean
}) { }) {
const pal = usePalette('default') const pal = usePalette('default')
const t = useTheme()
return ( return (
<CenteredView <CenteredView
style={[ style={[
@ -153,13 +158,30 @@ function DesktopWebHeader({
{ {
borderBottomWidth: showBorder ? 1 : 0, borderBottomWidth: showBorder ? 1 : 0,
}, },
{display: 'flex', flexDirection: 'column'},
]}> ]}>
<View>
<View style={styles.titleContainer} pointerEvents="none"> <View style={styles.titleContainer} pointerEvents="none">
<Text type="title-lg" style={[pal.text, styles.title]}> <Text type="title-lg" style={[pal.text, styles.title]}>
{title} {title}
</Text> </Text>
</View> </View>
{renderButton?.()} {renderButton?.()}
</View>
{subtitle ? (
<View>
<View style={[styles.titleContainer]} pointerEvents="none">
<Text
style={[
pal.text,
styles.subtitleDesktop,
t.atoms.text_contrast_medium,
]}>
{subtitle}
</Text>
</View>
</View>
) : null}
</CenteredView> </CenteredView>
) )
} }
@ -236,6 +258,9 @@ const styles = StyleSheet.create({
subtitle: { subtitle: {
fontSize: 13, fontSize: 13,
}, },
subtitleDesktop: {
fontSize: 15,
},
backBtn: { backBtn: {
width: 30, width: 30,
height: 30, height: 30,