useCamera.ts 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356
  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 Player from 'xgplayer';
  6. import HlsPlugin from 'xgplayer-hls';
  7. import FlvPlugin from 'xgplayer-flv';
  8. import 'xgplayer/dist/index.min.css';
  9. export function useCamera() {
  10. const cameraList = (params) => defHttp.get({ url: '/safety/ventanalyCamera/list', params });
  11. const cameraAddrList = (params) => defHttp.post({ url: '/ventanaly-device/camera/info', params });
  12. const cameraAddr = (params) => defHttp.get({ url: '/ventanaly-device/camera/queryByCameraCode', params });
  13. let webRtcServer = <any[]>[];
  14. const playerList = <any[]>[];
  15. const playerDoms = <(HTMLVideoElement | undefined | null)[]>[];
  16. const videoParentDomList: (HTMLElement | [string, { name: string; addr: string }])[] = [];
  17. async function getCamera(deviceid, parentPlayerDom?) {
  18. removeCamera();
  19. if (!parentPlayerDom) {
  20. parentPlayerDom = document.createElement('div');
  21. parentPlayerDom.setAttribute('style', `top:0px; left: 0px; width: 100%; height: 100%; position: fixed; z-index: 999;`);
  22. }
  23. const res = await cameraList({ deviceid });
  24. const cameras: [] = res.records || [];
  25. // const cameras: [] = [
  26. // {
  27. // name: '1111',
  28. // devicekind: 'toHKRtsp',
  29. // },
  30. // {
  31. // name: '2222',
  32. // devicekind: 'toHKRtsp',
  33. // },
  34. // // {
  35. // // name: '3333',
  36. // // devicekind: 'toHKRtsp',
  37. // // },
  38. // // {
  39. // // name: '4444',
  40. // // devicekind: 'toHKRtsp',
  41. // // },
  42. // ];
  43. const cameraAddrs: any[] = [],
  44. cameraNames: string[] = [];
  45. if (cameras.length > 0) {
  46. for (let i = 0; i < cameras.length; i++) {
  47. const item = cameras[i];
  48. if (item['devicekind'] === 'toHKRtsp') {
  49. // 从海康平台接口获取视频流
  50. const data = await cameraAddr({ cameraCode: item['addr'] });
  51. if (data && data['url']) {
  52. cameraAddrs.push({ name: item['name'], addr: data['url'] });
  53. }
  54. // // 从海康平台接口获取视频流测试
  55. // cameraAddrs.push({
  56. // name: item['name'],
  57. // addr: 'https://demo.unified-streaming.com/k8s/features/stable/video/tears-of-steel/tears-of-steel.mp4/.m3u8',
  58. // });
  59. } else if (item['devicekind'] == 'toHKR') {
  60. cameraNames.push(item['name']);
  61. } else {
  62. cameraAddrs.push({ name: item['name'], addr: item['addr'] });
  63. }
  64. }
  65. }
  66. if (cameraNames.length > 0) {
  67. // 请求接口从装备院拿数据
  68. const addrs: string[] = await cameraAddrList({ cameraNameList: cameraNames });
  69. for (let i = 0; i < addrs.length; i++) {
  70. cameraAddrs.push({ name: '摄像头' + i, addr: addrs[i] });
  71. }
  72. }
  73. await deviceCameraInit(cameraAddrs, parentPlayerDom);
  74. }
  75. function deviceCameraInit(cameraAddrs, player: HTMLElement) {
  76. const newWebRtcServer = [];
  77. let livePlayerDiv: HTMLElement | null = document.getElementById('LivePlayerBox');
  78. if (livePlayerDiv) {
  79. livePlayerDiv.remove();
  80. livePlayerDiv = null;
  81. }
  82. if (!livePlayerDiv) {
  83. const dom = document.createElement('div');
  84. dom.setAttribute('id', 'LivePlayerBox');
  85. livePlayerDiv = dom;
  86. player.appendChild(livePlayerDiv);
  87. }
  88. return new Promise((resolve) => {
  89. const playCamrea = () => {
  90. if (cameraAddrs.length > 0) {
  91. const promiseList: Promise<any>[] = [];
  92. cameraAddrs.forEach(async (cameraUrl: { name: string; addr: string }, index) => {
  93. const promise = new Promise(async (childResolve) => {
  94. let cameraNameDom: null | HTMLElement = null;
  95. console.log('摄像头地址--------->', cameraUrl, cameraUrl.addr.startsWith('rtsp://'), livePlayerDiv);
  96. if (cameraUrl.addr.includes('0.0.0.0')) {
  97. cameraUrl.addr = cameraUrl.addr.replace('0.0.0.0', window.location.hostname);
  98. }
  99. if (cameraUrl.addr && cameraUrl.addr.startsWith('rtsp://')) {
  100. const server = webRtcServer?.shift();
  101. let videoDom: HTMLVideoElement | null = null;
  102. if (server) {
  103. try {
  104. videoDom = server.videoElement as HTMLVideoElement;
  105. videoDom.volume = 0;
  106. const cameraNameDom = videoDom.parentElement?.getElementsByClassName('video-name')[0];
  107. if (cameraNameDom) cameraNameDom.innerText = cameraUrl.name;
  108. playerDoms.unshift(videoDom);
  109. newWebRtcServer.unshift(server);
  110. // videoParentDomList.unshift()
  111. await server.connect(cameraUrl['addr']);
  112. videoDom.play();
  113. childResolve(null);
  114. } catch (error) {
  115. playerDoms.unshift(undefined);
  116. childResolve(null);
  117. }
  118. } else {
  119. videoDom = document.createElement('video');
  120. videoDom.volume = 0;
  121. videoDom.setAttribute('class', 'rtspVideo');
  122. videoDom.setAttribute('muted', 'muted');
  123. videoDom.setAttribute('poster', '/src/assets/images/vent/noSinge.png');
  124. videoDom.autoplay = true;
  125. try {
  126. const server = new window['WebRtcStreamer'](videoDom, location.protocol + VUE_APP_URL.webRtcUrl);
  127. newWebRtcServer.unshift(server);
  128. await server.connect(cameraUrl.addr);
  129. videoDom.play();
  130. playerDoms.unshift(videoDom);
  131. childResolve(null);
  132. } catch (error) {
  133. console.log('WebRtcStreamer 抛出异常!!!!!!');
  134. playerDoms.unshift(null);
  135. childResolve(null);
  136. }
  137. }
  138. if (videoDom) {
  139. const videoParentDom: HTMLElement = document.createElement('div');
  140. videoParentDom.setAttribute('class', 'video-parent');
  141. videoParentDom.appendChild(videoDom);
  142. cameraNameDom = document.createElement('div');
  143. cameraNameDom.setAttribute('class', 'video-name');
  144. cameraNameDom.innerText = cameraUrl.name;
  145. videoParentDom.appendChild(cameraNameDom);
  146. videoParentDom.addEventListener('dblclick', () => {
  147. if (videoDom?.requestFullscreen) {
  148. videoDom.requestFullscreen();
  149. videoDom.play();
  150. }
  151. });
  152. videoParentDomList.push(videoParentDom);
  153. }
  154. } else {
  155. // try {
  156. // fetch(cameraUrl.addr, {
  157. // method: 'get',
  158. // mode: 'no-cors',
  159. // })
  160. // .then(() => {
  161. // videoParentDomList.push(['player', cameraUrl]);
  162. // childResolve(null);
  163. // })
  164. // .catch(() => {
  165. // videoParentDomList.push(['onPlayer' + index, cameraUrl]);
  166. // childResolve(null);
  167. // });
  168. // } catch (error) {
  169. // // console.log('可以捕获到异常吗?????');
  170. // childResolve(null);
  171. // }
  172. videoParentDomList.push(['player' + index, cameraUrl]);
  173. childResolve(null);
  174. }
  175. });
  176. promiseList.push(promise);
  177. });
  178. Promise.all(promiseList).then(() => {
  179. resolve(null);
  180. });
  181. } else {
  182. resolve(null);
  183. }
  184. };
  185. playCamrea();
  186. }).then(() => {
  187. videoParentDomList.forEach((videoParent) => {
  188. if (typeof videoParent[0] === 'string' && livePlayerDiv) {
  189. const videoParentDom: HTMLElement = document.createElement('div');
  190. videoParentDom.setAttribute('class', 'liveVideo');
  191. livePlayerDiv?.appendChild(videoParentDom);
  192. useDrag(videoParentDom);
  193. const videoDom: HTMLElement = document.createElement('div');
  194. videoDom.setAttribute('id', videoParent[0]);
  195. videoParentDom.appendChild(videoDom);
  196. if (videoParent[1] && videoParent[1].addr) {
  197. const fileExtension = videoParent[1].addr.split('.').pop();
  198. let player;
  199. if (fileExtension === 'flv') {
  200. player = new Player({
  201. id: videoParent[0],
  202. url: videoParent[1].addr,
  203. width: 294,
  204. height: 188,
  205. poster: '/src/assets/images/vent/noSinge.png',
  206. plugins: [FlvPlugin],
  207. fluid: true,
  208. autoplay: true,
  209. isLive: true,
  210. playsinline: false,
  211. screenShot: true,
  212. whitelist: [''],
  213. ignores: ['time'],
  214. closeVideoClick: true,
  215. customConfig: {
  216. isClickPlayBack: false,
  217. },
  218. flv: {
  219. retryCount: 3, // 重试 3 次,默认值
  220. retryDelay: 1000, // 每次重试间隔 1 秒,默认值
  221. loadTimeout: 10000, // 请求超时时间为 10 秒,默认值
  222. fetchOptions: {
  223. // 该参数会透传给 fetch,默认值为 undefined
  224. mode: 'cors',
  225. },
  226. targetLatency: 10, // 直播目标延迟,默认 10 秒
  227. maxLatency: 20, // 直播允许的最大延迟,默认 20 秒
  228. disconnectTime: 10, // 直播断流时间,默认 0 秒,(独立使用时等于 maxLatency)
  229. maxJumpDistance: 10,
  230. },
  231. });
  232. playerList.push(player);
  233. }
  234. if (fileExtension === 'm3u8') {
  235. let player;
  236. if (document.createElement('video').canPlayType('application/vnd.apple.mpegurl')) {
  237. // 原生支持 hls 播放
  238. player = new Player({
  239. id: videoParent[0],
  240. url: videoParent[1].addr,
  241. width: 294,
  242. height: 188,
  243. isLive: true,
  244. autoplay: true,
  245. autoplayMuted: true,
  246. cors: true,
  247. poster: '/src/assets/images/vent/noSinge.png',
  248. hls: {
  249. retryCount: 3, // 重试 3 次,默认值
  250. retryDelay: 1000, // 每次重试间隔 1 秒,默认值
  251. loadTimeout: 10000, // 请求超时时间为 10 秒,默认值
  252. fetchOptions: {
  253. // 该参数会透传给 fetch,默认值为 undefined
  254. mode: 'cors',
  255. },
  256. targetLatency: 10, // 直播目标延迟,默认 10 秒
  257. maxLatency: 20, // 直播允许的最大延迟,默认 20 秒
  258. disconnectTime: 10, // 直播断流时间,默认 0 秒,(独立使用时等于 maxLatency)
  259. maxJumpDistance: 10,
  260. },
  261. });
  262. } else if (HlsPlugin.isSupported()) {
  263. // 第一步
  264. player = new Player({
  265. id: videoParent[0],
  266. url: videoParent[1].addr,
  267. width: 294,
  268. height: 188,
  269. isLive: true,
  270. autoplay: true,
  271. autoplayMuted: true,
  272. plugins: [HlsPlugin], // 第二步
  273. poster: '/src/assets/images/vent/noSinge.png',
  274. hls: {
  275. retryCount: 3, // 重试 3 次,默认值
  276. retryDelay: 1000, // 每次重试间隔 1 秒,默认值
  277. loadTimeout: 10000, // 请求超时时间为 10 秒,默认值
  278. fetchOptions: {
  279. // 该参数会透传给 fetch,默认值为 undefined
  280. mode: 'cors',
  281. },
  282. targetLatency: 10, // 直播目标延迟,默认 10 秒
  283. maxLatency: 20, // 直播允许的最大延迟,默认 20 秒
  284. disconnectTime: 10, // 直播断流时间,默认 0 秒,(独立使用时等于 maxLatency)
  285. maxJumpDistance: 10,
  286. },
  287. });
  288. }
  289. playerList.push(player);
  290. }
  291. playerList.push(player);
  292. }
  293. } else {
  294. useDrag(videoParent as HTMLElement);
  295. livePlayerDiv?.appendChild(videoParent as Node);
  296. }
  297. });
  298. const players = livePlayerDiv?.getElementsByClassName('player');
  299. if (players && players.length) {
  300. for (let i = 0; i < players.length; i++) {
  301. try {
  302. const isCanPlayer = !players[i].getAttribute('id')?.startsWith('onPlayer');
  303. const dom = players[i].getElementsByTagName('video')[0];
  304. if (dom && isCanPlayer) {
  305. playerDoms.unshift(dom);
  306. } else {
  307. playerDoms.unshift(null);
  308. }
  309. } catch (error) {
  310. console.log('可以捕获到异常吗?????');
  311. playerDoms.unshift(null);
  312. }
  313. }
  314. }
  315. if (webRtcServer.length > 0) {
  316. for (let i = 0; i < webRtcServer.length; i++) {
  317. webRtcServer[i].videoElement.parentElement.remove();
  318. webRtcServer[i].disconnect();
  319. webRtcServer[i] = undefined;
  320. }
  321. }
  322. webRtcServer.length = 0;
  323. webRtcServer = newWebRtcServer;
  324. });
  325. }
  326. function removeCamera() {
  327. if (webRtcServer.length > 0) {
  328. webRtcServer.forEach((item) => {
  329. item.disconnect();
  330. });
  331. webRtcServer = [];
  332. }
  333. playerDoms.forEach((dom) => {
  334. dom?.remove();
  335. });
  336. videoParentDomList.length = 0;
  337. playerDoms.length = 0;
  338. }
  339. return {
  340. getCamera,
  341. webRtcServer,
  342. playerDoms,
  343. deviceCameraInit,
  344. removeCamera,
  345. };
  346. }