ToB企服应用市场:ToB评测及商务社交产业平台

标题: WebRtc 视频通话,语音通话实现方案 [打印本页]

作者: 卖不甜枣    时间: 2024-8-6 18:41
标题: WebRtc 视频通话,语音通话实现方案
先了解一下流程 和 流程图(chatGpt的回复)


 

 



实现 (底层代码实现, 可作为demo认识)

小demo

  1. <template>
  2.   <div>
  3.     <video ref="localVideo" autoplay muted></video> <!-- 本地视频元素,用于显示本地视频 -->
  4.     <video ref="remoteVideo" autoplay></video> <!-- 远程视频元素,用于显示远程视频 -->
  5.     <button @click="startCall">开始视频</button> <!-- 点击按钮开始呼叫 -->
  6.     <button @click="endCall">结束视频</button> <!-- 点击按钮结束通话 -->
  7.   </div>
  8. </template>
  9. <script lang="ts">
  10. import { ref, onMounted } from 'vue';
  11. // import WebSocket from 'websocket'
  12. export default {
  13.   setup() {
  14.     // 创建本地视频和远程视频的引用
  15.     const localVideo = ref(null); // 本地视频元素引用
  16.     const remoteVideo = ref(null); // 远程视频元素引用
  17.     // 保存本地媒体流和RTCPeerConnection对象
  18.     let localStream = null; // 本地媒体流
  19.     let peerConnection = null; // RTCPeerConnection对象
  20.     // 开始呼叫方法
  21.     const startCall = async () => {
  22.     console.log('开始');
  23.       try {
  24.         // 获取本地媒体流(视频和音频)
  25.         localStream = await navigator.mediaDevices.getUserMedia({ video: true, audio: true });
  26.         // 将本地媒体流绑定到本地视频元素
  27.         localVideo.value.srcObject = localStream;
  28.         // 创建RTCPeerConnection对象
  29.         peerConnection = new RTCPeerConnection();
  30.         // 将本地媒体流中的轨道添加到peerConnection中
  31.         localStream.getTracks().forEach(track => {
  32.           peerConnection.addTrack(track, localStream);
  33.         });
  34.         // 当远程流到达时,将其绑定到远程视频元素
  35.         peerConnection.ontrack = (event) => {
  36.           remoteVideo.value.srcObject = event.streams[0];
  37.         };
  38.         // 创建一个offer并设置为本地描述
  39.         const offer = await peerConnection.createOffer();
  40.         await peerConnection.setLocalDescription(offer);
  41.         // 发送offer给对方并等待对方的answer
  42.         // 在实际应用中,这部分需要与信令服务器交互来完成
  43.         // 示例:通过Socket 发送offer给对方
  44.         // Socket .send(JSON.stringify({ type: 'offer', offer: offer }));
  45.       } catch (error) {
  46.         console.error('Error starting call:', error);
  47.       }
  48.     };
  49.     // 结束通话方法
  50.     const endCall = () => {
  51.       // 停止本地媒体流中的所有轨道
  52.       localStream.getTracks().forEach(track => track.stop());
  53.       // 关闭peerConnection连接
  54.       peerConnection.close();
  55.       // 重置视频元素的srcObject
  56.       localVideo.value.srcObject = null;
  57.       remoteVideo.value.srcObject = null;
  58.     };
  59.     // 初始化操作,例如连接信令服务器等
  60.     onMounted(() => {
  61.       
  62.     });
  63.     // 返回给模板部分需要使用的变量和方法
  64.     return {
  65.       localVideo,
  66.       remoteVideo,
  67.       startCall,
  68.       endCall
  69.     };
  70.   }
  71. };
  72. </script>
复制代码
https://juejin.cn/post/7266417942182608955
https://juejin.cn/post/7266417942182608955https://juejin.cn/post/7170767923005358094
https://juejin.cn/post/7170767923005358094webrtc一对一多对多音视频通话开辟第一集_哔哩哔哩_bilibili项目地点:zou-hong-run/webrtc_one2one_many2many (github.com), 视频播放量 1177、弹幕量 0、点赞数 29、投硬币枚数 20、收藏人数 79、转发人数 4, 视频作者 red润, 作者简介 学习成份复杂男 学习讨论群:811710917,干系视频:webrtc一对一多对多音视频通话教程第八集,webrtc一对一多对多音视频通话教程第十集,webrtc多人音视频通话教程第十一集,webrtc多人音视频通话教程第十三集,webrtc多人音视频通话教程第十四集(完结),webrtc一对一多对多音视频通话教程第七集,webrtc一对一多对多音视频通话教程第五集,【手把手WebRTC音视频SDK】22-根本架构-封装采集数据为MediaFrame布局,webrtc一对一多对多音视频通话教程第六集,webrtc一对一多对多音视频通话教程第四集
https://www.bilibili.com/video/BV1gK411v7wy/?spm_id_from=333.788&vd_source=3e36960fd2cef2338d62a0f86944333aWebRTC 使用入门详解_webrtc教程-CSDN博客文章欣赏阅读1.8k次,点赞6次,收藏15次。文档来源https://webrtc.org/getting-started/media-devices?hl=zh-cn。_webrtc教程
https://blog.csdn.net/qq_47658204/article/details/130177016


 
实现 ( 这个是采用了 引入sip-0.13.6.min.js(已封装好的脚本实现的) )

封装的组件, 代码采用父子传参, pinia传参, 后续由于视频通话,音频通话免登录需求, 必要独立项目外, 做了 http 携带参数... 
已知题目: 项目上线后, http欣赏器不支持麦克风和摄像头(本地支持).
办理方案: 换 https 即可


 父
  1. const handleVideoPhone = async (type, row) => {
  2.   // 存储对象到 Pinia 中
  3.   // const myObject = { type, row }
  4.   // await myStore.setMyObject(myObject)
  5.   // console.log(myStore.videoData, '取到了-------------');
  6.   // 命名的路由
  7.   // router.push({ name: 'callVA', params: { userId: '123' } })
  8.   // message.alertError("当前设备不在线,无法进行视频通话!")
  9.   // 用户id 设备id 音视频类型type // 0 音屏,1 视频-------
  10.   const url = `https://www.XXXXXX.com/callVA?type=${type}&id=${row.aqmPkId}&deviceid=${row.deviceId}`
  11.   // const url = `http://localhost:8388/callVA?type=${type}&id=${row.aqmPkId}&deviceid=${row.deviceId}`
  12.   // console.log(url, 'url-------------------');
  13.   // window.location.href = url
  14.   // window.open(url, '_blank');  // 在新窗口中打开链接
  15.   window.open(url, 'video');  // 在新窗口中打开链接
  16. }
复制代码
子 
  1. <template>
  2.   <div style="width: 100%; height: 100%">
  3.     <!-- /* -webkit-background-clip: text; */ -->
  4.     <h1
  5.       style="
  6.         letter-spacing: 3px;
  7.         text-align: center;
  8.         background-image: linear-gradient(to top, #89ceeb, #00e7ee);
  9.         -webkit-text-fill-color: transparent;
  10.         background-clip: text;
  11.       "
  12.       >{{ status }}</h1
  13.     >
  14.     <div style="display: flex; align-items: center; justify-content: center">
  15.       <dv-decoration-6 v-if="status === '通话中...'" style="width: 300px; height: 30px" />
  16.       <!-- <dv-decoration-6 style="width:300px;height:30px;" /> -->
  17.     </div>
  18.     <div style="margin: 30px; text-align: center">
  19.       <!-- 开始呼叫 -->
  20.       <el-button type="primary" round @click="call()">{{
  21.         v_type == '1' ? '开启视频' : '开启音频'
  22.       }}</el-button>
  23.       &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;
  24.       <!-- 同意 -->
  25.       <!-- <button @click="acceptCall">同意视频</button> -->
  26.       &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;
  27.       <!-- 挂断 -->
  28.       <el-button type="danger" round @click="endCall">{{
  29.         v_type == '1' ? '挂断视频' : '挂断音频'
  30.       }}</el-button>
  31.       <!-- <p id="regUA_msg"> 用户代理状态 </p> -->
  32.     </div>
  33.     <div style="display: flex; justify-content: space-evenly; align-items: center; width: 100%">
  34.       <!-- 本地视频元素,用于显示本地视频 -->
  35.       <!-- <div
  36.         style="
  37.           width: 800px;
  38.           height: 450px;
  39.           padding: 10px;
  40.           margin: 10px;
  41.           border: 3px solid gray;
  42.           border-radius: 5px;
  43.           box-sizing: border-box;
  44.         "
  45.       >
  46.         <video style="width: 100%; height: 100%" ref="localVideo" autoplay muted></video>
  47.       </div> -->
  48.       <!-- 远程视频元素,用于显示远程视频 -->
  49.       <div
  50.         style="
  51.           /* width: 800px;
  52.           height: 450px; */
  53.           width: 80%;
  54.           /* height: 100%; */
  55.           padding: 10px;
  56.           margin: 10px;
  57.           border: 3px solid gray;
  58.           border-radius: 5px;
  59.           box-sizing: border-box;
  60.         "
  61.       >
  62.         <video style="width: 100%; height: 100%" ref="remoteVideo" autoplay></video>
  63.         <!-- <audio v-else-if="v_type == '0'" controls>
  64.           <source ref="remoteVideo" type="audio/mp3" />
  65.         </audio> -->
  66.       </div>
  67.     </div>
  68.   </div>
  69. </template>
  70. <script setup lang="ts">
  71. // import { useVideoStore } from '../../stores/myvideo' // pinia
  72. // import { useRouter } from 'vue-router' // router
  73. //  @/api/drone/ws
  74. // import * as wsApi from '../../api/ws/index' // api
  75. import { onMounted, onUnmounted, ref } from 'vue'
  76. import axios from 'axios'
  77. // const myStore = useVideoStore() // pinia
  78. // const router = useRouter()
  79. // 获取存储在 Pinia 中的对象
  80. // let myObject = myStore.videoData
  81. // console.log(myObject,myObject?.type, myObject?.row, 'ccccccccccccccccccc')
  82. /**
  83. * 发起视频通话
  84. */
  85. const status = ref<string>('等待开启通话...')
  86. const v_type = ref<string>() // 0 音屏,1 视频
  87. const userId = ref('') //用户id
  88. const deviceId = ref('') // 用户设备id
  89. // const access_token = ref('') // token
  90. const socket = ref() // new websocket
  91. const activeMessage = ref<any>() // websockt收到的信息
  92. let timer: any //心跳定时器
  93. // 创建本地视频和远程视频的引用-------------------------------
  94. // const localVideo = ref<any>(null) // 播放本地视频
  95. const remoteVideo = ref<any>(null) // 播放远程视频
  96. let userAgent: any // 注册 UA
  97. let sipsession: any // SIP 如果sipsession存在,则调用它的terminate()方法来终止会话
  98. const sip_id = ref<string>() // 用户设备 sip_id
  99. const sip_host = ref<string>() // 本地登录 sip服务器地址 sip_host
  100. // 组件加载
  101. const handleVideoPhone = async (
  102.   type: string,
  103.   row: { id?: string | null; deviceId: any; aqmPkId?: any }
  104. ) => {
  105.   // console.log(type, row)
  106.   userId.value = row?.aqmPkId // 用户id 1011
  107.   deviceId.value = row?.deviceId // 用户 设备id xxxxxxxxxxxxx
  108.   v_type.value = type // 0 音屏,1 视频-------
  109.   // 获取 设备sip_id
  110.   // let res = await wsApi.getUserDeviceSipId(`${deviceId.value}`)
  111.   // console.log(res, '===========');
  112.   // if (res) {
  113.   //   console.log(111);
  114.   //   sip_id.value = res.sip_id
  115.   //   console.log('用户id------', userId.value, res)
  116.   // }
  117.   const res = await axios.get(
  118.     `https://www.XXXXXX.com:1443/admin-api/drone/ws/getUserDeviceSipId/${deviceId.value}`
  119.   )
  120.   if (res.status === 200) {
  121.     // console.log('设备sip_id', res)
  122.     sip_id.value = res.data.data.sip_id
  123.   } else {
  124.     console.log('接口未连接')
  125.   }
  126.   websocketFun() //创建WebSocket连接
  127. }
  128. // --------------------------------------------------
  129. // 3.发送音视频通话请求,成功后配置通话参数,发起通话并监听拨打结果并做相应处理
  130. const call = async () => {
  131.   // 发送音视频通话请求通知设备
  132.   const video = {
  133.     act: 'ma_set_sip_info', // 请求标识
  134.     v_type: v_type.value, //视频 1 语音 0
  135.     user_id: userId.value // 用户id
  136.   }
  137.   status.value = '连接中...'
  138.   // console.log(video);
  139.   socket.value.send(JSON.stringify(video))
  140.   // 长链接发送报文指定设备开启推流
  141.   socket.value.send(JSON.stringify({ act: 'ma_open_rtsp', device_id: deviceId.value }))
  142.   // console.log('设备推流')
  143.   // 长链接发送报文指定设备关闭推流
  144.   // socket.value.send(JSON.stringify({ act: 'ma_stop_rtsp', device_id: deviceId.value }))
  145.   // 服务器地址
  146.   // var host = document.getElementById('sip_host').value
  147.   var host = sip_host.value
  148.   //呼叫目标,可以是设备的sip_id,或者群组通话的room_id
  149.   // var to = document.getElementById('device_sipId').value
  150.   // var to = deviceId.value
  151.   var to = String(sip_id.value)
  152.   console.log(host, to, '测试')
  153.   sipsession = await userAgent.invite(to + '@' + host, {
  154.     sessionDescriptionHandlerOptions: {
  155.       constraints: {
  156.         audio: true,
  157.         // video: true //音频通话则为false
  158.         video: v_type.value == '1' ? true : false //音频通话则为false
  159.       }
  160.     }
  161.   })
  162.   // 当呼叫被接受时触发的事件
  163.   sipsession.on('accepted', async function () {
  164.     status.value = '通话中...'
  165.     console.log('呼叫接收,开始通话')
  166.     // 我们需要检查 peer connection 来确定添加了哪个轨道
  167.     var pc = await sipsession.sessionDescriptionHandler.peerConnection
  168.     // 获取远程轨道
  169.     var remoteStream = new MediaStream()
  170.     // console.log(remoteStream, '远程轨道')
  171.     pc.getReceivers().forEach(function (receiver) {
  172.       remoteStream.addTrack(receiver.track)
  173.     })
  174.     //此处remoteVideo为一个video标签,将远程轨道绑定到它上面并播放
  175.     remoteVideo.value.srcObject = remoteStream
  176.     remoteVideo.value.play()
  177.     if (pc.getSenders()) {
  178.       // console.log('开启本地视频')
  179.       // var localStream = new MediaStream()
  180.       // // var localStream = await navigator.mediaDevices.getUserMedia({ video: true, audio: true })
  181.       // pc.getSenders().forEach(function (sender) {
  182.       //   localStream.addTrack(sender.track)
  183.       // })
  184.       // // localVideo 是一个 video 标签,将本地轨道绑定到它上面并播放
  185.       // localVideo.value.srcObject = localStream
  186.       // localVideo.value.play()
  187.     }
  188.   })
  189.   // xu获取本地媒体流(视频和音频)
  190.   // xu await getLocalStream()
  191. }
  192. // 获取本地媒体流(视频和音频)
  193. // const getLocalStream = async () => {
  194. //   const stream = await navigator.mediaDevices.getUserMedia({ video: true, audio: true })
  195. //   // 将本地媒体流绑定到本地视频元素
  196. //   localVideo.value.srcObject = stream
  197. //   localVideo.value.play()
  198. //   localStream.value = stream
  199. //   return stream
  200. // }
  201. // 挂断通话
  202. const endCall = async () => {
  203.   // 如果sipsession存在,则调用它的terminate()方法来终止会话
  204.   if (sipsession) {
  205.     await sipsession.terminate()
  206.     status.value = '已挂断...'
  207.   }
  208.   //
  209.   // path: 'user-device', name: 'DroneUserDevice', // 跳转回:人员设备绑定页面
  210.   console.log('挂断,关闭当前窗口')
  211.   // router.push({ name: 'DroneUserDevice', params: { userId: '123' } })
  212.   // 关闭当前窗口
  213.   // window.close();
  214. }
  215. //--------------------------------------------------
  216. // websocket 连接Socket信令服务器
  217. const websocketFun = async () => {
  218.   const res = await axios.get('https://www.XXXXXX.com:1443/admin-api/drone/ws/getCurrentAqmUser')
  219.   if (res.status !== 200) {
  220.     console.log('accessToken--------------')
  221.     return
  222.   }
  223.   socket.value = new WebSocket('wss://XXXXXX.com/wss')
  224.   socket.value.onopen = async function () {
  225.     // socket.value = Sock // websocket
  226.     // 管理员账号登录
  227.     const message = {
  228.       act: 'ma_login', // 管理员登录
  229.       // user_name: 'admin',
  230.       // access_token: (await wsApi.getCurrentAqmUserInfo()).accessToken // 接口返回token,会过期
  231.       access_token: res.data.data.accessToken // 接口返回token,会过期
  232.     }
  233.     console.log(message)
  234.     socket.value.send(JSON.stringify(message))
  235.     if (timer) clearInterval(timer) //清空上一个定时器
  236.     timer = setInterval(() => {
  237.       // 获取长链接实时、状态等心跳包
  238.       return socket.value.send(JSON.stringify({ act: 'ma_get_active_devices' }))
  239.     }, 5000)
  240.   }
  241.   socket.value.onmessage = async (event) => {
  242.     // 处理收到的消息
  243.     activeMessage.value = JSON.parse(event.data)
  244.     // console.log('收到消息:', activeMessage.value)
  245.     // 登录
  246.     if (activeMessage.value.cmd == 'ma_login') {
  247.       if (activeMessage.value.status == true) {
  248.         // console.log(activeMessage.value)
  249.         // 1. 获取sip对象注册信息
  250.         let sip_info = activeMessage.value.admin_info.sip_info
  251.         sip_host.value = sip_info.sip_host // 服务器地址sip_host 拨打电话
  252.         // sip_id.value = sip_info.sip_id // 呼叫目标
  253.         // console.log('登录成功: sip_info =', sip_info)
  254.         regUserAgent(sip_info) // 注册UA
  255.       } else {
  256.         console.log(activeMessage.value.msg)
  257.       }
  258.     }
  259.     // 心跳包()设备活跃  获取长链接实时、状态等心跳包
  260.     else if (activeMessage.value.cmd == 'ma_get_active_devices') {
  261.       if (activeMessage.value.status == true) {
  262.         // console.log(activeMessage.value.msg)
  263.         // console.log('心跳包:', activeMessage.value)
  264.         // let res = activeMessage.value.data.filter((item) => {
  265.         //   // console.log(item.user_info.user_id)
  266.         //   return item.user_info.user_id == userId.value
  267.         // })
  268.         // sip_id.value = res[0].user_info.sip_id // 设备 sip_id,心跳包
  269.         // console.log(res, sip_id.value, 'ccccccccccccccccccccc')
  270.       } else {
  271.         console.log(activeMessage.value.msg)
  272.       }
  273.     }
  274.     // 发送音视频通话请求通知设备
  275.     else if (activeMessage.value.cmd == 'ma_set_sip_info') {
  276.       if (activeMessage.value.status == true) {
  277.         // console.log('发送音视频通话请求通知设备', activeMessage.value.msg, activeMessage.value)
  278.         // flag.value = true // 标杆 视频电话是否接听
  279.       } else {
  280.         console.log(activeMessage.value.msg)
  281.         // flag.value = false // 标杆 视频电话是否接听
  282.         // console.log(flag.value)
  283.         endCall() // 挂断电话, 跳回页面
  284.       }
  285.     } else if (activeMessage.value.cmd == 'ma_open_rtsp') {
  286.       if (activeMessage.value.status == true) {
  287.         // console.log('发送报文指定设备推流', activeMessage.value)
  288.       } else {
  289.         activeMessage.value.status == true
  290.       }
  291.     } else if (activeMessage.value.cmd == 'ma_stop_rtsp' && activeMessage.value.status == true) {
  292.       if (activeMessage.value.status == true) {
  293.         // console.log('发送报文指定设备关闭推流', activeMessage.value)
  294.       } else {
  295.         console.log(activeMessage.value.msg)
  296.       }
  297.     }
  298.   }
  299.   socket.value.onclose = function (event) {
  300.     clearInterval(timer) // 停止心跳检测
  301.     console.log('Sock连接已关闭', event.code, event.reason)
  302.     // 连接已关闭,执行清理操作
  303.   }
  304.   socket.value.onerror = function (error) {
  305.     clearInterval(timer) // 停止心跳检测
  306.     console.error('Sock错误:', error)
  307.     // 处理Sock错误
  308.   }
  309. }
  310. // 2.使用sip_info中的参数注册UA对象,监听注册结果
  311. //注册UA
  312. const regUserAgent = async (sip_info) => {
  313.   let sip_id = sip_info.sip_id,
  314.     sip_pwd = sip_info.sip_pwd,
  315.     sip_host = sip_info.sip_host,
  316.     wss_url = sip_info.wss_url,
  317.     stun_host = sip_info.stun_host,
  318.     turn_host = sip_info.turn_host,
  319.     turn_pwd = sip_info.turn_pwd,
  320.     turn_user = sip_info.turn_user,
  321.     userAgentStatus = false
  322.   //配置参数
  323.   let config = {
  324.     uri: sip_id + '@' + sip_host, //此sip_id为管理员的sip_id
  325.     transportOptions: {
  326.       wsServers: [wss_url],
  327.       connectionTimeout: 30
  328.     },
  329.     authorizationUser: sip_id,
  330.     password: sip_pwd,
  331.     sessionDescriptionHandlerFactoryOptions: {
  332.       peerConnectionOptions: {
  333.         rtcConfiguration: {
  334.           iceServers: [
  335.             { urls: 'stun:' + stun_host },
  336.             {
  337.               urls: 'turn:' + turn_host,
  338.               username: turn_user,
  339.               credential: turn_pwd
  340.             }
  341.           ]
  342.         }
  343.       }
  344.     }
  345.   }
  346.   //创建user agent
  347.   userAgent = await new SIP.UA(config)
  348.   //注册成功监听处理
  349.   userAgent.on('registered', () => {
  350.     console.log('代理注册成功: registered ok')
  351.     userAgentStatus = true // 模拟注册成功
  352.     // let regUA_msgEl: any = document.getElementById('regUA_msg')
  353.     // regUA_msgEl.innerText = '用户代理注册成功!'
  354.   })
  355.   //注册失败监听处理
  356.   userAgent.on('registrationFailed', (response, cause) => {
  357.     console.log('代理注册失败: registrationFailed, ', response, cause)
  358.     userAgentStatus = false
  359.     // let regUA_msgEl: any = document.getElementById('regUA_msg')
  360.     // regUA_msgEl.innerText = '用户代理注册失败!'
  361.   })
  362.   userAgent.on('invite', function (session) {
  363.     var url = session.remoteIdentity.uri.toString() + '--->call'
  364.     var isaccept = confirm(url)
  365.     if (isaccept) {
  366.       //接收来电
  367.       session.accept({
  368.         sessionDescriptionHandlerOptions: {
  369.           constraints: {
  370.             audio: true,
  371.             video: true
  372.           }
  373.         }
  374.       })
  375.       sipsession = session
  376.       // 接听通话
  377.       session.on('accepted', function () {
  378.         // We need to check the peer connection to determine which track was added
  379.         var pc = session.sessionDescriptionHandler.peerConnection
  380.         // console.log(pc)
  381.         // console.log(pc.getLocalStreams())
  382.         // Gets remote tracks
  383.         var remoteStream = new MediaStream()
  384.         pc.getReceivers().forEach(function (receiver) {
  385.           remoteStream.addTrack(receiver.track)
  386.         })
  387.         remoteVideo.value.srcObject = remoteStream
  388.         remoteVideo.value.play()
  389.         if (pc.getSenders()) {
  390.           // var localStream = new MediaStream()
  391.           // pc.getSenders().forEach(function (sender) {
  392.           //   localStream.addTrack(sender.track)
  393.           // })
  394.           // localVideo.value.srcObject = localStream
  395.           // localVideo.value.play()
  396.         }
  397.       })
  398.     } else {
  399.       //拒绝来电
  400.       session.reject()
  401.     }
  402.   })
  403. }
  404. // watch(videoVisible, (newVal, oldVal) => {
  405. //   // console.log(newVal,oldVal);
  406. //   if (newVal == false) {
  407. //     socket.value.close() // 断开websocket
  408. //     endCall() // 关闭视频
  409. //   }
  410. // })
  411. /** 初始化 **/
  412. onMounted(async () => {
  413.   // getList()
  414.   // websocketFun() //创建WebSocket连接
  415.   // id.value = route.params.id
  416.   // let myObject = myStore.videoData
  417.   // console.log(myObject, myObject?.type,  myObject?.type, 'ccccccccccccccccccc')
  418.   // 获取当前页面的 URL 地址
  419.   const urlParams = new URLSearchParams(window.location.search)
  420.   // 获取参数值
  421.   const type = urlParams.get('type')
  422.   const id = urlParams.get('id')
  423.   const deviceId = urlParams.get('deviceid')
  424.   const myObject = {
  425.     type,
  426.     row: {
  427.       aqmPkId: id,
  428.       deviceId
  429.     }
  430.   }
  431.   // 现在 type、id 和 deviceId 分别包含了 URL 中对应的数值
  432.   handleVideoPhone(myObject.type, myObject.row)
  433.   // const ccc = 'xxxxxxxxxxxxxx';
  434.   // const url = `https://www.XXXXXX.com:1443/drone/ws/getUserDeviceSipId/?deviceId=${ccc}`;
  435.   // const res = await axios.get(url);
  436.   // if (res.status === 200) {
  437.   //   console.log('设备sip_id', res);
  438.   //   sip_id.value = res.data.sip_id;
  439.   // } else {
  440.   //   console.log('接口未连接');
  441.   // }
  442. })
  443. // 在组件离开时销毁定时器
  444. onUnmounted(() => {
  445.   // socket.value.close() // 清空websocket
  446.   clearInterval(timer)
  447. })
  448. </script>
  449. <style lang="scss" scoped></style>
复制代码


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。




欢迎光临 ToB企服应用市场:ToB评测及商务社交产业平台 (https://dis.qidao123.com/) Powered by Discuz! X3.4