index.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378
  1. window.onload = function () {
  2. const {
  3. AppId,
  4. AppKey
  5. } = window.config || {};
  6. // 此处使用固定的房间号的随机的用户ID,请自行替换
  7. const RoomId = "ssss02";
  8. const UserId = Math.floor(Math.random() * 1000000).toString();
  9. if (!AppId || !AppKey) {
  10. alert('请先设置 AppId 和 AppKey');
  11. return;
  12. }
  13. if (!RoomId) {
  14. alert('请先设置 RoomId');
  15. return;
  16. }
  17. console.log('UCloudRTC sdk version: ', UCloudRTC.version);
  18. canAutoplay.video().then(res => {
  19. if (!res.result) {
  20. const pullerElm = document.querySelector('#pullers-title');
  21. const hintElm = document.createElement('p');
  22. hintElm.className = 'hint';
  23. hintElm.textContent = '当前浏览器不支持自动播放视频,订阅远端流成功后,可点击对应的视频区域进行播放'
  24. pullerElm.appendChild(hintElm);
  25. }
  26. });
  27. // 用于维护应用内的状态
  28. const App = {
  29. state: {
  30. roomId: RoomId,
  31. userId: UserId,
  32. isJoinedRoom: false,
  33. selectedStream: null,
  34. localStreams: [],
  35. remoteStreams: []
  36. },
  37. client: null,
  38. setState: function (key, value) {
  39. const keys = Object.keys(this.state);
  40. if (keys.includes(key)) {
  41. this.state[key] = value;
  42. }
  43. switch (key) {
  44. case 'roomId':
  45. this.renderRoomId();
  46. break;
  47. case 'isJoinedRoom':
  48. this.renderRoomStatus();
  49. break;
  50. case 'selectedStream':
  51. this.renderSelectedStream();
  52. break;
  53. case 'localStream-add':
  54. this.loadStream(value, 'pusher');
  55. break;
  56. case 'localStream-remove':
  57. this.unloadStream(value, 'pusher');
  58. break;
  59. case 'remoteStream-add':
  60. this.loadStream(value, 'puller');
  61. break;
  62. case 'remoteStream-remove':
  63. this.unloadStream(value, 'puller');
  64. break;
  65. case 'remoteStream-update':
  66. this.rerenderStream(value);
  67. break;
  68. default:
  69. }
  70. },
  71. renderRoomId: function () {
  72. const { roomId } = this.state;
  73. const roomElem = document.querySelector('#roomId');
  74. roomElem.innerHTML = roomId;
  75. },
  76. renderRoomStatus: function () {
  77. const { isJoinedRoom } = this.state;
  78. const roomStatusElem = document.querySelector('#roomStatus');
  79. if (isJoinedRoom) {
  80. roomStatusElem.innerHTML = "已加入";
  81. } else {
  82. roomStatusElem.innerHTML = "未加入";
  83. }
  84. },
  85. renderSelectedStream: function () {
  86. const { selectedStream } = this.state;
  87. const selectedStreamElem = document.querySelector('#selectedStream');
  88. if (selectedStream) {
  89. selectedStreamElem.innerHTML = selectedStream.sid;
  90. } else {
  91. selectedStreamElem.innerHTML = "未选择";
  92. }
  93. },
  94. loadStream: function (stream, type) {
  95. let parent;
  96. let isPuller;
  97. switch (type) {
  98. case 'pusher':
  99. parent = document.querySelector('#pushers');
  100. break;
  101. case 'puller':
  102. parent = document.querySelector('#pullers');
  103. isPuller = true;
  104. break;
  105. default:
  106. return;
  107. }
  108. const player = document.createElement('div');
  109. player.className = 'media-player';
  110. player.id = 's' + stream.sid;
  111. const uIDElem = document.createElement('div');
  112. uIDElem.innerHTML = `用户ID:${stream.uid}`;
  113. uIDElem.style = 'overflow: hidden; text-overflow: ellipsis;';
  114. const sIDElem = document.createElement('div');
  115. sIDElem.innerHTML = `流ID:${stream.sid}`;
  116. sIDElem.style = 'overflow: hidden; text-overflow: ellipsis;';
  117. player.append(uIDElem);
  118. player.append(sIDElem);
  119. if (isPuller) {
  120. const pElem = document.createElement('p');
  121. pElem.innerHTML = 'unsubscribe';
  122. player.append(pElem);
  123. } else {
  124. const videoElem = document.createElement('video');
  125. videoElem.autoplay = true;
  126. videoElem.playsInline = true;
  127. videoElem['webkit-playsinline'] = 'true';
  128. videoElem.srcObject = stream.mediaStream;
  129. player.append(videoElem);
  130. }
  131. player.addEventListener('click', function() {
  132. this.handleSelectStream(stream);
  133. }.bind(this));
  134. parent.append(player);
  135. },
  136. unloadStream: function (stream, type) {
  137. let parent;
  138. switch (type) {
  139. case 'pusher':
  140. parent = document.querySelector('#pushers');
  141. break;
  142. case 'puller':
  143. parent = document.querySelector('#pullers');
  144. break;
  145. default:
  146. return;
  147. }
  148. const player = document.querySelector('#s' + stream.sid);
  149. parent.removeChild(player);
  150. },
  151. rerenderStream: function (stream) {
  152. const player = document.querySelector('#s' + stream.sid);
  153. if (stream.mediaStream) {
  154. const videoElem = document.createElement('video');
  155. videoElem.autoplay = true;
  156. videoElem.playsInline = true;
  157. videoElem['webkit-playsinline'] = 'true';
  158. videoElem.srcObject = stream.mediaStream;
  159. const pElem = player.querySelector('p');
  160. player.removeChild(pElem);
  161. player.appendChild(videoElem);
  162. videoElem.addEventListener('click', function() {
  163. if (videoElem.paused) {
  164. videoElem.play();
  165. }
  166. });
  167. } else {
  168. const videoElem = player.querySelector('video');
  169. const pElem = document.createElement('p');
  170. pElem.innerHTML = 'unsubscribe';
  171. player.removeChild(videoElem);
  172. player.appendChild(pElem);
  173. }
  174. },
  175. init: function () {
  176. this.renderRoomId();
  177. const token = UCloudRTC.generateToken(AppId, AppKey, RoomId, UserId);
  178. this.client = new UCloudRTC.Client(AppId, token);
  179. // 监听 publish 成功的事件
  180. this.client.on('stream-published', (localStream) => {
  181. console.info('stream-published: ', localStream);
  182. const { localStreams } = this.state;
  183. localStreams.push(localStream);
  184. this.setState('localStream-add', localStream);
  185. });
  186. this.client.on('stream-added', (remoteStream) => {
  187. console.info('stream-added: ', remoteStream);
  188. const { remoteStreams } = this.state;
  189. remoteStreams.push(remoteStream);
  190. // 自动订阅
  191. this.client.subscribe(remoteStream.sid, (err) => {
  192. console.error('自动订阅失败:', err);
  193. });
  194. this.setState('remoteStream-add', remoteStream);
  195. });
  196. this.client.on('stream-subscribed', (remoteStream) => {
  197. console.info('stream-subscribed: ', remoteStream);
  198. const { remoteStreams } = this.state;
  199. const idx = remoteStreams.findIndex(item => item.sid === remoteStream.sid);
  200. if (idx >= 0) {
  201. remoteStreams.splice(idx, 1, remoteStream);
  202. }
  203. this.setState('remoteStream-update', remoteStream);
  204. });
  205. this.client.on('stream-removed', (remoteStream) => {
  206. console.info('stream-removed: ', remoteStream);
  207. const { remoteStreams } = this.state;
  208. const idx = remoteStreams.findIndex(item => item.sid === remoteStream.sid);
  209. if (idx >= 0) {
  210. const p = remoteStreams.splice(idx, 1)[0];
  211. this.setState('remoteStream-remove', p);
  212. }
  213. });
  214. this.client.on('connection-state-change', ({ previous, current }) => {
  215. console.log(`连接状态 ${previous} -> ${current}`);
  216. });
  217. this.client.on('stream-reconnected', ({previous, current}) => {
  218. console.log(`流已断开重连`);
  219. if (previous.type === 'publish') {
  220. const { localStreams } = this.state;
  221. const idx = localStreams.findIndex(item => item.sid === previous.sid);
  222. if (idx >= 0) {
  223. const oldStream = localStreams.splice(idx, 1, current)[0];
  224. this.setState('localStream-remove', oldStream);
  225. localStreams.push(current);
  226. this.setState('localStream-add', current);
  227. }
  228. } else {
  229. const { remoteStreams } = this.state;
  230. const idx = remoteStreams.findIndex(item => item.sid === previous.sid);
  231. if (idx >= 0) {
  232. const oldStream = remoteStreams.splice(idx, 1, current)[0];
  233. this.setState('remoteStream-remove', oldStream);
  234. remoteStreams.push(current);
  235. this.setState('remoteStream-add', current);
  236. }
  237. }
  238. });
  239. document.querySelector('#joinRoomBtn').addEventListener('click', this.handleJoinRoom.bind(this));
  240. document.querySelector('#publishBtn').addEventListener('click', this.handlePublish.bind(this));
  241. document.querySelector('#publishScreenBtn').addEventListener('click', this.handlePublishScreen.bind(this));
  242. document.querySelector('#unPublishBtn').addEventListener('click', this.handleUnpublish.bind(this));
  243. document.querySelector('#subscribeBtn').addEventListener('click', this.handleSubscribe.bind(this));
  244. document.querySelector('#unSubscribeBtn').addEventListener('click', this.handleUnsubscribe.bind(this));
  245. document.querySelector('#leaveRoomBtn').addEventListener('click', this.handleLeaveRoom.bind(this));
  246. window.addEventListener('beforeunload', this.handleLeaveRoom.bind(this));
  247. },
  248. // 操作
  249. handleJoinRoom: function () {
  250. const {
  251. roomId,
  252. userId,
  253. isJoinedRoom
  254. } = this.state;
  255. if (isJoinedRoom) {
  256. alert('已经加入了房间');
  257. return;
  258. }
  259. if (!roomId) {
  260. alert('请先填写房间号');
  261. return;
  262. }
  263. this.client.joinRoom(roomId, userId, () => {
  264. console.info('加入房间成功');
  265. this.setState('isJoinedRoom', true);
  266. }, (err) => {
  267. console.error('加入房间失败: ', err);
  268. });
  269. },
  270. handlePublish: function () {
  271. this.client.publish(err => {
  272. console.error(`发布失败:错误码 - ${err.name},错误信息 - ${err.message}`);
  273. });
  274. },
  275. handlePublishScreen: function () {
  276. this.client.publish({ audio: false, video: false, screen: true }, err => {
  277. console.error(`发布失败:错误码 - ${err.name},错误信息 - ${err.message}`);
  278. });
  279. },
  280. handleUnpublish: function () {
  281. const {
  282. selectedStream
  283. } = this.state;
  284. if (!selectedStream) {
  285. alert('未选择需要取消发布的本地流');
  286. return;
  287. }
  288. this.client.unpublish(selectedStream.sid, (stream) => {
  289. console.info('取消发布本地流成功:', stream);
  290. const {
  291. localStreams
  292. } = this.state;
  293. const idx = localStreams.findIndex(item => item.sid === stream.sid);
  294. if (idx >= 0) {
  295. const p = localStreams.splice(idx, 1)[0];
  296. this.setState('selectedStream', null);
  297. this.setState('localStream-remove', p);
  298. }
  299. }, (err) => {
  300. console.error('取消发布本地流失败:', err);
  301. })
  302. },
  303. handleSubscribe: function () {
  304. const {
  305. selectedStream
  306. } = this.state;
  307. if (!selectedStream) {
  308. alert('未选择需要订阅的远端流');
  309. return;
  310. }
  311. this.client.subscribe(selectedStream.sid, (err) => {
  312. console.error('订阅失败:', err);
  313. });
  314. },
  315. handleUnsubscribe: function () {
  316. const {
  317. selectedStream
  318. } = this.state;
  319. if (!selectedStream) {
  320. alert('未选择需要取消订阅的远端流');
  321. return;
  322. }
  323. this.client.unsubscribe(selectedStream.sid, (stream) => {
  324. console.info('取消订阅成功:', stream);
  325. const {
  326. remoteStreams
  327. } = this.state;
  328. const idx = remoteStreams.findIndex(item => item.sid === stream.sid);
  329. if (idx >= 0) {
  330. remoteStreams.splice(idx, 1, stream);
  331. this.setState('remoteStream-update', stream);
  332. }
  333. }, (err) => {
  334. console.error('订阅失败:', err);
  335. });
  336. },
  337. handleLeaveRoom: function () {
  338. const {
  339. isJoinedRoom
  340. } = this.state;
  341. if (!isJoinedRoom) {
  342. return;
  343. }
  344. this.client.leaveRoom(() => {
  345. console.info('离开房间成功');
  346. this.setState('selectedStream', null);
  347. const {
  348. localStreams,
  349. remoteStreams
  350. } = this.state;
  351. localStreams.forEach(item => {
  352. this.setState('localStream-remove', item);
  353. });
  354. remoteStreams.forEach(item => {
  355. this.setState('remoteStream-remove', item);
  356. });
  357. this.setState('isJoinedRoom', false);
  358. }, (err) => {
  359. console.error('离开房间失败:', err);
  360. });
  361. },
  362. handleSelectStream: function (stream) {
  363. console.log('select stream: ', stream);
  364. this.setState('selectedStream', stream);
  365. }
  366. }
  367. App.init();
  368. }