index.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541
  1. <template>
  2. <view class="calling" v-show="callingDialog">
  3. <!-- 等待对方接受 -->
  4. <view class="calling-waiting" v-if="dialling">
  5. <view class="calling-waiting-header">
  6. <u-avatar :src="defaultAvatar" size="240" mode="square"></u-avatar>
  7. </view>
  8. <view class="calling-waiting-status">
  9. <view>正在呼叫{{ toAccount }}</view>
  10. </view>
  11. <view class="calling-waiting-button">
  12. <view class="calling-waiting-button-passed" @click="handleDebounce(leave, 500)">
  13. <view class="iconfont icon-guaduan"></view>
  14. </view>
  15. <view class="calling-waiting-button-mute" @click="micHandler">
  16. <view class="iconfont icon-mic-on-full" :class="isMicOn ? 'on' : 'off'"></view>
  17. </view>
  18. </view>
  19. </view>
  20. <!-- 接收方 -->
  21. <view class="calling-waiting" v-else-if="isDialled">
  22. <view class="calling-waiting-header">
  23. <u-avatar :src="defaultAvatar" size="240" mode="square"></u-avatar>
  24. </view>
  25. <view class="calling-waiting-status">
  26. <view>{{ sponsor }}来电</view>
  27. </view>
  28. <view class="calling-waiting-button">
  29. <view class="calling-waiting-button-mute" @click="handleDebounce(accept, 500)">
  30. <view class="iconfont icon-call"></view>
  31. </view>
  32. <view class="calling-waiting-button-passed" @click="handleDebounce(leave, 500)">
  33. <view class="iconfont icon-guaduan"></view>
  34. </view>
  35. </view>
  36. </view>
  37. <!-- 正在聊天 -->
  38. <view class="calling-waiting" v-else-if="calling">
  39. <view class="calling-waiting-userList">
  40. <view class="calling-waiting-userList-item" v-for="(item, index) in invitedUserInfo" :key="index">
  41. <u-avatar :src="item.avatar || defaultAvatar" size="120"></u-avatar>
  42. <view style="text-align: center;margin-top: 30rpx;">
  43. <text class="nick-text">{{ item.nick || item.userID }}</text>
  44. <text v-if="item.isInvitedMicOn === true || item.isInvitedMicOn == undefined"
  45. class="iconfont icon-yuyin"
  46. style="color: #006FFF;font-size: 50rpx;margin-left: 10rpx;"></text>
  47. <text v-else class="iconfont icon-maikefeng-jingyin"
  48. style="color: #fff;font-size: 50rpx;margin-left: 10rpx;"></text>
  49. </view>
  50. </view>
  51. </view>
  52. <view class="calling-waiting-duration">
  53. {{ formatDurationStr }}
  54. </view>
  55. <view class="calling-waiting-button">
  56. <view class="calling-waiting-button-passed" @click="handleDebounce(leave, 500)">
  57. <view class="iconfont icon-guaduan"></view>
  58. </view>
  59. <view class="calling-waiting-button-mute" @click="micHandler">
  60. <view class="iconfont icon-mic-on-full" :class="isMicOn ? 'on' : 'off'"></view>
  61. </view>
  62. </view>
  63. </view>
  64. </view>
  65. </template>
  66. <script>
  67. export default {
  68. data() {
  69. return {
  70. timeout: null,
  71. callType: 1, //1:audio,2:video
  72. Trtc: undefined,
  73. isCamOn: true,
  74. isMicOn: true,
  75. isInvitedMicOn: true,
  76. maskShow: false,
  77. isLocalMain: true, // 本地视频是否是主屏幕显示
  78. start: 0,
  79. end: 0,
  80. duration: 0,
  81. hangUpTimer: 0, // 通话计时id
  82. ready: false,
  83. dialling: true, // 是否拨打电话中
  84. calling: false, // 是否通话中
  85. isDialled: false, // 是否被呼叫
  86. inviteID: '',
  87. inviteData: {},
  88. sponsor: '', // 发起者
  89. toAccount: '222', // 接收者
  90. invitedUserID: [], //被邀请者
  91. invitedNick: '',
  92. invitedUserInfo: [],
  93. defaultAvatar: 'https://imgcache.qq.com/open/qcloud/video/act/webim-avatar/avatar-3.png',
  94. viewLocalDomID: '',
  95. callingUserList: [], // 参加通话的人 ,不包括自己
  96. callingType: 'C2C', //区分多人和C2C通话的UI样式
  97. isStartLocalView: false, //本地是否开启
  98. callingTips: {
  99. callEnd: 1, //通话结束
  100. callTimeout: 5
  101. },
  102. callingDialog: false,
  103. callingInfo: {
  104. type: 'C2C'
  105. }
  106. }
  107. },
  108. // onShow() {
  109. // this.initListener();
  110. // },
  111. onHide() {
  112. this.removeListener();
  113. },
  114. watch: {
  115. callingUserList: {
  116. handler(newValue) {
  117. console.log(newValue)
  118. },
  119. deep: true,
  120. immediate: true
  121. }
  122. },
  123. computed: {
  124. formatDurationStr() {
  125. return this.$format.formatDuration(this.duration)
  126. }
  127. },
  128. methods: {
  129. handleDebounce(func, wait) {
  130. let context = this
  131. let args = arguments
  132. if (this.timeout) clearTimeout(this.timeout)
  133. this.timeout = setTimeout(() => {
  134. func.apply(context, args)
  135. }, wait)
  136. },
  137. accept() {
  138. this.trtcCalling.accept({
  139. inviteID: this.inviteID,
  140. roomID: this.inviteData.roomID,
  141. callType: this.inviteData.callType
  142. }).then((res) => {
  143. this.changeState('calling', true);
  144. this.dialling = false;
  145. this.calling = true;
  146. this.isDialled = false;
  147. console.log('接收信息res', res)
  148. // this.currentUserProfile.nick = res.data.message.nick
  149. })
  150. },
  151. leave() { // 离开房间,发起方挂断
  152. this.isMicOn = true
  153. this.isCamOn = true
  154. this.maskShow = false
  155. this.isStartLocalView = false
  156. this.dialling = false;
  157. this.calling = false;
  158. this.isDialled = false;
  159. this.callingDialog = false
  160. if (!this.calling) { // 还没有通话,单方面挂断
  161. this.trtcCalling.hangup().then((res) => {
  162. // this.currentUserProfile.nick = res.data.message.nick
  163. this.changeState('dialling', false)
  164. clearTimeout(this.timer)
  165. })
  166. return
  167. }
  168. this.hangUp() // 通话一段时间之后,某一方面结束通话
  169. },
  170. hangUp() { // 通话一段时间之后,某一方挂断电话
  171. this.changeState('calling', false)
  172. this.trtcCalling.hangup();
  173. this.callingDialog = false
  174. uni.showToast({
  175. title: '已挂断',
  176. icon: 'none'
  177. })
  178. },
  179. openDialog() {
  180. this.callingDialog = true;
  181. this.dialling = true;
  182. this.calling = false;
  183. this.isDialled = false;
  184. },
  185. initListener() {
  186. // sdk内部发生了错误
  187. this.trtcCalling.on(this.TRTCCalling.EVENT.ERROR, this.handleError)
  188. // 被邀请进行通话
  189. this.trtcCalling.on(this.TRTCCalling.EVENT.INVITED, this.handleNewInvitationReceived)
  190. // 有用户同意进入通话,那么会收到此回调
  191. this.trtcCalling.on(this.TRTCCalling.EVENT.USER_ENTER, this.handleUserEnter)
  192. // 如果有用户同意离开通话,那么会收到此回调
  193. this.trtcCalling.on(this.TRTCCalling.EVENT.USER_LEAVE, this.handleUserLeave)
  194. // 用户拒绝通话
  195. this.trtcCalling.on(this.TRTCCalling.EVENT.REJECT, this.handleInviteeReject)
  196. //邀请方忙线
  197. this.trtcCalling.on(this.TRTCCalling.EVENT.LINE_BUSY, this.handleInviteeLineBusy)
  198. // 作为被邀请方会收到,收到该回调说明本次通话被取消了
  199. this.trtcCalling.on(this.TRTCCalling.EVENT.CALLING_CANCEL, this.handleInviterCancel)
  200. // 重复登陆,收到该回调说明被踢出房间
  201. this.trtcCalling.on(this.TRTCCalling.EVENT.KICKED_OUT, this.handleKickedOut)
  202. // 作为邀请方会收到,收到该回调说明本次通话超时未应答
  203. this.trtcCalling.on(this.TRTCCalling.EVENT.CALLING_TIMEOUT, this.handleCallTimeout)
  204. // 邀请用户无应答
  205. this.trtcCalling.on(this.TRTCCalling.EVENT.NO_RESP, this.handleNoResponse)
  206. // 收到该回调说明本次通话结束了
  207. this.trtcCalling.on(this.TRTCCalling.EVENT.CALLING_END, this.handleCallEnd)
  208. // 远端用户开启/关闭了摄像头, 会收到该回调
  209. this.trtcCalling.on(this.TRTCCalling.EVENT.USER_VIDEO_AVAILABLE, this.handleUserVideoChange)
  210. // 远端用户开启/关闭了麦克风, 会收到该回调
  211. this.trtcCalling.on(this.TRTCCalling.EVENT.USER_AUDIO_AVAILABLE, this.handleUserAudioChange)
  212. },
  213. removeListener() {
  214. this.trtcCalling.off(this.TRTCCalling.EVENT.ERROR, this.handleError)
  215. this.trtcCalling.off(this.TRTCCalling.EVENT.INVITED, this.handleNewInvitationReceived)
  216. this.trtcCalling.off(this.TRTCCalling.EVENT.USER_ENTER, this.handleUserEnter)
  217. this.trtcCalling.off(this.TRTCCalling.EVENT.USER_LEAVE, this.handleUserLeave)
  218. this.trtcCalling.off(this.TRTCCalling.EVENT.REJECT, this.handleInviteeReject)
  219. this.trtcCalling.off(this.TRTCCalling.EVENT.LINE_BUSY, this.handleInviteeLineBusy)
  220. this.trtcCalling.off(this.TRTCCalling.EVENT.CALLING_CANCEL, this.handleInviterCancel)
  221. this.trtcCalling.off(this.TRTCCalling.EVENT.KICKED_OUT, this.handleKickedOut)
  222. this.trtcCalling.off(this.TRTCCalling.EVENT.CALLING_TIMEOUT, this.handleCallTimeout)
  223. this.trtcCalling.off(this.TRTCCalling.EVENT.NO_RESP, this.handleNoResponse)
  224. this.trtcCalling.off(this.TRTCCalling.EVENT.CALLING_END, this.handleCallEnd)
  225. this.trtcCalling.off(this.TRTCCalling.EVENT.USER_VIDEO_AVAILABLE, this.handleUserVideoChange)
  226. this.trtcCalling.off(this.TRTCCalling.EVENT.USER_AUDIO_AVAILABLE, this.handleUserAudioChange)
  227. },
  228. handleError() {
  229. console.log('Error')
  230. },
  231. handleUserVideoChange() {},
  232. handleUserAudioChange(payload) {
  233. const _index = this.invitedUserInfo.findIndex(item => item.userID === payload.userID)
  234. if (_index >= 0) {
  235. this.invitedUserInfo[_index].isInvitedMicOn = payload.isAudioAvailable
  236. }
  237. },
  238. // 双方,通话已建立, 通话结束
  239. handleCallEnd({
  240. userID,
  241. callEnd
  242. }) {
  243. // 自己挂断的要补充消息 被邀请者都无应答时结束
  244. // 历史消息中没有通话结束
  245. if (userID === this.userID && this.invitedUserID.length === 0 || this.callingUserList === 0) {
  246. this.sendMessage(userID, callEnd, this.callingTips.callEnd)
  247. }
  248. this.changeState('dialling', false)
  249. this.isMicOn = true
  250. this.isCamOn = true
  251. this.maskShow = false
  252. this.isStartLocalView = false
  253. },
  254. // 自己超时且是邀请发起者,需主动挂断,并通知上层对端无应答
  255. handleNoResponse({
  256. sponsor,
  257. userIDList
  258. }) { //邀请者
  259. if (sponsor === this.userID) {
  260. userIDList.forEach((userID) => {
  261. this.setCallingstatus(userID)
  262. })
  263. if (userIDList.indexOf(this.userID) === -1) { //当超时者是自己时,添加消息
  264. this.sendMessage(userIDList, '', this.callingTips.callTimeout)
  265. }
  266. }
  267. },
  268. // 当自己收到对端超时的信令时,或者当我是被邀请者但自己超时了,通知上层通话超时
  269. // case: A呼叫B,B在线,B超时未响应,B会触发该事件,A也会触发该事件
  270. handleCallTimeout({
  271. userIDList
  272. }) {
  273. if (this.calling) {
  274. return
  275. }
  276. if (this.userID === this.sponsor) { // 该用户是邀请者
  277. userIDList.forEach((userID) => {
  278. this.setCallingstatus(userID) //超时未接听
  279. })
  280. return
  281. }
  282. //用户是被邀请者
  283. if (userIDList.indexOf(this.userID) > -1) { //当超时者是自己时,添加消息
  284. //会话列表切换后发消息
  285. this.toAccount && this.sendMessage(this.userID, '', this.callingTips.callTimeout)
  286. }
  287. this.changeState('isDialled', false)
  288. },
  289. handleKickedOut() {
  290. // 重复登陆,被踢出房间
  291. },
  292. // 自己超时且是邀请发起者,需主动挂断,并通知上层对端无应答
  293. handleNoResponse({
  294. sponsor,
  295. userIDList
  296. }) { //邀请者
  297. if (sponsor === this.userID) {
  298. userIDList.forEach((userID) => {
  299. this.setCallingstatus(userID)
  300. })
  301. if (userIDList.indexOf(this.userID) === -1) { // 当超时者是自己时,添加消息
  302. this.sendMessage(userIDList, '', this.callingTips.callTimeout)
  303. }
  304. }
  305. },
  306. // 当自己收到对端超时的信令时,或者当我是被邀请者但自己超时了,通知上层通话超时
  307. // case: A呼叫B,B在线,B超时未响应,B会触发该事件,A也会触发该事件
  308. handleCallTimeout({
  309. userIDList
  310. }) {
  311. if (this.calling) {
  312. return
  313. }
  314. if (this.userID === this.sponsor) { // 该用户是邀请者
  315. userIDList.forEach((userID) => {
  316. this.setCallingstatus(userID) //超时未接听
  317. })
  318. return
  319. }
  320. //用户是被邀请者
  321. if (userIDList.indexOf(this.userID) > -1) { //当超时者是自己时,添加消息
  322. //会话列表切换后发消息
  323. this.toAccount && this.sendMessage(this.userID, '', this.callingTips.callTimeout)
  324. }
  325. this.changeState('isDialled', false)
  326. },
  327. handleKickedOut() {},
  328. // 通知被呼叫方,邀请被取消,未接通
  329. handleInviterCancel() {
  330. // 邀请被取消
  331. this.changeState('isDialled', false)
  332. uni.showToast({
  333. title: '通话已取消',
  334. icon: 'none'
  335. })
  336. },
  337. // 通知呼叫方,对方在忙碌,未接通
  338. handleInviteeLineBusy({
  339. sponsor,
  340. userID
  341. }) {
  342. // A call B,C call A, A在忙线, 拒绝通话,对于呼叫者C收到通知,XXX在忙线
  343. if (sponsor === this.userID) {
  344. this.setCallingstatus(userID)
  345. uni.showToast({
  346. title: '对方忙线',
  347. icon: 'none'
  348. })
  349. }
  350. },
  351. setCallingstatus(userID) {
  352. const _index = this.invitedUserID.indexOf(userID)
  353. if (_index >= 0) {
  354. this.invitedUserID.splice(_index, 1)
  355. }
  356. if (this.invitedUserID.length === 0) {
  357. this.changeState('isDialled', false)
  358. this.changeState('dialling', false)
  359. }
  360. },
  361. // 通知呼叫方,未接通
  362. //userID:invitee(被邀请者)
  363. handleInviteeReject({
  364. userID
  365. }) {
  366. if (this.userID === this.sponsor) {
  367. // 发起者
  368. this.setCallingstatus(userID)
  369. uni.showToast({
  370. title: '用户拒绝通话',
  371. icon: 'none'
  372. })
  373. }
  374. },
  375. // 用户离开
  376. handleUserLeave({
  377. userID
  378. }) {
  379. console.log(this.callType)
  380. if (this.callType === this.TRTCCalling.CALL_TYPE.AUDIO_CALL) {
  381. // 语音通话
  382. const _index = this.invitedUserInfo.findIndex(item => item.userID === userID)
  383. if (_index >= 0) {
  384. this.invitedUserInfo.splice(_index, 1)
  385. }
  386. return
  387. }
  388. const index = this.callingUserList.findIndex(item => item === userID)
  389. if (index >= 0) {
  390. this.callingUserList.splice(index, 1)
  391. }
  392. this.dialling = true;
  393. this.calling = false;
  394. this.isDialled = false;
  395. this.callingDialog = false
  396. },
  397. // 被呼叫 接听方
  398. async handleNewInvitationReceived(payload) {
  399. console.log('接听方', payload)
  400. this.inviteID = payload.inviteID
  401. this.callingDialog = true
  402. this.dialling = false;
  403. this.calling = false;
  404. this.isDialled = true;
  405. },
  406. // 双方建立连接
  407. handleUserEnter({
  408. userID
  409. }) {
  410. this.changeState('dialling', true)
  411. this.isAccept()
  412. // 判断是否为多人通话
  413. if (this.callingUserList.length >= 2) {
  414. this.callingType = this.$TIM.TYPES.CONV_GROUP
  415. }
  416. if (this.callingUserList.indexOf(userID) === -1) {
  417. if (this.callType === this.TRTCCalling.CALL_TYPE.AUDIO_CALL) {
  418. this.getUserAvatar(userID)
  419. } else {
  420. this.callingUserList.push(userID)
  421. }
  422. }
  423. if (this.callType === this.TRTCCalling.CALL_TYPE.VIDEO_CALL) {
  424. this.$nextTick(() => {
  425. if (!this.isStartLocalView) {
  426. this.startLocalView() //本地只开启一次
  427. }
  428. this.startRemoteView(userID) //远端多次拉流
  429. })
  430. }
  431. },
  432. /**
  433. * 播放本地流
  434. */
  435. startLocalView() {
  436. this.trtcCalling.startLocalView({
  437. userID: this.userID,
  438. videoViewDomID: 'local'
  439. }).then(() => {
  440. this.isStartLocalView = true
  441. })
  442. },
  443. async sendMessage(userId, callEnd, callText) {
  444. let call_text = ''
  445. userId = Array.isArray(userId) ? userId.join(',') : userId
  446. let messageData = {
  447. to: this.toAccount,
  448. from: userId,
  449. conversationType: this.currentConversationType,
  450. payload: {
  451. data: '',
  452. description: '',
  453. extension: ''
  454. }
  455. }
  456. const message = await this.$TIM.createCustomMessage(messageData)
  457. },
  458. /**
  459. * 播放远端流
  460. * @param {Object} userID
  461. */
  462. startRemoteView(userID) {
  463. this.trtcCalling.startRemoteView({
  464. userID: userID,
  465. videoViewDomID: `video-${userID}`
  466. }).then(() => {
  467. })
  468. },
  469. /**
  470. * 获取被呼叫者信息
  471. * @param {Object} userID
  472. */
  473. getUserAvatar(userID) {
  474. const _index = this.invitedUserInfo.findIndex(item => item.userID === userID)
  475. if (_index >= 0) {
  476. return
  477. }
  478. let _userIDList = [userID]
  479. let promise = this.tim.getUserProfile({
  480. userIDList: _userIDList // 请注意:即使只拉取一个用户的资料,也需要用数组类型,例如:userIDList: ['user1']
  481. })
  482. promise.then((imResponse) => {
  483. if (imResponse.data[0]) {
  484. this.invitedUserInfo.push(imResponse.data[0])
  485. }
  486. }).catch(() => {})
  487. },
  488. /**
  489. * 对方接听自己发起的电话
  490. */
  491. isAccept() {
  492. clearTimeout(this.timer)
  493. this.changeState('calling', true)
  494. clearTimeout(this.hangUpTimer)
  495. this.resetDuration(0)
  496. this.start = new Date()
  497. },
  498. resetDuration(duration) {
  499. this.duration = duration
  500. this.hangUpTimer = setTimeout(() => {
  501. let now = new Date()
  502. this.resetDuration(parseInt((now - this.start) / 1000))
  503. }, 1000)
  504. },
  505. /**
  506. * 修改状态
  507. * @param {Object} state
  508. * @param {Object} boolean
  509. */
  510. changeState(state, boolean) {
  511. let stateList = ['dialling', 'isDialled', 'calling']
  512. stateList.forEach(item => {
  513. this[item] = item === state ? boolean : false
  514. })
  515. this.$store.commit('UPDATE_ISBUSY', stateList.some(item => this[
  516. item]))
  517. // 若stateList 中存在 true , isBusy 为 true
  518. },
  519. /**
  520. * 是否打开麦克风
  521. */
  522. micHandler() {
  523. if (this.isMicOn) {
  524. this.trtcCalling.setMicMute(true)
  525. this.isMicOn = false
  526. } else {
  527. this.trtcCalling.setMicMute(false)
  528. this.isMicOn = true
  529. }
  530. }
  531. }
  532. }
  533. </script>
  534. <style lang="scss" scoped>
  535. @import './index.scss';
  536. </style>