import {useMemo} from 'react' import {View} from 'react-native' import { type AppBskyActorDefs, type ModerationCause, type ModerationDecision, } from '@atproto/api' import {useLingui} from '@lingui/react/macro' import {useNavigation} from '@react-navigation/native' import {createSanitizedDisplayName} from '#/lib/moderation/create-sanitized-display-name' import {makeProfileLink} from '#/lib/routes/links' import {type NavigationProp} from '#/lib/routes/types' import {logger} from '#/logger' import {type Shadow} from '#/state/cache/profile-shadow' import { type ActiveConvoStates, isConvoActive, useConvo, } from '#/state/messages/convo' import {type ConvoItem} from '#/state/messages/convo/types' import {useSession} from '#/state/session' import {PreviewableUserAvatar} from '#/view/com/util/UserAvatar' import {atoms as a, useTheme} from '#/alf' import {AvatarBubbles} from '#/components/AvatarBubbles' import {Button, ButtonIcon} from '#/components/Button' import {ConvoMenu} from '#/components/dms/ConvoMenu' import {Bell2Off_Filled_Corner0_Rounded as BellOffIcon} from '#/components/icons/Bell2' import {DotGrid3x1_Stroke2_Corner0_Rounded as DotsHorizontalIcon} from '#/components/icons/DotGrid' import * as Layout from '#/components/Layout' import {Link} from '#/components/Link' import {ProfileBadges} from '#/components/ProfileBadges' import {Text} from '#/components/Typography' import {IS_LIQUID_GLASS, IS_WEB} from '#/env' const PFP_SIZE = IS_WEB ? 40 : Layout.HEADER_SLOT_SIZE export function MessagesListHeader({ profile, moderation, }: { profile?: Shadow moderation?: ModerationDecision | null }) { const t = useTheme() const convoState = useConvo() const isGroupChat = convoState?.isGroup?.() const blockInfo = useMemo(() => { if (!moderation) return const modui = moderation.ui('profileView') const blocks = modui.alerts.filter(alert => alert.type === 'blocking') const listBlocks = blocks.filter(alert => alert.source.type === 'list') const userBlock = blocks.find(alert => alert.source.type === 'user') return { listBlocks, userBlock, } }, [moderation]) return ( {isConvoActive(convoState) ? ( moderation && blockInfo && profile && !isGroupChat ? ( ) : ( ) ) : ( <> )} ) } function ProfileHeaderReady({ convoState, profile, moderation, blockInfo, }: { convoState: ActiveConvoStates profile: Shadow moderation: ModerationDecision blockInfo: { listBlocks: ModerationCause[] userBlock?: ModerationCause } }) { const {t: l} = useLingui() const {currentAccount} = useSession() const isDeletedAccount = profile?.handle === 'missing.invalid' const displayName = isDeletedAccount ? l`Deleted Account` : createSanitizedDisplayName(profile, true, moderation.ui('displayName')) const latestMessageFromOther = convoState.items.findLast( (item: ConvoItem) => item.type === 'message' && item.message.sender.did !== currentAccount?.did, ) const latestReportableMessage = latestMessageFromOther?.type === 'message' ? latestMessageFromOther.message : undefined return ( {displayName} } muted={convoState.convo?.muted} settings={ isConvoActive(convoState) ? ( ) : null } /> ) } function GroupHeaderReady({ convoState, profile, moderation, }: { convoState: ActiveConvoStates profile?: Shadow moderation?: ModerationDecision | null }) { const {t: l} = useLingui() const navigation = useNavigation() const groupInfo = convoState.getGroupInfo?.() const isDeletedAccount = profile?.handle === 'missing.invalid' const displayName = isDeletedAccount ? l`Deleted Account` : profile ? createSanitizedDisplayName(profile, true, moderation?.ui('displayName')) : undefined const groupName = groupInfo?.name ?? (displayName ? l`${displayName}’s group chat` : l`Group chat`) const handleNavigateToSettings = () => { const convoId = convoState.convo?.id if (convoId) { navigation.navigate('MessagesConversationSettings', { conversation: convoId, }) } else { logger.error(`handleNavigateToSettings: missing convo ID`) } } return ( {groupName} } muted={convoState.convo?.muted} settings={ isConvoActive(convoState) ? ( ) : null } /> ) } function Wrapper({ heading, muted, settings, }: { heading: React.ReactNode muted: boolean settings: React.ReactNode }) { return ( {heading} {settings} ) } function MuteStatus({muted}: {muted: boolean}) { const t = useTheme() return muted ? ( <> · ) : undefined }