* implement AppBskyEmbedVideo lexicon in player * add alt to native player * add prerelease package * update prerelease * add video embed view manually from record * fix type error on example video * black bg + use aspect ratio on web * add video to feeds * fix video overflowing aspect ratio * remove prerelease package --------- Co-authored-by: Samuel Newman <10959775+mozzius@users.noreply.github.com>
713 lines
22 KiB
TypeScript
713 lines
22 KiB
TypeScript
import React, {useRef} from 'react'
|
|
import {StyleSheet, useWindowDimensions, View} from 'react-native'
|
|
import {runOnJS} from 'react-native-reanimated'
|
|
import Animated from 'react-native-reanimated'
|
|
import {useSafeAreaInsets} from 'react-native-safe-area-context'
|
|
import {AppBskyFeedDefs, AppBskyFeedThreadgate} from '@atproto/api'
|
|
import {msg, Trans} from '@lingui/macro'
|
|
import {useLingui} from '@lingui/react'
|
|
|
|
import {moderatePost_wrapped as moderatePost} from '#/lib/moderatePost_wrapped'
|
|
import {clamp} from '#/lib/numbers'
|
|
import {ScrollProvider} from '#/lib/ScrollContext'
|
|
import {isAndroid, isNative, isWeb} from '#/platform/detection'
|
|
import {useModerationOpts} from '#/state/preferences/moderation-opts'
|
|
import {
|
|
fillThreadModerationCache,
|
|
sortThread,
|
|
ThreadBlocked,
|
|
ThreadModerationCache,
|
|
ThreadNode,
|
|
ThreadNotFound,
|
|
ThreadPost,
|
|
usePostThreadQuery,
|
|
} from '#/state/queries/post-thread'
|
|
import {usePreferencesQuery} from '#/state/queries/preferences'
|
|
import {useSession} from '#/state/session'
|
|
import {useComposerControls} from '#/state/shell'
|
|
import {useMergedThreadgateHiddenReplies} from '#/state/threadgate-hidden-replies'
|
|
import {useInitialNumToRender} from 'lib/hooks/useInitialNumToRender'
|
|
import {useMinimalShellFabTransform} from 'lib/hooks/useMinimalShellTransform'
|
|
import {useSetTitle} from 'lib/hooks/useSetTitle'
|
|
import {useWebMediaQueries} from 'lib/hooks/useWebMediaQueries'
|
|
import {sanitizeDisplayName} from 'lib/strings/display-names'
|
|
import {cleanError} from 'lib/strings/errors'
|
|
import {CenteredView} from 'view/com/util/Views'
|
|
import {atoms as a, useTheme} from '#/alf'
|
|
import {ListFooter, ListMaybePlaceholder} from '#/components/Lists'
|
|
import {Text} from '#/components/Typography'
|
|
import {List, ListMethods} from '../util/List'
|
|
import {ViewHeader} from '../util/ViewHeader'
|
|
import {PostThreadComposePrompt} from './PostThreadComposePrompt'
|
|
import {PostThreadItem} from './PostThreadItem'
|
|
import {PostThreadLoadMore} from './PostThreadLoadMore'
|
|
import {PostThreadShowHiddenReplies} from './PostThreadShowHiddenReplies'
|
|
|
|
// FlatList maintainVisibleContentPosition breaks if too many items
|
|
// are prepended. This seems to be an optimal number based on *shrug*.
|
|
const PARENTS_CHUNK_SIZE = 15
|
|
|
|
const MAINTAIN_VISIBLE_CONTENT_POSITION = {
|
|
// We don't insert any elements before the root row while loading.
|
|
// So the row we want to use as the scroll anchor is the first row.
|
|
minIndexForVisible: 0,
|
|
}
|
|
|
|
const REPLY_PROMPT = {_reactKey: '__reply__'}
|
|
const LOAD_MORE = {_reactKey: '__load_more__'}
|
|
const SHOW_HIDDEN_REPLIES = {_reactKey: '__show_hidden_replies__'}
|
|
const SHOW_MUTED_REPLIES = {_reactKey: '__show_muted_replies__'}
|
|
|
|
enum HiddenRepliesState {
|
|
Hide,
|
|
Show,
|
|
ShowAndOverridePostHider,
|
|
}
|
|
|
|
type YieldedItem =
|
|
| ThreadPost
|
|
| ThreadBlocked
|
|
| ThreadNotFound
|
|
| typeof SHOW_HIDDEN_REPLIES
|
|
| typeof SHOW_MUTED_REPLIES
|
|
type RowItem =
|
|
| YieldedItem
|
|
// TODO: TS doesn't actually enforce it's one of these, it only enforces matching shape.
|
|
| typeof REPLY_PROMPT
|
|
| typeof LOAD_MORE
|
|
|
|
type ThreadSkeletonParts = {
|
|
parents: YieldedItem[]
|
|
highlightedPost: ThreadNode
|
|
replies: YieldedItem[]
|
|
}
|
|
|
|
const keyExtractor = (item: RowItem) => {
|
|
return item._reactKey
|
|
}
|
|
|
|
export function PostThread({uri}: {uri: string | undefined}) {
|
|
const {hasSession, currentAccount} = useSession()
|
|
const {_} = useLingui()
|
|
const t = useTheme()
|
|
const {isMobile, isTabletOrMobile} = useWebMediaQueries()
|
|
const initialNumToRender = useInitialNumToRender()
|
|
const {height: windowHeight} = useWindowDimensions()
|
|
const [hiddenRepliesState, setHiddenRepliesState] = React.useState(
|
|
HiddenRepliesState.Hide,
|
|
)
|
|
|
|
const {data: preferences} = usePreferencesQuery()
|
|
const {
|
|
isFetching,
|
|
isError: isThreadError,
|
|
error: threadError,
|
|
refetch,
|
|
data: {thread, threadgate} = {},
|
|
} = usePostThreadQuery(uri)
|
|
|
|
const treeView = React.useMemo(
|
|
() =>
|
|
!!preferences?.threadViewPrefs?.lab_treeViewEnabled &&
|
|
hasBranchingReplies(thread),
|
|
[preferences?.threadViewPrefs, thread],
|
|
)
|
|
const rootPost = thread?.type === 'post' ? thread.post : undefined
|
|
const rootPostRecord = thread?.type === 'post' ? thread.record : undefined
|
|
const threadgateRecord = threadgate?.record as
|
|
| AppBskyFeedThreadgate.Record
|
|
| undefined
|
|
const threadgateHiddenReplies = useMergedThreadgateHiddenReplies({
|
|
threadgateRecord,
|
|
})
|
|
|
|
const moderationOpts = useModerationOpts()
|
|
const isNoPwi = React.useMemo(() => {
|
|
const mod =
|
|
rootPost && moderationOpts
|
|
? moderatePost(rootPost, moderationOpts)
|
|
: undefined
|
|
return !!mod
|
|
?.ui('contentList')
|
|
.blurs.find(
|
|
cause =>
|
|
cause.type === 'label' &&
|
|
cause.labelDef.identifier === '!no-unauthenticated',
|
|
)
|
|
}, [rootPost, moderationOpts])
|
|
|
|
// Values used for proper rendering of parents
|
|
const ref = useRef<ListMethods>(null)
|
|
const highlightedPostRef = useRef<View | null>(null)
|
|
const [maxParents, setMaxParents] = React.useState(
|
|
isWeb ? Infinity : PARENTS_CHUNK_SIZE,
|
|
)
|
|
const [maxReplies, setMaxReplies] = React.useState(50)
|
|
|
|
useSetTitle(
|
|
rootPost && !isNoPwi
|
|
? `${sanitizeDisplayName(
|
|
rootPost.author.displayName || `@${rootPost.author.handle}`,
|
|
)}: "${rootPostRecord!.text}"`
|
|
: '',
|
|
)
|
|
|
|
// On native, this is going to start out `true`. We'll toggle it to `false` after the initial render if flushed.
|
|
// This ensures that the first render contains no parents--even if they are already available in the cache.
|
|
// We need to delay showing them so that we can use maintainVisibleContentPosition to keep the main post on screen.
|
|
// On the web this is not necessary because we can synchronously adjust the scroll in onContentSizeChange instead.
|
|
const [deferParents, setDeferParents] = React.useState(isNative)
|
|
|
|
const currentDid = currentAccount?.did
|
|
const threadModerationCache = React.useMemo(() => {
|
|
const cache: ThreadModerationCache = new WeakMap()
|
|
if (thread && moderationOpts) {
|
|
fillThreadModerationCache(cache, thread, moderationOpts)
|
|
}
|
|
return cache
|
|
}, [thread, moderationOpts])
|
|
|
|
const [justPostedUris, setJustPostedUris] = React.useState(
|
|
() => new Set<string>(),
|
|
)
|
|
|
|
const skeleton = React.useMemo(() => {
|
|
const threadViewPrefs = preferences?.threadViewPrefs
|
|
if (!threadViewPrefs || !thread) return null
|
|
|
|
return createThreadSkeleton(
|
|
sortThread(
|
|
thread,
|
|
threadViewPrefs,
|
|
threadModerationCache,
|
|
currentDid,
|
|
justPostedUris,
|
|
threadgateHiddenReplies,
|
|
),
|
|
currentDid,
|
|
treeView,
|
|
threadModerationCache,
|
|
hiddenRepliesState !== HiddenRepliesState.Hide,
|
|
threadgateHiddenReplies,
|
|
)
|
|
}, [
|
|
thread,
|
|
preferences?.threadViewPrefs,
|
|
currentDid,
|
|
treeView,
|
|
threadModerationCache,
|
|
hiddenRepliesState,
|
|
justPostedUris,
|
|
threadgateHiddenReplies,
|
|
])
|
|
|
|
const error = React.useMemo(() => {
|
|
if (AppBskyFeedDefs.isNotFoundPost(thread)) {
|
|
return {
|
|
title: _(msg`Post not found`),
|
|
message: _(msg`The post may have been deleted.`),
|
|
}
|
|
} else if (skeleton?.highlightedPost.type === 'blocked') {
|
|
return {
|
|
title: _(msg`Post hidden`),
|
|
message: _(
|
|
msg`You have blocked the author or you have been blocked by the author.`,
|
|
),
|
|
}
|
|
} else if (threadError?.message.startsWith('Post not found')) {
|
|
return {
|
|
title: _(msg`Post not found`),
|
|
message: _(msg`The post may have been deleted.`),
|
|
}
|
|
} else if (isThreadError) {
|
|
return {
|
|
message: threadError ? cleanError(threadError) : undefined,
|
|
}
|
|
}
|
|
|
|
return null
|
|
}, [thread, skeleton?.highlightedPost, isThreadError, _, threadError])
|
|
|
|
// construct content
|
|
const posts = React.useMemo(() => {
|
|
if (!skeleton) return []
|
|
|
|
const {parents, highlightedPost, replies} = skeleton
|
|
let arr: RowItem[] = []
|
|
if (highlightedPost.type === 'post') {
|
|
// We want to wait for parents to load before rendering.
|
|
// If you add something here, you'll need to update both
|
|
// maintainVisibleContentPosition and onContentSizeChange
|
|
// to "hold onto" the correct row instead of the first one.
|
|
|
|
if (!highlightedPost.ctx.isParentLoading && !deferParents) {
|
|
// When progressively revealing parents, rendering a placeholder
|
|
// here will cause scrolling jumps. Don't add it unless you test it.
|
|
// QT'ing this thread is a great way to test all the scrolling hacks:
|
|
// https://bsky.app/profile/www.mozzius.dev/post/3kjqhblh6qk2o
|
|
|
|
// Everything is loaded
|
|
let startIndex = Math.max(0, parents.length - maxParents)
|
|
for (let i = startIndex; i < parents.length; i++) {
|
|
arr.push(parents[i])
|
|
}
|
|
}
|
|
arr.push(highlightedPost)
|
|
if (!highlightedPost.post.viewer?.replyDisabled) {
|
|
arr.push(REPLY_PROMPT)
|
|
}
|
|
for (let i = 0; i < replies.length; i++) {
|
|
arr.push(replies[i])
|
|
if (i === maxReplies) {
|
|
break
|
|
}
|
|
}
|
|
}
|
|
return arr
|
|
}, [skeleton, deferParents, maxParents, maxReplies])
|
|
|
|
// This is only used on the web to keep the post in view when its parents load.
|
|
// On native, we rely on `maintainVisibleContentPosition` instead.
|
|
const didAdjustScrollWeb = useRef<boolean>(false)
|
|
const onContentSizeChangeWeb = React.useCallback(() => {
|
|
// only run once
|
|
if (didAdjustScrollWeb.current) {
|
|
return
|
|
}
|
|
// wait for loading to finish
|
|
if (thread?.type === 'post' && !!thread.parent) {
|
|
function onMeasure(pageY: number) {
|
|
ref.current?.scrollToOffset({
|
|
animated: false,
|
|
offset: pageY,
|
|
})
|
|
}
|
|
// Measure synchronously to avoid a layout jump.
|
|
const domNode = highlightedPostRef.current
|
|
if (domNode) {
|
|
const pageY = (domNode as any as Element).getBoundingClientRect().top
|
|
onMeasure(pageY)
|
|
}
|
|
didAdjustScrollWeb.current = true
|
|
}
|
|
}, [thread])
|
|
|
|
// On native, we reveal parents in chunks. Although they're all already
|
|
// loaded and FlatList already has its own virtualization, unfortunately FlatList
|
|
// has a bug that causes the content to jump around if too many items are getting
|
|
// prepended at once. It also jumps around if items get prepended during scroll.
|
|
// To work around this, we prepend rows after scroll bumps against the top and rests.
|
|
const needsBumpMaxParents = React.useRef(false)
|
|
const onStartReached = React.useCallback(() => {
|
|
if (skeleton?.parents && maxParents < skeleton.parents.length) {
|
|
needsBumpMaxParents.current = true
|
|
}
|
|
}, [maxParents, skeleton?.parents])
|
|
const bumpMaxParentsIfNeeded = React.useCallback(() => {
|
|
if (!isNative) {
|
|
return
|
|
}
|
|
if (needsBumpMaxParents.current) {
|
|
needsBumpMaxParents.current = false
|
|
setMaxParents(n => n + PARENTS_CHUNK_SIZE)
|
|
}
|
|
}, [])
|
|
const onScrollToTop = bumpMaxParentsIfNeeded
|
|
const onMomentumEnd = React.useCallback(() => {
|
|
'worklet'
|
|
runOnJS(bumpMaxParentsIfNeeded)()
|
|
}, [bumpMaxParentsIfNeeded])
|
|
|
|
const onEndReached = React.useCallback(() => {
|
|
if (isFetching || posts.length < maxReplies) return
|
|
setMaxReplies(prev => prev + 50)
|
|
}, [isFetching, maxReplies, posts.length])
|
|
|
|
const onPostReply = React.useCallback(
|
|
(postUri: string | undefined) => {
|
|
refetch()
|
|
if (postUri) {
|
|
setJustPostedUris(set => {
|
|
const nextSet = new Set(set)
|
|
nextSet.add(postUri)
|
|
return nextSet
|
|
})
|
|
}
|
|
},
|
|
[refetch],
|
|
)
|
|
|
|
const {openComposer} = useComposerControls()
|
|
const onPressReply = React.useCallback(() => {
|
|
if (thread?.type !== 'post') {
|
|
return
|
|
}
|
|
openComposer({
|
|
replyTo: {
|
|
uri: thread.post.uri,
|
|
cid: thread.post.cid,
|
|
text: thread.record.text,
|
|
author: thread.post.author,
|
|
embed: thread.post.embed,
|
|
},
|
|
onPost: onPostReply,
|
|
})
|
|
}, [openComposer, thread, onPostReply])
|
|
|
|
const canReply = !error && rootPost && !rootPost.viewer?.replyDisabled
|
|
const hasParents =
|
|
skeleton?.highlightedPost?.type === 'post' &&
|
|
(skeleton.highlightedPost.ctx.isParentLoading ||
|
|
Boolean(skeleton?.parents && skeleton.parents.length > 0))
|
|
const showHeader =
|
|
isNative || (isTabletOrMobile && (!hasParents || !isFetching))
|
|
|
|
const renderItem = ({item, index}: {item: RowItem; index: number}) => {
|
|
if (item === REPLY_PROMPT && hasSession) {
|
|
return (
|
|
<View>
|
|
{!isMobile && (
|
|
<PostThreadComposePrompt onPressCompose={onPressReply} />
|
|
)}
|
|
</View>
|
|
)
|
|
} else if (item === SHOW_HIDDEN_REPLIES || item === SHOW_MUTED_REPLIES) {
|
|
return (
|
|
<PostThreadShowHiddenReplies
|
|
type={item === SHOW_HIDDEN_REPLIES ? 'hidden' : 'muted'}
|
|
onPress={() =>
|
|
setHiddenRepliesState(
|
|
item === SHOW_HIDDEN_REPLIES
|
|
? HiddenRepliesState.Show
|
|
: HiddenRepliesState.ShowAndOverridePostHider,
|
|
)
|
|
}
|
|
hideTopBorder={index === 0}
|
|
/>
|
|
)
|
|
} else if (isThreadNotFound(item)) {
|
|
return (
|
|
<View
|
|
style={[
|
|
a.p_lg,
|
|
index !== 0 && a.border_t,
|
|
t.atoms.border_contrast_low,
|
|
t.atoms.bg_contrast_25,
|
|
]}>
|
|
<Text style={[a.font_bold, a.text_md, t.atoms.text_contrast_medium]}>
|
|
<Trans>Deleted post.</Trans>
|
|
</Text>
|
|
</View>
|
|
)
|
|
} else if (isThreadBlocked(item)) {
|
|
return (
|
|
<View
|
|
style={[
|
|
a.p_lg,
|
|
index !== 0 && a.border_t,
|
|
t.atoms.border_contrast_low,
|
|
t.atoms.bg_contrast_25,
|
|
]}>
|
|
<Text style={[a.font_bold, a.text_md, t.atoms.text_contrast_medium]}>
|
|
<Trans>Blocked post.</Trans>
|
|
</Text>
|
|
</View>
|
|
)
|
|
} else if (isThreadPost(item)) {
|
|
if (!treeView && item.ctx.hasMoreSelfThread) {
|
|
return <PostThreadLoadMore post={item.post} />
|
|
}
|
|
const prev = isThreadPost(posts[index - 1])
|
|
? (posts[index - 1] as ThreadPost)
|
|
: undefined
|
|
const next = isThreadPost(posts[index + 1])
|
|
? (posts[index + 1] as ThreadPost)
|
|
: undefined
|
|
const showChildReplyLine = (next?.ctx.depth || 0) > item.ctx.depth
|
|
const showParentReplyLine =
|
|
(item.ctx.depth < 0 && !!item.parent) || item.ctx.depth > 1
|
|
const hasUnrevealedParents =
|
|
index === 0 && skeleton?.parents && maxParents < skeleton.parents.length
|
|
|
|
return (
|
|
<View
|
|
ref={item.ctx.isHighlightedPost ? highlightedPostRef : undefined}
|
|
onLayout={deferParents ? () => setDeferParents(false) : undefined}>
|
|
<PostThreadItem
|
|
post={item.post}
|
|
record={item.record}
|
|
threadgateRecord={threadgateRecord ?? undefined}
|
|
moderation={threadModerationCache.get(item)}
|
|
treeView={treeView}
|
|
depth={item.ctx.depth}
|
|
prevPost={prev}
|
|
nextPost={next}
|
|
isHighlightedPost={item.ctx.isHighlightedPost}
|
|
hasMore={item.ctx.hasMore}
|
|
showChildReplyLine={showChildReplyLine}
|
|
showParentReplyLine={showParentReplyLine}
|
|
hasPrecedingItem={showParentReplyLine || !!hasUnrevealedParents}
|
|
overrideBlur={
|
|
hiddenRepliesState ===
|
|
HiddenRepliesState.ShowAndOverridePostHider &&
|
|
item.ctx.depth > 0
|
|
}
|
|
onPostReply={onPostReply}
|
|
hideTopBorder={index === 0 && !item.ctx.isParentLoading}
|
|
/>
|
|
</View>
|
|
)
|
|
}
|
|
return null
|
|
}
|
|
|
|
if (!thread || !preferences || error) {
|
|
return (
|
|
<ListMaybePlaceholder
|
|
isLoading={!error}
|
|
isError={Boolean(error)}
|
|
noEmpty
|
|
onRetry={refetch}
|
|
errorTitle={error?.title}
|
|
errorMessage={error?.message}
|
|
/>
|
|
)
|
|
}
|
|
|
|
return (
|
|
<CenteredView style={[a.flex_1]} sideBorders={true}>
|
|
{showHeader && (
|
|
<ViewHeader
|
|
title={_(msg({message: `Post`, context: 'description'}))}
|
|
showBorder
|
|
/>
|
|
)}
|
|
|
|
<ScrollProvider onMomentumEnd={onMomentumEnd}>
|
|
<List
|
|
ref={ref}
|
|
data={posts}
|
|
renderItem={renderItem}
|
|
keyExtractor={keyExtractor}
|
|
onContentSizeChange={isNative ? undefined : onContentSizeChangeWeb}
|
|
onStartReached={onStartReached}
|
|
onEndReached={onEndReached}
|
|
onEndReachedThreshold={2}
|
|
onScrollToTop={onScrollToTop}
|
|
maintainVisibleContentPosition={
|
|
isNative && hasParents
|
|
? MAINTAIN_VISIBLE_CONTENT_POSITION
|
|
: undefined
|
|
}
|
|
// @ts-ignore our .web version only -prf
|
|
desktopFixedHeight
|
|
removeClippedSubviews={isAndroid ? false : undefined}
|
|
ListFooterComponent={
|
|
<ListFooter
|
|
// Using `isFetching` over `isFetchingNextPage` is done on purpose here so we get the loader on
|
|
// initial render
|
|
isFetchingNextPage={isFetching}
|
|
error={cleanError(threadError)}
|
|
onRetry={refetch}
|
|
// 300 is based on the minimum height of a post. This is enough extra height for the `maintainVisPos` to
|
|
// work without causing weird jumps on web or glitches on native
|
|
height={windowHeight - 200}
|
|
/>
|
|
}
|
|
initialNumToRender={initialNumToRender}
|
|
windowSize={11}
|
|
sideBorders={false}
|
|
/>
|
|
</ScrollProvider>
|
|
{isMobile && canReply && hasSession && (
|
|
<MobileComposePrompt onPressReply={onPressReply} />
|
|
)}
|
|
</CenteredView>
|
|
)
|
|
}
|
|
|
|
function MobileComposePrompt({onPressReply}: {onPressReply: () => unknown}) {
|
|
const safeAreaInsets = useSafeAreaInsets()
|
|
const fabMinimalShellTransform = useMinimalShellFabTransform()
|
|
return (
|
|
<Animated.View
|
|
style={[
|
|
styles.prompt,
|
|
fabMinimalShellTransform,
|
|
{
|
|
bottom: clamp(safeAreaInsets.bottom, 15, 30),
|
|
},
|
|
]}>
|
|
<PostThreadComposePrompt onPressCompose={onPressReply} />
|
|
</Animated.View>
|
|
)
|
|
}
|
|
|
|
function isThreadPost(v: unknown): v is ThreadPost {
|
|
return !!v && typeof v === 'object' && 'type' in v && v.type === 'post'
|
|
}
|
|
|
|
function isThreadNotFound(v: unknown): v is ThreadNotFound {
|
|
return !!v && typeof v === 'object' && 'type' in v && v.type === 'not-found'
|
|
}
|
|
|
|
function isThreadBlocked(v: unknown): v is ThreadBlocked {
|
|
return !!v && typeof v === 'object' && 'type' in v && v.type === 'blocked'
|
|
}
|
|
|
|
function createThreadSkeleton(
|
|
node: ThreadNode,
|
|
currentDid: string | undefined,
|
|
treeView: boolean,
|
|
modCache: ThreadModerationCache,
|
|
showHiddenReplies: boolean,
|
|
threadgateRecordHiddenReplies: Set<string>,
|
|
): ThreadSkeletonParts | null {
|
|
if (!node) return null
|
|
|
|
return {
|
|
parents: Array.from(flattenThreadParents(node, !!currentDid)),
|
|
highlightedPost: node,
|
|
replies: Array.from(
|
|
flattenThreadReplies(
|
|
node,
|
|
currentDid,
|
|
treeView,
|
|
modCache,
|
|
showHiddenReplies,
|
|
threadgateRecordHiddenReplies,
|
|
),
|
|
),
|
|
}
|
|
}
|
|
|
|
function* flattenThreadParents(
|
|
node: ThreadNode,
|
|
hasSession: boolean,
|
|
): Generator<YieldedItem, void> {
|
|
if (node.type === 'post') {
|
|
if (node.parent) {
|
|
yield* flattenThreadParents(node.parent, hasSession)
|
|
}
|
|
if (!node.ctx.isHighlightedPost) {
|
|
yield node
|
|
}
|
|
} else if (node.type === 'not-found') {
|
|
yield node
|
|
} else if (node.type === 'blocked') {
|
|
yield node
|
|
}
|
|
}
|
|
|
|
// The enum is ordered to make them easy to merge
|
|
enum HiddenReplyType {
|
|
None = 0,
|
|
Muted = 1,
|
|
Hidden = 2,
|
|
}
|
|
|
|
function* flattenThreadReplies(
|
|
node: ThreadNode,
|
|
currentDid: string | undefined,
|
|
treeView: boolean,
|
|
modCache: ThreadModerationCache,
|
|
showHiddenReplies: boolean,
|
|
threadgateRecordHiddenReplies: Set<string>,
|
|
): Generator<YieldedItem, HiddenReplyType> {
|
|
if (node.type === 'post') {
|
|
// dont show pwi-opted-out posts to logged out users
|
|
if (!currentDid && hasPwiOptOut(node)) {
|
|
return HiddenReplyType.None
|
|
}
|
|
|
|
// handle blurred items
|
|
if (node.ctx.depth > 0) {
|
|
const modui = modCache.get(node)?.ui('contentList')
|
|
if (modui?.blur || modui?.filter) {
|
|
if (!showHiddenReplies || node.ctx.depth > 1) {
|
|
if ((modui.blurs[0] || modui.filters[0]).type === 'muted') {
|
|
return HiddenReplyType.Muted
|
|
}
|
|
return HiddenReplyType.Hidden
|
|
}
|
|
}
|
|
|
|
if (!showHiddenReplies) {
|
|
const hiddenByThreadgate = threadgateRecordHiddenReplies.has(
|
|
node.post.uri,
|
|
)
|
|
const authorIsViewer = node.post.author.did === currentDid
|
|
if (hiddenByThreadgate && !authorIsViewer) {
|
|
return HiddenReplyType.Hidden
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!node.ctx.isHighlightedPost) {
|
|
yield node
|
|
}
|
|
|
|
if (node.replies?.length) {
|
|
let hiddenReplies = HiddenReplyType.None
|
|
for (const reply of node.replies) {
|
|
let hiddenReply = yield* flattenThreadReplies(
|
|
reply,
|
|
currentDid,
|
|
treeView,
|
|
modCache,
|
|
showHiddenReplies,
|
|
threadgateRecordHiddenReplies,
|
|
)
|
|
if (hiddenReply > hiddenReplies) {
|
|
hiddenReplies = hiddenReply
|
|
}
|
|
if (!treeView && !node.ctx.isHighlightedPost) {
|
|
break
|
|
}
|
|
}
|
|
|
|
// show control to enable hidden replies
|
|
if (node.ctx.depth === 0) {
|
|
if (hiddenReplies === HiddenReplyType.Muted) {
|
|
yield SHOW_MUTED_REPLIES
|
|
} else if (hiddenReplies === HiddenReplyType.Hidden) {
|
|
yield SHOW_HIDDEN_REPLIES
|
|
}
|
|
}
|
|
}
|
|
} else if (node.type === 'not-found') {
|
|
yield node
|
|
} else if (node.type === 'blocked') {
|
|
yield node
|
|
}
|
|
return HiddenReplyType.None
|
|
}
|
|
|
|
function hasPwiOptOut(node: ThreadPost) {
|
|
return !!node.post.author.labels?.find(l => l.val === '!no-unauthenticated')
|
|
}
|
|
|
|
function hasBranchingReplies(node?: ThreadNode) {
|
|
if (!node) {
|
|
return false
|
|
}
|
|
if (node.type !== 'post') {
|
|
return false
|
|
}
|
|
if (!node.replies) {
|
|
return false
|
|
}
|
|
if (node.replies.length === 1) {
|
|
return hasBranchingReplies(node.replies[0])
|
|
}
|
|
return true
|
|
}
|
|
|
|
const styles = StyleSheet.create({
|
|
prompt: {
|
|
// @ts-ignore web-only
|
|
position: isWeb ? 'fixed' : 'absolute',
|
|
left: 0,
|
|
right: 0,
|
|
},
|
|
})
|