218 lines
7.3 KiB
TypeScript
218 lines
7.3 KiB
TypeScript
import React, {useCallback} from 'react'
|
|
import {Keyboard, Pressable, View} from 'react-native'
|
|
import {
|
|
AppBskyActorDefs,
|
|
ChatBskyConvoDefs,
|
|
ModerationCause,
|
|
} from '@atproto/api'
|
|
import {msg, Trans} from '@lingui/macro'
|
|
import {useLingui} from '@lingui/react'
|
|
import {useNavigation} from '@react-navigation/native'
|
|
|
|
import {NavigationProp} from '#/lib/routes/types'
|
|
import {Shadow} from '#/state/cache/types'
|
|
import {
|
|
useConvoQuery,
|
|
useMarkAsReadMutation,
|
|
} from '#/state/queries/messages/conversation'
|
|
import {useMuteConvo} from '#/state/queries/messages/mute-conversation'
|
|
import {useProfileBlockMutationQueue} from '#/state/queries/profile'
|
|
import * as Toast from '#/view/com/util/Toast'
|
|
import {atoms as a, useTheme} from '#/alf'
|
|
import {BlockedByListDialog} from '#/components/dms/BlockedByListDialog'
|
|
import {LeaveConvoPrompt} from '#/components/dms/LeaveConvoPrompt'
|
|
import {ReportConversationPrompt} from '#/components/dms/ReportConversationPrompt'
|
|
import {ArrowBoxLeft_Stroke2_Corner0_Rounded as ArrowBoxLeft} from '#/components/icons/ArrowBoxLeft'
|
|
import {DotGrid_Stroke2_Corner0_Rounded as DotsHorizontal} from '#/components/icons/DotGrid'
|
|
import {Flag_Stroke2_Corner0_Rounded as Flag} from '#/components/icons/Flag'
|
|
import {Mute_Stroke2_Corner0_Rounded as Mute} from '#/components/icons/Mute'
|
|
import {Person_Stroke2_Corner0_Rounded as Person} from '#/components/icons/Person'
|
|
import {PersonCheck_Stroke2_Corner0_Rounded as PersonCheck} from '#/components/icons/PersonCheck'
|
|
import {PersonX_Stroke2_Corner0_Rounded as PersonX} from '#/components/icons/PersonX'
|
|
import {SpeakerVolumeFull_Stroke2_Corner0_Rounded as Unmute} from '#/components/icons/Speaker'
|
|
import * as Menu from '#/components/Menu'
|
|
import * as Prompt from '#/components/Prompt'
|
|
import {Bubble_Stroke2_Corner2_Rounded as Bubble} from '../icons/Bubble'
|
|
|
|
let ConvoMenu = ({
|
|
convo: initialConvo,
|
|
profile,
|
|
control,
|
|
currentScreen,
|
|
showMarkAsRead,
|
|
hideTrigger,
|
|
triggerOpacity,
|
|
blockInfo,
|
|
}: {
|
|
convo: ChatBskyConvoDefs.ConvoView
|
|
profile: Shadow<AppBskyActorDefs.ProfileViewBasic>
|
|
control?: Menu.MenuControlProps
|
|
currentScreen: 'list' | 'conversation'
|
|
showMarkAsRead?: boolean
|
|
hideTrigger?: boolean
|
|
triggerOpacity?: number
|
|
blockInfo: {
|
|
listBlocks: ModerationCause[]
|
|
userBlock?: ModerationCause
|
|
}
|
|
}): React.ReactNode => {
|
|
const navigation = useNavigation<NavigationProp>()
|
|
const {_} = useLingui()
|
|
const t = useTheme()
|
|
const leaveConvoControl = Prompt.usePromptControl()
|
|
const reportControl = Prompt.usePromptControl()
|
|
const blockedByListControl = Prompt.usePromptControl()
|
|
const {mutate: markAsRead} = useMarkAsReadMutation()
|
|
|
|
const {listBlocks, userBlock} = blockInfo
|
|
const isBlocking = userBlock || !!listBlocks.length
|
|
|
|
const {data: convo} = useConvoQuery(initialConvo)
|
|
|
|
const onNavigateToProfile = useCallback(() => {
|
|
navigation.navigate('Profile', {name: profile.did})
|
|
}, [navigation, profile.did])
|
|
|
|
const {mutate: muteConvo} = useMuteConvo(convo?.id, {
|
|
onSuccess: data => {
|
|
if (data.convo.muted) {
|
|
Toast.show(_(msg`Chat muted`))
|
|
} else {
|
|
Toast.show(_(msg`Chat unmuted`))
|
|
}
|
|
},
|
|
onError: () => {
|
|
Toast.show(_(msg`Could not mute chat`))
|
|
},
|
|
})
|
|
|
|
const [queueBlock, queueUnblock] = useProfileBlockMutationQueue(profile)
|
|
|
|
const toggleBlock = React.useCallback(() => {
|
|
if (listBlocks.length) {
|
|
blockedByListControl.open()
|
|
return
|
|
}
|
|
|
|
if (userBlock) {
|
|
queueUnblock()
|
|
} else {
|
|
queueBlock()
|
|
}
|
|
}, [userBlock, listBlocks, blockedByListControl, queueBlock, queueUnblock])
|
|
|
|
return (
|
|
<>
|
|
<Menu.Root control={control}>
|
|
{!hideTrigger && (
|
|
<View style={{opacity: triggerOpacity}}>
|
|
<Menu.Trigger label={_(msg`Chat settings`)}>
|
|
{({props, state}) => (
|
|
<Pressable
|
|
{...props}
|
|
onPress={() => {
|
|
Keyboard.dismiss()
|
|
// eslint-disable-next-line react/prop-types -- eslint is confused by the name `props`
|
|
props.onPress()
|
|
}}
|
|
style={[
|
|
a.p_sm,
|
|
a.rounded_full,
|
|
(state.hovered || state.pressed) && t.atoms.bg_contrast_25,
|
|
// make sure pfp is in the middle
|
|
{marginLeft: -10},
|
|
]}>
|
|
<DotsHorizontal size="md" style={t.atoms.text} />
|
|
</Pressable>
|
|
)}
|
|
</Menu.Trigger>
|
|
</View>
|
|
)}
|
|
<Menu.Outer>
|
|
<Menu.Group>
|
|
{showMarkAsRead && (
|
|
<Menu.Item
|
|
label={_(msg`Mark as read`)}
|
|
onPress={() =>
|
|
markAsRead({
|
|
convoId: convo?.id,
|
|
})
|
|
}>
|
|
<Menu.ItemText>
|
|
<Trans>Mark as read</Trans>
|
|
</Menu.ItemText>
|
|
<Menu.ItemIcon icon={Bubble} />
|
|
</Menu.Item>
|
|
)}
|
|
<Menu.Item
|
|
label={_(msg`Go to user's profile`)}
|
|
onPress={onNavigateToProfile}>
|
|
<Menu.ItemText>
|
|
<Trans>Go to profile</Trans>
|
|
</Menu.ItemText>
|
|
<Menu.ItemIcon icon={Person} />
|
|
</Menu.Item>
|
|
<Menu.Item
|
|
label={_(msg`Mute conversation`)}
|
|
onPress={() => muteConvo({mute: !convo?.muted})}>
|
|
<Menu.ItemText>
|
|
{convo?.muted ? (
|
|
<Trans>Unmute conversation</Trans>
|
|
) : (
|
|
<Trans>Mute conversation</Trans>
|
|
)}
|
|
</Menu.ItemText>
|
|
<Menu.ItemIcon icon={convo?.muted ? Unmute : Mute} />
|
|
</Menu.Item>
|
|
</Menu.Group>
|
|
<Menu.Divider />
|
|
<Menu.Group>
|
|
<Menu.Item
|
|
label={
|
|
isBlocking ? _(msg`Unblock account`) : _(msg`Block account`)
|
|
}
|
|
onPress={toggleBlock}>
|
|
<Menu.ItemText>
|
|
{isBlocking ? _(msg`Unblock account`) : _(msg`Block account`)}
|
|
</Menu.ItemText>
|
|
<Menu.ItemIcon icon={isBlocking ? PersonX : PersonCheck} />
|
|
</Menu.Item>
|
|
<Menu.Item
|
|
label={_(msg`Report conversation`)}
|
|
onPress={reportControl.open}>
|
|
<Menu.ItemText>
|
|
<Trans>Report conversation</Trans>
|
|
</Menu.ItemText>
|
|
<Menu.ItemIcon icon={Flag} />
|
|
</Menu.Item>
|
|
</Menu.Group>
|
|
<Menu.Divider />
|
|
<Menu.Group>
|
|
<Menu.Item
|
|
label={_(msg`Leave conversation`)}
|
|
onPress={leaveConvoControl.open}>
|
|
<Menu.ItemText>
|
|
<Trans>Leave conversation</Trans>
|
|
</Menu.ItemText>
|
|
<Menu.ItemIcon icon={ArrowBoxLeft} />
|
|
</Menu.Item>
|
|
</Menu.Group>
|
|
</Menu.Outer>
|
|
</Menu.Root>
|
|
|
|
<LeaveConvoPrompt
|
|
control={leaveConvoControl}
|
|
convoId={convo.id}
|
|
currentScreen={currentScreen}
|
|
/>
|
|
<ReportConversationPrompt control={reportControl} />
|
|
<BlockedByListDialog
|
|
control={blockedByListControl}
|
|
listBlocks={listBlocks}
|
|
/>
|
|
</>
|
|
)
|
|
}
|
|
ConvoMenu = React.memo(ConvoMenu)
|
|
|
|
export {ConvoMenu}
|