使用uniapp完成一个客服聊天功能
起因:
最近在工作中,老大让我重构一下商城的客服聊天页面(因为之前是买来的代码---php渲染恶心的一比,现在要用uniapp做多端应用,所以前端这块的重构基本上就是重做)
规划:
- 既然要做实时聊天呢,那自然少不了websocket的参与,当然了,在uniapp中,对websocket做了封装和处理,我们直接使用它提供的api就行;
- 布局方面,其实只需要区分,自己发送的消息和别人发送的消息,自己发送的展示在右边,别人发送的展示在左边(这快比较简单),然后你要展示什么样的内容,那必须要有对应的标签占位先(比如你要发送一个图片,后端返回的是一个url,那你就需要留一个image标签)
- 各种交互,需要考虑的是,每发送一个消息,都要滑动到最底部(刚进来之后同理)、还有输入较多字符后,输入框要增高一定的高度、点击更多后要在底下展示发送图片,拍照等按钮、图片预览等(这块很烦,h5,小程序,客户端三端兼容很痛苦~)
实际效果展示:

代码展示:
嗯...,代码好像有点多(采用的是uniapp + uview,所以组件、路由跳转、网络请求这些都是采用uview的方式)
<template>
<view class="service">
<!-- 头部 -->
<u-navbar :is-back="false" title=" " title-width="0" :background="background" :border-bottom="false" class="header"
height="44">
<view class="serviceTitle">
<u-image :src="fanhui" alt @click="goBack" width="24rpx" height="40rpx" />
<view v-if="!isSysMessage">{{userInfo.shopName}}</view>
<view v-else>通知消息</view>
</view>
<view class="goShopBtn" @click="goShopDetail" v-if="!isSysMessage">进店</view>
</u-navbar>
<!-- 分割线 -->
<view class="hrline"></view>
<view class="goodHref" v-if="!isSysMessage && goodsId" @click="goGoodsDetail">
<view class="goodContent">
<u-image :src="goodsInfo.goodsImg?url + goodsInfo.goodsImg:''" width="90rpx" height="90rpx" border-radius="12rpx"></u-image>
<text>{{goodsInfo.goodsName}}</text>
</view>
<!-- <view class="goHref" @click="sendGoodsHref">发送链接</view> -->
</view>
<!-- 聊天窗口区域 -->
<scroll-view :class="{'chatBox':true,'isGoods':Boolean(goodsId)}" :scroll-y="true" id="chatBoxScroll" :style="{height: style.contentViewHeight + 'px'}"
:scroll-with-animation="true" :scroll-top="scrollTop" @click="noShowMore" refresher-enabled="true"
:refresher-triggered="triggered" :refresher-threshold="100" refresher-background="#f2f2f2" @refresherrefresh="onRefresh"
@refresherrestore="onRestore" @refresherpulling="isLoadmore=false">
<!-- 加载中与没有更多展示 -->
<u-loadmore :status="status" margin-top="60" class="loadmore" v-show="isLoadmore" />
<!-- 再包一层,方便计算高度 -->
<view id="chatContent" :class="{isPadding:!isSysMessage}">
<!-- 商品链接发送 -->
<!-- 聊天 -->
<view class="chat">
<!-- 这里再包一层壳子,方便整体循环 -->
<!-- 这里过滤,只展示图片和文字,订单和商品不展示 -->
<view class="chatMessage" v-for="(item,index) in messageList" :key="index" v-if="!isSysMessage">
<!-- 如果不是系统消息 -->
<view class="timeShow" v-if="((item.content&&(item.content.type=='image' || !item.content.type)))">{{item.createTime}}</view>
<!-- 别人发的信息展示 -->
<view v-if="item.type=='chat' && item.groupName=='店铺客服'" class="isText">店铺客服接待了你</view>
<view v-if="item.type=='message'" class="isText">当前无客服在线,您可以留言,我们会尽快回复您</view>
<view class="chatY" v-if="item.from && Number(item.from)!==userInfo.userId && ((item.content&&(item.content.type=='image' || !item.content.type)))">
<u-image :src="userInfo.shopImg" width="72rpx" height="72rpx" border-radius="50%"></u-image>
<view class="messagey" v-if="item.content && !item.content.type">
<text selectable="true">{{item.content}}</text>
</view>
<u-image class="messageyImg" :src="item.content?item.content.content:''" width="300rpx" height="300rpx"
border-radius="12rpx" v-if="item.content && item.content.type=='image'" @click="checkImg(item)"></u-image>
</view>
<!-- 自己发的信息展示 -->
<view class="chatM" v-if="(!item.from || (Number(item.from)==userInfo.userId)) && ((item.content&&(item.content.type=='image' || !item.content.type)))">
<view class="messagex" v-if="item.content && !item.content.type">
<text selectable="true">{{item.content}}</text>
</view>
<u-image class="messagexImg" :src="item.content?item.content.content:''" width="300rpx" height="300rpx"
border-radius="12rpx" v-if="item.content && item.content.type=='image'" @click="checkImg(item)"></u-image>
<u-image :src="userInfo.userPhoto" width="72rpx" height="72rpx" border-radius="50%"></u-image>
</view>
</view>
<!-- 如果是系统消息 -->
<view class="chatMessage" v-for="(item,index) in messageList" :key="index" v-if="isSysMessage && item.msgContent">
<!-- 别人发的信息展示 -->
<view class="timeShow">{{item.createTime}}</view>
<view class="chatY">
<u-image :src="sysImg" width="72rpx" height="72rpx" border-radius="50%"></u-image>
<view class="messagey">{{item.msgContent}}</view>
</view>
</view>
</view>
</view>
</scroll-view>
<!-- 输入框 -->
<view class="inputBox" v-if="!isSysMessage" id="inputBox">
<u-field v-model="message" label=" " label-width="0" placeholder="在此输入文字" type="textarea" maxlength="100" :clearable="false"
clear-size="0" placeholder-style="color: #9A9A9A;font-size:28rpx" @click="isFocus" @blur="isBlur">
</u-field>
<u-image :src="jiahao" width="54rpx" height="54rpx" v-show="!message" @click="sendImage"></u-image>
<view v-show="message" class="sendStn" @click="sendText">发送</view>
</view>
<!-- 更多 -->
<!-- #ifdef H5 -->
<view class="more" style="background: #f2f2f2; height: 200rpx;" v-if="!isSysMessage && isf">
<!-- #endif -->
<!-- #ifndef H5 -->
<view class="more" style="background: #f2f2f2; height: 200rpx;" v-if="!isSysMessage">
<!-- #endif -->
<u-image :src="xiangji" width="90rpx" height="90rpx" @click="goCamera"></u-image>
<u-image :src="tupian" width="90rpx" height="90rpx" @click="goAlbum"></u-image>
<!-- <u-image :src="shangpin" width="90rpx" height="90rpx"></u-image> -->
</view>
<!-- toast弹窗 -->
<u-toast ref="uToast" />
<!-- 图片预览 -->
<previewImage ref="previewImage" :imgs="messgaeImgList" :index="indexImg" @setIndexImg="indexImg=0"></previewImage>
<!-- 无网络提示 -->
<u-no-network></u-no-network>
</view>
</template>
<script>
// 用户图片预览
import config from "common/config.js";
import previewImage from '@/components/kxj-previewImage/kxj-previewImage.vue';
// 引入,为解决小程序不支持eval和new Function
import {
Function,
evaluate
} from 'eval5';
export default {
data() {
return {
url: config.baseUrl,
fanhui: '',
jiahao: '',
xiangji: '',
tupian: '',
shangpin: '',
sysImg: '',
background: {
backgroundColor: '#f2f2f2'
},
message: '',
messageList: [],
scrollTop: 0,
// 聊天时页面滚动样式
style: {
pageHeight: 0,
contentViewHeight: 0,
footViewHeight: 90,
mitemHeight: 0
},
messgaeImgList: [],
indexImg: 0,
// 这个状态为了判断更多是否显示
morenStyle: false,
triggered: false,
_freshing: false,
isSysMessage: false,
initHeight: 0,
// 下拉加载更多相关
lastPage: 1,
page: 1,
status: 'loadmore',
isLoadmore: false,
shopId: 0,
// shopInfo: {},
goodsId: 0,
goodsInfo: {},
userInfo: {},
// 聊天分页
chatPage: 1,
chatLastPage: 1,
wssServer: '',
tokenId: '',
socketOpen: false,
// 订单相关
orderNo: 0,
orderId: 0,
isKill: false,
killId: 0,
isf: true,
windowSize: 0,
// isNetWork: false
}
},
components: {
previewImage,
},
watch: {
// 这个通过监听输入值的改变,改变聊天区域的高度
message(val, oldVal) {
// 小程序中,因为加了setTimeout,所以先触发了监听,导致that.initHeight!==res[0].height,最终高度计算错误
// 所以,在这个做判断,如果清空输入值时,不触发监听
if (val !== '') {
let that = this;
let query = uni.createSelectorQuery();
query.select('#inputBox').boundingClientRect();
setTimeout(() => {
query.exec((res) => {
if (that.initHeight !== res[0].height) {
that.style.contentViewHeight = that.style.contentViewHeight - (res[0].height - that.initHeight);
that.initHeight = res[0].height;
}
})
}, 100);
}
},
windowSize(val, oldVal) {
if (val > oldVal) {
this.isf = true;
}
},
isNetWork(val, oldVal) {
// 防止进入自己店铺后也会触发
if (this.wssServer) {
if (val) {
this.connectWS();
} else {
uni.closeSocket();
}
}
}
},
onUnload() {
uni.setStorageSync('isNet', false);
uni.closeSocket();
// 将未读变为已读
if (this.isSysMessage) {
this.updateMessages()
} else {
this.updataSysMessage()
}
},
onLoad(data) {
uni.setStorageSync('isNet', true);
// this.isNetWork = uni.getStorageSync('isNetwork');
if (data.sys) {
this.isSysMessage = data.sys;
}
this.shopId = data.shopId;
this.goodsId = data.goodsId;
this.orderId = data.orderId;
this.orderNo = data.orderNo;
this.isKill = data.isKill;
this.killId = data.id;
this.tokenId = uni.getStorageSync('tokenId');
// 获取用户信息
this.queryUserDetails()
// 进来先请求一次数据(历史记录)
this.$nextTick(() => {
this.onRefresh();
this.scrollToBottom();
});
//监听服务器打开
uni.onSocketOpen(() => {
this.socketOpen = true;
let sendData = {
type: 'login',
uid: this.userInfo.userId,
userName: this.userInfo.loginName,
role: 'user',
platform: 2,
shopId: this.userInfo.shopId,
group: this.userInfo.receiveId
}
uni.sendSocketMessage({
data: JSON.stringify(sendData),
})
// 获取带出商品信息
if (this.goodsId) {
this.getGoodsInfo(this.goodsId);
}
// 如果是订单,则发送订单
if (this.orderId) {
let ordersData = {
content: `{"type":"orders","orderId":"${this.orderId}","content": "订单号:${this.orderNo}"}`,
role: 'user',
type: 'say',
to: this.userInfo.receiveId
}
uni.sendSocketMessage({
data: JSON.stringify(ordersData)
})
}
// 接受服务器传来的数据
uni.onSocketMessage((res) => {
let data = JSON.parse(res.data);
// 如果是图片相关,则转换为json对象,重构数据
try {
// 这里不处理数字,否则回变为科学计数法
if (isNaN(data.content)) {
// let text = eval("(" + data.content + ")");
let text = (new Function("return" + data.content))();
data.content = text;
}
if (data.content && data.content.type == 'image') {
data.content.content = this.url + data.content.content;
}
if ((data.type == 'say' && data.role == 'user') || (data.type == 'chat' && data.groupName) || (data.type ==
'message')) {
this.messageList.push(data)
// 图片预览相关
this.messgaeImgList = [];
this.messageList.forEach((item, index) => {
if (item.content && item.content.type == 'image') {
this.messgaeImgList.push(item.content.content);
}
})
};
// 如果是文字相关,则直接push
} catch (e) {
if ((data.type == 'say' && data.role == 'user') || (data.type == 'chat' && data.groupName) || (data.type ==
'message')) {
this.messageList.push(data)
}
}
this.$nextTick(function() {
this.scrollToBottom();
})
});
});
//监听服务器打开失败重连
uni.onSocketError((res) => {
console.log('WebSocket连接打开失败,请检查!');
});
//监听服务器关闭
uni.onSocketClose((res) => {
console.log("webSocket on 关闭连接");
});
// 获取底部输入框高度
this.$nextTick(() => {
let that = this;
let query = uni.createSelectorQuery();
query.select('#inputBox').boundingClientRect();
query.exec((res) => {
that.initHeight = res[0].height;
})
})
// 聊天区域高度获取
this.getchatHeight()
if (this.goodsId) {
this.style.contentViewHeight = this.style.contentViewHeight - 70
}
// 将未读变为已读
if (this.isSysMessage) {
this.updateMessages()
} else {
this.updataSysMessage()
}
uni.onWindowResize((res) => {
this.windowSize = res.size.windowHeight;
// alert('变化后的窗口高度=' + res.size.windowHeight)
})
},
methods: {
//创建wekSocket长连接
connectWS() {
// 创建Socket
let that = this;
this.$u.vuex('SocketTask', uni.connectSocket({
url: that.wssServer,
header: {
'content-type': 'application/json'
},
method: 'post',
success: function(res) {
console.log('WebSocket连接创建success', res);
},
fail: function(err) {
setTimeout(() => {
that.connectWS();
}, 500);
console.log('WebSocket连接创建fail', err)
},
}));
},
goBack() {
this.$u.route({
type: 'navigateBack'
})
},
isJson(obj) {
let isjson = typeof(obj) == "object" && Object.prototype.toString.call(obj).toLowerCase() == "[object object]" && !
obj.length;
return isjson;
},
// 聊天区域高度获取
getchatHeight() {
// 获取页面高度相关
const res = uni.getSystemInfoSync(); //获取手机可使用窗口高度
this.style.pageHeight = res.windowHeight - res.statusBarHeight;
if (this.isSysMessage) {
// 如果是系统消息
// #ifndef MP-WEIXIN
this.style.contentViewHeight = this.style.pageHeight - uni.getSystemInfoSync().screenWidth / 750 * (100) + 10 //像素 因为给出的是像素高度 然后我们用的是rpx 所以换算一下
// #endif
// #ifdef MP-WEIXIN
this.style.contentViewHeight = this.style.pageHeight - uni.getSystemInfoSync().screenWidth / 750 * (100);
// #endif
} else {
// 否则
// #ifndef MP-WEIXIN
this.style.contentViewHeight = this.style.pageHeight - uni.getSystemInfoSync().screenWidth / 750 * (100) - 40; //像素 因为给出的是像素高度 然后我们用的是rpx 所以换算一下
// #endif
// #ifdef MP-WEIXIN
this.style.contentViewHeight = this.style.pageHeight - uni.getSystemInfoSync().screenWidth / 750 * (100) - 40;
// #endif
}
},
// 下拉相关
onRefresh(pagesize, page) {
// 下拉被触发
// 这里下拉刷新逻辑有点坑,参照:https://ask.dcloud.net.cn/article/37181
this.isLoadmore = false;
if (this._freshing) return;
this._freshing = true;
if (!this.triggered) { //界面下拉触发,triggered可能不是true,要设为true
this.triggered = true;
}
if (this.isSysMessage) {
// 请求系统消息
this.queryMessages(4, this.page);
} else {
// 这里加载更多聊天历史记录
this.getHistory(this.shopId, this.chatPage);
}
},
onRestore() {
// 下拉刷新被复位
console.log("onRestore");
},
goShopDetail() {
this.$u.route({
url: "pagesShopping/shop/index",
params: {
shopId: this.shopId,
isAgain: true
},
type: 'redirect'
})
},
// 发送图片
sendImage() {
this.isf = true;
// #ifndef APP-PLUS
if (!this.morenStyle) {
this.morenStyle = true;
/*
这里是为了给下面的更多区域一个从下面滑上来的效果,
主要是通过改变回话区域的高度,来控制下方是否展示,下方隐藏展示同理
*/
let i = 0;
const setTime = setInterval(() => {
i = i + 10;
this.style.contentViewHeight = this.style.contentViewHeight - 10;
if (i == 100) {
clearInterval(setTime)
this.scrollTop = this.scrollTop + 100;
}
}, 2)
}
// #endif
// #ifdef APP-PLUS
if (!this.morenStyle) {
this.morenStyle = true;
this.style.contentViewHeight = this.style.contentViewHeight - 100;
this.scrollTop = this.scrollTop + 100;
}
// #endif
},
// 点击回话框其他区域,隐藏下方更多展示
noShowMore() {
// #ifndef APP-PLUS
if (this.morenStyle) {
this.morenStyle = false;
let n = 0;
const setTime1 = setInterval(() => {
n = n + 10;
this.style.contentViewHeight = this.style.contentViewHeight + 10;
if (n == 100) {
clearInterval(setTime1)
this.scrollTop = this.scrollTop - 100;
}
}, 5)
}
// #endif
// #ifdef APP-PLUS
if (this.morenStyle) {
this.morenStyle = false;
this.style.contentViewHeight = this.style.contentViewHeight + 100;
this.scrollTop = this.scrollTop - 100;
}
// #endif
},
// 获取焦点
isFocus() {
this.isf = false;
},
// 失去焦点
isBlur() {},
// 预览图片
checkImg(item) {
if (item.content && item.content.type == 'image') {
this.indexImg = this.messgaeImgList.indexOf(item.content.content);
this.$refs.previewImage.show = true;
}
},
// 发送文字按钮
sendText() {
if (this.message) {
let message = {
content: this.message,
role: 'user',
type: 'say',
to: this.userInfo.receiveId
};
//发送会话内容
if (this.socketOpen) {
uni.sendSocketMessage({
data: JSON.stringify(message),
success: () => {
this.$nextTick(function() {
// 发送完之后清空输入框
this.message = '';
// 重新获取底部高度
let that = this;
let query = uni.createSelectorQuery();
query.select('#inputBox').boundingClientRect();
setTimeout(() => {
query.exec((res) => {
that.initHeight = res[0].height;
// 重新获取内容区高度
this.getchatHeight();
if (this.goodsId) {
this.style.contentViewHeight = this.style.contentViewHeight - 70
}
})
}, 150)
this.morenStyle = false;
})
},
fail: (err) => {
console.log(err)
}
});
} else {
this.$refs.uToast.show({
title: '服务未开启,暂不能发送消息'
})
}
}
},
// 调起相机方法
goCamera() {
uni.chooseImage({
count: 1, //默认9
sizeType: ['compressed'],
sourceType: ['camera'],
success: (res) => {
const tempFilePaths = res.tempFilePaths;
uni.uploadFile({
url: this.url + '/weapp/user/uploadPicture',
filePath: tempFilePaths[0],
name: 'img',
formData: {
dir: 'users',
isTumb: 1,
isLocation: 1,
id: 'WU_FILE_0',
},
header: {
tokenid: this.tokenId
},
success: (ress) => {
let data = JSON.parse(ress.data);
let message = {
content: `{"content":"${data.data}","type": "image"}`,
role: 'user',
type: 'say',
to: this.userInfo.receiveId
};
uni.sendSocketMessage({
data: JSON.stringify(message)
})
}
});
}
});
},
// 选择图片方法
goAlbum() {
uni.chooseImage({
count: 1, //默认9
sizeType: ['original', 'compressed'],
sourceType: ['album'],
success: (res) => {
const tempFilePaths = res.tempFilePaths;
uni.uploadFile({
url: this.url + '/weapp/user/uploadPicture',
filePath: tempFilePaths[0],
name: 'img',
formData: {
dir: 'users',
isTumb: 1,
isLocation: 1,
id: 'WU_FILE_0',
},
header: {
tokenid: this.tokenId
},
success: (ress) => {
let data = JSON.parse(ress.data);
let message = {
content: `{"content":"${data.data}","type": "image"}`,
role: 'user',
type: 'say',
to: this.userInfo.receiveId
};
uni.sendSocketMessage({
data: JSON.stringify(message)
})
}
});
}
});
},
// 下滑到底部的方法
scrollToBottom() {
let that = this;
let query = uni.createSelectorQuery();
query.select('#chatContent').boundingClientRect();
query.select('#chatBoxScroll').boundingClientRect();
query.exec((res) => {
that.style.mitemHeight = 0;
that.style.mitemHeight = res[0].height;
// res[0].forEach((rect) => that.style.mitemHeight = that.style.mitemHeight + rect.height + 40) //获取所有内部子元素的高度
if (that.style.mitemHeight > (that.style.contentViewHeight)) { //判断子元素高度是否大于显示高度
that.scrollTop = that.style.mitemHeight - that.style.contentViewHeight + 100 //用子元素的高度减去显示的高度就获益获得序言滚动的高度
}
})
},
// 点击进入商品详情
goGoodsDetail() {
if (!this.orderNo && !this.isKill) {
this.$u.route({
url: 'pagesShopping/commodity/index',
params: {
goodsId: this.goodsId,
isAgain: true
},
type: 'redirect'
})
} else if (this.isKill) {
this.$u.route({
url: 'pagesShopping/seckill/seckillDetail',
params: {
id: this.killId,
goodsId: this.goodsId,
isAgain: true
},
type: 'redirect'
})
} else {
this.$u.route({
url: 'pageOrders/orderDetail/index',
params: {
orderId: this.orderId,
shopId: this.shopId,
isAgain: true
},
type: 'redirect'
})
}
},
// 网络请求相关
queryMessages(pagesize, page) {
// 获取系统消息
this.$u.post('/messages/queryMessages', {
pagesize,
page
}).then(res => {
if (res.status == 200) {
this.lastPage = res.data.last_page;
if (this.page <= this.lastPage) {
this.lastPage == 1 ? this.status = 'nomore' : this.status = 'loading';
// 这里调转一下数组之后再保存
let newList = [];
newList.push(...res.data.data);
newList.reverse();
this.messageList.unshift(...newList);
this.page++
this._freshing = false;
this.triggered = false
} else {
this.status = 'nomore';
this.isLoadmore = true;
this._freshing = false;
this.triggered = false
}
} else {
this._freshing = false;
this.triggered = false;
}
})
},
// 获取历史聊天记录
getHistory(receiveId, page) {
this.$u.post('/addon/wstim-chats-pagequery', {
receiveId,
page,
}).then(res => {
if (res.status == 1 || res.status == 200) {
this.chatLastPage = res.data.last_page;
if (this.chatPage <= this.chatLastPage) {
this.chatLastPage == 1 ? this.status = 'nomore' : this.status = 'loading';
this.messageList.unshift(...res.data.data);
this.chatPage++
this._freshing = false;
this.triggered = false;
// 图片预览相关
res.data.data.forEach((item, index) => {
if (item.content && item.content.type == 'image') {
this.messgaeImgList.push(item.content.content);
}
})
} else {
this.status = 'nomore';
this.isLoadmore = true;
this._freshing = false;
this.triggered = false
}
} else {
this._freshing = false;
this.triggered = false;
}
})
},
updateMessages() {
// 修改系统消息状态(全部变为已读)
this.$u.post('/messages/updateMessages', {}).then(res => {
if (res.status == 200) {
return;
}
})
},
updataSysMessage() {
// 修改回话消息状态(全部变为已读)
this.$u.post('/addon/wstim-chats-setRead', {
shopId: this.shopId
}).then(res => {
if (res.status == 1 || res.status == 200) {
return;
}
})
},
// 带出商品信息获取
getGoodsInfo(goodsId) {
this.$u.post('/goods_details/getProductDetails', {
goodsId
}).then(res => {
if (res.status == 200) {
this.goodsInfo = res.data;
this.sendGoodsHref()
}
})
},
// 发送商品信息
sendGoodsHref() {
let goodsData = {
content: `{"type":"goods","goodsId":"${this.goodsId}","content": "${this.goodsInfo.goodsName}"}`,
role: 'user',
type: 'say',
to: this.userInfo.receiveId
}
uni.sendSocketMessage({
data: JSON.stringify(goodsData),
})
},
// 获取基础基础数据(用户,店铺,联系信息)
queryUserDetails() {
this.$u.get('/addon/wstim-chats-getBaseData', {
shopId: this.shopId,
}).then(res => {
if (res.status == 1 || res.status == 200) {
this.userInfo = res.data;
this.wssServer = res.data.server;
//连接websoket
if (this.tokenId) {
this.connectWS();
}
}
})
}
}
}
</script>
<style lang="scss">
.service {
background: $bg_color;
height: 100vh;
width: 100vw;
overflow: hidden;
// 头部
.header {
/deep/ .u-slot-content {
display: flex;
justify-content: space-between;
padding-right: 24rpx;
}
.serviceTitle {
display: flex;
/* #ifdef MP-WEIXIN */
.u-image {
margin-left: 24rpx;
}
/* #endif */
&>view {
font-size: 32rpx;
font-family: PingFang SC;
font-weight: bold;
color: #343434;
margin-left: 24rpx;
}
}
.goShopBtn {
font-size: 28rpx;
font-family: PingFang SC;
font-weight: bold;
color: $white_color;
width: 87rpx;
height: 48rpx;
background: #CA3232;
border-radius: 12rpx;
text-align: center;
line-height: 48rpx;
}
}
// 分割线
.hrline {
height: 2rpx;
width: 100vh;
background: #E5E5E5;
// margin-bottom: 69rpx;
}
.goodHref {
position: fixed;
left: 0;
right: 0;
z-index: 10;
top: 120rpx;
/* #ifdef MP-WEIXIN */
top: 150rpx;
/* #endif */
/* #ifdef APP-PLUS */
top: 180rpx;
/* #endif */
margin: 0 36rpx;
// margin-top: 69rpx;
background: $white_color;
box-shadow: 1rpx 6rpx 24rpx 0rpx rgba(186, 186, 186, 0.11);
border-radius: 30rpx;
.goodContent {
display: flex;
padding: 36rpx;
text {
margin-left: 24rpx;
flex: 1;
font-size: 28rpx;
font-family: PingFang SC;
color: #343434;
line-height: 42rpx;
}
}
.goHref {
text-align: center;
font-size: 28rpx;
font-family: PingFang SC;
font-weight: bold;
color: #343434;
padding: 30rpx 0;
border-top: 2px solid #F5F5F5;
}
}
.isGoods {
margin-top: 140rpx;
}
.loadmore {
padding-bottom: 0rpx;
}
.chatBox {
height: calc(100vh - 170rpx);
#chatContent {
.chat {
margin-top: 72rpx;
padding: 0 36rpx;
.chatMessage {
@mixin textShow {
margin: 0 auto;
background: #E3E3E3;
border-radius: 8rpx;
font-size: 24rpx;
font-family: PingFang SC;
font-weight: bold;
color: #939393;
width: 300rpx;
height: 42rpx;
line-height: 42rpx;
text-align: center;
margin-bottom: 52rpx;
}
.timeShow {
@include textShow;
}
.isText {
@include textShow;
background: transparent;
}
.chatY {
display: flex;
position: relative;
margin-bottom: 68rpx;
&>.messagey {
// flex: 1;
margin-left: 24rpx;
padding: 23rpx 26rpx;
background: $white_color;
max-width: 558rpx;
border-radius: 12rpx;
word-break: break-all;
&::before {
content: '';
display: inline-block;
border: 12rpx solid;
border-color: transparent $white_color transparent transparent;
position: absolute;
left: 75rpx;
top: 30rpx;
}
}
&>.messageyImg {
/* #ifndef MP-WEIXIN */
&::before {
content: '';
display: inline-block;
border: 12rpx solid;
border-color: transparent $white_color transparent transparent;
position: absolute;
left: 4rpx;
top: 30rpx;
}
/deep/ .u-image__image {
padding: 20rpx;
background: $white_color;
margin-left: 24rpx;
}
/* #endif */
.u-image {
&::before {
content: '';
display: inline-block;
border: 12rpx solid;
border-color: transparent $white_color transparent transparent;
position: absolute;
left: 4rpx;
top: 30rpx;
}
}
/* #ifdef MP-WEIXIN */
&::before {
content: '';
display: inline-block;
border: 12rpx solid;
border-color: transparent $white_color transparent transparent;
position: absolute;
left: 0rpx;
top: 30rpx;
}
/deep/ .u-image__image {
padding: 20rpx;
background: $white_color;
margin-left: 24rpx;
}
/* #endif */
}
}
.chatM {
display: flex;
position: relative;
justify-content: flex-end;
margin-bottom: 68rpx;
&>.messagex {
margin-right: 24rpx;
padding: 23rpx 26rpx;
max-width: 558rpx;
border-radius: 12rpx;
background: #CA3232;
color: $white_color;
word-break: break-all;
&::after {
content: '';
display: inline-block;
border: 12rpx solid;
border-color: transparent transparent transparent #CA3232;
position: absolute;
right: 75rpx;
top: 30rpx;
}
}
&>.messagexImg {
display: flex;
// 微信小程序样式
/* #ifdef MP-WEIXIN */
.u-image {
margin-right: 64rpx;
/deep/ .u-image__image {
padding: 20rpx;
background: #CA3232;
}
&::after {
content: '';
display: inline-block;
border: 12rpx solid;
border-color: transparent transparent transparent #CA3232;
position: absolute;
right: -62rpx;
top: 30rpx;
}
}
/* #endif */
/* #ifndef MP-WEIXIN */
/deep/ .u-image__image {
padding: 20rpx;
background: #CA3232;
margin-left: -65rpx;
}
&::after {
content: '';
display: inline-block;
border: 12rpx solid;
border-color: transparent transparent transparent #CA3232;
position: absolute;
right: 4rpx;
top: 30rpx;
}
/* #endif */
}
}
}
}
}
.isPadding {
padding-top: 69rpx;
}
}
.inputBox {
background: #FFFFFF;
// position: fixed;
// bottom: 0;
// right: 0;
// left: 0;
display: flex;
align-items: center;
position: relative;
/deep/ .u-field {
padding: 0;
width: 650rpx;
margin-left: 24rpx;
.u-flex {
margin: 0 !important;
.u-textarea-class {
// display: flex;
// align-items: center;
min-height: 30rpx !important;
max-height: 150rpx;
padding: 30rpx;
.uni-textarea-textarea {
overflow-y: auto !important;
}
}
}
}
&>.u-image {
position: absolute;
right: 24rpx;
}
&>.sendStn {
position: absolute;
right: 28rpx;
background: #CA3232;
color: $white_color;
font-weight: bold;
padding: 10rpx 18rpx;
border-radius: 12rpx;
top: 50%;
transform: translateY(-50%);
}
}
.more {
display: flex;
.u-image {
margin: 0 24rpx;
margin-top: 36rpx;
}
}
}
</style>

浙公网安备 33010602011771号