123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254 |
- <template>
- <el-aside class="kefu pt-5px h-100%" width="260px">
- <div class="color-[#999] font-bold my-10px">
- 会话记录({{ kefuStore.getConversationList.length }})
- </div>
- <div
- v-for="item in kefuStore.getConversationList"
- :key="item.id"
- :class="{ active: item.id === activeConversationId, pinned: item.adminPinned }"
- class="kefu-conversation px-10px flex items-center"
- @click="openRightMessage(item)"
- @contextmenu.prevent="rightClick($event as PointerEvent, item)"
- >
- <div class="flex justify-center items-center w-100%">
- <div class="flex justify-center items-center w-50px h-50px">
- <!-- 头像 + 未读 -->
- <el-badge
- :hidden="item.adminUnreadMessageCount === 0"
- :max="99"
- :value="item.adminUnreadMessageCount"
- >
- <el-avatar :src="item.userAvatar" alt="avatar" />
- </el-badge>
- </div>
- <div class="ml-10px w-100%">
- <div class="flex justify-between items-center w-100%">
- <span class="username">{{ item.userNickname }}</span>
- <span class="color-[#999]" style="font-size: 13px">
- {{ lastMessageTimeMap.get(item.id) ?? '计算中' }}
- </span>
- </div>
- <!-- 最后聊天内容 -->
- <div
- v-dompurify-html="
- getConversationDisplayText(item.lastMessageContentType, item.lastMessageContent)
- "
- class="last-message flex items-center color-[#999]"
- >
- </div>
- </div>
- </div>
- </div>
- <!-- 右键,进行操作(类似微信) -->
- <ul v-show="showRightMenu" :style="rightMenuStyle" class="right-menu-ul">
- <li
- v-show="!rightClickConversation.adminPinned"
- class="flex items-center"
- @click.stop="updateConversationPinned(true)"
- >
- <Icon class="mr-5px" icon="ep:top" />
- 置顶会话
- </li>
- <li
- v-show="rightClickConversation.adminPinned"
- class="flex items-center"
- @click.stop="updateConversationPinned(false)"
- >
- <Icon class="mr-5px" icon="ep:bottom" />
- 取消置顶
- </li>
- <li class="flex items-center" @click.stop="deleteConversation">
- <Icon class="mr-5px" color="red" icon="ep:delete" />
- 删除会话
- </li>
- <li class="flex items-center" @click.stop="closeRightMenu">
- <Icon class="mr-5px" color="red" icon="ep:close" />
- 取消
- </li>
- </ul>
- </el-aside>
- </template>
- <script lang="ts" setup>
- import { KeFuConversationApi, KeFuConversationRespVO } from '@/api/mall/promotion/kefu/conversation'
- import { useEmoji } from './tools/emoji'
- import { formatPast } from '@/utils/formatTime'
- import { KeFuMessageContentTypeEnum } from './tools/constants'
- import { useAppStore } from '@/store/modules/app'
- import { useMallKefuStore } from '@/store/modules/mall/kefu'
- import { jsonParse } from '@/utils'
- defineOptions({ name: 'KeFuConversationList' })
- const message = useMessage() // 消息弹窗
- const appStore = useAppStore()
- const kefuStore = useMallKefuStore() // 客服缓存
- const { replaceEmoji } = useEmoji()
- const activeConversationId = ref(-1) // 选中的会话
- const collapse = computed(() => appStore.getCollapse) // 折叠菜单
- /** 计算消息最后发送时间距离现在过去了多久 */
- const lastMessageTimeMap = ref<Map<number, string>>(new Map<number, string>())
- const calculationLastMessageTime = () => {
- kefuStore.getConversationList?.forEach((item) => {
- lastMessageTimeMap.value.set(item.id, formatPast(item.lastMessageTime, 'YYYY-MM-DD'))
- })
- }
- defineExpose({ calculationLastMessageTime })
- /** 打开右侧的消息列表 */
- const emits = defineEmits<{
- (e: 'change', v: KeFuConversationRespVO): void
- }>()
- const openRightMessage = (item: KeFuConversationRespVO) => {
- // 同一个会话则不处理
- if (activeConversationId.value === item.id) {
- return
- }
- activeConversationId.value = item.id
- emits('change', item)
- }
- /** 获得消息类型 */
- const getConversationDisplayText = computed(
- () => (lastMessageContentType: number, lastMessageContent: string) => {
- switch (lastMessageContentType) {
- case KeFuMessageContentTypeEnum.SYSTEM:
- return '[系统消息]'
- case KeFuMessageContentTypeEnum.VIDEO:
- return '[视频消息]'
- case KeFuMessageContentTypeEnum.IMAGE:
- return '[图片消息]'
- case KeFuMessageContentTypeEnum.PRODUCT:
- return '[产品消息]'
- case KeFuMessageContentTypeEnum.ORDER:
- return '[订单消息]'
- case KeFuMessageContentTypeEnum.VOICE:
- return '[语音消息]'
- case KeFuMessageContentTypeEnum.TEXT:
- return replaceEmoji(jsonParse(lastMessageContent).text || lastMessageContent)
- default:
- return ''
- }
- }
- )
- //======================= 右键菜单 =======================
- const showRightMenu = ref(false) // 显示右键菜单
- const rightMenuStyle = ref<any>({}) // 右键菜单 Style
- const rightClickConversation = ref<KeFuConversationRespVO>({} as KeFuConversationRespVO) // 右键选中的会话对象
- /** 打开右键菜单 */
- const rightClick = (mouseEvent: PointerEvent, item: KeFuConversationRespVO) => {
- rightClickConversation.value = item
- // 显示右键菜单
- showRightMenu.value = true
- rightMenuStyle.value = {
- top: mouseEvent.clientY - 110 + 'px',
- left: collapse.value ? mouseEvent.clientX - 80 + 'px' : mouseEvent.clientX - 210 + 'px'
- }
- }
- /** 关闭右键菜单 */
- const closeRightMenu = () => {
- showRightMenu.value = false
- }
- /** 置顶会话 */
- const updateConversationPinned = async (adminPinned: boolean) => {
- // 1. 会话置顶/取消置顶
- await KeFuConversationApi.updateConversationPinned({
- id: rightClickConversation.value.id,
- adminPinned
- })
- message.notifySuccess(adminPinned ? '置顶成功' : '取消置顶成功')
- // 2. 关闭右键菜单,更新会话列表
- closeRightMenu()
- await kefuStore.updateConversation(rightClickConversation.value.id)
- }
- /** 删除会话 */
- const deleteConversation = async () => {
- // 1. 删除会话
- await message.confirm('您确定要删除该会话吗?')
- await KeFuConversationApi.deleteConversation(rightClickConversation.value.id)
- // 2. 关闭右键菜单,更新会话列表
- closeRightMenu()
- kefuStore.deleteConversation(rightClickConversation.value.id)
- }
- /** 监听右键菜单的显示状态,添加点击事件监听器 */
- watch(showRightMenu, (val) => {
- if (val) {
- document.body.addEventListener('click', closeRightMenu)
- } else {
- document.body.removeEventListener('click', closeRightMenu)
- }
- })
- const timer = ref<any>()
- /** 初始化 */
- onMounted(() => {
- timer.value = setInterval(calculationLastMessageTime, 1000 * 10) // 十秒计算一次
- })
- /** 组件卸载前 */
- onBeforeUnmount(() => {
- clearInterval(timer.value)
- })
- </script>
- <style lang="scss" scoped>
- .kefu {
- background-color: var(--app-content-bg-color);
- &-conversation {
- height: 60px;
- //background-color: #fff;
- //transition: border-left 0.05s ease-in-out; /* 设置过渡效果 */
- .username {
- min-width: 0;
- max-width: 60%;
- }
- .last-message {
- font-size: 13px;
- }
- .last-message,
- .username {
- overflow: hidden;
- text-overflow: ellipsis;
- display: -webkit-box;
- -webkit-box-orient: vertical;
- -webkit-line-clamp: 1;
- }
- }
- .active {
- background-color: rgba(128, 128, 128, 0.5); // 透明色,暗黑模式下也能体现
- }
- .right-menu-ul {
- position: absolute;
- background-color: var(--app-content-bg-color);
- padding: 5px;
- margin: 0;
- list-style-type: none; /* 移除默认的项目符号 */
- border-radius: 12px;
- box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1); /* 阴影效果 */
- width: 130px;
- li {
- padding: 8px 16px;
- cursor: pointer;
- border-radius: 12px;
- transition: background-color 0.3s; /* 平滑过渡 */
- &:hover {
- background-color: var(--left-menu-bg-active-color); /* 悬停时的背景颜色 */
- }
- }
- }
- }
- </style>
|