useCamera.ts 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243
  1. import { defHttp } from '/@/utils/http/axios';
  2. import { render, h, nextTick } from 'vue';
  3. import LivePlayer from '@liveqing/liveplayer-v3';
  4. import { useDrag } from '../event/useDrag';
  5. import { useGlobSetting } from '/@/hooks/setting';
  6. export function useCamera() {
  7. const cameraList = (params) => defHttp.get({ url: '/safety/ventanalyCamera/list', params });
  8. const cameraAddrList = (params) => defHttp.post({ url: '/ventanaly-device/camera/info', params });
  9. const cameraAddr = (params) => defHttp.get({ url: '/ventanaly-device/camera/queryByCameraCode', params });
  10. let webRtcServer = <any[]>[];
  11. let playerDoms = <(HTMLVideoElement | undefined | null)[]>[];
  12. async function getCamera(deviceid, parentPlayerDom?) {
  13. if (!parentPlayerDom) {
  14. parentPlayerDom = document.createElement('div');
  15. parentPlayerDom.setAttribute('style', `top:0px; left: 0px; width: 100%; height: 100%; position: fixed; z-index: 999;`);
  16. }
  17. const res = await cameraList({ deviceid });
  18. const cameras: [] = res.records || [];
  19. const cameraAddrs: any[] = [],
  20. cameraNames: string[] = [];
  21. if (cameras.length > 0) {
  22. cameras.forEach(async (item) => {
  23. if (item['devicekind'] === 'toHKRtsp') {
  24. // 从海康平台接口获取视频流
  25. // cameraAddrs.push({ name: item['name'], addr: 'http://devimages.apple.com/iphone/samples/bipbop/gear1/prog_index.m3u8' });
  26. const data = await cameraAddr({ cameraCode: item['addr'] });
  27. if (data && data['url']) {
  28. cameraAddrs.push({ name: item['name'], addr: data['url'] });
  29. }
  30. } else {
  31. cameraAddrs.push({ name: item['name'], addr: item['addr'] });
  32. }
  33. });
  34. }
  35. if (cameraNames.length > 0) {
  36. // 请求接口从装备院拿数据
  37. const addrs: string[] = await cameraAddrList({ cameraNameList: cameraNames });
  38. for (let i = 0; i < addrs.length; i++) {
  39. cameraAddrs.push({ name: '摄像头' + i, addr: addrs[i] });
  40. }
  41. }
  42. const obj = await deviceCameraInit(cameraAddrs, parentPlayerDom, webRtcServer);
  43. webRtcServer = obj.webRtcServerList;
  44. playerDoms = obj.playerDoms;
  45. }
  46. function deviceCameraInit(cameraAddrs, player: HTMLElement, webRtcServerList: any[] = []) {
  47. const playerDoms: (HTMLVideoElement | undefined | null)[] = [];
  48. const webRtcServer: any[] = [];
  49. let livePlayerDiv: HTMLElement | null = document.getElementById('LivePlayerBox');
  50. if (livePlayerDiv) {
  51. livePlayerDiv.remove();
  52. livePlayerDiv = null;
  53. }
  54. if (!livePlayerDiv) {
  55. const dom = document.createElement('div');
  56. dom.setAttribute('id', 'LivePlayerBox');
  57. livePlayerDiv = dom;
  58. player.appendChild(livePlayerDiv);
  59. }
  60. const videoParentDomList: (HTMLElement | [string, { name: string; addr: string }])[] = [];
  61. return new Promise((resolve) => {
  62. const playCamrea = () => {
  63. if (cameraAddrs.length > 0) {
  64. const promiseList: Promise<any>[] = [];
  65. cameraAddrs.forEach(async (cameraUrl: { name: string; addr: string }, index) => {
  66. const promise = new Promise(async (childResolve) => {
  67. let cameraNameDom: null | HTMLElement = null;
  68. console.log('摄像头地址--------->', cameraUrl, cameraUrl.addr.startsWith('rtsp://'), livePlayerDiv);
  69. if (cameraUrl.addr.includes('0.0.0.0')) {
  70. cameraUrl.addr = cameraUrl.addr.replace('0.0.0.0', window.location.hostname);
  71. }
  72. if (cameraUrl.addr && cameraUrl.addr.startsWith('rtsp://')) {
  73. const server = webRtcServerList.shift();
  74. let videoDom: HTMLVideoElement | null = null;
  75. if (server) {
  76. try {
  77. videoDom = server.videoElement as HTMLVideoElement;
  78. videoDom.volume = 0;
  79. const cameraNameDom = videoDom.parentElement?.getElementsByClassName('video-name')[0];
  80. if (cameraNameDom) cameraNameDom.innerText = cameraUrl.name;
  81. playerDoms.unshift(videoDom);
  82. webRtcServer.unshift(server);
  83. // videoParentDomList.unshift()
  84. await server.connect(cameraUrl['addr']);
  85. videoDom.play();
  86. childResolve(null);
  87. } catch (error) {
  88. playerDoms.unshift(undefined);
  89. childResolve(null);
  90. }
  91. } else {
  92. videoDom = document.createElement('video');
  93. videoDom.volume = 0;
  94. videoDom.setAttribute('class', 'rtspVideo');
  95. videoDom.setAttribute('muted', 'muted');
  96. videoDom.setAttribute('poster', '/src/assets/images/vent/noSinge.png');
  97. videoDom.autoplay = true;
  98. try {
  99. const server = new window['WebRtcStreamer'](videoDom, location.protocol + VUE_APP_URL.webRtcUrl);
  100. webRtcServer.unshift(server);
  101. await server.connect(cameraUrl.addr);
  102. videoDom.play();
  103. playerDoms.unshift(videoDom);
  104. childResolve(null);
  105. } catch (error) {
  106. console.log('WebRtcStreamer 抛出异常!!!!!!');
  107. playerDoms.unshift(null);
  108. childResolve(null);
  109. }
  110. }
  111. if (videoDom) {
  112. const videoParentDom: HTMLElement = document.createElement('div');
  113. videoParentDom.setAttribute('class', 'video-parent');
  114. videoParentDom.appendChild(videoDom);
  115. cameraNameDom = document.createElement('div');
  116. cameraNameDom.setAttribute('class', 'video-name');
  117. cameraNameDom.innerText = cameraUrl.name;
  118. videoParentDom.appendChild(cameraNameDom);
  119. videoParentDom.addEventListener('dblclick', () => {
  120. if (videoDom?.requestFullscreen) {
  121. videoDom.requestFullscreen();
  122. videoDom.play();
  123. }
  124. });
  125. videoParentDomList.push(videoParentDom);
  126. }
  127. } else {
  128. try {
  129. fetch(cameraUrl.addr, {
  130. method: 'get',
  131. mode: 'no-cors',
  132. })
  133. .then(() => {
  134. videoParentDomList.push(['player', cameraUrl]);
  135. childResolve(null);
  136. })
  137. .catch(() => {
  138. videoParentDomList.push(['onPlayer' + index, cameraUrl]);
  139. childResolve(null);
  140. });
  141. } catch (error) {
  142. // console.log('可以捕获到异常吗?????');
  143. childResolve(null);
  144. }
  145. }
  146. });
  147. promiseList.push(promise);
  148. });
  149. Promise.all(promiseList).then(() => {
  150. resolve(null);
  151. });
  152. } else {
  153. resolve(null);
  154. }
  155. };
  156. playCamrea();
  157. }).then(() => {
  158. videoParentDomList.forEach((videoParentDom) => {
  159. if (typeof videoParentDom[0] === 'string' && livePlayerDiv) {
  160. const videoDom: HTMLElement = document.createElement('div');
  161. videoDom.setAttribute('class', 'liveVideo');
  162. livePlayerDiv?.appendChild(videoDom);
  163. useDrag(videoDom);
  164. if (videoParentDom[0].startsWith('onPlayer')) {
  165. render(
  166. h(LivePlayer, {
  167. class: 'player',
  168. id: videoParentDom[0],
  169. muted: 'muted',
  170. autoplay: true,
  171. live: true,
  172. videoUrl: videoParentDom[1].addr,
  173. videoTitle: videoParentDom[1].name,
  174. alt: '无信号',
  175. poster: '/src/assets/images/vent/noSinge.png',
  176. }),
  177. videoDom
  178. );
  179. } else {
  180. render(
  181. h(LivePlayer, {
  182. class: 'player',
  183. id: videoParentDom[0],
  184. muted: 'muted',
  185. autoplay: true,
  186. live: true,
  187. videoUrl: videoParentDom[1].addr,
  188. videoTitle: videoParentDom[1].name,
  189. alt: '无信号',
  190. poster: '/src/assets/images/vent/noSinge.png',
  191. }),
  192. videoDom
  193. );
  194. }
  195. } else {
  196. useDrag(videoParentDom as HTMLElement);
  197. livePlayerDiv?.appendChild(videoParentDom as Node);
  198. }
  199. });
  200. const players = livePlayerDiv?.getElementsByClassName('player');
  201. if (players && players.length) {
  202. for (let i = 0; i < players.length; i++) {
  203. try {
  204. const isCanPlayer = !players[i].getAttribute('id')?.startsWith('onPlayer');
  205. const dom = players[i].getElementsByTagName('video')[0];
  206. if (dom && isCanPlayer) {
  207. playerDoms.unshift(dom);
  208. } else {
  209. playerDoms.unshift(null);
  210. }
  211. } catch (error) {
  212. console.log('可以捕获到异常吗?????');
  213. playerDoms.unshift(null);
  214. }
  215. }
  216. }
  217. if (webRtcServerList.length > 0) {
  218. for (let i = 0; i < webRtcServerList.length; i++) {
  219. webRtcServerList[i].videoElement.parentElement.remove();
  220. webRtcServerList[i].disconnect();
  221. webRtcServerList[i] = undefined;
  222. }
  223. }
  224. return { webRtcServerList: webRtcServer, playerDoms };
  225. });
  226. }
  227. return {
  228. getCamera,
  229. webRtcServer,
  230. playerDoms,
  231. deviceCameraInit,
  232. };
  233. }