即时通信嵌入测试

This commit is contained in:
2024-10-21 16:54:36 +08:00
parent 2d5e601636
commit b9045d5d4c
30 changed files with 1224 additions and 21 deletions

View File

@@ -0,0 +1,44 @@
import request from '@/utils/request'
// 查询通信目录列表
export function listContact(query) {
return request({
url: '/system/contact/list',
method: 'get',
params: query
})
}
// 查询通信目录详细
export function getContact(id) {
return request({
url: '/system/contact/' + id,
method: 'get'
})
}
// 新增通信目录
export function addContact(data) {
return request({
url: '/system/contact',
method: 'post',
data: data
})
}
// 修改通信目录
export function updateContact(data) {
return request({
url: '/system/contact',
method: 'put',
data: data
})
}
// 删除通信目录
export function delContact(id) {
return request({
url: '/system/contact/' + id,
method: 'delete'
})
}

View File

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.2 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.3 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.0 KiB

View File

@@ -35,7 +35,7 @@ export default {
},
data() {
return {
title: '项目管理系统',
title: '福安德综合办公系统',
logo: logoImg
}
}

View File

@@ -40,6 +40,8 @@ import VueMeta from 'vue-meta'
// 字典数据组件
import DictData from '@/components/DictData'
import webSocket from "./utils/websocket";
//打印组件 添加时间2024年3月9日
import Print from 'vue-print-nb'
@@ -55,6 +57,7 @@ Vue.prototype.selectDictLabel = selectDictLabel
Vue.prototype.selectDictLabels = selectDictLabels
Vue.prototype.download = download
Vue.prototype.handleTree = handleTree
Vue.prototype.$webSocket = webSocket
// 全局组件挂载
Vue.component('DictTag', DictTag)
@@ -91,9 +94,32 @@ Vue.use(Element, {
Vue.config.productionTip = false
new Vue({
let newVue = new Vue({
el: '#app',
created() {
//监听用户窗口是否关闭
window.addEventListener('beforeunload', this.closeSocket);
},
destroyed() {
window.removeEventListener('beforeunload', this.closeSocket);
},
methods: {
onBeforeUnload(event) {
// 在这里编写你想要执行的代码
// 例如:发送数据到服务器或者显示警告信息
// 设置event.returnValue以显示浏览器默认的警告信息
event.returnValue = '您可能有未保存的更改!';
},
closeSocket() {
//关闭websocket连接
this.$websocket.close();
}
},
router,
store,
render: h => h(App)
})
export default newVue

View File

@@ -1,6 +1,6 @@
import { login, logout, getInfo } from '@/api/login'
import { getToken, setToken, removeToken } from '@/utils/auth'
import newVue from '@/main'
const user = {
state: {
token: getToken(),
@@ -25,7 +25,10 @@ const user = {
},
SET_PERMISSIONS: (state, permissions) => {
state.permissions = permissions
}
},
SET_ID: (state, id) => {
state.id = id
},
},
actions: {
@@ -51,6 +54,7 @@ const user = {
return new Promise((resolve, reject) => {
getInfo().then(res => {
const user = res.data.user
console.log(user)
const avatar = (user.avatar == "" || user.avatar == null) ? require("@/assets/images/profile.jpg") : user.avatar;
if (res.data.roles && res.data.roles.length > 0) { // 验证返回的roles是否是一个非空数组
commit('SET_ROLES', res.data.roles)
@@ -58,8 +62,11 @@ const user = {
} else {
commit('SET_ROLES', ['ROLE_DEFAULT'])
}
commit('SET_ID',user.userId)
commit('SET_NAME', user.userName)
commit('SET_AVATAR', avatar)
// TODO 获取用户信息时检查socket连接状态并进行连接
newVue.$webSocket.initWebSocket()
resolve(res)
}).catch(error => {
reject(error)
@@ -75,6 +82,8 @@ const user = {
commit('SET_ROLES', [])
commit('SET_PERMISSIONS', [])
removeToken()
// TODO 用户推出登陆后 关闭socket连接
newVue.$webSocket.close()
resolve()
}).catch(error => {
reject(error)

View File

@@ -0,0 +1,123 @@
import { Notification } from "element-ui";
import { getToken } from "./auth";
import store from '../store'
var socket = null;//实例对象
var lockReconnect = false; //是否真正建立连接
var timeout = 20 * 1000; //20秒一次心跳
var timeoutObj = null; //心跳倒计时
var serverTimeoutObj = null; //服务心跳倒计时
var timeoutnum = null; //断开 重连倒计时
const initWebSocket = async () => {
if ("WebSocket" in window) {
if (!store.state.user.id) {
console.log("未登录websocket工具获取不到userId")
}else {
const wsUrl = process.env.VUE_APP_SOCKET_SERVER + store.state.user.id;
console.log("连接已开启")
socket = new WebSocket(wsUrl);
socket.onerror = webSocketOnError;
socket.onmessage = webSocketOnMessage;
socket.onclose = closeWebsocket;
socket.onopen = openWebsocket;
console.log(socket)
}
} else {
Notification.error({
title: "错误",
message: "您的浏览器不支持websocket请更换Chrome或者Firefox",
});
}
}
//建立连接
const openWebsocket = (e) => {
start();
}
const start = ()=> {
//开启心跳
timeoutObj && clearTimeout(timeoutObj);
serverTimeoutObj && clearTimeout(serverTimeoutObj);
timeoutObj = setTimeout(function() {
//这里发送一个心跳,后端收到后,返回一个心跳消息
if (socket.readyState == 1) {
//如果连接正常
// socket.send("heartbeat");
} else {
//否则重连
reconnect();
}
serverTimeoutObj = setTimeout(function() {
//超时关闭
socket.close();
}, timeout);
}, timeout);
}
//重新连接
const reconnect =() => {
if (lockReconnect) {
return;
}
lockReconnect = true;
//没连接上会一直重连,设置延迟避免请求过多
timeoutnum && clearTimeout(timeoutnum);
timeoutnum = setTimeout(function() {
//新连接
initWebSocket();
lockReconnect = false;
}, 1000);
}
//重置心跳
const reset =() => {
//清除时间
clearTimeout(timeoutObj);
clearTimeout(serverTimeoutObj);
//重启心跳
start();
}
const sendWebsocket = (message) =>{
socket.send(message);
}
const webSocketOnError = (e) => {
initWebSocket();
reconnect();
}
//服务器返回的数据
const webSocketOnMessage = (e)=> {
//判断是否登录
console.log("111111111111"+e)
if (getToken()) {
//window自定义事件
window.dispatchEvent(
new CustomEvent("onmessageWS", {
detail: {
data: JSON.parse(e?.data)
},
})
);
}
// socket.onmessage(e)
reset();
}
const closeWebsocket=(e) => {
reconnect();
}
//断开连接
const close =() => {
//WebSocket对象也有发送和关闭的两个方法只需要在自定义方法中分别调用send()和close()即可实现。
socket.close();
}
//具体问题具体分析,把需要用到的方法暴露出去
export default { initWebSocket, sendWebsocket, webSocketOnMessage, close };

View File

@@ -1,7 +1,7 @@
<template>
<div class="login">
<el-form ref="loginForm" :model="loginForm" :rules="loginRules" class="login-form">
<h3 class="title">工程项目进度及成本控制管理系统</h3>
<h3 class="title">福安德综合办公系统</h3>
<el-form-item prop="username">
<el-input
v-model="loginForm.username"
@@ -56,7 +56,7 @@
</el-form>
<!-- 底部 -->
<div class="el-login-footer">
<span>Copyright © 2021-2023 KonBAI All Rights Reserved.</span>
<span>Copyright © 2024 spark All Rights Reserved.</span>
</div>
</div>
</template>

View File

@@ -0,0 +1,361 @@
<template>
<div class="app-container">
<el-container class="app">
<el-aside width="calc(30% - 20px)" style="background-color: white">
<el-container>
<el-header>
<el-row>
<el-col :span="24">
<el-input suffix-icon="el-icon-search" placeholder="Enter 回车搜索联系人" v-model="contactQueryParams.userName" @keyup.enter.native="getContactList"/>
</el-col>
</el-row>
<el-row style="margin-top: 5px">
<!-- TODO 这里搞一个弹窗去搜索用户表 从而进行添加好友操作 -->
<el-button size="mini">添加联系人</el-button>
</el-row>
<el-row style="margin-top: 5px">
<el-button size="mini">全部</el-button>
<el-button size="mini">个人</el-button>
<el-button size="mini">群聊</el-button>
</el-row>
</el-header>
<div v-loading="contactListLoading" style="margin-top: 30px">
<el-main v-if="contactList.length > 0" v-infinite-scroll="contactLoadMore" :infinite-scroll-distance="750" :infinite-scroll-disabled="contactListTotal < 10" class="msgListMain">
<el-row class="msgUserList" v-for="(item, index) in contactList" :key="item.contactUserId" :style="index > 0 && 'margin-top: 10px'" @click.native="loadMessage(item.id)">
<el-col :span="6">
<el-image :src="(item.user.avatar === '' || item.user.avatar == null) ? require('@/assets/images/profile.jpg') : item.user.avatar" fit="fill" style="width: 70%;border-radius: 50%;"/>
</el-col>
<el-col :span="18">
<el-row>
<el-col :span="15"><span style="font-weight: 500; font-size: 16px">{{item.user.nickName}}</span></el-col>
<el-col :span="5">
<el-divider direction="vertical"/>
</el-col>
</el-row>
<el-row>
<el-col :span="5" style="font-size: 13px; text-overflow: ellipsis; white-space: nowrap">
<span><i class="el-icon-circle-check"></i> {{item.endMsg}}</span>
</el-col>
<el-col :span="5" style="float: right">
<el-dropdown class="hover_down_menu">
<span class="el-dropdown-link">
<i class="el-icon-arrow-down el-icon-more"></i>
</span>
<el-dropdown-menu slot="dropdown">
<el-dropdown-item>置顶</el-dropdown-item>
<el-dropdown-item>删除</el-dropdown-item>
</el-dropdown-menu>
</el-dropdown>
</el-col>
</el-row>
</el-col>
</el-row>
</el-main>
<el-main v-else class="msgListMain_empty">
<el-row>
<el-col :span="24">
<img src="@/assets/images/contact.png" style="width: 80%; height: 80%"/>
</el-col>
</el-row>
<el-row>
<el-col :span="24">
<span style="color: gray">暂无联系人</span>
</el-col>
</el-row>
</el-main>
</div>
</el-container>
</el-aside>
<el-main :class="currentContact.id ? 'main' : 'main_empty'" v-loading="msgListLoading">
<div v-if="currentContact.id">
<el-row>
<el-col :span="8" style="color: #666">
<span style="font-weight: 500; font-size: 16px">{{currentContact.userName}}</span>
<span style="font-size: 16px; margin-left: 30px">{{currentContact.industry}}</span>
<el-divider direction="vertical"/>
<span style="font-size: 16px;">{{currentContact.job}}</span>
</el-col>
</el-row>
<el-row>
<el-col :span="24">
<span>{{currentContact.major}}</span>
<span style="color: red; font-size: 17px; margin-left: 20px">{{currentContact.salary}}</span>
<span style="margin-left: 20px">{{currentContact.city}}</span>
</el-col>
</el-row>
<el-row>
<el-col :span="24" class="msg_content" id="message_content">
<el-row v-for="(item, index) in msgList" :key="item.id" :style="index > 0 && 'margin-top: 30px'">
<div v-if="item.userId === currentContact.contactUserId">
<el-col :span="2" style="text-align: center">
<el-image :src="currentContact.avatar" fit="cover" style="width: 40%;border-radius: 50%"/>
</el-col>
<el-col :span="10" style="font-size: 16px; line-height: 40px;">
<span>{{item.content}}</span>
<span style="font-size: 11px; color: gray; margin-left: 5px">{{item.createTime}}</span>
</el-col>
</div>
<div v-else>
<el-col :span="24" style="font-size: 16px;">
<div class="chat_bubble">
<span>{{item.content}}</span>
</div>
<i class="el-icon-circle-check" style="float: right; margin-right: 5px; color: lightgray; vertical-align: bottom; margin-top: 23px"></i>
<span style="font-size: 11px; color: gray; margin-right: 5px; float: right; margin-top: 25px">{{item.createTime}}</span>
</el-col>
</div>
</el-row>
<el-row id="message_content_end" style="height: 15px"><el-col></el-col></el-row>
</el-col>
</el-row>
<el-row>
<el-col :span="24">
<el-row>
<el-col :span="24">
<el-popover
placement="top-start"
trigger="click">
<div>
<VEmojiPicker :showSearch="false" @select="insertEmoji" />
</div>
<img slot="reference" src="@/assets/images/emoji.png" title="表情" class="input_top_menu_img"/>
</el-popover>
</el-col>
</el-row>
<el-input type="textarea" :rows="3" v-model="inputVal" style="font-size: 17px; color: black" @keyup.enter.native="send" placeholder="Enter 回车发送消息"/>
</el-col>
</el-row>
</div>
<div v-else>
<el-row>
<el-col :span="24">
<img src="@/assets/images/message.png"/>
</el-col>
</el-row>
<el-row>
<el-col :span="24">
<span style="color: gray">与您进行过沟通的联系人都会在左侧列表中显示</span>
</el-col>
</el-row>
</div>
</el-main>
</el-container>
</div>
</template>
<script>
import { VEmojiPicker } from 'v-emoji-picker';
import { parseTime } from '@/utils/ruoyi';
import { listContact, getContact } from "@/api/system/contact";
import { addMessage } from "@/api/system/message";
export default {
name: "chat",
components: {
VEmojiPicker
},
data() {
return {
//联系人列表
contactList: [],
contactListTotal: 0,
contactListLoading: false,
//消息记录
msgList: [],
msgListTotal: 0,
msgListLoading: false,
inputVal: '',
search: '',
contactUserId: null,
userId: null,
contactQueryParams: {
pageSize: 10,
pageNum: 1
},
currentContact: {}
}
},
mounted() {
window.addEventListener("onmessageWS", this.subscribeMessage);
},
created() {
this.userId = this.$store.state.user.id;
this.subscribeMessage();
this.getContactList();
},
methods: {
getContactList() {
this.contactListLoading = true;
this.contactQueryParams.userId = this.userId;
listContact(this.contactQueryParams).then(response => {
if (response.code === 200) {
this.contactList = response.rows;
this.contactListTotal = response.total;
const contactUserId = this.$route.query.userId;
if (contactUserId) {
this.contactUserId = contactUserId;
let contact = response.rows.find(row => row.contactUserId == contactUserId);
this.loadMessage(contact.id);
}
}
this.contactListLoading = false;
})
},
contactLoadMore() {
// this.contactQueryParams.pageSize = 5;
// this.contactQueryParams.pageNum++;
this.getContactList();
},
loadMessage(concatId) {
this.msgListLoading = true;
getContact(concatId).then(response => {
if (response.code === 200) {
this.currentContact = response.data;
this.msgList = response.data.messages;
}
this.msgListLoading = false;
this.fleshScroll();
})
},
insertEmoji(emoji) {
this.inputVal += emoji.data;
},
send() {
const message = {
contactId: this.currentContact.id,
userId: this.userId,
content: this.inputVal,
roomId: this.currentContact.roomId
}
this.msgList.push({
...message,
id: this.msgList.length + 1,
createTime: parseTime(new Date())
})
this.fleshLastMsg();
addMessage(message);
const msg = {
sendUserId: this.userId,
sendUserName: this.$store.state.user.name,
userId: this.currentContact.contactUserId,
type: "chat",
detail: this.inputVal
}
this.$websocket.sendWebsocket(JSON.stringify(msg));
this.inputVal = '';
this.fleshScroll();
},
subscribeMessage(res) {
console.log(res);
if (res) {
const { sendUserId, sendUserName, userId, type, detail } = res.detail.data;
const message = {
id: 1,
contactId: userId,
userId: sendUserId,
content: detail,
roomId: this.currentContact.roomId,
createTime: parseTime(new Date())
}
this.msgList.push(message);
this.fleshLastMsg();
this.fleshScroll();
}
},
fleshLastMsg() {
const index = this.contactList.findIndex(e => e.id === this.currentContact.id);
this.contactList[index].endMsg = this.msgList[this.msgList.length - 1].content;
},
fleshScroll() {
this.$nextTick(() => {
document.getElementById("message_content_end").scrollIntoView();
})
}
}
}
</script>
<style scoped lang="scss">
.app-container {
background: linear-gradient(180deg, rgba(0, 190, 189, .1), rgba(136, 255, 254, .2) 50%, rgba(242, 244, 247, .1));
}
.app {
background-color: white;
border-radius: 12px 12px 0 0;
}
.msgListMain {
height: 500px;
overflow-y:auto;
margin-top: 15px;
}
.msgUserList {
border-radius: 10px;
}
.msgListMain_empty {
display: flex;
height: 500px;
flex-direction: column;
justify-content: center;
align-items: center;
text-align: center;
}
.hover_down_menu {
display: none;
}
.msgUserList:hover {
background-color: #f2f2f2;
cursor: pointer;
}
.msgUserList:hover .hover_down_menu{
display: block;
}
.el-dropdown-link {
cursor: pointer;
}
.el-icon-arrow-down {
font-size: 15px;
font-weight: 500;
}
.main {
background-color: white;
height: 600px;
margin-left: 5px;
}
.main_empty {
display: flex;
background-color: white;
height: 600px;
flex-direction: column;
justify-content: center;
align-items: center;
}
.main_empty .el-row {
text-align: center;
}
.main_empty img {
width: 25%;
}
.msg_content {
margin-top: 30px;
height: 390px;
overflow: auto;
//background-color: gray;
}
.chat_bubble {
float: right;
margin-right: 35px;
color: #333;
background-color: rgba(0, 190, 189, .2);
height: 40px;
line-height: 40px;
padding: 0 12px 0 12px;
border-radius: 5px;
}
.input_top_menu_img{
width: 22px;
height: 22px;
cursor: pointer;
}
</style>

View File

@@ -1,7 +1,7 @@
<template>
<div class="register">
<el-form ref="registerForm" :model="registerForm" :rules="registerRules" class="register-form">
<h3 class="title">工程项目进度及成本控制管理系统</h3>
<h3 class="title">福安德综合办公系统</h3>
<el-form-item prop="username">
<el-input v-model="registerForm.username" type="text" auto-complete="off" placeholder="账号">
<svg-icon slot="prefix" icon-class="user" class="el-input__icon input-icon" />
@@ -61,7 +61,7 @@
</el-form>
<!-- 底部 -->
<div class="el-register-footer">
<span>Copyright © 2018-2023 KonBAI All Rights Reserved.</span>
<span>Copyright © 2018-2023 spark All Rights Reserved.</span>
</div>
</div>
</template>