import {useMemo} from 'react' import {Pressable, View} from 'react-native' import {type AppBskyUnspeccedDefs, moderateProfile} from '@atproto/api' import {msg, plural, Trans} from '@lingui/macro' import {useLingui} from '@lingui/react' import {logger} from '#/logger' import {useModerationOpts} from '#/state/preferences/moderation-opts' import {useTrendingSettings} from '#/state/preferences/trending' import {useGetTrendsQuery} from '#/state/queries/trending/useGetTrendsQuery' import {useTrendingConfig} from '#/state/trending-config' import {LoadingPlaceholder} from '#/view/com/util/LoadingPlaceholder' import {formatCount} from '#/view/com/util/numeric/format' import {atoms as a, useGutters, useTheme, type ViewStyleProp, web} from '#/alf' import {AvatarStack} from '#/components/AvatarStack' import {type Props as SVGIconProps} from '#/components/icons/common' import {Flame_Stroke2_Corner1_Rounded as FlameIcon} from '#/components/icons/Flame' import {Trending3_Stroke2_Corner1_Rounded as TrendingIcon} from '#/components/icons/Trending' import {Link} from '#/components/Link' import {SubtleHover} from '#/components/SubtleHover' import {Text} from '#/components/Typography' const TOPIC_COUNT = 5 export function ExploreTrendingTopics() { const {enabled} = useTrendingConfig() const {trendingDisabled} = useTrendingSettings() return enabled && !trendingDisabled ? : null } function Inner() { const {data: trending, error, isLoading, isRefetching} = useGetTrendsQuery() const noTopics = !isLoading && !error && !trending?.trends?.length return isLoading || isRefetching ? ( Array.from({length: TOPIC_COUNT}).map((__, i) => ( )) ) : error || !trending?.trends || noTopics ? null : ( <> {trending.trends.map((trend, index) => ( { logger.metric('trendingTopic:click', {context: 'explore'}) }} /> ))} ) } export function TrendRow({ trend, rank, children, onPress, }: ViewStyleProp & { trend: AppBskyUnspeccedDefs.TrendView rank: number children?: React.ReactNode onPress?: () => void }) { const t = useTheme() const {_, i18n} = useLingui() const gutters = useGutters([0, 'base']) const category = useCategoryDisplayName(trend?.category || 'other') const age = Math.floor( (Date.now() - new Date(trend.startedAt || Date.now()).getTime()) / (1000 * 60 * 60), ) const badgeType = trend.status === 'hot' ? 'hot' : age < 2 ? 'new' : age const postCount = trend.postCount ? _( plural(trend.postCount, { other: `${formatCount(i18n, trend.postCount)} posts`, }), ) : null const actors = useModerateTrendingActors(trend.actors) return ( {({hovered, pressed}) => ( <> {rank}. {trend.displayName} {actors.length > 0 && ( )} {postCount} {postCount && category && <> · } {category} {children} )} ) } type TrendingIndicatorType = 'hot' | 'new' | number function TrendingIndicator({type}: {type: TrendingIndicatorType | 'skeleton'}) { const t = useTheme() const {_} = useLingui() const pillStyles = [ a.flex_row, a.align_center, a.gap_xs, a.rounded_full, a.px_sm, { height: 28, }, ] let Icon: React.ComponentType | null = null let text: string | null = null let color: string | null = null let backgroundColor: string | null = null switch (type) { case 'skeleton': { return ( ) } case 'hot': { Icon = FlameIcon color = t.scheme === 'light' ? t.palette.negative_500 : t.palette.negative_950 backgroundColor = t.scheme === 'light' ? t.palette.negative_50 : t.palette.negative_200 text = _(msg`Hot`) break } case 'new': { Icon = TrendingIcon text = _(msg`New`) color = t.palette.positive_700 backgroundColor = t.palette.positive_50 break } default: { text = _( msg({ message: `${type}h ago`, comment: 'trending topic time spent trending. should be as short as possible to fit in a pill', }), ) color = t.atoms.text_contrast_medium.color backgroundColor = t.atoms.bg_contrast_25.backgroundColor break } } return ( {Icon && } {text} ) } function useCategoryDisplayName( category: AppBskyUnspeccedDefs.TrendView['category'], ) { const {_} = useLingui() switch (category) { case 'sports': return _(msg`Sports`) case 'politics': return _(msg`Politics`) case 'video-games': return _(msg`Video Games`) case 'pop-culture': return _(msg`Entertainment`) case 'news': return _(msg`News`) case 'other': default: return null } } export function TrendingTopicRowSkeleton({}: {withPosts: boolean}) { const t = useTheme() const gutters = useGutters([0, 'base']) return ( ) } function useModerateTrendingActors( actors: AppBskyUnspeccedDefs.TrendView['actors'], ) { const moderationOpts = useModerationOpts() return useMemo(() => { if (!moderationOpts) return [] return actors .filter(actor => { const decision = moderateProfile(actor, moderationOpts) return !decision.ui('avatar').filter && !decision.ui('avatar').blur }) .slice(0, 3) }, [actors, moderationOpts]) }