【代码优化】客服聊天

pull/146/head
puhui999 2025-05-08 16:18:27 +08:00
parent 7e6a3bbad3
commit c149a54e83
4 changed files with 243 additions and 212 deletions

View File

@ -7,7 +7,11 @@
:clearable="false" :clearable="false"
v-model="message" v-model="message"
placeholder="请输入你要咨询的问题" placeholder="请输入你要咨询的问题"
:maxlength="maxLength"
:focus="autoFocus"
@focus="handleFocus"
></uni-easyinput> ></uni-easyinput>
<text v-if="showCharCount" class="char-count">{{ message.length }}/{{ maxLength }}</text>
</view> </view>
<text class="sicon-basic bq" @tap.stop="onTools('emoji')"></text> <text class="sicon-basic bq" @tap.stop="onTools('emoji')"></text>
<text <text
@ -16,14 +20,21 @@
:class="{ 'is-active': toolsMode === 'tools' }" :class="{ 'is-active': toolsMode === 'tools' }"
@tap.stop="onTools('tools')" @tap.stop="onTools('tools')"
></text> ></text>
<button v-if="message" class="ss-reset-button send-btn" @tap="sendMessage"> <button
发送 v-if="message"
class="ss-reset-button send-btn"
@tap="sendMessage"
:disabled="isDisabled || sending"
:class="{ 'disabled': isDisabled || sending }"
>
<text v-if="sending"></text>
<text v-else></text>
</button> </button>
</view> </view>
</template> </template>
<script setup> <script setup>
import { computed } from 'vue'; import { computed, ref, onUnmounted } from 'vue';
/** /**
* 消息发送组件 * 消息发送组件
*/ */
@ -38,8 +49,25 @@
type: String, type: String,
default: '', default: '',
}, },
//
autoFocus: {
type: Boolean,
default: false
},
//
maxLength: {
type: Number,
default: 500
},
//
showCharCount: {
type: Boolean,
default: true
}
}); });
const emits = defineEmits(['update:modelValue', 'onTools', 'sendMessage']); const emits = defineEmits(['update:modelValue', 'onTools', 'sendMessage']);
const message = computed({ const message = computed({
get() { get() {
return props.modelValue; return props.modelValue;
@ -49,16 +77,55 @@
} }
}); });
//
const sending = ref(false);
//
const isDisabled = computed(() => {
return !message.value.trim() || message.value.length > props.maxLength;
});
//
const handleFocus = () => {
//
if (props.toolsMode !== '') {
onTools('');
}
};
// //
function onTools(mode) { function onTools(mode) {
emits('onTools', mode); emits('onTools', mode);
} }
//
let sendTimer = null;
// //
function sendMessage() { function sendMessage() {
emits('sendMessage'); //
if (sending.value || isDisabled.value) return;
//
if (sendTimer) clearTimeout(sendTimer);
//
sending.value = true;
//
sendTimer = setTimeout(() => {
emits('sendMessage');
//
setTimeout(() => {
sending.value = false;
}, 300);
}, 300);
} }
//
onUnmounted(() => {
if (sendTimer) clearTimeout(sendTimer);
});
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@ -70,6 +137,16 @@
height: 64rpx; height: 64rpx;
border-radius: 32rpx; border-radius: 32rpx;
background: var(--ui-BG-1); background: var(--ui-BG-1);
position: relative;
.char-count {
position: absolute;
right: 15rpx;
top: 50%;
transform: translateY(-50%);
font-size: 22rpx;
color: #999;
}
} }
.bq { .bq {
@ -97,6 +174,12 @@
font-size: 26rpx; font-size: 26rpx;
color: #fff; color: #fff;
margin-left: 11rpx; margin-left: 11rpx;
transition: all 0.3s;
&.disabled {
opacity: 0.6;
background: #cccccc;
}
} }
} }
</style> </style>

View File

@ -1,240 +1,169 @@
<template> <template>
<!-- 聊天列表使用scroll-view原生组件 --> <!-- 聊天列表使用scroll-view原生组件整体倒置 -->
<scroll-view <scroll-view ref="scrollRef" class="chat-scroll-view" scroll-y :refresher-enabled="false"
ref="scrollRef" @scroll="onScroll" style="transform: scaleY(-1);">
class="chat-scroll-view" <!-- 消息列表容器 -->
scroll-y <view class="message-container">
:scroll-top="scrollTop" <!-- 消息列表 -->
:refresher-enabled="true" <view class="message-list">
:refresher-triggered="isTriggered" <view v-for="(item, index) in messageList" :key="item.id" class="message-item"
@scrolltolower="onScrollToLower" style="transform: scaleY(-1);">
@refresherrefresh="onRefresh" <!-- 消息渲染 -->
@scroll="onScroll" <MessageListItem :message="item" :message-index="index"
> :message-list="messageList"></MessageListItem>
<!-- 撑一下顶部导航 --> </view>
<view :style="{ height: sys_navBar + 'px' }"></view>
<!-- 消息列表 -->
<view class="message-list" :style="{ transform: 'scaleY(-1)' }">
<view v-for="(item, index) in messageList" :key="item.id" style="transform: scaleY(-1)">
<!-- 消息渲染 -->
<MessageListItem
:message="item"
:message-index="index"
:message-list="messageList"
></MessageListItem>
</view> </view>
<!-- 加载更多提示 -->
<view v-if="loadingMore" class="loading-more">...</view>
</view>
<!-- 底部聊天输入框 -->
<su-fixed bottom>
<slot name="bottom"></slot>
</su-fixed>
<!-- 查看最新消息提示 -->
<view v-if="showNewMessageTip" class="new-message-tip" @click="scrollToBottom">
<text>有新消息</text>
</view> </view>
</scroll-view> </scroll-view>
<!-- 底部聊天输入框 -->
<su-fixed bottom>
<view v-if="showNewMessageTip" :style="backToTopStyle">
<text>有新消息</text>
</view>
<slot name="bottom"></slot>
</su-fixed>
</template> </template>
<script setup> <script setup>
import MessageListItem from '@/pages/chat/components/messageListItem.vue'; import MessageListItem from '@/pages/chat/components/messageListItem.vue';
import { reactive, ref, nextTick, watch } from 'vue'; import { onMounted, reactive, ref } from 'vue';
import KeFuApi from '@/sheep/api/promotion/kefu'; import KeFuApi from '@/sheep/api/promotion/kefu';
import { isEmpty } from '@/sheep/helper/utils'; import { isEmpty } from '@/sheep/helper/utils';
import sheep from '@/sheep';
import { formatDate } from '@/sheep/util'; import { formatDate } from '@/sheep/util';
const sys_navBar = sheep.$platform.navbar;
const scrollRef = ref(null); // scroll-view
const messageList = ref([]); // const messageList = ref([]); //
const showNewMessageTip = ref(false); // const showNewMessageTip = ref(false); //
const isRefreshing = ref(false); // const refreshMessage = ref(false); //
const isTriggered = ref(false); //
const loadingMore = ref(false); //
const scrollTop = ref(0); //
const lastScrollTop = ref(0); //
const hasReachedBottom = ref(true); //
const forceUpdate = ref(0); //
const queryParams = reactive({ const queryParams = reactive({
no: 1, // no: 1, //
limit: 20, limit: 20,
createTime: undefined, createTime: undefined,
}); });
const backToTopStyle = reactive({
// height: '30px',
watch(messageList, (newVal, oldVal) => { width: '100px',
// ( ) 'background-color': '#fff',
if (newVal.length > oldVal.length && (hasReachedBottom.value || oldVal.length === 0)) { 'border-radius': '30px',
nextTick(() => { 'box-shadow': '0 2px 4px rgba(0, 0, 0, 0.1)',
scrollToBottom(true); display: 'flex',
}); justifyContent: 'center',
} alignItems: 'center',
}); }); //
// const pagingRef = ref(null); //
// const queryList = async (no, limit) => {
const getMessageList = async (isRefresh = false) => { //
if (isRefresh) { queryParams.no = no;
queryParams.createTime = undefined; queryParams.limit = limit;
queryParams.no = 1;
}
loadingMore.value = true;
try {
const { data } = await KeFuApi.getKefuMessageList(queryParams);
if (isEmpty(data)) {
loadingMore.value = false;
return;
}
if (isRefresh) {
messageList.value = data;
} else {
messageList.value = [...messageList.value, ...data];
}
if (data.slice(-1).length > 0) {
// createTime
queryParams.createTime = formatDate(data.slice(-1)[0].createTime);
}
//
if (isRefresh) {
setTimeout(() => {
scrollToBottom(true);
}, 300);
}
} catch (error) {
console.error('获取消息列表失败:', error);
} finally {
loadingMore.value = false;
isTriggered.value = false;
}
};
//
const onRefresh = async () => {
isTriggered.value = true;
isRefreshing.value = true;
await getMessageList(true);
isRefreshing.value = false;
setTimeout(() => {
isTriggered.value = false;
}, 500);
};
//
const onScrollToLower = async () => {
if (loadingMore.value) return;
queryParams.no++;
await getMessageList(); await getMessageList();
}; };
// //
const onScroll = (e) => { const getMessageList = async () => {
lastScrollTop.value = e.detail.scrollTop; const { data } = await KeFuApi.getKefuMessageList(queryParams);
if (isEmpty(data)) {
// - // pagingRef.value.completeByNoMore([], true);
const scrollHeight = e.detail.scrollHeight; return;
const scrollTop = e.detail.scrollTop;
const clientHeight = e.detail.scrollHeight - scrollHeight; //
//
hasReachedBottom.value = scrollTop <= 10;
//
if (!hasReachedBottom.value) {
showNewMessageTip.value = true;
} else {
showNewMessageTip.value = false;
} }
if (queryParams.no > 1 && refreshMessage.value) {
const newMessageList = [];
for (const message of data) {
if (messageList.value.some((val) => val.id === message.id)) {
continue;
}
newMessageList.push(message);
}
//
messageList.value = [...newMessageList, ...messageList.value];
// pagingRef.value.updateCache(); //
refreshMessage.value = false; //
return;
}
if (data.slice(-1).length > 0) {
// createTime
queryParams.createTime = formatDate(data.slice(-1)[0].createTime);
}
messageList.value = data;
// pagingRef.value.completeByNoMore(data, false);
}; };
// /** 刷新消息列表 */
const scrollToBottom = (force = false) => {
//
if (force) {
// scrollTop
// 使
forceUpdate.value++;
setTimeout(() => {
scrollTop.value = 0;
}, 50);
} else {
scrollTop.value = 0;
}
showNewMessageTip.value = false;
hasReachedBottom.value = true;
};
// -
const refreshMessageList = async (message = undefined) => { const refreshMessageList = async (message = undefined) => {
if (typeof message !== 'undefined') { if (typeof message !== 'undefined') {
// //
messageList.value = [message, ...messageList.value]; messageList.value.map(message);
// pagingRef.value.addChatRecordData([message], false);
//
if (hasReachedBottom.value) {
// DOM
nextTick(() => {
scrollToBottom(true);
});
} else {
showNewMessageTip.value = true;
}
} else { } else {
// queryParams.createTime = undefined;
await getMessageList(true); refreshMessage.value = true;
await getMessageList();
}
//
if (queryParams.no > 1) {
showNewMessageTip.value = true;
} else {
onScrollToUpper();
} }
}; };
// /** 滚动到最新消息 */
getMessageList(true); const onBackToTopClick = (event) => {
event(false); //
// pagingRef.value.scrollToBottom();
};
// /** 监听滚动到底部事件(因为 scroll 翻转了顶就是底) */
defineExpose({ const onScrollToUpper = () => {
getMessageList, //
refreshMessageList if (queryParams.no === 1) {
return;
}
showNewMessageTip.value = false;
};
defineExpose({ getMessageList, refreshMessageList });
/** 监听消息列表滚动 */
const onScroll = (e) => {
const { scrollTop } = e.detail;
console.log('scrollTop', scrollTop);
};
onMounted(() => {
getMessageList();
}); });
</script> </script>
<style> <style>
.chat-scroll-view { .chat-scroll-view {
height: calc(100vh - 100px); /* 减去底部输入框的高度 */ height: calc(100vh - 100px);
/* 减去底部输入框的高度 */
width: 100%; width: 100%;
position: relative; position: relative;
background-color: #f8f8f8; background-color: #f8f8f8;
z-index: 1;
/* 确保层级正确 */
}
.message-container {
width: 100%;
min-height: 100vh;
/* 确保容器至少有一屏高度 */
display: flex;
flex-direction: column;
justify-content: flex-end;
/* 默认内容放到底部 */
} }
.message-list { .message-list {
transform: scaleY(-1); /* 聊天列表倒置,让最新消息在底部 */
width: 100%; width: 100%;
padding-bottom: 20rpx; display: flex;
flex-direction: column;
padding-bottom: 20px;
/* 底部留出一些空间 */
} }
.loading-more { .message-item {
text-align: center; margin-bottom: 10px;
color: #999;
font-size: 24rpx;
padding: 20rpx 0;
}
.new-message-tip {
position: fixed;
bottom: 140rpx;
left: 50%;
transform: translateX(-50%);
background-color: #fff;
padding: 10rpx 30rpx;
border-radius: 30rpx;
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
z-index: 100;
font-size: 24rpx;
color: #333;
} }
</style> </style>

View File

@ -42,11 +42,12 @@
sheep.$url.static('/static/img/shop/chat/default.png') sheep.$url.static('/static/img/shop/chat/default.png')
" "
mode="aspectFill" mode="aspectFill"
lazy-load
></image> ></image>
<!-- 内容 --> <!-- 内容 -->
<template v-if="message.contentType === KeFuMessageContentTypeEnum.TEXT"> <template v-if="message.contentType === KeFuMessageContentTypeEnum.TEXT">
<view class="message-box" :class="{ admin: message.senderType === UserTypeEnum.ADMIN }"> <view class="message-box" :class="{ admin: message.senderType === UserTypeEnum.ADMIN }">
<mp-html :content="replaceEmoji(getMessageContent(message).text || message.content)" /> <mp-html :content="processedContent" :domain="sheep.$url.cdn('')" lazy-load />
</view> </view>
</template> </template>
<template v-if="message.contentType === KeFuMessageContentTypeEnum.IMAGE"> <template v-if="message.contentType === KeFuMessageContentTypeEnum.IMAGE">
@ -138,7 +139,16 @@
return false; return false;
}); });
// //
const emojiMap = computed(() => {
const map = new Map();
emojiList.forEach(emoji => {
map.set(emoji.name, emoji.file);
});
return map;
});
// -
function replaceEmoji(data) { function replaceEmoji(data) {
let newData = data; let newData = data;
if (typeof newData !== 'object') { if (typeof newData !== 'object') {
@ -146,27 +156,28 @@
let zhEmojiName = newData.match(reg); let zhEmojiName = newData.match(reg);
if (zhEmojiName) { if (zhEmojiName) {
zhEmojiName.forEach((item) => { zhEmojiName.forEach((item) => {
let emojiFile = selEmojiFile(item); const emojiFile = emojiMap.value.get(item) || '';
newData = newData.replace( if (emojiFile) {
item, newData = newData.replace(
`<img class="chat-img" style="width: 24px;height: 24px;margin: 0 3px;vertical-align: middle;" src="${sheep.$url.cdn( item,
'/static/img/chat/emoji/' + emojiFile, `<img class="chat-img" style="width: 24px;height: 24px;margin: 0 3px;vertical-align: middle;" src="${sheep.$url.cdn(
)}"/>`, '/static/img/chat/emoji/' + emojiFile,
); )}"/>`,
);
}
}); });
} }
} }
return newData; return newData;
} }
function selEmojiFile(name) { //
for (let index in emojiList) { const processedContent = computed(() => {
if (emojiList[index].name === name) { if (props.message.contentType === KeFuMessageContentTypeEnum.TEXT) {
return emojiList[index].file; return replaceEmoji(getMessageContent.value(props.message).text || props.message.content);
}
} }
return false; return props.message.content;
} });
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">

View File

@ -13,6 +13,9 @@
v-model="chat.msg" v-model="chat.msg"
@on-tools="onTools" @on-tools="onTools"
@send-message="onSendMessage" @send-message="onSendMessage"
:auto-focus="false"
:show-char-count="true"
:max-length="500"
></message-input> ></message-input>
</template> </template>
</MessageList> </MessageList>
@ -29,6 +32,9 @@
v-model="chat.msg" v-model="chat.msg"
@on-tools="onTools" @on-tools="onTools"
@send-message="onSendMessage" @send-message="onSendMessage"
:auto-focus="false"
:show-char-count="true"
:max-length="500"
></message-input> ></message-input>
</tools-popup> </tools-popup>
<!-- 商品订单选择 --> <!-- 商品订单选择 -->
@ -181,6 +187,8 @@
// 2.3 KEFU_MESSAGE_ADMIN_READ // 2.3 KEFU_MESSAGE_ADMIN_READ
if (type === WebSocketMessageTypeConstants.KEFU_MESSAGE_ADMIN_READ) { if (type === WebSocketMessageTypeConstants.KEFU_MESSAGE_ADMIN_READ) {
console.log('管理员已读消息'); console.log('管理员已读消息');
//
sheep.$helper.toast('客服已读您的消息');
} }
}, },
}); });