Vue通过配置WebSocket并实现群聊功能
程序员文章站
2022-07-06 18:06:01
原文链接:
写jquery项目时,使用websocket很简单,不用去考虑模块化,组件之间的访问问题,面向文档编程即可,在vue项目中使用时,远远没有想象中的那么简单,需要考虑很多场...
原文链接:
写jquery项目时,使用websocket很简单,不用去考虑模块化,组件之间的访问问题,面向文档编程即可,在vue项目中使用时,远远没有想象中的那么简单,需要考虑很多场景,本篇文章将与各位开发者分享下 vue-native-websocket 库的使用以及配置,用其实现群聊功能。先看下最终实现的效果
安装依赖
本文中对于 vue-native-websocket 库的讲解,项目中配置了vuex,对其不了解的开发者请移步官方文档,如果选择继续阅读本篇文章会比较吃力。
vue-native-websocket安装 # yarn | npm 安装 yarn add vue-native-websocket | npm install vue-native-websocket --save
安装成功
配置插件
在 main.js 中进行导入
import vuenativesock from 'vue-native-websocket'
使用 vuenativesock 插件,并进行相关配置
// main.js // base.lkwebsocket为你服务端websocket地址 vue.use(vuenativesock,base.lkwebsocket,{ // 启用vuex集成,store的值为你的vuex store: store, // 数据发送/接收使用使用json格式 format: "json", // 开启自动重连 reconnection: true, // 尝试重连的次数 reconnectionattempts: 5, // 重连间隔时间 reconnectiondelay: 3000, // 将数据进行序列化,由于启用了json格式的数据传输这里需要进行重写 passtostorehandler: function (eventname, event) { if (!eventname.startswith('socket_')) { return } let method = 'commit'; let target = eventname.touppercase(); let msg = event; if (this.format === 'json' && event.data) { msg = json.parse(event.data); if (msg.mutation) { target = [msg.namespace || '', msg.mutation].filter((e) => !!e).join('/'); } else if (msg.action) { method = 'dispatch'; target = [msg.namespace || '', msg.action].filter((e) => !!e).join('/'); } } this.store[method](target, msg); this.store.state.socket.message = msg; } });
vuex的相关配置:mutations和actions添加相关函数
// vuex配置文件 import vue from 'vue' import vuex from 'vuex' vue.use(vuex); export default new vuex.store({ state: { token:"", userid:"", // 用户头像 profilepicture: "", socket: { // 连接状态 isconnected: false, // 消息内容 message: '', // 重新连接错误 reconnecterror: false } }, mutations: { socket_onopen (state, event) { // 连接打开触发的函数 vue.prototype.$socket = event.currenttarget; state.socket.isconnected = true }, socket_onclose (state, event) { // 连接关闭触发的函数 state.socket.isconnected = false; console.log(event); }, socket_onerror (state, event) { // 连接发生错误触发的函数 console.error(state, event) }, socket_onmessage (state, message) { // 收到消息时触发的函数 state.socket.message = message }, socket_reconnect(state, count) { // 重新连接触发的函数 console.info(state, count) }, socket_reconnect_error(state) { // 重新连接失败触发的函数 state.socket.reconnecterror = true; }, }, actions: { customeradded (context) { // 新连接添加函数 console.log('action received: customeradded'); console.log(context) } }, modules: { } })
至此 vue-native-websocket 配置结束,如需了解更多配置方法,请移步 npm仓库
使用插件并实现群聊
在消息发送接收组件中添加 onmessage 监听(mounted生命周期中)
// 监听消息接收 this.$options.sockets.onmessage = (res)=>{ // res.data为服务端返回的数据 const data = json.parse(res.data); // 200为服务端连接建立成功时返回的状态码(此处根据真实后端返回值进行相应的修改) if(data.code===200){ // 连接建立成功 console.log(data.msg); }else{ // 获取服务端推送的消息 const msgobj = { msg: data.msg, avatarsrc: data.avatarsrc, userid: data.userid }; // 渲染页面:如果msgarray存在则转json if(lodash.isempty(localstorage.getitem("msgarray"))){ this.renderpage([],msgobj,0); }else{ this.renderpage(json.parse(localstorage.getitem("msgarray")),msgobj,0); } } };
实现消息发送
// 消息发送函数 sendmessage: function (event) { if (event.keycode === 13) { // 阻止编辑框默认生成div事件 event.preventdefault(); let msgtext = ""; // 获取输入框下的所有子元素 let allnodes = event.target.childnodes; for(let item of allnodes){ // 判断当前元素是否为img元素 if(item.nodename==="img"){ msgtext += `/${item.alt}/`; } else{ // 获取text节点的值 if(item.nodevalue!==null){ msgtext += item.nodevalue; } } } // 消息发送: 消息内容、状态码、当前登录用户的头像地址、用户id this.$socket.sendobj({msg: msgtext,code: 0,avatarsrc: this.$store.state.profilepicture,userid: this.$store.state.userid}); // 清空输入框中的内容 event.target.innerhtml = ""; } }
实现页面渲染
// 渲染页面函数 renderpage: function(msgarray,msgobj,status){ if(status===1){ // 页面第一次加载,如果本地存储中有数据则渲染至页面 let msgarray = []; if(localstorage.getitem("msgarray")!==null){ msgarray = json.parse(localstorage.getitem("msgarray")); for (let i = 0; i<msgarray.length;i++){ const thissendermessageobj = { "msgtext": msgarray[i].msg, "msgid": i, "avatarsrc": msgarray[i].avatarsrc, "userid": msgarray[i].userid }; // 解析并渲染 this.messageparsing(thissendermessageobj); } } }else{ // 判断本地存储中是否有数据 if(localstorage.getitem("msgarray")===null){ // 新增记录 msgarray.push(msgobj); localstorage.setitem("msgarray",json.stringify(msgarray)); for (let i = 0; i <msgarray.length; i++){ const thissendermessageobj = { "msgtext": msgarray[i].msg, "msgid": i, "avatarsrc": msgarray[i].avatarsrc, "userid": msgarray[i].userid, }; // 解析并渲染 this.messageparsing(thissendermessageobj); } }else{ // 更新记录 msgarray = json.parse(localstorage.getitem("msgarray")); msgarray.push(msgobj); localstorage.setitem("msgarray",json.stringify(msgarray)); const thissendermessageobj = { "msgtext": msgobj.msg, "msgid": date.now(), "avatarsrc": msgobj.avatarsrc, "userid": msgobj.userid }; // 解析并渲染 this.messageparsing(thissendermessageobj); } } }
实现消息解析
// 消息解析 messageparsing: function(msgobj){ // 解析接口返回的数据进行渲染 let separatereg = /(\/[^/]+\/)/g; let msgtext = msgobj.msgtext; let finalmsgtext = ""; // 将符合条件的字符串放到数组里 const resultarray = msgtext.match(separatereg); if(resultarray!==null){ for (let item of resultarray){ // 删除字符串中的/符号 item = item.replace(/\//g,""); for (let emojiitem of this.emojilist){ // 判断捕获到的字符串与配置文件中的字符串是否相同 if(emojiitem.info === item){ const imgsrc = require(`../assets/img/emoji/${emojiitem.hover}`); const imgtag = `<img src="${imgsrc}" width="28" height="28" alt="${item}">`; // 替换匹配的字符串为img标签:全局替换 msgtext = msgtext.replace(new regexp(`/${item}/`,'g'),imgtag); } } } finalmsgtext = msgtext; }else{ finalmsgtext = msgtext; } msgobj.msgtext = finalmsgtext; // 渲染页面 this.sendermessagelist.push(msgobj); // 修改滚动条位置 this.$nexttick(function () { this.$refs.messagescontainer.scrolltop = this.$refs.messagescontainer.scrollheight; }); }
dom结构
通过每条消息的userid和vuex中的存储的当前用户的userid来判断当前消息是否为对方发送
<!--消息显示--> <div class="messages-panel" ref="messagescontainer"> <div class="row-panel" v-for="item in sendermessagelist" :key="item.msgid"> <!--发送者消息样式--> <div class="sender-panel" v-if="item.userid===userid"> <!--消息--> <div class="msg-body"> <!--消息尾巴--> <div class="tail-panel"> <svg class="icon" aria-hidden="true"> <use xlink:href="#icon-zbds30duihuakuangyou" rel="external nofollow" ></use> </svg> </div> <!--消息内容--> <p v-html="item.msgtext"/> </div> <!--头像--> <div class="avatar-panel"> <img :src="item.avatarsrc" alt=""> </div> </div> <!--对方消息样式--> <div class="otherside-panel" v-else> <!--头像--> <div class="avatar-panel"> <img :src="item.avatarsrc" alt=""> </div> <!--消息--> <div class="msg-body"> <!--消息尾巴--> <div class="tail-panel"> <svg class="icon" aria-hidden="true"> <use xlink:href="#icon-zbds30duihuakuangzuo" rel="external nofollow" ></use> </svg> </div> <!--消息内容--> <p v-html="item.msgtext"/> </div> </div> </div> </div>
总结
以上所述是小编给大家介绍的vue通过配置websocket并实现群聊功能,希望对大家有所帮助!