Files
open-anylink-web/src/views/message/MessageLayout.vue

546 lines
15 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<!-- eslint-disable prettier/prettier -->
<script setup>
import { ref, onMounted, computed, watch, nextTick } from 'vue'
import {
Phone,
VideoCamera,
Position,
CirclePlus,
Setting,
LocationInformation,
Clock,
Picture,
FolderAdd,
CreditCard,
PictureRounded
} from '@element-plus/icons-vue'
import DragLine from '@/components/common/DragLine.vue'
import SearchBox from '@/components/common/SearchBox.vue'
import AddBotton from '@/components/common/AddBotton.vue'
import SessionBox from '@/components/message/SessionBox.vue'
import InputTool from '@/components/message/InputTool.vue'
import InputEditor from '@/components/message/InputEditor.vue'
import MessageItem from '@/components/message/MessageItem.vue'
import { userStore, settingStore, messageStore } from '@/stores'
import backgroupImage from '@/assets/messagebx_bg.webp'
import { msgChatSessionListService, msgChatPullMsgService } from '@/api/message'
import { MsgType } from '@/proto/msg'
import wsConnect from '@/js/websocket/wsConnect'
const userData = userStore()
const settingData = settingStore()
const messageData = messageStore()
const asideWidth = ref(0)
const asideWidthMin = 200
const asideWidthMax = 500
const inputBoxHeight = ref(0)
const inputBoxHeightMin = 200
const inputBoxHeightMax = 500
const isShowTopLoading = ref(true)
const isTopLoading = ref(false)
const loadMoreTips = ref('查看更多消息')
const loadCursor = computed(() => {
return isTopLoading.value ? 'auto' : 'pointer'
})
const choosedSession = computed(() => {
if (userData.curSessionId)
return messageData.sessionList[userData.curSessionId]
else
return {}
})
const msgRecords = computed(() => {
if (userData.curSessionId) {
return messageData.msgRecordsList[userData.curSessionId]
}
else {
return []
}
})
const msgListDiv = ref()
onMounted(async () => {
asideWidth.value = settingData.sessionListDrag[userData.user.account] || 300
inputBoxHeight.value = settingData.inputBoxDrag[userData.user.account] || 300
const res = await msgChatSessionListService()
messageData.setSessionList(res.data.data) //入缓存
if (userData.curSessionId) {
pullMsg()
}
if (msgListDiv.value) msgListDiv.value.scrollTop = msgListDiv.value.scrollHeight
})
// 把sessionList转成数组并按照lastMsgTime排序
const sessionListSorted = computed(() => {
if (!messageData.sessionList) {
return []
}
else {
let sessionArr = Object.values(messageData.sessionList)
return sessionArr.sort((a, b) => b.lastMsgTime - a.lastMsgTime)
}
})
const showName = computed(() => {
switch (choosedSession.value?.sessionType) {
case MsgType.CHAT:
return choosedSession.value.objectInfo.nickName
case MsgType.GROUP_CHAT:
return choosedSession.value.objectInfo.groupName
default:
return ''
}
})
const showId = computed(() => {
switch (choosedSession.value?.sessionType) {
case MsgType.CHAT:
return choosedSession.value.objectInfo.account
case MsgType.GROUP_CHAT:
return choosedSession.value.objectInfo.groupId
default:
return ''
}
})
const getLastMsgTime = (index) => {
if (index > 0) {
return msgRecords.value[index - 1].msgTime;
} else {
return null;
}
}
const onAsideDragUpdate = ({ width }) => {
asideWidth.value = width
settingData.setSessionListDrag({
...settingData.sessionListDrag,
[userData.user.account]: width
})
}
const onInputBoxDragUpdate = ({ height }) => {
inputBoxHeight.value = height
msgListReachBottom()
settingData.setInputBoxDrag({
...settingData.inputBoxDrag,
[userData.user.account]: height
})
}
const pullMsg = () => {
// 1.如果session中存在未读信息需要从服务器拉取否则就使用本地缓存的消息避免频繁查询服务器
// 2.如果这个会话没有缓存过消息,需要从服务器拉取一定数量的历史消息
if (!msgRecords.value || choosedSession.value.unreadCount > 0) {
msgChatPullMsgService({
sessionId: userData.curSessionId,
readMsgId: choosedSession.value.readMsgId,
readTime: choosedSession.value.readTime,
pageSize: 20
})
.then((res) => {
messageData.addMsgRecords(userData.curSessionId, res.data.data.msgList)
const msgCount = res.data.data.msgList.length
messageData.updateSession({
sessionId: userData.curSessionId,
readMsgId: res.data.data.lastMsgId,
readTime: new Date(),
lastMsgId: res.data.data.lastMsgId,
lastMsgContent: res.data.data.msgList[msgCount - 1].content,
lastMsgTime: res.data.data.msgList[msgCount - 1].msgTime,
unreadCount: 0
})
})
}
}
// 表示有个session被选中了
const handleIsChoosed = (exportSession) => {
if (userData.curSessionId !== exportSession.sessionId) {
userData.setCurSessionId(exportSession.sessionId)
}
else {
// TODO 这个是为了临时消除接收端当前session下出现的未读图标,后面要通过""已读消息"来消除的
messageData.updateSession({
sessionId: userData.curSessionId,
readMsgId: choosedSession.value.lastMsgId,
readTime: new Date(),
unreadCount: 0
})
}
pullMsg()
}
const handleSwitchTag = (obj) => {
messageData.updateSession(obj)
}
const handleExportContent = (content) => {
// TODO 这里还要考虑失败情况1消息发不出去2消息发出去了服务器不发“已发送”
wsConnect.sendMsg(showId.value, choosedSession.value.sessionType, content, (deliveredMsg) => {
const now = new Date()
messageData.updateSession({
sessionId: userData.curSessionId,
readMsgId: deliveredMsg.body.msgId, // 发消息视为已经读到最后一条消息(自己发的)
readTime: now,
lastMsgId: deliveredMsg.body.msgId, // lastMsgId = 最后一条消息(自己发的)
lastMsgContent: content,
lastMsgTime: now,
unreadCount: 0, // readMsgId = lastMsgId = 最后一条消息自己发的因此未读是0
draft: '' //草稿意味着要清空
})
// 如果当前sessionid和这个“已发送”消息的sessionId更新到msgRecords中
if (userData.curSessionId === deliveredMsg.body.sessionId) {
messageData.addMsgRecords(userData.curSessionId, [{
sessionId: userData.curSessionId,
msgId: deliveredMsg.body.msgId,
fromId: userData.user.account,
msgType: choosedSession.value.sessionType,
content: content,
msgTime: now
}])
}
})
}
const onLoadMore = () => {
isTopLoading.value = true
loadMoreTips.value = ''
}
watch(msgRecords, () => {
msgListReachBottom()
}, {deep: true})
const msgListReachBottom = () => {
nextTick(() => {
msgListDiv.value.scrollTo({
top: msgListDiv.value.scrollHeight,
behavior: 'smooth'
})
})
}
</script>
<template>
<el-container class="msg-container-hole">
<el-aside class="msg-aside bdr-r" :style="{ width: asideWidth + 'px' }">
<div class="msg-aside-main">
<div class="header">
<SearchBox></SearchBox>
<AddBotton></AddBotton>
</div>
<div class="session-list my-scrollbar">
<SessionBox
v-for="item in sessionListSorted"
:key="item.sessionId"
:sessionId="item.sessionId"
@isChoosed="handleIsChoosed"
@switchTag="handleSwitchTag"
></SessionBox>
</div>
</div>
<DragLine
direction="right"
:min="asideWidthMin"
:max="asideWidthMax"
:origin-size="asideWidth"
@drag-update="onAsideDragUpdate"
></DragLine>
</el-aside>
<el-main class="msg-box">
<el-image
class="backgroup-image"
v-if="!userData.curSessionId"
:src="backgroupImage"
fit="cover"
></el-image>
<el-container v-else class="container">
<el-header class="header bdr-b">
<div class="show-name-id">
<span class="show-name">{{ showName }}</span>
<span v-if="choosedSession?.sessionType === MsgType.CHAT" class="show-id">{{
showId
}}</span>
</div>
<div class="action-set">
<el-button class="action-button" :icon="Phone" circle />
<el-button class="action-button" :icon="VideoCamera" circle />
<el-button class="action-button" :icon="Position" circle />
<el-button class="action-button" :icon="CirclePlus" circle />
<el-button class="action-button" :icon="Setting" circle />
</div>
</el-header>
<el-main class="body">
<div v-if="isShowTopLoading" class="top-loading">
<div
v-loading="isTopLoading"
:fullscreen="false"
class="loading-box"
@click="onLoadMore"
:style="{ cursor: loadCursor }"
>
{{ loadMoreTips }}
</div>
</div>
<div class="show-box my-scrollbar" ref="msgListDiv">
<div class="message-main">
<span class="no-more-message">当前无更多消息</span>
<MessageItem
v-for="(item, index) in msgRecords"
:key="index"
:msg="item"
:obj="choosedSession?.objectInfo"
:lastMsgTime="getLastMsgTime(index)"
></MessageItem>
</div>
</div>
<div class="input-box bdr-t" :style="{ height: inputBoxHeight + 'px' }">
<el-container class="input-box-container">
<el-header class="input-box-header">
<DragLine
direction="top"
:min="inputBoxHeightMin"
:max="inputBoxHeightMax"
:origin-size="inputBoxHeight"
@drag-update="onInputBoxDragUpdate"
></DragLine>
<div class="tool-set">
<div class="left-tools">
<InputTool tips="表情">
<template #iconSlot>
<PictureRounded />
</template>
</InputTool>
<InputTool tips="图片">
<template #iconSlot>
<Picture />
</template>
</InputTool>
<InputTool tips="文件">
<template #iconSlot>
<FolderAdd />
</template>
</InputTool>
<InputTool tips="代码">
<template #iconSlot>
<CreditCard />
</template>
</InputTool>
<InputTool tips="位置">
<template #iconSlot>
<LocationInformation />
</template>
</InputTool>
</div>
<div class="right-tools">
<InputTool tips="历史记录">
<template #iconSlot>
<Clock />
</template>
</InputTool>
</div>
</div>
</el-header>
<el-main class="input-box-main">
<InputEditor
:draft="choosedSession?.draft"
@exportContent="handleExportContent"
></InputEditor>
</el-main>
</el-container>
</div>
</el-main>
</el-container>
</el-main>
</el-container>
</template>
<style lang="scss" scoped>
.msg-container-hole {
height: 100%;
user-select: none;
.msg-aside {
height: 100%;
position: relative;
.msg-aside-main {
width: 100%;
height: 100%;
display: flex; // 需要flex布局否则session-list的滚动条会有问题
flex-direction: column;
overflow: hidden; // 禁用它的滚动条
.header {
margin-top: 10px;
margin-bottom: 10px;
display: flex;
}
.session-list {
width: 100%;
overflow-y: scroll; // 用它的滚动条
}
}
}
.msg-box {
padding: 0;
overflow: hidden; // 禁用它的滚动条
.backgroup-image {
width: 100%;
height: 100%;
}
.container {
width: 100%;
height: 100%;
.header {
width: 100%;
height: 50px;
display: flex;
align-items: center;
justify-content: space-between;
.show-name-id {
display: flex;
align-items: center;
user-select: text;
.show-name {
font-size: 16px;
font-weight: bold;
}
.show-id {
margin-left: 10px;
font-size: 14px;
color: gray;
}
}
.action-set {
margin-right: 20px;
.action-button {
border: 0;
}
}
}
.body {
width: 100%;
height: 100%;
padding: 0;
display: flex;
flex-direction: column;
overflow: hidden; // 禁用它的滚动条
position: relative;
.top-loading {
width: 100%;
height: 30px;
position: absolute;
top: 0;
display: flex;
justify-content: center;
align-items: center;
.loading-box {
color: #409eff;
font-size: 14px;
}
:deep(.circular) {
width: 24px;
height: 24px;
position: absolute;
top: 12px;
left: -12px;
}
}
.show-box {
width: 100%;
display: flex;
flex: 1;
overflow-y: scroll; // 用它的滚动条
.message-main {
width: 100%;
height: 100%;
padding: 20px;
padding-right: 15px;
.no-more-message {
width: 100%;
height: 40px;
display: flex;
justify-content: center;
align-items: center;
font-size: 14px;
color: gray;
user-select: text;
}
}
}
.input-box {
width: 100%;
display: flex;
position: relative;
.input-box-header {
width: 100%;
height: auto;
padding: 0;
display: flex;
flex-direction: column;
.tool-set {
display: flex;
justify-content: space-between;
.left-tools {
display: flex;
}
.right-tools {
margin-right: 10px;
}
}
}
.input-box-main {
width: 100%;
padding: 0;
}
}
}
}
}
}
</style>