Top/Latest for hashtags (#3625)
* Split HashtagScreen into two components * Hashtag tabs * Visual fixeszio/stable
parent
c0ca891501
commit
d3c0b48da3
|
@ -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,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
|
@ -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,
|
||||||
|
|
Loading…
Reference in New Issue