极乐门资源网 Design By www.ioogu.com
效果
初始化滚动条高度
var keyHeight = 0;
数据格式
const CHAT_DATA=[ { type:0,//0客服1用户 content:'欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎欢迎', headImg:'../../assets/common/images/headHortrait.jpeg',//头像 creatTime:'2019-01-01',//创建时间 contentType:'text' }, { type: 0,//0客服1用户 content: '1111111', headImg: '../../assets/common/images/headHortrait.jpeg',//头像 creatTime: '2019-01-01',//创建时间 contentType: 'text' }, { type: 1,//0客服1用户 content: '222222', headImg: '../../assets/common/images/headHortrait.jpeg',//头像 creatTime: '2019-01-01',//创建时间 contentType: 'text' }, { type: 0,//0客服1用户 content: '333333', headImg: '../../assets/common/images/headHortrait.jpeg',//头像 creatTime: '2019-01-01',//创建时间 contentType: 'text' }, { type: 1,//0客服1用户 content: '4444444', headImg: '../../assets/common/images/headHortrait.jpeg',//头像 creatTime: '2019-01-01',//创建时间 contentType: 'text', }, { type: 0,//0客服1用户 content: 'http://tmp/wxc79c66d8b0ed19a8.o6zAJs6QE8L8FKq645ts4e3LoKzI.pGakaVHKmbQ3160aa57e2bf33cb576fbabf691cd890b.durationTime=3706.aac', headImg: '../../assets/common/images/headHortrait.jpeg',//头像 creatTime: '2019-01-01',//创建时间 contentType: 'voice', duration: '3706', },{ type: 1,//0客服1用户 content: 'http://tmp/wxc79c66d8b0ed19a8.o6zAJs6QE8L8FKq645ts4e3LoKzI.pGakaVHKmbQ3160aa57e2bf33cb576fbabf691cd890b.durationTime=3706.aac', headImg: '../../assets/common/images/headHortrait.jpeg',//头像 creatTime: '2019-01-01',//创建时间 contentType: 'voice', duration:'3706' }, { type: 1,//0客服1用户 content: 'https://img.yzcdn.cn/vant/cat.jpeg', headImg: '../../assets/common/images/headHortrait.jpeg',//头像 creatTime: '2019-01-01',//创建时间 contentType: 'img' }, { type: 1,//0客服1用户 content: 'https://img.yzcdn.cn/vant/cat.jpeg', headImg: '../../assets/common/images/headHortrait.jpeg',//头像 creatTime: '2019-01-01',//创建时间 contentType: 'img' } ];
wxml对话框
<block wx:key wx:for='{{chatData}}' wx:for-index="index"> <!-- 单个消息1 客服发出(左) --> <view wx:if='{{item.type==0}}' id='msg-{{index}}' class="contentLeft" style=''> <view class="head"> <image class="headImg" src='{{item.headImg}}' mode='widthFix'></image> </view> <view class='leftMsg' wx:if="{{item.contentType==='text'}}">{{item.content}}</view> <view class='leftMsg' wx:if="{{item.contentType==='voice'}}" data-duration="{{item.content}}">{{item.duration}}s</view> <view class='leftMsg img' wx:if="{{item.contentType==='img'}}"><image src="/UploadFiles/2021-04-02/{{item.content}}">wxml底部输入框
<view class='inputRoom' style='bottom: {{inputBottom}};height: {{bottomHeight}}'> <van-row class="bottomRow"> <van-col span="2"wx:if="{{show}}"> <van-icon bindtap="startRecord" class="iconfont icon" class-prefix='icon' size="40rpx" name="yuyin" ></van-icon></van-col> <van-col span="2" wx:if="{{!show}}"> <van-icon bindtap="startRecord" class="iconfont icon" class-prefix='icon' size="40rpx" name="fabiaowenzhang" ></van-icon></van-col> <van-col span="18" wx:if="{{show}}"> <input bindconfirm='sendClick'bind:input="inputValue" adjust-position='{{false}}' value='{{inputVal}}' confirm-type='send' bindfocus='focus' bindblur='blur'></input></van-col> <van-col span="18" wx:if="{{!show}}"> <view class="holdTape" bind:touchstart="startTalk" bind:touchend='stopRecord'>按住请说话</view></van-col> <van-col span="2"><van-icon class="iconfont icon" class-prefix='icon' size="40rpx" name='biaoqing' bindtap="getEmoji"></van-icon></van-col> <van-col span="2"><van-uploader use-slot accept='image' bind:after-read="uploadeImg"> <van-icon class="iconfont icon" class-prefix='icon' size="40rpx" name='icon02' ></van-icon></van-uploader></van-col> </van-row> <view wx:if="{{showEmoji}}" class="emoji"> <emoji bind:clickEmoji="clickEmoji" data-key="inputVal" value="{{inputVal}}" /> </view> </view> </view> <view class="recordDailog" wx:if="{{showDailog}}" > <view class="show"> <image src="/UploadFiles/2021-04-02/record.png">css
#page{ height: 90%; overflow-y: auto; } .content{ background: white; } .inputRoom { width: 100vw; /* height: 16vw; */ border-top: 1px solid #cdcdcd; position: fixed; bottom: 0; display: flex; align-items: center; z-index: 20; background: white; flex-direction: column; } .bottomRow{ width: 100%; height: 16vw; display: flex; align-items: center; flex-direction: row } .bottomRow .van-row{ width: 100%; } .emoji{ height: 30vw; } input { width: 90%; height: 9.33vw; background-color: #EEF4FA; border-radius: 6rpx; font-size: 28rpx; color: #444; padding: 0 3%; margin-left: 2%; } .leftMsg { font-size: 26rpx; color: #333333; line-height: 6vw; padding: 2vw 2.5vw; background-color: #EEF4FA; border-radius: 10rpx; z-index: 10; } .rightMsg { font-size: 26rpx; color: white; line-height: 6vw; padding: 2vw 2.5vw; background-color: #496DFF; border-radius: 10rpx; z-index: 10; } .chatFrame{ background: white; height: 100% } .icon{ line-height: 8vw; } .head{ display: flex; align-items: center } .headImg{ border-radius: 50%; width: 60rpx;height: 60rpx; } .holdTape{ width: 90%; height: 9.33vw; background-color: #EEF4FA; border-radius: 6rpx; padding: 0 3%; margin-left: 2%; display: flex; align-items: center; justify-content: center; } .recordDailog{ -webkit-transition-duration: 300ms; transition-duration: 300ms; z-index: 1000; position: fixed; top: 50%; left: 50%; width: -webkit-fit-content; width: fit-content; -webkit-transform: translate(-50%,-50%); transform: translate(-50%,-50%); max-width: var(--toast-max-width,70%); } .show{ width: var(--toast-default-width,90px); min-height: var(--toast-default-min-height,90px); padding: var(--toast-default-padding,16px); display: flex; -webkit-flex-direction: column; flex-direction: column; -webkit-align-items: center; align-items: center; -webkit-justify-content: center; justify-content: center; box-sizing: initial; color: var(--toast-text-color,#fff); font-size: var(--toast-font-size,14px); line-height: var(--toast-line-height,20px); white-space: pre-wrap; word-wrap: break-word; background-color: var(--toast-background-color,rgba(50,50,51,.88)); border-radius: var(--toast-border-radius,4px); } .show image{ width: 24px; height: 24px } image{ /* max-width: 88vw; max-height: 400rpx; */ width: 100% } .img{ background: none; width: 90% } .contentRight{ display: flex; justify-content: flex-end; padding: 2vw 2vw 2vw 11vw;width: 86%; } .contentLeft{ display: flex; padding: 2vw 11vw 2vw 2vw;width: 86%; }js
// pages/contact/contact.js const { pageFunc } = require('../../utils/util.js'); const app = getApp(); var windowWidth = wx.getSystemInfoSync().windowWidth; var windowHeight = wx.getSystemInfoSync().windowHeight; var keyHeight = 0; const { CHAT_DATA}=require("../../data/customerService.js"); import Toast from '../../components/vant/toast/toast'; const recorderManager = wx.getRecorderManager(); const innerAudioContext = wx.createInnerAudioContext(); const db = wx.cloud.database(); /** * 初始化数据 */ /** * 计算msg总高度 */ function calScrollHeight(that, keyHeight) { var query = wx.createSelectorQuery(); query.select('.scrollMsg').boundingClientRect(function(rect) { }).exec(); } Page({ /** * 页面的初始数据 */ data: { scrollHeight: '100vh', inputVal:"", inputBottom: 0, chatData:[], show:true, showDailog:false, bottomHeight:"18vw", sendData:{}, pagination: { pageSize: 5, currentPage: 1, total: 0, }, showEmoji:false, toastTitle:"录音中...." }, /** * 生命周期函数--监听页面加载 */ onLoad: function (options) { // this.setData({ // cusHeadIcon: app.globalData.userInfo.avatarUrl, // }); const { pagination } = this.data this.getData({ param: CHAT_DATA, pagination }); wx.pageScrollTo({ scrollTop: 1000 }) }, getData(params) { const { chatData } = this.data; const { param, pagination: { pageSize = 10, currentPage = 1 }, } = params; this.setData({ pagination: { pageSize, currentPage } }); const { data, pagination } = pageFunc(param, currentPage, pageSize); data.forEach((item) => { if (item.duration) { item.duration = Math.ceil(item.duration / 1000) } }); this.setData({ 'chatData': data.concat(chatData) }); }, startRecord(){//开始录音 const {show}=this.data if (show){ this.setData({ show: false, }) }else{ this.setData({ show: true, }) } }, startTalk(e){//开始说话 this.setData({ showDailog:true, }) const options = { duration: 60000, sampleRate: 44100, numberOfChannels: 1, encodeBitRate: 192000, format: 'aac', frameSize: 50 } recorderManager.start(options) recorderManager.onStart((res) => { }) }, stopRecord(){//停止说话 const that=this this.setData({ showDailog:false, }) recorderManager.stop(); recorderManager.onStop((res) => { const { sendData, chatData } = that.data; let { tempFilePath, duration, fileSize} = res sendData.tempFilePathData=res duration = Math.ceil(duration / 1000) const data = { content: tempFilePath, duration , fileSize, contentType: 'voice', type: 1}; chatData.push(data); wx.createSelectorQuery().select('.content').boundingClientRect(function (rect) { // 使页面滚动到底部 wx.pageScrollTo({ scrollTop: rect.bottom + 5000 }) }).exec(); that.setData({ 'tempFilePath': tempFilePath, sendData, chatData, scrollHeight: (windowHeight - 0) + 'px', toView: 'msg-' + (chatData.length - 1), inputBottom: '0px' }) }) }, playRecord(e){//播放语音 const { currentTarget: { dataset: { duration } }}=e; const { tempFilePath} = this.data innerAudioContext.autoplay = true; innerAudioContext.src = tempFilePath , innerAudioContext.onPlay(() => { this.setData({ toastTitle: "播放中....", showDailog: true, }) }) innerAudioContext.onEnded((res) => { this.setData({ toastTitle: "录音中....", showDailog: false, }) }) innerAudioContext.onError((res) => { }); innerAudioContext.play() }, uploadeImg(e){ const { file: { path,size:fileSize} } = e.detail; const { chatData } = this.data; const data = { content: path, fileSize, contentType: 'img', type: 1 }; chatData.push(data); wx.createSelectorQuery().select('.content').boundingClientRect(function (rect) { // 使页面滚动到底部 wx.pageScrollTo({ scrollTop: rect.bottom + 5000 }) }).exec() this.setData({ chatData, scrollHeight: (windowHeight - 0) + 'px', toView: 'msg-' + (chatData.length - 1), inputBottom:'0px' }) }, getEmoji(){//获取表情包 wx.createSelectorQuery().select('.content').boundingClientRect(function (rect) { // 使页面滚动到底部 wx.pageScrollTo({ scrollTop: rect.bottom + 5000 }) }).exec() this.setData({ showEmoji:true, bottomHeight:"48vw" }) }, clickEmoji: function (e) {//选择表情包 const { detail: { value }, currentTarget: { dataset: { key } } } = e; this.setData({ [key]: value }) }, onPreview(e){ const { currentTarget: { dataset: { src } } } = e; const urls = [src] wx.previewImage({ current: src, urls }) }, /** * 生命周期函数--监听页面显示 */ onShow: function () { }, /** * 页面相关事件处理函数--监听用户下拉动作 */ onPullDownRefresh: function () { let { pagination: { currentPage } } = this.data this.getData({ param: CHAT_DATA, pagination: { pageSize: 5, currentPage: currentPage + 1, } }); }, /** * 页面上拉触底事件的处理函数 */ onReachBottom: function () { }, /** * 获取聚焦 */ inputValue(e){ const {detail:{value}}=e this.setData({ inputVal:value }) }, focus: function (e) { const { chatData}=this.data keyHeight = e.detail.height; wx.pageScrollTo({ scrollTop: windowHeight - keyHeight }) this.setData({ toView: 'msg-' + (chatData.length - 1), inputBottom: keyHeight + 'px', scrollHeight: (windowHeight - keyHeight) + 'px', showEmoji:false, bottomHeight: "18vw" }) //计算msg高度 // calScrollHeight(this, keyHeight); }, //失去聚焦(软键盘消失) blur: function (e) { const { chatData } = this.data this.setData({ scrollHeight: '100vh', inputBottom: 0 }) this.setData({ toView: 'msg-' + (chatData.length - 1) }) }, /** * 发送点击监听 */ sendClick: function (e) { const { chatData, scrollHeight}=this.data; wx.createSelectorQuery().select('.content').boundingClientRect(function (rect) { // 使页面滚动到底部 wx.pageScrollTo({ scrollTop: rect.bottom + 5000 }) }).exec() chatData.push({ type: 1, contentType: 'text', content: e.detail.value, headImg: '../../assets/common/images/headHortrait.jpeg', }) this.setData({ chatData, inputVal:'' }); }, /** * 退回上一页 */ toBackClick: function () { wx.navigateBack({}) } })// pages/contact/contact.js const { pageFunc } = require('../../utils/util.js'); const app = getApp(); var windowWidth = wx.getSystemInfoSync().windowWidth; var windowHeight = wx.getSystemInfoSync().windowHeight; var keyHeight = 0; const { CHAT_DATA}=require("../../data/customerService.js"); import Toast from '../../components/vant/toast/toast'; const recorderManager = wx.getRecorderManager(); const innerAudioContext = wx.createInnerAudioContext(); const db = wx.cloud.database(); /** * 初始化数据 */ /** * 计算msg总高度 */ function calScrollHeight(that, keyHeight) { var query = wx.createSelectorQuery(); query.select('.scrollMsg').boundingClientRect(function(rect) { }).exec(); } Page({ /** * 页面的初始数据 */ data: { scrollHeight: '100vh', inputVal:"", inputBottom: 0, chatData:[], show:true, showDailog:false, bottomHeight:"18vw", sendData:{}, pagination: { pageSize: 5, currentPage: 1, total: 0, }, showEmoji:false, toastTitle:"录音中...." }, /** * 生命周期函数--监听页面加载 */ onLoad: function (options) { // this.setData({ // cusHeadIcon: app.globalData.userInfo.avatarUrl, // }); const { pagination } = this.data this.getData({ param: CHAT_DATA, pagination }); wx.pageScrollTo({ scrollTop: 1000 }) }, getData(params) { const { chatData } = this.data; const { param, pagination: { pageSize = 10, currentPage = 1 }, } = params; this.setData({ pagination: { pageSize, currentPage } }); const { data, pagination } = pageFunc(param, currentPage, pageSize); data.forEach((item) => { if (item.duration) { item.duration = Math.ceil(item.duration / 1000) } }); this.setData({ 'chatData': data.concat(chatData) }); }, startRecord(){//开始录音 const {show}=this.data if (show){ this.setData({ show: false, }) }else{ this.setData({ show: true, }) } }, startTalk(e){//开始说话 this.setData({ showDailog:true, }) const options = { duration: 60000, sampleRate: 44100, numberOfChannels: 1, encodeBitRate: 192000, format: 'aac', frameSize: 50 } recorderManager.start(options) recorderManager.onStart((res) => { }) }, stopRecord(){//停止说话 const that=this this.setData({ showDailog:false, }) recorderManager.stop(); recorderManager.onStop((res) => { const { sendData, chatData } = that.data; let { tempFilePath, duration, fileSize} = res sendData.tempFilePathData=res duration = Math.ceil(duration / 1000) const data = { content: tempFilePath, duration , fileSize, contentType: 'voice', type: 1}; chatData.push(data); wx.createSelectorQuery().select('.content').boundingClientRect(function (rect) { // 使页面滚动到底部 wx.pageScrollTo({ scrollTop: rect.bottom + 5000 }) }).exec(); that.setData({ 'tempFilePath': tempFilePath, sendData, chatData, scrollHeight: (windowHeight - 0) + 'px', toView: 'msg-' + (chatData.length - 1), inputBottom: '0px' }) }) }, playRecord(e){//播放语音 const { currentTarget: { dataset: { duration } }}=e; const { tempFilePath} = this.data innerAudioContext.autoplay = true; innerAudioContext.src = tempFilePath , innerAudioContext.onPlay(() => { this.setData({ toastTitle: "播放中....", showDailog: true, }) }) innerAudioContext.onEnded((res) => { this.setData({ toastTitle: "录音中....", showDailog: false, }) }) innerAudioContext.onError((res) => { }); innerAudioContext.play() }, uploadeImg(e){ const { file: { path,size:fileSize} } = e.detail; const { chatData } = this.data; const data = { content: path, fileSize, contentType: 'img', type: 1 }; chatData.push(data); wx.createSelectorQuery().select('.content').boundingClientRect(function (rect) { // 使页面滚动到底部 wx.pageScrollTo({ scrollTop: rect.bottom + 5000 }) }).exec() this.setData({ chatData, scrollHeight: (windowHeight - 0) + 'px', toView: 'msg-' + (chatData.length - 1), inputBottom:'0px' }) }, getEmoji(){//获取表情包 wx.createSelectorQuery().select('.content').boundingClientRect(function (rect) { // 使页面滚动到底部 wx.pageScrollTo({ scrollTop: rect.bottom + 5000 }) }).exec() this.setData({ showEmoji:true, bottomHeight:"48vw" }) }, clickEmoji: function (e) {//选择表情包 const { detail: { value }, currentTarget: { dataset: { key } } } = e; this.setData({ [key]: value }) }, onPreview(e){ const { currentTarget: { dataset: { src } } } = e; const urls = [src] wx.previewImage({ current: src, urls }) }, /** * 生命周期函数--监听页面显示 */ onShow: function () { }, /** * 页面相关事件处理函数--监听用户下拉动作 */ onPullDownRefresh: function () { let { pagination: { currentPage } } = this.data this.getData({ param: CHAT_DATA, pagination: { pageSize: 5, currentPage: currentPage + 1, } }); }, /** * 页面上拉触底事件的处理函数 */ onReachBottom: function () { }, /** * 获取聚焦 */ inputValue(e){ const {detail:{value}}=e this.setData({ inputVal:value }) }, focus: function (e) { const { chatData}=this.data keyHeight = e.detail.height; wx.pageScrollTo({ scrollTop: windowHeight - keyHeight }) this.setData({ toView: 'msg-' + (chatData.length - 1), inputBottom: keyHeight + 'px', scrollHeight: (windowHeight - keyHeight) + 'px', showEmoji:false, bottomHeight: "18vw" }) //计算msg高度 // calScrollHeight(this, keyHeight); }, //失去聚焦(软键盘消失) blur: function (e) { const { chatData } = this.data this.setData({ scrollHeight: '100vh', inputBottom: 0 }) this.setData({ toView: 'msg-' + (chatData.length - 1) }) }, /** * 发送点击监听 */ sendClick: function (e) { const { chatData, scrollHeight}=this.data; wx.createSelectorQuery().select('.content').boundingClientRect(function (rect) { // 使页面滚动到底部 wx.pageScrollTo({ scrollTop: rect.bottom + 5000 }) }).exec() chatData.push({ type: 1, contentType: 'text', content: e.detail.value, headImg: '../../assets/common/images/headHortrait.jpeg', }) this.setData({ chatData, inputVal:'' }); }, /** * 退回上一页 */ toBackClick: function () { wx.navigateBack({}) } })以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
标签:
小程序,聊天,小程序,聊天功能
极乐门资源网 Design By www.ioogu.com
极乐门资源网
免责声明:本站文章均来自网站采集或用户投稿,网站不提供任何软件下载或自行开发的软件!
如有用户或公司发现本站内容信息存在侵权行为,请邮件告知! 858582#qq.com
极乐门资源网 Design By www.ioogu.com
暂无微信小程序聊天功能的示例代码的评论...
P70系列延期,华为新旗舰将在下月发布
3月20日消息,近期博主@数码闲聊站 透露,原定三月份发布的华为新旗舰P70系列延期发布,预计4月份上市。
而博主@定焦数码 爆料,华为的P70系列在定位上已经超过了Mate60,成为了重要的旗舰系列之一。它肩负着重返影像领域顶尖的使命。那么这次P70会带来哪些令人惊艳的创新呢?
根据目前爆料的消息来看,华为P70系列将推出三个版本,其中P70和P70 Pro采用了三角形的摄像头模组设计,而P70 Art则采用了与上一代P60 Art相似的不规则形状设计。这样的外观是否好看见仁见智,但辨识度绝对拉满。
更新日志
2025年01月23日
2025年01月23日
- 小骆驼-《草原狼2(蓝光CD)》[原抓WAV+CUE]
- 群星《欢迎来到我身边 电影原声专辑》[320K/MP3][105.02MB]
- 群星《欢迎来到我身边 电影原声专辑》[FLAC/分轨][480.9MB]
- 雷婷《梦里蓝天HQⅡ》 2023头版限量编号低速原抓[WAV+CUE][463M]
- 群星《2024好听新歌42》AI调整音效【WAV分轨】
- 王思雨-《思念陪着鸿雁飞》WAV
- 王思雨《喜马拉雅HQ》头版限量编号[WAV+CUE]
- 李健《无时无刻》[WAV+CUE][590M]
- 陈奕迅《酝酿》[WAV分轨][502M]
- 卓依婷《化蝶》2CD[WAV+CUE][1.1G]
- 群星《吉他王(黑胶CD)》[WAV+CUE]
- 齐秦《穿乐(穿越)》[WAV+CUE]
- 发烧珍品《数位CD音响测试-动向效果(九)》【WAV+CUE】
- 邝美云《邝美云精装歌集》[DSF][1.6G]
- 吕方《爱一回伤一回》[WAV+CUE][454M]