908 lines
32 KiB
Vue
908 lines
32 KiB
Vue
<template>
|
||
<!-- 时间分隔条:列表第一条 / 距上一条超过阈值时居中显示灰色时间 -->
|
||
<div
|
||
v-if="shouldShowTimeTip"
|
||
class="flex items-center justify-center px-4 py-2 text-12px text-[var(--el-text-color-disabled)]"
|
||
>
|
||
{{ formatTimeTip(message.sendTime) }}
|
||
</div>
|
||
|
||
<!-- 好友会话事件(FRIEND_ADD / FRIEND_DELETE):跟群广播事件同灰色样式,文案固定 -->
|
||
<div
|
||
v-if="isFriendChatTipMessage"
|
||
class="flex items-center justify-center px-4 py-2 text-12px text-[var(--el-text-color-secondary)]"
|
||
>
|
||
<TipSegments :segments="friendChatTipSegments" />
|
||
</div>
|
||
|
||
<!-- 群广播事件:跟好友事件同灰色样式,mention 段挂点击弹 UserInfoCard -->
|
||
<div
|
||
v-else-if="isGroupNotificationMessage"
|
||
class="flex items-center justify-center px-4 py-2 text-12px text-[var(--el-text-color-secondary)]"
|
||
>
|
||
<TipSegments :segments="groupNotificationSegments" />
|
||
</div>
|
||
|
||
<!-- 撤回消息:整行灰色 tip,sender 名段可点击 -->
|
||
<div
|
||
v-else-if="isRecall"
|
||
class="flex items-center justify-center px-4 py-2 text-12px text-[var(--el-text-color-secondary)]"
|
||
>
|
||
<TipSegments :segments="recallTipSegments" />
|
||
</div>
|
||
|
||
<!-- 普通消息气泡 -->
|
||
<div
|
||
v-else
|
||
class="flex gap-2 items-start px-4 py-2"
|
||
:class="{ 'cursor-pointer': isInMultiSelect }"
|
||
@contextmenu.prevent="handleContextMenu"
|
||
@click.capture="handleMultiSelectClick"
|
||
>
|
||
<!-- 多选指示:永远在整行最左侧(自己 / 对方消息一致),不参与 selfSend reverse -->
|
||
<span
|
||
v-if="isInMultiSelect"
|
||
class="flex flex-shrink-0 items-center justify-center mt-3 w-5 h-5 rounded-full transition-colors"
|
||
:class="
|
||
isMessageChecked
|
||
? 'bg-[#07c160]'
|
||
: 'border border-[var(--el-border-color)] bg-[var(--el-bg-color)]'
|
||
"
|
||
>
|
||
<Icon v-if="isMessageChecked" icon="ant-design:check-outlined" :size="12" color="#fff" />
|
||
</span>
|
||
|
||
<!-- 消息行:头像 + 气泡内部按 selfSend reverse -->
|
||
<div
|
||
class="flex flex-1 min-w-0 gap-2 items-start"
|
||
:class="{ 'flex-row-reverse': message.selfSend }"
|
||
>
|
||
<!-- 头像:点击弹 UserInfoCard 由 UserAvatar 内部承接 -->
|
||
<UserAvatar
|
||
:id="message.selfSend ? userStore.getUser?.id : message.senderId"
|
||
:name="senderRealNickname"
|
||
:url="message.selfSend ? userStore.getUser?.avatar : senderAvatar"
|
||
:size="36"
|
||
/>
|
||
|
||
<div class="flex flex-col gap-0.5 max-w-[70%]" :class="{ 'items-end': message.selfSend }">
|
||
<!-- 群聊对方消息:气泡上方显示发送者昵称 -->
|
||
<div
|
||
v-if="showSenderName"
|
||
class="mb-0.5 text-12px text-[var(--el-text-color-secondary)] leading-tight"
|
||
>
|
||
{{ senderDisplayName }}
|
||
</div>
|
||
<div class="flex gap-1.5 items-center" :class="{ 'flex-row-reverse': message.selfSend }">
|
||
<!-- 消息内容:按 type 走 9 类气泡,统一由 MessageBubble 渲染 -->
|
||
<MessageBubble
|
||
:type="message.type"
|
||
:content="message.content"
|
||
:self-send="message.selfSend"
|
||
:upload-progress="message.uploadProgress"
|
||
:mentions="textMentions"
|
||
@click-card="handleCardClick"
|
||
@open-merge="handleMergeOpen"
|
||
/>
|
||
|
||
<!-- 状态区:自己消息展示发送状态 + 已读/群回执;对方消息 + @自己时展示 @徽标 -->
|
||
<div class="flex gap-1.5 items-center text-base">
|
||
<template v-if="message.selfSend">
|
||
<Icon
|
||
v-if="showSendingLoading"
|
||
icon="ant-design:loading-outlined"
|
||
class="im-loading-spin"
|
||
/>
|
||
<Icon
|
||
v-else-if="message.status === ImMessageStatus.FAILED"
|
||
icon="ant-design:warning-filled"
|
||
color="#f56c6c"
|
||
class="cursor-pointer"
|
||
title="发送失败,点击重试"
|
||
@click="handleResend"
|
||
/>
|
||
<!-- 已读态(私聊) -->
|
||
<span
|
||
v-else-if="privateReadLabel"
|
||
class="text-12px whitespace-nowrap"
|
||
:class="
|
||
message.status === ImMessageStatus.READ
|
||
? 'text-[#409eff]'
|
||
: 'text-[var(--el-text-color-secondary)]'
|
||
"
|
||
>
|
||
{{ privateReadLabel }}
|
||
</span>
|
||
<!-- 群回执:点击弹 popover 展示已读 / 未读成员列表 -->
|
||
<MessageReadStatus
|
||
v-else-if="showGroupReadStatus"
|
||
:message="message"
|
||
:group-id="conversationStore.activeConversation?.targetId || 0"
|
||
:group-members="groupMembersForReadStatus"
|
||
class="text-12px whitespace-nowrap text-[var(--el-text-color-secondary)]"
|
||
/>
|
||
</template>
|
||
<!-- 群 @ 我提示 -->
|
||
<el-tag
|
||
v-if="!message.selfSend && isAtMe"
|
||
type="danger"
|
||
size="small"
|
||
effect="plain"
|
||
class="ml-1"
|
||
>
|
||
@我
|
||
</el-tag>
|
||
</div>
|
||
</div>
|
||
<!-- 引用块:气泡下方,selfSend 时竖线在右侧 -->
|
||
<ReplyPreview
|
||
v-if="quote"
|
||
:quote="quote"
|
||
clickable
|
||
:mirrored="message.selfSend"
|
||
class="max-w-[280px]"
|
||
@locate="emit('locate', $event)"
|
||
/>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script lang="ts" setup>
|
||
import { computed, inject } from 'vue'
|
||
import { useClipboard } from '@vueuse/core'
|
||
import Icon from '@/components/Icon/src/Icon.vue'
|
||
import { useMessage } from '@/hooks/web/useMessage'
|
||
import { ElMessageBox } from 'element-plus'
|
||
|
||
import {
|
||
ImForwardMode,
|
||
ImMessageType,
|
||
ImMessageStatus,
|
||
ImGroupReceiptStatus,
|
||
ImConversationType,
|
||
ImFriendAddSource,
|
||
ImGroupMemberRole,
|
||
isFriendChatTip,
|
||
isGroupNotification,
|
||
isMediaMessageType,
|
||
isNormalMessage
|
||
} from '@/views/im/utils/constants'
|
||
import { MESSAGE_TIME_TIP_GAP_MS, MESSAGE_RECALL_WINDOW_MS } from '@/views/im/utils/config'
|
||
import { pinGroupMessage as apiPinGroupMessage, cancelMuteMember } from '@/api/im/group'
|
||
import { removeGroupMember } from '@/api/im/group/member'
|
||
import {
|
||
buildQuoteFromMessage,
|
||
extractAddableFace,
|
||
getQuoteFromMessage,
|
||
parseMessage,
|
||
type CardMessage,
|
||
type MentionCandidate,
|
||
type TextMessage
|
||
} from '@/views/im/utils/message'
|
||
import { buildRecallTipSegments } from '@/views/im/utils/conversation'
|
||
import { formatTimeTip } from '@/views/im/utils/time'
|
||
import { useUserStore } from '@/store/modules/user'
|
||
import { useConversationStore } from '../../../../store/conversationStore'
|
||
import { useGroupStore } from '../../../../store/groupStore'
|
||
import { useFriendStore } from '../../../../store/friendStore'
|
||
import { useDraftStore } from '../../../../store/draftStore'
|
||
import { useFaceStore } from '../../../../store/faceStore'
|
||
import {
|
||
getMemberDisplayName,
|
||
getMentionCandidates,
|
||
getSenderDisplayName,
|
||
getSenderRealNickname,
|
||
resolveFriendNotificationSegments,
|
||
resolveGroupNotificationSegments
|
||
} from '@/views/im/utils/user'
|
||
import { useImUiStore } from '../../../../store/uiStore'
|
||
import { useMessageSender } from '../../../../composables/useMessageSender'
|
||
import { mediaTypeHandlers, useMediaUploader } from '../../../../composables/useMediaUploader'
|
||
import { useMuteOverlay } from '../../../../composables/useMuteOverlay'
|
||
import type { Message } from '../../../../types'
|
||
import MessageReadStatus from './MessageReadStatus.vue'
|
||
import ReplyPreview from './ReplyPreview.vue'
|
||
import TipSegments from './TipSegments.vue'
|
||
import UserAvatar from '../../../../components/user/UserAvatar.vue'
|
||
import MessageBubble from './MessageBubble.vue'
|
||
import { IM_FORWARD_DIALOG_KEY, IM_MERGE_DETAIL_DIALOG_KEY } from './forward/keys'
|
||
import { useMessageMultiSelect } from '../../../../composables/useMessageMultiSelect'
|
||
import type { GroupMemberLite } from '../../../../components/group/GroupMember.vue'
|
||
|
||
defineOptions({ name: 'ImMessageItem' })
|
||
|
||
const props = defineProps<{
|
||
message: Message
|
||
/** 列表中的上一条消息,用于判断是否要在当前消息上方渲染时间分隔条;不传按列表第一条处理 */
|
||
prevMessage?: Message
|
||
}>()
|
||
|
||
const emit = defineEmits<{
|
||
/** 引用块点击 → MessagePanel 滚定位 + 高亮 */
|
||
locate: [messageId: number]
|
||
/** 禁言:需要父组件打开时长选择弹窗 */
|
||
mute: [groupId: number, userId: number, displayName: string]
|
||
/** 数据变更后刷新群信息 */
|
||
reload: []
|
||
}>()
|
||
|
||
// ==================== Stores / Hooks ====================
|
||
|
||
const userStore = useUserStore()
|
||
const conversationStore = useConversationStore()
|
||
const groupStore = useGroupStore()
|
||
const friendStore = useFriendStore()
|
||
const draftStore = useDraftStore()
|
||
const faceStore = useFaceStore()
|
||
const uiStore = useImUiStore()
|
||
const { recall, sendRaw } = useMessageSender()
|
||
const { uploadAndSendMedia } = useMediaUploader()
|
||
const muteOverlay = useMuteOverlay()
|
||
// 仅用 confirm,避免 message 跟 props.message 同名冲突(vue/no-dupe-keys)
|
||
const { confirm: confirmDialog, success: successMessage } = useMessage()
|
||
// legacy:true 兼容 HTTP 环境,没有 navigator.clipboard 时降级到 execCommand
|
||
const { copy: copyToClipboard } = useClipboard({ legacy: true })
|
||
|
||
// ==================== 消息类型判断 ====================
|
||
|
||
/** 是否在当前消息上方渲染时间分隔条:列表第一条 / 距上一条超过阈值;缺 sendTime 不渲染 */
|
||
const shouldShowTimeTip = computed(() => {
|
||
if (!props.message.sendTime) {
|
||
return false
|
||
}
|
||
if (!props.prevMessage?.sendTime) {
|
||
return true
|
||
}
|
||
return props.message.sendTime - props.prevMessage.sendTime > MESSAGE_TIME_TIP_GAP_MS
|
||
})
|
||
|
||
/** 仅 MessageItem 自身仍要用到的 type 判定(其它分支已下沉到 MessageBubble) */
|
||
const isVoice = computed(() => props.message.type === ImMessageType.VOICE)
|
||
const isMerge = computed(() => props.message.type === ImMessageType.MERGE)
|
||
|
||
|
||
// ==================== 事件消息(撤回 / 好友 / 群广播) ====================
|
||
// 这三类不走普通气泡,渲染成居中灰色 tip;判断 + 文案配对放一起,新增第四类事件只需在本块改完
|
||
|
||
/** 是否已撤回:pull / WS 两路都会调 recallMessage 把原消息更新为 type=RECALL,渲染只需识别 type */
|
||
const isRecall = computed(() => props.message.type === ImMessageType.RECALL)
|
||
|
||
/** 撤回提示 segments:依赖 activeConversation 实时算 sender 名 */
|
||
const recallTipSegments = computed(() => {
|
||
const conversation = conversationStore.activeConversation
|
||
return buildRecallTipSegments(
|
||
props.message.senderId,
|
||
props.message.selfSend,
|
||
conversation?.type ?? 0,
|
||
conversation?.targetId ?? 0
|
||
)
|
||
})
|
||
|
||
/** 是否会话内好友事件气泡(FRIEND_ADD / FRIEND_DELETE) */
|
||
const isFriendChatTipMessage = computed(() => isFriendChatTip(props.message.type))
|
||
|
||
/** 好友事件 segments */
|
||
const friendChatTipSegments = computed(() => resolveFriendNotificationSegments(props.message))
|
||
|
||
/** 是否群广播事件(GROUP_CREATE..GROUP_BANNED 段位,排除 GROUP_MEMBER_SETTING_UPDATE 个人信号) */
|
||
const isGroupNotificationMessage = computed(() => isGroupNotification(props.message.type))
|
||
|
||
/** 群广播事件 segments */
|
||
const groupNotificationSegments = computed(() => resolveGroupNotificationSegments(props.message))
|
||
|
||
// ==================== 消息内容解析 / payload ====================
|
||
|
||
/** 引用对象:气泡内嵌入展示;非引用消息返回 null,模板 v-if 不渲染 */
|
||
const quote = computed(() => getQuoteFromMessage(props.message.content))
|
||
|
||
/** MessagePanel 注入的弹窗触发函数 */
|
||
const openForwardDialog = inject(IM_FORWARD_DIALOG_KEY)
|
||
const openMergeDetail = inject(IM_MERGE_DETAIL_DIALOG_KEY)
|
||
|
||
/** 多选模式:模块级单例 composable */
|
||
const multiSelect = useMessageMultiSelect()
|
||
|
||
/** 合并消息气泡点击:打开详情弹窗(嵌套合并由弹窗内部 push 栈) */
|
||
function handleMergeOpen(content: string) {
|
||
openMergeDetail?.(content)
|
||
}
|
||
|
||
/** 文本气泡 @ mention 候选名字:仅群消息有效,按 atUserIds 反查群成员真实昵称;非 TEXT 不走 store 读,让 getMentionCandidates 直接返回稳定空数组 */
|
||
const textMentions = computed<MentionCandidate[]>(() => {
|
||
if (props.message.type !== ImMessageType.TEXT) {
|
||
return getMentionCandidates(undefined, null)
|
||
}
|
||
return getMentionCandidates(props.message.atUserIds, conversationStore.activeConversation)
|
||
})
|
||
|
||
/** 名片点击:用户名片弹 UserInfoCard,群名片弹 GroupInfoCard;其它 targetType(含改包脏数据)忽略 */
|
||
function handleCardClick(card: CardMessage, e: MouseEvent) {
|
||
if (!card?.targetId) {
|
||
return
|
||
}
|
||
if (card.targetType === ImConversationType.PRIVATE) {
|
||
uiStore.openUserInfoCardAtEvent(
|
||
{ id: card.targetId, nickname: card.name, avatar: card.avatar },
|
||
e,
|
||
ImFriendAddSource.CARD
|
||
)
|
||
return
|
||
}
|
||
if (card.targetType === ImConversationType.GROUP) {
|
||
uiStore.openGroupInfoCardAtEvent(
|
||
{
|
||
id: card.targetId,
|
||
name: card.name,
|
||
showImage: card.avatar,
|
||
memberCount: card.memberCount
|
||
},
|
||
e
|
||
)
|
||
}
|
||
}
|
||
|
||
/** 媒体上传中:MessageBubble 自渲染遮罩 / 进度条;外层只用作 showSendingLoading 判定 */
|
||
const isUploading = computed(() => props.message.uploadProgress != null)
|
||
|
||
/**
|
||
* 是否在气泡尾部显示「发送中」loading 转圈
|
||
*
|
||
* 图片 / 视频 / 文件气泡内嵌已有进度反馈(遮罩 / 进度条),外层 loading 不再叠加;
|
||
* 语音气泡只有麦克风 + 时长,无内嵌进度,必须保留外层 loading 让用户感知正在发送
|
||
*/
|
||
const showSendingLoading = computed(
|
||
() => props.message.status === ImMessageStatus.SENDING && (!isUploading.value || isVoice.value)
|
||
)
|
||
|
||
|
||
// ==================== 发送人 / 已读 / @ ====================
|
||
|
||
/** 群聊 + 对方消息 时,在气泡上方显示发送者昵称 */
|
||
const showSenderName = computed(() => {
|
||
if (props.message.selfSend) {
|
||
return false
|
||
}
|
||
return conversationStore.activeConversation?.type === ImConversationType.GROUP
|
||
})
|
||
|
||
/** 发送者头像;私聊的 conversation.avatar 就是对方头像(openConversation 入参约定) */
|
||
const senderAvatar = computed(() => {
|
||
const conversation = conversationStore.activeConversation
|
||
if (!conversation || props.message.selfSend) {
|
||
return ''
|
||
}
|
||
if (conversation.type === ImConversationType.GROUP) {
|
||
const group = groupStore.getGroup(conversation.targetId)
|
||
return group?.members?.find((member) => member.userId === props.message.senderId)?.avatar || ''
|
||
}
|
||
return conversation.avatar || ''
|
||
})
|
||
|
||
/** 头像色卡 fallback 文本:永远是真实昵称,不掺备注 */
|
||
const senderRealNickname = computed(() => {
|
||
const conversation = conversationStore.activeConversation
|
||
return getSenderRealNickname(
|
||
props.message.senderId,
|
||
conversation?.type ?? 0,
|
||
conversation?.targetId ?? 0
|
||
)
|
||
})
|
||
|
||
/** 气泡上方发送人显示名(仅群聊对方消息显示):好友备注 > 群备注 > 真实昵称 */
|
||
const senderDisplayName = computed(() => {
|
||
const conversation = conversationStore.activeConversation
|
||
return getSenderDisplayName(
|
||
props.message.senderId,
|
||
conversation?.type ?? 0,
|
||
conversation?.targetId ?? 0
|
||
)
|
||
})
|
||
|
||
/** 私聊「已读 / 未读」态(仅对自己发送的私聊消息展示) */
|
||
const privateReadLabel = computed(() => {
|
||
if (!props.message.selfSend) {
|
||
return ''
|
||
}
|
||
if (conversationStore.activeConversation?.type !== ImConversationType.PRIVATE) {
|
||
return ''
|
||
}
|
||
if (props.message.status === ImMessageStatus.READ) {
|
||
return '已读'
|
||
}
|
||
if (props.message.status === ImMessageStatus.UNREAD) {
|
||
return '未读'
|
||
}
|
||
return ''
|
||
})
|
||
|
||
/**是否需要显示群回执 popover:自己发的群消息且后端开启了回执(NO_RECEIPT 表示发送时未要求回执,不渲染) */
|
||
const showGroupReadStatus = computed(() => {
|
||
if (!props.message.selfSend) {
|
||
return false
|
||
}
|
||
if (conversationStore.activeConversation?.type !== ImConversationType.GROUP) {
|
||
return false
|
||
}
|
||
const status = props.message.receiptStatus
|
||
if (status === undefined || status === null) {
|
||
return false
|
||
}
|
||
return status !== ImGroupReceiptStatus.NO_RECEIPT
|
||
})
|
||
|
||
/** 当前群成员(供 MessageReadStatus 计算未读名单;未加载完时兜底空数组不渲染) */
|
||
const groupMembersForReadStatus = computed<GroupMemberLite[]>(() => {
|
||
const conversation = conversationStore.activeConversation
|
||
if (!conversation || conversation.type !== ImConversationType.GROUP) {
|
||
return []
|
||
}
|
||
const group = groupStore.getGroup(conversation.targetId)
|
||
return (group?.members || []).map((member) => {
|
||
const friend = friendStore.getFriend(member.userId)
|
||
return {
|
||
userId: member.userId,
|
||
showName: getMemberDisplayName(member, friend),
|
||
nickname: member.nickname,
|
||
avatar: member.avatar,
|
||
status: member.status
|
||
}
|
||
})
|
||
})
|
||
|
||
/** 是否 @我(群消息展示小徽标) */
|
||
const isAtMe = computed(() => {
|
||
const myId = Number(userStore.getUser?.id) || 0
|
||
if (!myId) {
|
||
return false
|
||
}
|
||
return (props.message.atUserIds || []).includes(myId)
|
||
})
|
||
|
||
// ==================== 右键菜单 / 操作 ====================
|
||
|
||
/** 右键菜单 key 常量;push 端和分发端从同一处取,typo 编译期就能抓 */
|
||
const MENU_KEYS = {
|
||
COPY: 'COPY',
|
||
REPLY: 'REPLY',
|
||
FORWARD: 'FORWARD',
|
||
MULTI_SELECT: 'MULTI_SELECT',
|
||
PIN: 'PIN',
|
||
ADD_TO_FACE: 'ADD_TO_FACE',
|
||
MUTE: 'MUTE',
|
||
UNMUTE: 'UNMUTE',
|
||
KICK: 'KICK',
|
||
RECALL: 'RECALL',
|
||
DELETE: 'DELETE'
|
||
} as const
|
||
type MenuKey = (typeof MENU_KEYS)[keyof typeof MENU_KEYS]
|
||
|
||
/**
|
||
* 右键菜单项:
|
||
* - 引用:已落库(id≠0)+ 未撤回的消息可引用,引用块写入 draftStore.reply
|
||
* - 撤回 / 删除:互斥;自己发送 + 已落库 + 未撤回 + 2 分钟内显示「撤回」(推服务器),其它显示「删除」(仅本地清)
|
||
*
|
||
* 好友事件气泡态不弹菜单
|
||
*/
|
||
async function handleContextMenu(e: MouseEvent) {
|
||
if (isFriendChatTipMessage.value) {
|
||
return
|
||
}
|
||
// 多选模式下不弹菜单:右键点击就当切换勾选,与单击行为一致
|
||
if (isInMultiSelect.value) {
|
||
if (canForward.value) {
|
||
multiSelect.toggle(props.message)
|
||
}
|
||
return
|
||
}
|
||
|
||
const items: Array<{
|
||
key: MenuKey
|
||
name: string
|
||
disabled?: boolean
|
||
divided?: boolean
|
||
danger?: boolean
|
||
icon?: string
|
||
}> = []
|
||
// 「复制」:仅文本消息支持;放在第一项,对齐微信桌面右键习惯
|
||
if (props.message.type === ImMessageType.TEXT) {
|
||
items.push({
|
||
key: MENU_KEYS.COPY,
|
||
name: '复制',
|
||
icon: 'ant-design:copy-outlined'
|
||
})
|
||
}
|
||
// 「引用」:已落库(id≠0)+ 未撤回 + 非合并转发;MERGE 内嵌快照在引用预览里无法降级展示
|
||
if (!!props.message.id && !isRecall.value && !isMerge.value) {
|
||
items.push({
|
||
key: MENU_KEYS.REPLY,
|
||
name: '引用',
|
||
icon: 'bxs:quote-alt-left'
|
||
})
|
||
}
|
||
// 「转发」「多选」:已落库(id≠0)+ 普通消息 + 未撤回;触发 ForwardDialog / 进入多选模式
|
||
if (canForward.value) {
|
||
items.push({
|
||
key: MENU_KEYS.FORWARD,
|
||
name: '转发',
|
||
icon: 'ant-design:share-alt-outlined'
|
||
})
|
||
items.push({
|
||
key: MENU_KEYS.MULTI_SELECT,
|
||
name: '多选',
|
||
icon: 'ant-design:check-square-outlined'
|
||
})
|
||
}
|
||
// 「置顶」:仅群聊 + 普通消息 + 已落库 + 未撤回 + 群主或管理员;已置顶不再展示,由置顶面板的「移除」入口承接
|
||
// 不本地预判上限,让后端校验,超限时通过 error toast 反馈
|
||
if (canPin.value) {
|
||
items.push({
|
||
key: MENU_KEYS.PIN,
|
||
name: '置顶',
|
||
icon: 'ant-design:pushpin-outlined'
|
||
})
|
||
}
|
||
// 「添加到表情」:FACE / IMAGE 消息 + 已落库 + 未撤回;写入个人表情包,对照微信「添加到表情」
|
||
if (canAddToFace.value) {
|
||
items.push({
|
||
key: MENU_KEYS.ADD_TO_FACE,
|
||
name: '添加到表情',
|
||
icon: 'ant-design:smile-outlined'
|
||
})
|
||
}
|
||
// 「禁言 / 解禁 / 移除」:群聊 + 非自己消息 + 我是群主或管理员
|
||
if (currentGroup.value && !props.message.selfSend && canManageSender.value) {
|
||
const senderMember = currentGroup.value.members?.find(
|
||
(m) => m.userId === props.message.senderId
|
||
)
|
||
const isMuted = senderMember?.muteEndTime && new Date(senderMember.muteEndTime) > new Date()
|
||
if (isMuted) {
|
||
items.push({
|
||
key: MENU_KEYS.UNMUTE,
|
||
name: '解除禁言',
|
||
icon: 'ant-design:audio-outlined',
|
||
divided: true
|
||
})
|
||
} else {
|
||
items.push({
|
||
key: MENU_KEYS.MUTE,
|
||
name: '禁言',
|
||
icon: 'ant-design:audio-muted-outlined',
|
||
divided: true
|
||
})
|
||
}
|
||
items.push({
|
||
key: MENU_KEYS.KICK,
|
||
name: '移除',
|
||
icon: 'ant-design:user-delete-outlined',
|
||
danger: true
|
||
})
|
||
}
|
||
// 「撤回 / 删除」二选一:
|
||
// - 自己发送 + 已落库(id≠0)+ 未撤回 + 在撤回窗口内 → 撤回(推服务器把消息态置 RECALL)
|
||
// - 其它(对方消息 / 已撤回 / 超出撤回窗口)→ 删除(仅本地清,不动后端)
|
||
// divided 把这一项和上面的「引用」隔开,danger 显红对齐微信
|
||
const canRecall =
|
||
props.message.selfSend &&
|
||
!!props.message.id &&
|
||
!isRecall.value &&
|
||
Date.now() - props.message.sendTime <= MESSAGE_RECALL_WINDOW_MS
|
||
if (canRecall) {
|
||
items.push({
|
||
key: MENU_KEYS.RECALL,
|
||
name: '撤回',
|
||
icon: 'ant-design:undo-outlined',
|
||
divided: true,
|
||
danger: true
|
||
})
|
||
} else {
|
||
items.push({
|
||
key: MENU_KEYS.DELETE,
|
||
name: '删除',
|
||
icon: 'ant-design:delete-outlined',
|
||
divided: true,
|
||
danger: true
|
||
})
|
||
}
|
||
|
||
// 把菜单渲染交给全局 uiStore(单例,避免每条消息都挂一份菜单 DOM)
|
||
const menuHandlers: Record<MenuKey, () => void | Promise<void>> = {
|
||
[MENU_KEYS.COPY]: handleCopy,
|
||
[MENU_KEYS.REPLY]: handleReply,
|
||
[MENU_KEYS.FORWARD]: handleForward,
|
||
[MENU_KEYS.MULTI_SELECT]: handleEnterMultiSelect,
|
||
[MENU_KEYS.PIN]: handlePin,
|
||
[MENU_KEYS.ADD_TO_FACE]: handleAddToFace,
|
||
[MENU_KEYS.MUTE]: handleMute,
|
||
[MENU_KEYS.UNMUTE]: handleUnmute,
|
||
[MENU_KEYS.KICK]: handleKick,
|
||
[MENU_KEYS.RECALL]: handleRecall,
|
||
[MENU_KEYS.DELETE]: handleDelete
|
||
}
|
||
uiStore.openContextMenu({ x: e.clientX, y: e.clientY }, items, async (item) => {
|
||
await menuHandlers[item.key as MenuKey]?.()
|
||
})
|
||
}
|
||
|
||
/** 是否可「添加到表情」:FACE / IMAGE 消息 + 已落库 + 未撤回(GIF / 静图都允许) */
|
||
const canAddToFace = computed(() => {
|
||
if (isRecall.value || !props.message.id) {
|
||
return false
|
||
}
|
||
return extractAddableFace(props.message) !== null
|
||
})
|
||
|
||
/** 添加到个人表情:从 message 抽 url + 尺寸 + name 写入个人表情库;幂等失败时返回 false 走 toast 兜底 */
|
||
async function handleAddToFace() {
|
||
const payload = extractAddableFace(props.message)
|
||
if (!payload) {
|
||
return
|
||
}
|
||
const data = await faceStore.addFaceUserItem(payload)
|
||
if (data) {
|
||
successMessage('已添加到表情')
|
||
}
|
||
}
|
||
|
||
/** 当前激活会话对应的群(私聊场景为 undefined) */
|
||
const currentGroup = computed(() => {
|
||
const conversation = conversationStore.activeConversation
|
||
if (!conversation || conversation.type !== ImConversationType.GROUP) {
|
||
return undefined
|
||
}
|
||
return groupStore.getGroup(conversation.targetId)
|
||
})
|
||
|
||
/** 当前用户在该群里的角色;私聊或非群成员 → undefined */
|
||
const myGroupRole = computed(() => {
|
||
const myId = Number(userStore.getUser?.id) || 0
|
||
return currentGroup.value?.members?.find((m) => m.userId === myId)?.role
|
||
})
|
||
|
||
/** 是否可管理该消息发送人:我的角色高于目标角色(群主 > 管理员 > 普通成员);目标角色未知时不展示 */
|
||
const canManageSender = computed(() => {
|
||
if (!currentGroup.value || !myGroupRole.value) {
|
||
return false
|
||
}
|
||
const senderMember = currentGroup.value.members?.find((m) => m.userId === props.message.senderId)
|
||
// 成员缓存不完整时不展示管理操作,等成员数据补齐后再判断
|
||
if (!senderMember?.role) {
|
||
return false
|
||
}
|
||
// 角色数值:1=群主 2=管理员 3=普通成员;数值越小权限越高
|
||
return myGroupRole.value < senderMember.role
|
||
})
|
||
|
||
/** 是否允许转发 / 多选:普通消息 + 已落库(id≠0)+ 未撤回;本地占位 / 撤回 / 事件消息一律不可 */
|
||
const canForward = computed(
|
||
() => isNormalMessage(props.message.type) && !!props.message.id && !isRecall.value
|
||
)
|
||
|
||
/** 多选模式是否激活;切换会话由 index.vue 主动 exitMultiSelect */
|
||
const isInMultiSelect = computed(() => multiSelect.state.active)
|
||
|
||
/** 当前消息是否已勾选 */
|
||
const isMessageChecked = computed(() => {
|
||
if (!isInMultiSelect.value) {
|
||
return false
|
||
}
|
||
return multiSelect.selectedIdSet.value.has(props.message.clientMessageId)
|
||
})
|
||
|
||
/** 多选模式下点击气泡:可转发的消息切换勾选;事件 / 撤回 / 占位等不响应(直接吃掉点击避免触发图片预览等) */
|
||
function handleMultiSelectClick(e: MouseEvent) {
|
||
if (!isInMultiSelect.value) {
|
||
return
|
||
}
|
||
e.preventDefault()
|
||
e.stopPropagation()
|
||
if (!canForward.value) {
|
||
return
|
||
}
|
||
multiSelect.toggle(props.message)
|
||
}
|
||
|
||
/** 是否允许置顶(已置顶消息不再展示菜单项,由置顶面板的「移除」入口承接):群聊 + 普通消息 + 已落库 + 未撤回 + 群主或管理员 + 未置顶 */
|
||
const canPin = computed(
|
||
() =>
|
||
!!currentGroup.value &&
|
||
isNormalMessage(props.message.type) &&
|
||
!!props.message.id &&
|
||
!isRecall.value &&
|
||
(myGroupRole.value === ImGroupMemberRole.OWNER ||
|
||
myGroupRole.value === ImGroupMemberRole.ADMIN) &&
|
||
!currentGroup.value.pinnedMessages?.some((m) => m.id === props.message.id)
|
||
)
|
||
|
||
/** 置顶消息:二次确认 → 调后端 pin-message;后端广播 GROUP_MESSAGE_PIN,本端 dispatcher 拉最新 pinnedMessages */
|
||
async function handlePin() {
|
||
const group = currentGroup.value
|
||
if (!group) {
|
||
return
|
||
}
|
||
try {
|
||
await ElMessageBox.confirm('将在当前群成员的聊天中置顶', '置顶消息', {
|
||
confirmButtonText: '置顶',
|
||
cancelButtonText: '取消',
|
||
type: 'warning'
|
||
})
|
||
await apiPinGroupMessage({ groupId: group.id, messageId: props.message.id })
|
||
successMessage('已置顶')
|
||
} catch {}
|
||
}
|
||
|
||
/** 复制文本消息:解出 content 字段写入剪贴板,提示「内容已复制到剪贴板」 */
|
||
async function handleCopy() {
|
||
const text = parseMessage<TextMessage>(props.message.content)?.content
|
||
if (!text) {
|
||
return
|
||
}
|
||
await copyToClipboard(text)
|
||
successMessage('内容已复制到剪贴板')
|
||
}
|
||
|
||
/** 进入引用模式:把当前消息构造成 QuoteMessage 写入 draftStore,MessageInput 顶部引用条响应式出现 */
|
||
function handleReply() {
|
||
const conversation = conversationStore.activeConversation
|
||
if (!conversation) {
|
||
return
|
||
}
|
||
draftStore.setReply(conversation, buildQuoteFromMessage(props.message))
|
||
}
|
||
|
||
/** 转发当前消息:打开 ForwardDialog(单条模式;mode=single 即原样转) */
|
||
function handleForward() {
|
||
const conversation = conversationStore.activeConversation
|
||
if (!conversation) {
|
||
return
|
||
}
|
||
openForwardDialog?.({
|
||
mode: ImForwardMode.SINGLE,
|
||
messages: [props.message],
|
||
sourceConversation: conversation
|
||
})
|
||
}
|
||
|
||
/** 进入多选模式,初始勾选当前消息 */
|
||
function handleEnterMultiSelect() {
|
||
multiSelect.enter(props.message)
|
||
}
|
||
|
||
/**
|
||
* 撤回消息:弹确认框 → 调 useMessageSender.recall → 后端通过 WS RECALL 事件推回,
|
||
* websocketStore 把对应 message 的 type 改成 RECALL,UI 自动切到"XX 撤回了一条消息"
|
||
*
|
||
* 不做乐观撤回:失败 / 超时 / 后端拒绝时本端状态可能与服务端漂移,统一让 WS 回推最稳
|
||
*/
|
||
async function handleRecall() {
|
||
try {
|
||
await confirmDialog('确定要撤回这条消息吗?', '撤回消息')
|
||
await recall(props.message)
|
||
} catch {}
|
||
}
|
||
|
||
/**
|
||
* 失败消息点击重试
|
||
*
|
||
* - 媒体消息(image / file / voice / video):_localFile 在内存就重走 uploadAndSendMedia(重新上传 + 占位 + 进度)
|
||
* - 文本消息:移除 FAILED 占位 + 用原 content 走一遍 sendRaw 新建占位
|
||
*
|
||
* 媒体类型若 _localFile 已丢(理论上 IDB 恢复阶段就被 drop,进不到这里;保险起见仍走文本兜底)则按 sendRaw 重发,
|
||
* 后端拒绝失效 blob URL 时再次 FAILED,用户可右键删除
|
||
*
|
||
* 不还原原 receipt:群回执是发送时的扩展选项、不会持久化到 message,强行猜测可能与原意不符;
|
||
* 默认按"无回执"重发,绝大多数场景符合预期,要回执就重新发一次更直观
|
||
*/
|
||
async function handleResend() {
|
||
if (props.message.status !== ImMessageStatus.FAILED) {
|
||
return
|
||
}
|
||
const conversation = conversationStore.activeConversation
|
||
if (!conversation) {
|
||
return
|
||
}
|
||
// 禁言 / 封禁时拦截:避免重试绕过 MessageInput 的 muteOverlay 又走一次 sendRaw 让后端拒
|
||
if (muteOverlay.value) {
|
||
return
|
||
}
|
||
const message = props.message
|
||
const file = message._localFile
|
||
|
||
// 媒体类型 + _localFile 在 → 重走 uploadAndSendMedia;type 分发 + 旧元数据复用统一在 mediaTypeHandlers 表里
|
||
if (isMediaMessageType(message.type) && file) {
|
||
const handler = mediaTypeHandlers[message.type]
|
||
if (handler) {
|
||
const oldQuote = getQuoteFromMessage(message.content) ?? undefined
|
||
const context = handler.extractResendContext(message.content)
|
||
conversationStore.removeMessage(conversation.type, conversation.targetId, {
|
||
id: message.id,
|
||
clientMessageId: message.clientMessageId
|
||
})
|
||
await uploadAndSendMedia({
|
||
file,
|
||
type: message.type,
|
||
quote: oldQuote,
|
||
conversation,
|
||
context
|
||
})
|
||
return
|
||
}
|
||
}
|
||
|
||
// 文本类型 / 媒体类型但 _localFile 已丢:原 content 走 sendRaw 重发
|
||
conversationStore.removeMessage(conversation.type, conversation.targetId, {
|
||
id: message.id,
|
||
clientMessageId: message.clientMessageId
|
||
})
|
||
await sendRaw(message.type, message.content, {
|
||
atUserIds: message.atUserIds
|
||
})
|
||
}
|
||
|
||
/**
|
||
* 删除消息:本地软删,仅从 conversationStore.messages 移除,不调后端
|
||
* 区别于"撤回":服务端没动,多端登录时其它客户端 / 群里其他人依然能看到这条
|
||
*/
|
||
/** 禁言:emit 给父组件打开时长选择弹窗(避免 MessageItem 过重) */
|
||
function handleMute() {
|
||
const group = currentGroup.value
|
||
if (!group) {
|
||
return
|
||
}
|
||
const name = senderDisplayName.value || '该成员'
|
||
emit('mute', group.id, props.message.senderId, name)
|
||
}
|
||
|
||
/** 解除禁言 */
|
||
async function handleUnmute() {
|
||
const group = currentGroup.value
|
||
if (!group) {
|
||
return
|
||
}
|
||
try {
|
||
await confirmDialog('确定解除该成员的禁言吗?', '解除禁言')
|
||
await cancelMuteMember({ groupId: group.id, userId: props.message.senderId })
|
||
successMessage('已解除禁言')
|
||
emit('reload')
|
||
} catch {}
|
||
}
|
||
|
||
/** 移除群成员 */
|
||
async function handleKick() {
|
||
const group = currentGroup.value
|
||
if (!group) {
|
||
return
|
||
}
|
||
const name = senderDisplayName.value || '该成员'
|
||
try {
|
||
await confirmDialog(`确定将「${name}」移出群聊吗?`, '移除成员')
|
||
await removeGroupMember({ groupId: group.id, memberUserIds: [props.message.senderId] })
|
||
successMessage('已移除')
|
||
emit('reload')
|
||
} catch {}
|
||
}
|
||
|
||
function handleDelete() {
|
||
const conversation = conversationStore.activeConversation
|
||
if (!conversation) {
|
||
return
|
||
}
|
||
conversationStore.removeMessage(conversation.type, conversation.targetId, {
|
||
id: props.message.id,
|
||
clientMessageId: props.message.clientMessageId
|
||
})
|
||
}
|
||
</script>
|
||
|
||
<style scoped>
|
||
/* SENDING 状态的转圈动画 */
|
||
.im-loading-spin {
|
||
animation: im-loading-spin 1s linear infinite;
|
||
}
|
||
@keyframes im-loading-spin {
|
||
to {
|
||
transform: rotate(360deg);
|
||
}
|
||
}
|
||
</style>
|