vue+websokect实现实时聊天,可单聊、可群聊(一)

这篇具有很好参考价值的文章主要介绍了vue+websokect实现实时聊天,可单聊、可群聊(一)。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

效果图

vue+websokect实现实时聊天,可单聊、可群聊(一)

效果网站链接,安全性不符合实际上线使用,仅供学习交流

https://livequeen.top 

效果小程序二维码 (需现在web端获取账号)

vue+websokect实现实时聊天,可单聊、可群聊(一)

vue+websokect实现实时聊天,可单聊、可群聊(一)        vue+websokect实现实时聊天,可单聊、可群聊(一) 

思路

 一个实时聊天功能

第一,肯定要能够实现单聊或者群聊,所以不能无差别广播,要实现向指定用户发送广播。

第二,区分在线和离线,所以必须实现聊天信息要能保存到MYSQL数据库,即便离线,也能再次上线时收到消息。

分别新建3个表:

1、mess_user_list,用户存储对话的用户列表,type区分是否是群聊,2为是,1反之。

vue+websokect实现实时聊天,可单聊、可群聊(一)

 2、mess_group_list群关系表,储存群内成员的id集合。

vue+websokect实现实时聊天,可单聊、可群聊(一)

 3、mess_content存储聊天记录表,元素从左往右,依次为聊天信息的随机码(主键)、发送人id、接收人/群id、发送人用户名、内容、发送时间。

vue+websokect实现实时聊天,可单聊、可群聊(一)

第一步

一个实时聊天的功能,其核心就是websokect,进行双向平等对话,从后端主动推送消息到前端,实现实时推送消息的功能。关于websokect相关介绍,就不过多介绍,都能百度到,我这里推荐两个:

阮一峰老师的一篇文章

MDN文档

 第二步

写后端核心代码,conn是用户的连接信息,在用户连接上ws端口时,将用户id信息发送过来,并以其id为key存储conn连接信息,用于指定用户发送广播,实现单聊指定群聊,而broadcast()就是我们写的指定用户实现单聊的广播的方法。

let conns2 = {}// 实时聊天用户连接属性集

// websocket实时聊天
const wss2 = new WebSocket.Server({ port: 8082 })
wss2.on('connection', function connection (ws, req) {
  // 获取传递来的参数,req.url为端口号后的地址,需要处理只留下参数
  let index = req.url.indexOf('=')
  let id = req.url.substr(index + 1, req.url.length)
  console.log('成功连接实时聊天,用户:' + id)
  // 储存用户连接属性,用于识别发送对象
  conns2[id] = ws

  // 收到前端传来的消息
  ws.on('message', function incoming (message) {
    const obj = JSON.parse(message)
    // 广播给前端
    broadcast(obj)
  })

  // 关闭
  ws.on('close', function (e) {
    console.log(e, '服务端连接关闭')
  })

  // 错误
  ws.on('error', function (e) {
    console.log(e, '服务端异常')
  })
})

 forEach遍历循环,利用typeof判断conns已连接用户集合(在线集合)中是否有该用户,来判断是否在线,然后使用sendText()发送消息。

// 实时聊天发送消息处理
function broadcast (obj) {
  // 储存需要广播的对象id
  let users = []
  // 判断是否为群聊,1为否,2为是
  if (obj.type === '1') {
    users.push(obj.send_id)
    users.push(obj.accept_id)
  } else if (obj.type === '2') {
    users = obj.accept_group
  }
  // 判断是否存在发送对象
  if (users && users.length) {
    users.forEach(item => {
      // 判断是否’正在连接中‘是否有该用户
      if (typeof conns2[item] === 'undefined') {
        console.log(item + '---该用户已离线')
      } else {
        // 发送消息给前端
        conns2[item].send(JSON.stringify(obj))
      }
    })
  }
}

 第三步

写前端样式

<template>
  <!--对话总体-->
  <div class="mess">
    <!--用户列表-->
    <div class="mess_user_list">
      <!--用户本人-->
      <div class="user">
        <el-avatar :size="40" :src="userAvatar" style="margin: 5px;"></el-avatar>
        <span>{{userName}}</span>
      </div>
      <!--其他用户或群-->
      <div class="user_list">
        <div v-for="(item, index) in userList" :key="index" @click="showmessdlog(item)" class="user_list_item">
          <el-avatar :size="40" :src="userAvatar" style="margin: 5px;"></el-avatar>
          <div>
            <span>{{item.name}}</span>
            <p v-if="messlst[item.id][messlst[item.id].length-1].type !== 'null'" style="font-size: 10px;margin: 0px;margin-top: 4px;">[{{messlst[item.id][messlst[item.id].length-1].send_name}}]{{messlst[item.id][messlst[item.id].length-1].content}}</p>
            <p v-else style="font-size: 10px;margin: 0px;margin-top: 4px;">暂无消息</p>
          </div>
        </div>
      </div>
    </div>
    <!--有对话时,对话框-->
    <div v-if="acceptUser !== ''" class="mess_dialog">
      <!--对话框头部-->
      <div class="dlog_header">
        <span style="margin-left: 25px;">{{acceptUser}}</span>
      </div>
      <!--对话框内容-->
      <div class="dlog_content">
        <div v-for="(item, index) in messnowList" :key="index" class="dlog_content_item" style="margin-left: 5px;">
          <!--其他用户的消息展示-->
          <div v-if="item.send_id !== userId" class="content_other">
              <span style="font-size: 8px;">{{item.send_name}}   {{item.send_date}}</span>
              <p style="margin: 0px;margin-top: 4px;">{{item.content}}</p>
            </div>
          <!--本用户的消息展示-->
          <div v-else class="content_me">
              <span style="font-size: 8px;">{{item.send_date}}   {{item.send_name}}</span>
              <p style="margin: 0px;margin-top: 4px;">{{item.content}}</p>
            </div>
        </div>
      </div>
      <!--对话框底部-->
      <div class="dlog_footer">
        <el-input type="textarea" :rows="5" v-model="mess"></el-input>
        <el-button type="primary" @click="Wssendmess" style="float: right;margin-top: 5px;">发送</el-button>
      </div>
    </div>
    <!--无对话时,对话框-->
    <div v-else class="mess_dialog_false">
      <span>暂无消息,请选择用户对象</span>
    </div>
  </div>
</template>

CSS样式

<style scoped>
.mess{
  border-radius: 5px;
  background-clip: padding-box;
  margin:20px auto;
  width: 950px;
  height: 600px;
  border: 1px #8a8282;
  box-shadow: 0 0 10px #9b9393;
  background-color: white;
  display: flex;
}
.mess_user_list{
  width: 270px;
  height: 100%;
  background-color: #9f9c9c;
}
.mess_dialog{
  width: 680px;
  height: 600px;
}
.mess_dialog_false{
  width: 680px;
  height: 600px;
  text-align: center;
  line-height: 600px;
}
.dlog_header{
  width: 680px;
  height: 50px;
  border-bottom: 1px solid #8a8282;
  display: flex;
  align-items: center;
}
.dlog_content{
  width: 680px;
  height: 370px;
  border-bottom: 1px solid #8a8282;
  overflow-y: scroll;
}
.dlog_footer{
  width: 680px;
  height: 180px;
}
.user{
  height: 60px;
  width: 270px;
  /*垂直居中*/
  display: flex;
  align-items: center;
  border-bottom: 1px solid #0a0a0a;
}
.user_list{
  height: 540px;
  overflow-y: scroll;
}
.user_list_item{
  height: 60px;
  background-color: #b3b3b4;
  border-bottom: 1px solid #0a0a0a;
  display: flex;
  align-items: center;
}
.content_other{
  width: 650px;
}
.content_me{
  width: 650px;
  text-align: right;
}
</style>

第四步

前端js代码,连接后端ws端口,监听ws的四个状态,并各种绑定事件:

open-连接成功

close-连接关闭

error-连接错误

message-收到后端传来的消息

const ws = new WebSocket('ws://localhost:8082?id=' + id)


mounted () {
    ws.addEventListener('open', this.handleWsOpen.bind(this), false)
    ws.addEventListener('close', this.handleWsClose.bind(this), false)
    ws.addEventListener('error', this.handleWsError.bind(this), false)
    ws.addEventListener('message', this.handleWsMessage.bind(this), false)
  },
methods: {
    handleWsOpen () {
      console.log('WebSocket2已经打开 ')
    },
    handleWsClose (e) {
      console.log('WebSocket2关闭')
      console.log(e)
    },
    handleWsError (e) {
      console.log('WebSocket2发生错误')
      console.log(e)
    },
    handleWsMessage (e) {
      console.log('WebSocket2收到消息:' + e.data)
    }
}

 在连接成功ws后端端口后,前端就可以使用ws.send()发送消息给后端,同时存储消息于mysql,然后后端处理完成后,再按需发送给需要发送的前端对象即可。

附前端js完整代码:文章来源地址https://www.toymoban.com/news/detail-415612.html

<script>
import userAvatar from '@/assets/image/userAvatar.png'
import moment from 'moment'
const id = JSON.parse(localStorage.getItem('user')).id
const ws = new WebSocket('ws://localhost:8082?id=' + id)
export default {
  name: 'test6',
  data () {
    return {
      mess: '', // 输入的信息
      userAvatar: userAvatar, // 用户头像
      userName: '用户名', // 用户名
      userId: '', // 用户id
      acceptUser: '', // 对话的用户名或群名
      acceptUserId: '', // 对话用户的id
      acceptUserType: '', // 对话用户类型(个人、群)
      acceptUserList: [], // 接受群内用户集
      userList: [], // 对话用户列表
      messList: [], // 聊天内容列表
      messlst: [], // 延时复刻-聊天内容列表
      messnowList: [] // 当前对话用户的-聊天内容列表
    }
  },
  // 初始化
  created () {
    this.getuserData()
    this.getuserList()
  },
  mounted () {
    ws.addEventListener('open', this.handleWsOpen.bind(this), false)
    ws.addEventListener('close', this.handleWsClose.bind(this), false)
    ws.addEventListener('error', this.handleWsError.bind(this), false)
    ws.addEventListener('message', this.handleWsMessage.bind(this), false)
  },
  methods: {
    // 发送按钮点击事件
    async Wssendmess () {
      // 判断是否全是空格
      var message = this.mess.replace(/\s+/g, '')
      if (message !== '') {
        // 判断是单聊(1)还是群聊(2)
        if (this.acceptUserType === '2') {
          // 获取群聊成员id
          await this.getGroupUserId(this.acceptUserId)
        }
        // 发送消息格式
        var obj = JSON.stringify({
          send_id: this.userId,
          accept_id: this.acceptUserId,
          accept_group: this.acceptUserList,
          send_name: this.userName,
          content: this.mess,
          send_date: moment().format('YYYY-MM-DD HH:mm:ss'),
          mess_id: 'GP' + this.userId + Date.now(),
          type: this.acceptUserType
        })
        // 发送消息
        ws.send(obj)
        // 保存消息到数据库
        this.$api.addMessContent(JSON.parse(obj))
          .then((res) => {
            if (res.data === true) {
              console.log('对话消息添加成功!')
            } else {
              console.log('对话消息添加失败!')
            }
          })
          .catch((err) => {
            console.log(err)
          })
        // 发送完消息,重新输入框
        this.mess = ''
      }
    },
    handleWsOpen () {
      console.log('WebSocket2已经打开: ')
    },
    handleWsClose (e) {
      console.log('WebSocket2关闭: ')
      console.log(e)
    },
    handleWsError (e) {
      console.log('WebSocket2发生错误: ')
      console.log(e)
    },
    handleWsMessage (e) {
      console.log('WebSocket2收到消息' + e.data)
      var obj = JSON.parse(e.data)
      if (obj.accept_id !== this.userId) {
        // 强制刷新
        this.$forceUpdate()
        this.messlst[obj.accept_id].push(obj)
      } else {
        // 强制刷新
        this.$forceUpdate()
        this.messlst[obj.send_id].push(obj)
      }
    },
    // 获取当前用户信息
    async getuserData () {
      const id = JSON.parse(localStorage.getItem('user')).id
      const pwd = JSON.parse(localStorage.getItem('user')).password
      this.userId = id
      // 默认情况下,axios将JavaScript对象序列化为JSON。要以application/x-www-form-urlencoded格式发送数据,可以使用下面qs库对数据进行编码
      await this.$api.userinformation(id, pwd)
        .then((response) => { // 请求成功处理
          // 赋值用户名
          this.userName = response.data[0].name
          if (response.data[0].avatar_url !== '' && this.isurl(response.data[0].avatar_url) === true) {
            this.userAvatar = require('../../server/upload/' + response.data[0].avatar_url)
          }
        })
        .catch((error) => { // 请求失败处理
          console.log(error)
        })
    },
    // 获取对话列表
    async getuserList () {
      await this.$api.getMessUserList(this.userId)
        .then((res) => {
          if (res.data !== '' && res.data !== false) {
            console.log(res.data)
            // 赋值用户列表
            this.userList = res.data
            // 根据用户列表获取对应对话信息
            this.userList.forEach((item, index) => {
              this.getitemcontent(item, index)
            })
            setTimeout(() => {
              console.log(this.messList)
              this.messlst = this.messList
            }, 800)
          }
        })
    },
    // 展示对话框
    showmessdlog (item) {
      this.acceptUser = item.name
      this.acceptUserId = item.id
      this.acceptUserType = item.type
      this.messnowList = this.messlst[this.acceptUserId]
    },
    // 获取某个用户或群聊聊天的内容并赋值
    async getitemcontent (item, index) {
      await this.$api.getMessContent(item.type, this.userId, item.id)
        .then((res) => {
          // 判断是否有消息不为空
          if (res.data !== '' & res.data !== false) {
            // 赋值给消息数组
            this.messList[item.id] = res.data
          } else if (res.data === false) {
            this.messList[item.id] = [{type: 'null'}]
          }
        })
    },
    // 获取群成员id
    async getGroupUserId (groupid) {
      await this.$api.getGroupUserList(groupid)
        .then((res) => {
          if (res.data !== '' & res.data !== false) {
            var data = res.data
            this.acceptUserList = []
            data.forEach((item) => {
              this.acceptUserList.push(item.user_id)
            })
          }
        })
    }
  }
}
</script>

到了这里,关于vue+websokect实现实时聊天,可单聊、可群聊(一)的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 基于Springboot+WebSocket+Netty实现在线聊天、群聊系统

    此文主要实现在好友添加、建群、聊天对话、群聊功能,使用Java作为后端语言进行支持,界面友好,开发简单。 2.1、下载安装IntelliJ IDEA(后端语言开发工具),Mysql数据库,微信Web开发者工具。 1.创建maven project 先创建一个名为SpringBootDemo的项目,选择【New Project】 然后在弹出

    2024年02月14日
    浏览(42)
  • 使用 vue-3-socket.io 插件以及node.js实现实时聊天(1)

     这篇文章使用选项式API的写法,以实现群聊和私聊为主 客户端自然是对应使用vue3框架,服务端使用node.js配合express、http、socket.io、file等库来实现,具体如下: 1、下载所需的依赖 2、做socket客户端配置 注:\\\"http://localhost:3000\\\",该地址端口是对应后面配置服务端时所开放的端

    2024年02月05日
    浏览(43)
  • uniapp:聊天消息列表(好友列表+私人单聊)支持App、H5、小程序

          🎬 江城开朗的豌豆 :个人主页  🔥 个人专栏  :《 VUE 》 《 javaScript 》  📝  个人网站  :《 江城开朗的豌豆🫛 》  ⛺️ 生活的理想,就是为了理想的生活 ! 目录  ⭐  文章简介(效果图展示)           📟 插件传送门:聊天消息列表  📘  文章背景

    2024年04月15日
    浏览(55)
  • 网站在线客服系统实时语音视频聊天实战开发,利用peerjs vue.js实现webRTC网页音视频客服系统...

    webRTC机制和peerjs库的介绍在其他博客中已经有了很多介绍,这里我直接搬运过来 WebRTC(Web Real-Time Communication)即:网页即时通信。 简单点讲,它可以实现浏览器网页与网页之间的音视频实时通信(或传输其它任何数据),目前主流浏览器都支持该API,WebRTC现在已经纳入W3C标准。

    2024年02月04日
    浏览(59)
  • TCP案例-实时群聊

    服务器端循环获取所有的客户端socket,放到一个 socket的list 中,等到需要通信的时候,调用相对应的管道就可以了 客户端和服务器端一样,纪要发送消息,也要接受消息 读取线程是子线程 只能实现群发,无法端到端 也会发给自己

    2024年04月28日
    浏览(33)
  • 使用PHP实现实时聊天功能的匿名聊天与加密传输

    使用PHP实现实时聊天功能的匿名聊天与加密传输 随着互联网的发展,人与人之间的交流方式也发生了天翻地覆的变化。其中,实时聊天功能成为了一种越来越受欢迎的交流方式。对于很多网站来说,提供匿名聊天功能能够吸引更多的用户参与,但也需要考虑到用户信息的安全

    2024年02月13日
    浏览(62)
  • webSocket及使用webSocket实现实时聊天通信

    webSocket在开始的时候依旧使用的是http协议,只不过后面保持tcp持久链接,是一种全双工通信。webSockets和http很像,它的请求url用的是ws、wss,对应http、https 初始化 npm init -y 安装ws依赖 npm i --save ws 写服务器代码 服务器启动 node server.js 配置前端代码,即告诉浏览器这个请求不要

    2023年04月11日
    浏览(41)
  • 【微信开发】微信小程序实现实时聊天功能

    最近在做一个项目,需要运用到实时聊天功能,分享一下。      分为: 1.界面如何布局以及细节; 2.如何实现实时更新; 3.全部代码展示;         一、界面如何布局以及细节:         1.说到底,聊天界面就是循环一个数组,每一行为一个单位(item),循环(wx.for)我

    2024年02月04日
    浏览(59)
  • js和java交互websocket实现实时聊天

    目录 1.运用场景(实现效果):  2.代码及其讲解: js配置: (1)时间判断函数: (2)建立ws连接: 结束!         前端运用原生js来与后端提供的ws进行交互,实现在线实时聊天,离线储存数据方便读取聊天记录的效果,效果图如下: (1)时间判断函数: (2)建立ws连接:

    2024年02月12日
    浏览(38)
  • 使用WebSocket方式能将群聊信息实时群发给所有在线用户

    1.1 什么是WebSocket WebSocket是一种在单个TCP连接上进行全双工通信的网络协议。它是为了在Web浏览器和Web服务器之间提供实时、双向的通信而设计的。传统的HTTP协议是一种单向通信协议,客户端发送请求,服务器响应,然后连接就关闭了。而WebSocket允许在客户端和服务器之间建

    2024年02月03日
    浏览(38)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包