vue集成環信IM

vue 集成環信im 簡單demovue

 

環信AppKey:1106190415055331#testgit

測試帳號: test1  123456  test2  123456  test3  123456github

默認登陸test1vuex

用vuex統一管理聊天對象、聊天信息等api

 點擊相應的對象,發送文本瀏覽器

1. 引入環信依賴服務器

 

2. vuex 環信前期準備全局狀態控制網絡

import {dotData} from '@/utils'
import Vue from 'vue'
// im聊天狀態
const IM_CHART_DIALOG = {
state: {
visible: false, // 顯示隱藏
rosterList: [], // 好友列表
groupList: [], // 羣組列表
groupMessageMap: {}, // 聊天信息集合
groupMessageHistoryMap: {}, // 歷史消息
currentGroupId: '', // 當前活動
extMap: {}, // 附加信息
eNameMap: {}, // 用戶名信息
disabledGroupId: [],
groupIdMap: {}, // 羣消息
readonlyGroupIdArr: [] // 只讀的groupId
},
mutations: { // 修改狀態
SET_IM_CHART_DIALOG_VISIBLE(state, {visible}) { // 彈窗可見/隱藏
state.visible = visible
},
SET_IM_CHART_ROSTERLIST(state, rosterList) {
state.rosterList = rosterList
},
SET_IM_CHART_GROUPLIST(state, groupList) {
state.groupList = groupList
},
PUSH_IM_CHART_DIALOG_GROUP_MESSAGE(state, {message, groupId}) { // 用戶聊天信息
let chart = state.groupMessageMap[groupId] || []
chart.push(message)
Vue.set(state.groupMessageMap, groupId, chart)
},
UN_SHIFT_IM_CHART_DIALOG_GROUP_MESSAGE(state, {message, groupId}) { // 用戶聊天信息
let chart = state.groupMessageMap[groupId] || []
chart.unshift(message)
Vue.set(state.groupMessageMap, groupId, chart)
},
REMOVE_IM_CHART_DIALOG_GROUP_MESSAGE(state, groupId) { // 移除一個羣聊
let chart = state.groupMessageMap
delete chart[groupId]
state.groupMessageMap = chart
},
SET_IM_CHART_DIALOG_GROUP_ID(state, groupId) { // 設置當前用戶組
state.currentGroupId = groupId
},
SET_IM_CHART_EXT_MAP(state, {groupId, ext}) { // 設置group 附加信息
if (!state.extMap[groupId]) {
Vue.set(state.extMap, groupId, ext)
}
},
REMOVE_IM_CHART_EXT_MAP(state, groupId) { // 移除一個羣聊追加消息
let ext = state.extMap
delete ext[groupId]
state.extMap = ext
},
REMOVE_IM_CHART_GROUP_ID(state, groupId) {
let groupIdMap = state.groupIdMap
delete groupIdMap[groupId]
state.groupIdMap = groupIdMap
},
// eName 和用戶進行映射
SET_IM_CHART_DIALOG_MEMBER_E_NAME(state, {eName, member}) {
Vue.set(state.eNameMap, eName, member)
},
// 使當前用戶組消息已讀
SET_IM_CHART_DIALOG_GROUP_ID_READ(state, groupId) {
if (state.visible) {
let chart = state.groupMessageMap[groupId] || []
chart.forEach(v => {
Vue.set(v, 'isRead', true)
})
Vue.set(state.groupMessageMap, groupId, chart)
}
},
// 禁用的groupId
PUSH_IM_CHART_DIALOG_DISABLE_GROUP_ID(state, groupId) {
state.disabledGroupId.push(groupId)
},
// 羣組信息
SET_IM_CHART_DIALOG_GROUP_ID_INFO(state, {groupId, info}) {
if (groupId) {
Vue.set(state.groupIdMap, groupId, info)
}
},
// 清除羣消息
CLEAR_IM_CHART_DIALOG_GROUP_ID_INFO(state) {
state.groupIdMap = {}
},
// 歷史記錄拉取 第一次拉取爲真
SET_IM_GROUP_MESSAGE_HISTORY_FIRST(state, {groupId}) {
let obj = state.groupMessageHistoryMap[groupId] || {}
Vue.set(obj, 'first', true)
Vue.set(state.groupMessageHistoryMap, groupId, obj)
},
// 歷史記錄拉取 已經沒有數據了
SET_IM_GROUP_MESSAGE_HISTORY_OVER(state, {groupId}) {
let obj = state.groupMessageHistoryMap[groupId] || {}
Vue.set(obj, 'over', true)
Vue.set(state.groupMessageHistoryMap, groupId, obj)
},
// 只讀的消息列表
SET_IM_READ_ONLY_GROUP_ID(state, {groupId}) {
state.readonlyGroupIdArr.push(groupId)
}
},
actions: { // 調用方法(接收傳進來的參數)
// 展現彈窗
SET_IM_CHART_DIALOG_VISIBLE({commit}, {visible}) { // 彈窗可見
commit('SET_IM_CHART_DIALOG_VISIBLE', {visible})
},
// 好友列表
SET_IM_CHART_ROSTERLIST({commit}, rosterList) {
commit('SET_IM_CHART_ROSTERLIST', rosterList)
},
// 羣組列表
SET_IM_CHART_GROUPLIST({commit}, groupList) {
commit('SET_IM_CHART_GROUPLIST', groupList)
},
// 添加消息
PUSH_IM_CHART_DIALOG_GROUP_MESSAGE({commit}, message) { // 聊天信息組
const groupId = dotData(message, 'groupId')
const ext = dotData(message, 'ext')
delete message.ext
commit('PUSH_IM_CHART_DIALOG_GROUP_MESSAGE', {groupId, message})
commit('SET_IM_CHART_EXT_MAP', {groupId, ext}) // 設置附加消息
},
UN_SHIFT_IM_CHART_DIALOG_GROUP_MESSAGE({commit}, message) { // 聊天信息組
const groupId = dotData(message, 'groupId')
const ext = dotData(message, 'ext')
delete message.ext
commit('UN_SHIFT_IM_CHART_DIALOG_GROUP_MESSAGE', {groupId, message})
commit('SET_IM_CHART_EXT_MAP', {groupId, ext}) // 設置附加消息
},
REMOVE_IM_CHART_DIALOG_GROUP_ID({commit, getters}, groupId) {
commit('REMOVE_IM_CHART_DIALOG_GROUP_MESSAGE', groupId) // 移除消息
commit('REMOVE_IM_CHART_EXT_MAP', groupId) // 移除附加消息
// 移除一個羣
commit('REMOVE_IM_CHART_GROUP_ID', groupId)
commit('PUSH_IM_CHART_DIALOG_DISABLE_GROUP_ID', groupId)
if (groupId === getters.GET_IM_CHART_DIALOG_GROUP_ID) {
commit('SET_IM_CHART_DIALOG_GROUP_ID', '')
}
},
// 添加追加信息
SET_IM_CHART_EXT_MAP({commit}, {groupId, ext}) {
commit('SET_IM_CHART_EXT_MAP', {groupId, ext})
},
// 設置當前groupId
SET_IM_CHART_DIALOG_GROUP_ID({commit}, groupId) {
commit('SET_IM_CHART_DIALOG_GROUP_ID', groupId)
commit('SET_IM_CHART_DIALOG_GROUP_ID_READ', groupId)
},
// 添加禁用的groupId
PUSH_IM_CHART_DIALOG_DISABLE_GROUP_ID({commit, getters}, groupId) {
if (groupId) {
commit('PUSH_IM_CHART_DIALOG_DISABLE_GROUP_ID', groupId)
if (groupId === getters.GET_IM_CHART_DIALOG_GROUP_ID) {
commit('SET_IM_CHART_DIALOG_GROUP_ID', '')
}
}
},
SET_IM_CHART_DIALOG_GROUP_ID_INFO({commit}, {groupId, info}) { // 設置羣聊信息
commit('SET_IM_CHART_DIALOG_GROUP_ID_INFO', {groupId, info})
if (Array.isArray(info.members)) {
info.members.forEach(v => {
commit('SET_IM_CHART_DIALOG_MEMBER_E_NAME', {eName: v.userEasemobName, member: v})
})
}
},
CLEAR_IM_CHART_DIALOG_GROUP_ID_INFO({commit}) { // 清空羣信息
commit('CLEAR_IM_CHART_DIALOG_GROUP_ID_INFO')
},
SET_IM_GROUP_MESSAGE_HISTORY_FIRST({commit}, {groupId}) { // 第一次訪問歷史消息
commit('SET_IM_GROUP_MESSAGE_HISTORY_FIRST', {groupId})
},
SET_IM_GROUP_MESSAGE_HISTORY_OVER({commit}, {groupId}) { // 已無歷史消息
commit('SET_IM_GROUP_MESSAGE_HISTORY_OVER', {groupId})
},
SET_IM_READ_ONLY_GROUP_ID({commit}, {groupId}) { // 只讀消息
commit('SET_IM_READ_ONLY_GROUP_ID', {groupId})
}
},
getters: { // 彈窗展現
GET_IM_CHART_DIALOG_VISIBLE: state => {
return state.visible
},
GET_IM_CHART_ROSTERLIST: state => {
return state.rosterList || []
},
GET_IM_CHART_GROUPLIST: state => {
return state.groupList || []
},
GET_IM_CHART_DIALOG_GROUP_ID: state => { // 獲取當前groupId
return state.currentGroupId
},
GET_IM_CHART_EXT_MAP: (state) => (groupId) => { // 根據用戶組獲取
if (state.extMap[groupId]) {
return state.extMap[groupId]
}
},
GET_IM_CHART_DIALOG_GROUP_MESSAGE: (state) => (groupId) => { // 獲取消息組
if (groupId) {
return state.groupMessageMap[groupId] || []
}
return []
},
GET_IM_CHART_DIALOG_MEMBER_E_NAME: (state) => (eName) => { // 根據環信name 獲取用戶信息
if (eName) {
return state.eNameMap[eName] || {}
}
return state.eNameMap
},
// 獲取未讀消息條數
GET_IM_CHART_DIALOG_UN_READ_LENGTH: (state, getters) => (groupId) => { // 根據環信name 獲取用戶信息
if (Array.isArray(groupId)) {
let temp = 0
groupId.forEach(id => {
temp += getters.GET_IM_CHART_DIALOG_GROUP_MESSAGE(groupId).filter(v => v.isRead === false).length
})
return temp
} else {
return getters.GET_IM_CHART_DIALOG_GROUP_MESSAGE(groupId).filter(v => v.isRead === false).length
}
},
// 所有未讀消息
GET_IM_CHART_DIALOG_ALL_UN_READ_LENGTH: (state) => { // 根據環信name 獲取用戶信息
let temp = 0
for (let k in state.groupMessageMap) {
if (Array.isArray(state.groupMessageMap[k])) {
temp += state.groupMessageMap[k].filter(v => v.isRead === false).length
}
}
return temp
},
// 獲取最後一條消息
GET_IM_CHART_DIALOG_LAST_MESSAGE: (state, getters) => (groupId) => {
let temp = ''
if (Array.isArray(groupId)) {
groupId.forEach(id => {
let charts = getters.GET_IM_CHART_DIALOG_GROUP_MESSAGE(id)
let lastChart = ''
if (charts.length) {
lastChart = charts[charts.length - 1]
}
if (temp) {
if (lastChart) {
if (temp.timeStr < lastChart.timeStr) {
temp = lastChart
}
}
} else {
if (lastChart) {
temp = lastChart
}
}
})
} else {
let charts = getters.GET_IM_CHART_DIALOG_GROUP_MESSAGE(groupId)
if (charts.length) {
temp = charts[charts.length - 1]
}
}
if (temp) {
switch (temp.fileType) {
case 'audio':
return '[語音]'
case 'image':
return '[圖片]'
default :
return temp.sourceMsg
}
}
},
// 歷史消息對象
GET_IM_GROUP_MESSAGE_HISTORY: (state) => (groupId) => {
return dotData(state.groupMessageHistoryMap, groupId) || {}
},
// 是隻讀消息
IS_IM_READ_ONLY_GROUP_ID: (state) => (groupId) => {
return state.readonlyGroupIdArr.indexOf(groupId) >= 0
}
}
}

export default IM_CHART_DIALOG

3. 構建環信實例app

import store from '@/store'
import {alert, dotData} from '@/utils'
import moment from 'moment'

export const Easemob = class Easemob {
static instance = null
connection = null
config = {}
username = ''
WebIM = window.WebIM

// 初始化
constructor() {
if (Easemob.instance) {
return Easemob.instance
}
this.config = window.WebIM.config
Easemob.instance = this
}

// 獲取當前時間
_getTimeString() {
return moment().format('YYYY-MM-DD HH:mm:ss')
}

// 獲取鏈接
getConnection() {
if (!this.connection) {
const Connection = window.WebIM.connection
this.connection = new Connection({
isMultiLoginSessions: this.config.isMultiLoginSessions,
https: typeof this.config.https === 'boolean' ? this.config.https : location.protocol === 'https:',
url: this.config.xmppURL,
isAutoLogin: true,
heartBeatWait: this.config.heartBeatWait,
autoReconnectNumMax: this.config.autoReconnectNumMax,
autoReconnectInterval: this.config.autoReconnectInterval,
apiUrl: this.config.apiURL
})
}
return this.connection
}

// 發送時追加信息
appendMessage(message, className = 'in') {
message = this.createClassNameAndTime(message, className)
message = this._setExt(message)
return message
}

// 添加時間和來源
createClassNameAndTime(message, className = 'in') {
message.className = className
message.timeStr = message.timeStr || this._getTimeString()
message.loginUserName = this.username
return message
}

// 設置追加信息
_setExt(message) {
return message
}

// 發送已閱回執
readMessage(message) {
let ImMessage = this.WebIM.message
const id = this.getConnection().getUniqueId()
let msg = new ImMessage('read', id)
msg.set({
id: message.id,
to: message.from
})
this.getConnection().send(msg.body)
}

// 登錄
login(username, password) {
this.getConnection().open({
apiUrl: this.config.apiURL,
user: username,
pwd: password,
appKey: this.config.appkey
})

const that = this

this.connection.listen({
onOpened(message) {
that.username = username

// 獲取好友
this.getRoster({
success: roster => {
store.dispatch('SET_IM_CHART_ROSTERLIST', roster)
},
error: e => {
console.log(e)
}
})
// 獲取羣組
this.getGroup({
success: group => {
store.dispatch('SET_IM_CHART_GROUPLIST', group.data)
},
error: e => {
console.log(e)
}
})
},
// 文本消息
onTextMessage(message) {
message = that.createClassNameAndTime(message)
// 處理表情
if (typeof message.sourceMsg === 'string') {
Object.keys(that.config.emoConfig.map).forEach(v => {
message.data = message.sourceMsg = message.sourceMsg.replace(new RegExp(v, 'g'), `<img src="${that.config.emoConfig.path + that.config.emoConfig.map[v]}" />`)
})
}
let type = dotData(message, 'type')
if (type === 'groupchat') { // 羣組消息
message.groupId = message.to
if (store.getters.GET_IM_CHART_DIALOG_GROUP_ID !== message.groupId || store.getters.GET_IM_CHART_DIALOG_VISIBLE === false) {
message.isRead = false
}
store.dispatch('PUSH_IM_CHART_DIALOG_GROUP_MESSAGE', message)
} else {
message.groupId = message.from
store.dispatch('PUSH_IM_CHART_DIALOG_GROUP_MESSAGE', message)
}
},
// 穿透消息
onCmdMessage(message) {
message.sourceMsg = '發起對話'
message.data = '發起對話'
message = that.createClassNameAndTime(message)
let type = dotData(message, 'type')
if (type === 'groupchat') {
message.from = message.to
}
store.dispatch('PUSH_IM_CHART_DIALOG_GROUP_MESSAGE', message)
},
// 圖片消息
onPictureMessage(message) {
message = that.createClassNameAndTime(message)
message.fileType = 'image'
let type = dotData(message, 'type')
if (type === 'groupchat') { // 羣組消息
message.groupId = message.to
if (store.getters.GET_IM_CHART_DIALOG_GROUP_ID !== message.groupId || store.getters.GET_IM_CHART_DIALOG_VISIBLE === false) {
message.isRead = false
}
store.dispatch('PUSH_IM_CHART_DIALOG_GROUP_MESSAGE', message)
}
},
// 音頻消息
onAudioMessage(message) {
let options = {url: message.url}

options.onFileDownloadComplete = (response) => {
message.objectURL = that.WebIM.utils.parseDownloadResponse.call(that.getConnection(), response)
message = that.createClassNameAndTime(message)
message.fileType = 'audio'
let type = dotData(message, 'type')
if (type === 'groupchat') { // 羣組消息
message.groupId = message.to
if (store.getters.GET_IM_CHART_DIALOG_GROUP_ID !== message.groupId || store.getters.GET_IM_CHART_DIALOG_VISIBLE === false) {
message.isRead = false
}
store.dispatch('PUSH_IM_CHART_DIALOG_GROUP_MESSAGE', message)
}
}

options.onFileDownloadError = () => {
// 音頻下載失敗
}

// 通知服務器將音頻轉爲mp3
options.headers = {
'Accept': 'audio/mp3'
}

that.WebIM.utils.download.call(that.getConnection(), options)
},
onEmojiMessage: (message) => {
let data = message.data
for (let i = 0, l = data.length; i < l; i++) {
}
},
onError(message) {
that.handError(message)
}
})
return this.connection
}

// 處理報錯
handError(error) {
let err = 'im報錯:'
console.error(error)
switch (error.type) {
case 7 :
alert(err + '客戶端網絡中斷')
break
case 8 :
alert(err + '多端登陸,被踢下線')
break
case 16 :
alert(err + '服務端關閉了連接')
break
case 31 :
alert(err + '處理下行消息出錯,try/catch拋出異常')
break
case 32 :
alert(err + '客戶端斷線')
break
case 33 :
alert(err + '客戶端退出登陸')
break
case 34 :
alert(err + '同一瀏覽器打開標籤頁超過上限')
break
case 402 :
alert(err + '取到token 可是沒有連上xmpp server')
break
default:
alert(err + JSON.stringify(error))
}
}

// 登出
logout() {
if (this.connection) {
this.connection.close()
}
}

// 發送消息
sendMessage(message, to, config = {
type: 'txt',
success() {
},
fail() {
},
chatType: ''
}) {
const that = this
const id = this.getConnection().getUniqueId()
let ImMessage = this.WebIM.message
const type = config.type
const chatType = config.chatType
let msg = new ImMessage(type, id)
var option = {
msg: message,
to,
roomType: false,
success: (messageId, msgId) => {
let temp = {
id: messageId,
type,
from: that.username,
to,
data: message,
sourceMsg: message,
error: false,
msgId
}
if (config.isGroup) {
temp.groupId = to
} else {
temp.groupId = temp.to
}
temp = that.appendMessage(temp, 'out')
store.dispatch('PUSH_IM_CHART_DIALOG_GROUP_MESSAGE', temp)
config.success()
},
fail: () => {
config.fail()
}
}
if (chatType === 'chatRoom') {
option.chatType = 'chatRoom'
}
option = this._setExt(option)
msg.set(option)
if (chatType === 'chatRoom') {
msg.setGroup('groupchat')
}
this.getConnection().send(msg.body)
}

// 添加消息
unShiftMessage(message) {
const className = message.from === this.username ? 'out' : 'in'
console.log(message.from, this.username, className)
const temp = this.appendMessage(message, className)
store.dispatch('UN_SHIFT_IM_CHART_DIALOG_GROUP_MESSAGE', temp)
}

// 發送文本消息
sendTextMessage(message, to, success = () => {
}, fail = () => {
}) {
this.sendMessage(message, to, {type: 'txt', success, fail})
}

// 發送組消息
sendTextMessageGroup(message, groupId, success) {
this.sendMessage(message, groupId, {type: 'txt', success, chatType: 'chatRoom', isGroup: true})
}

// 發送cmd 消息
sendCmdMessage(message, to, success = () => {
}, fail = () => {
}, action = '[CMDMessage]') {
// todo
}

/**
* 發送圖片
* @param input
* @param to
*/
sendGroupImageMessage(input, to, success = () => {
}, fail = () => {
}, chatType = 'chatRoom') {
const conn = this.getConnection()
const id = conn.getUniqueId() // 生成本地消息id
let ImMessage = this.WebIM.message
const msg = new ImMessage('img', id) // 建立圖片消息
// var input = document.getElementById('image') // 選擇圖片的input
const file = this.WebIM.utils.getFileUrl(input) // 將圖片轉化爲二進制文件
const allowType = {
'jpg': true,
'gif': true,
'png': true,
'bmp': true,
'jpeg': true
}
const that = this
if (file.filetype.toLowerCase() in allowType) {
let option = {
apiUrl: this.config.apiURL,
file: file,
to, // 接收消息對象
roomType: false,
chatType,
onFileUploadError: () => { // 消息上傳失敗
alert('圖片發送失敗')
fail()
},
onFileUploadComplete: (file) => { // 消息上傳成功
if (Array.isArray(file.entities)) {
for (let v of file.entities) {
let url = file.uri + '/' + v.uuid
let temp = {
id,
type: 'chat',
from: that.username,
to,
url,
error: false,
groupId: to
}
temp = that.appendMessage(temp, 'out')
temp.fileType = 'image'
store.dispatch('PUSH_IM_CHART_DIALOG_GROUP_MESSAGE', temp)
success()
}
}
},
flashUpload: this.WebIM.flashUpload
}
option = this._setExt(option)
msg.set(option)
if (chatType === 'chatRoom') {
msg.setGroup('groupchat')
} else {
msg.body.chatType = chatType
}
conn.send(msg.body)
} else {
alert('只支持:' + allowType.join(',') + '的圖片格式')
}
}
}

export default Easemob

 

4. 初始化 登陸,獲取好友分組以及聊天信息,使用場景系統用戶自接登陸,因此這裏沒有單獨作註冊登陸ide

computed: {
  // 用戶
rosterList () {
return this.GET_IM_CHART_ROSTERLIST || []
},
groupList () {
return this.GET_IM_CHART_GROUPLIST || []
},
// 聊天數據
charts () {
return this.GET_IM_CHART_DIALOG_GROUP_MESSAGE(this.imTo.toId) || []
},
...mapGetters({
GET_IM_CHART_ROSTERLIST: 'GET_IM_CHART_ROSTERLIST',
GET_IM_CHART_GROUPLIST: 'GET_IM_CHART_GROUPLIST',
GET_IM_CHART_DIALOG_GROUP_MESSAGE: 'GET_IM_CHART_DIALOG_GROUP_MESSAGE',
GET_IM_CHART_DIALOG_GROUP_ID: 'GET_IM_CHART_DIALOG_GROUP_ID'
})
},

methods: {
initIm () {
// im 登錄
this.im.login('test1', '123456')
},
// 發送消息
sendMessage () {
if (this.imTo.chatType === 'single') {
// 會話
this.im.sendTextMessage(this.txt, this.imTo.toId, () => {
this.txt = ''
})
} else {
// 組
this.im.sendTextMessageGroup(this.txt, this.imTo.toId, () => {
this.txt = ''
})
}
},
// 發送圖片消息
sendImageMessage () {
if (!this.imTo.toId) {
alert('發送對象必須')
return
}
this.im.sendGroupImageMessage(this.$refs.imageInput, this.imTo.toId, () => {
this.fileTrigger = !this.fileTrigger
this.$nextTick(_ => {
this.fileTrigger = !this.fileTrigger
})
})
},
toChats (opt) {
if (opt.groupid) {
this.imTo = {
chatType: 'group',
toId: opt.groupid,
toName: opt.groupname
}
} else {
this.imTo = {
chatType: 'single',
toId: opt.name,
toName: opt.name
}
}
},
scrollHandler () {},
// 滾動到底
scrollToBottom () {
this.$nextTick(_ => {
this.$refs['chattingContent'].wrap.scrollTop = this.$refs['chattingContent'].wrap.scrollHeight
})
}
}

git地址:

https://github.com/DarkSide7H/vue-chat

示例截圖

相關文章
相關標籤/搜索