index.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338
  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. // 用于维护应用内的状态
  19. const App = {
  20. state: {
  21. roomId: RoomId,
  22. userId: UserId,
  23. isJoinedRoom: false,
  24. selectedStream: null,
  25. localStreams: [],
  26. remoteStreams: []
  27. },
  28. client: null,
  29. setState: function (key, value) {
  30. const keys = Object.keys(this.state);
  31. if (keys.includes(key)) {
  32. this.state[key] = value;
  33. }
  34. switch (key) {
  35. case 'roomId':
  36. this.renderRoomId();
  37. break;
  38. case 'isJoinedRoom':
  39. this.renderRoomStatus();
  40. break;
  41. case 'selectedStream':
  42. this.renderSelectedStream();
  43. break;
  44. case 'localStream-add':
  45. this.loadStream(value, 'pusher');
  46. break;
  47. case 'localStream-remove':
  48. this.unloadStream(value, 'pusher');
  49. break;
  50. case 'remoteStream-add':
  51. this.loadStream(value, 'puller');
  52. break;
  53. case 'remoteStream-remove':
  54. this.unloadStream(value, 'puller');
  55. break;
  56. case 'remoteStream-update':
  57. this.rerenderStream(value);
  58. break;
  59. default:
  60. }
  61. },
  62. renderRoomId: function () {
  63. const { roomId } = this.state;
  64. const roomElem = document.querySelector('#roomId');
  65. roomElem.innerHTML = roomId;
  66. },
  67. renderRoomStatus: function () {
  68. const { isJoinedRoom } = this.state;
  69. const roomStatusElem = document.querySelector('#roomStatus');
  70. if (isJoinedRoom) {
  71. roomStatusElem.innerHTML = "已加入";
  72. } else {
  73. roomStatusElem.innerHTML = "未加入";
  74. }
  75. },
  76. renderSelectedStream: function () {
  77. const { selectedStream } = this.state;
  78. const selectedStreamElem = document.querySelector('#selectedStream');
  79. if (selectedStream) {
  80. selectedStreamElem.innerHTML = selectedStream.sid;
  81. } else {
  82. selectedStreamElem.innerHTML = "未选择";
  83. }
  84. },
  85. loadStream: function (stream, type) {
  86. let parent;
  87. let isPuller;
  88. switch (type) {
  89. case 'pusher':
  90. parent = document.querySelector('#pushers');
  91. break;
  92. case 'puller':
  93. parent = document.querySelector('#pullers');
  94. isPuller = true;
  95. break;
  96. default:
  97. return;
  98. }
  99. const player = document.createElement('div');
  100. player.className = 'media-player';
  101. player.id = stream.sid;
  102. const uIDElem = document.createElement('div');
  103. uIDElem.innerHTML = `用户ID:${stream.uid}`;
  104. uIDElem.style = 'overflow: hidden; text-overflow: ellipsis;';
  105. const sIDElem = document.createElement('div');
  106. sIDElem.innerHTML = `流ID:${stream.sid}`;
  107. sIDElem.style = 'overflow: hidden; text-overflow: ellipsis;';
  108. player.append(uIDElem);
  109. player.append(sIDElem);
  110. if (isPuller) {
  111. const pElem = document.createElement('p');
  112. pElem.innerHTML = 'unsubscribe';
  113. player.append(pElem);
  114. } else {
  115. const videoElem = document.createElement('video');
  116. videoElem.autoplay = true;
  117. videoElem.playsinline = true;
  118. videoElem.srcObject = stream.mediaStream;
  119. player.append(videoElem);
  120. }
  121. player.addEventListener('click', function() {
  122. this.handleSelectStream(stream);
  123. }.bind(this));
  124. parent.append(player);
  125. },
  126. unloadStream: function (stream, type) {
  127. let parent;
  128. switch (type) {
  129. case 'pusher':
  130. parent = document.querySelector('#pushers');
  131. break;
  132. case 'puller':
  133. parent = document.querySelector('#pullers');
  134. break;
  135. default:
  136. return;
  137. }
  138. const player = document.querySelector('#' + stream.sid);
  139. parent.removeChild(player);
  140. },
  141. rerenderStream: function (stream) {
  142. const player = document.querySelector('#' + stream.sid);
  143. if (stream.mediaStream) {
  144. const videoElem = document.createElement('video');
  145. videoElem.autoplay = true;
  146. videoElem.playsinline = true;
  147. videoElem.srcObject = stream.mediaStream;
  148. const pElem = player.querySelector('p');
  149. player.removeChild(pElem);
  150. player.appendChild(videoElem);
  151. } else {
  152. const videoElem = player.querySelector('video');
  153. const pElem = document.createElement('p');
  154. pElem.innerHTML = 'unsubscribe';
  155. player.removeChild(videoElem);
  156. player.appendChild(pElem);
  157. }
  158. },
  159. init: function () {
  160. const token = UCloudRTC.generateToken(AppId, AppKey, RoomId, UserId);
  161. this.client = new UCloudRTC.Client(AppId, token);
  162. // 监听 publish 成功的事件
  163. this.client.on('stream-published', (localStream) => {
  164. console.info('stream-published: ', localStream);
  165. const { localStreams } = this.state;
  166. localStreams.push(localStream);
  167. this.setState('localStream-add', localStream);
  168. });
  169. this.client.on('stream-added', (remoteStream) => {
  170. console.info('stream-added: ', remoteStream);
  171. const { remoteStreams } = this.state;
  172. remoteStreams.push(remoteStream);
  173. // 自动订阅
  174. this.client.subscribe(remoteStream.sid, (err) => {
  175. console.error('自动订阅失败:', err);
  176. });
  177. this.setState('remoteStream-add', remoteStream);
  178. });
  179. this.client.on('stream-subscribed', (remoteStream) => {
  180. console.info('stream-subscribed: ', remoteStream);
  181. const { remoteStreams } = this.state;
  182. const idx = remoteStreams.findIndex(item => item.sid === remoteStream.sid);
  183. if (idx >= 0) {
  184. remoteStreams.splice(idx, 1, remoteStream);
  185. }
  186. this.setState('remoteStream-update', remoteStream);
  187. });
  188. this.client.on('stream-removed', (remoteStream) => {
  189. console.info('stream-removed: ', remoteStream);
  190. const { remoteStreams } = this.state;
  191. const idx = remoteStreams.findIndex(item => item.sid === remoteStream.sid);
  192. if (idx >= 0) {
  193. const p = remoteStreams.splice(idx, 1)[0];
  194. this.setState('remoteStream-remove', p);
  195. }
  196. });
  197. document.querySelector('#joinRoomBtn').addEventListener('click', this.handleJoinRoom.bind(this));
  198. document.querySelector('#publishBtn').addEventListener('click', this.handlePublish.bind(this));
  199. document.querySelector('#publishScreenBtn').addEventListener('click', this.handlePublishScreen.bind(this));
  200. document.querySelector('#unPublishBtn').addEventListener('click', this.handleUnpublish.bind(this));
  201. document.querySelector('#subscribeBtn').addEventListener('click', this.handleSubscribe.bind(this));
  202. document.querySelector('#unSubscribeBtn').addEventListener('click', this.handleUnsubscribe.bind(this));
  203. document.querySelector('#leaveRoomBtn').addEventListener('click', this.handleLeaveRoom.bind(this));
  204. window.addEventListener('beforeunload', this.handleLeaveRoom.bind(this));
  205. },
  206. // 操作
  207. handleJoinRoom: function () {
  208. const {
  209. roomId,
  210. userId,
  211. isJoinedRoom
  212. } = this.state;
  213. if (isJoinedRoom) {
  214. alert('已经加入了房间');
  215. return;
  216. }
  217. if (!roomId) {
  218. alert('请先填写房间号');
  219. return;
  220. }
  221. this.client.joinRoom(roomId, userId, () => {
  222. console.info('加入房间成功');
  223. this.setState('isJoinedRoom', true);
  224. }, (err) => {
  225. console.error('加入房间失败: ', err);
  226. });
  227. },
  228. handlePublish: function () {
  229. this.client.publish(err => {
  230. console.error('发布失败:', err);
  231. });
  232. },
  233. handlePublishScreen: function () {
  234. this.client.publish({
  235. audio: true,
  236. video: false,
  237. screen: true
  238. }, err => {
  239. console.error('发布失败:', err);
  240. });
  241. },
  242. handleUnpublish: function () {
  243. const {
  244. selectedStream
  245. } = this.state;
  246. if (!selectedStream) {
  247. alert('未选择需要取消发布的本地流');
  248. return;
  249. }
  250. this.client.unpublish(selectedStream.sid, (stream) => {
  251. console.info('取消发布本地流成功:', stream);
  252. const {
  253. localStreams
  254. } = this.state;
  255. const idx = localStreams.findIndex(item => item.sid === stream.sid);
  256. if (idx >= 0) {
  257. const p = localStreams.splice(idx, 1)[0];
  258. this.setState('selectedStream', null);
  259. this.setState('localStream-remove', p);
  260. }
  261. }, (err) => {
  262. console.error('取消发布本地流失败:', err);
  263. })
  264. },
  265. handleSubscribe: function () {
  266. const {
  267. selectedStream
  268. } = this.state;
  269. if (!selectedStream) {
  270. alert('未选择需要订阅的远端流');
  271. return;
  272. }
  273. this.client.subscribe(selectedStream.sid, (err) => {
  274. console.error('订阅失败:', err);
  275. });
  276. },
  277. handleUnsubscribe: function () {
  278. const {
  279. selectedStream
  280. } = this.state;
  281. if (!selectedStream) {
  282. alert('未选择需要取消订阅的远端流');
  283. return;
  284. }
  285. this.client.unsubscribe(selectedStream.sid, (stream) => {
  286. console.info('取消订阅成功:', stream);
  287. const {
  288. remoteStreams
  289. } = this.state;
  290. const idx = remoteStreams.findIndex(item => item.sid === stream.sid);
  291. if (idx >= 0) {
  292. remoteStreams.splice(idx, 1, stream);
  293. this.setState('remoteStream-update', stream);
  294. }
  295. }, (err) => {
  296. console.error('订阅失败:', err);
  297. });
  298. },
  299. handleLeaveRoom: function () {
  300. const {
  301. isJoinedRoom
  302. } = this.state;
  303. if (!isJoinedRoom) {
  304. return;
  305. }
  306. this.client.leaveRoom(() => {
  307. console.info('离开房间成功');
  308. this.setState('selectedStream', null);
  309. const {
  310. localStreams,
  311. remoteStreams
  312. } = this.state;
  313. localStreams.forEach(item => {
  314. this.setState('localStream-remove', item);
  315. });
  316. remoteStreams.forEach(item => {
  317. this.setState('remoteStream-remove', item);
  318. });
  319. this.setState('isJoinedRoom', false);
  320. }, (err) => {
  321. console.error('离开房间失败:', err);
  322. });
  323. },
  324. handleSelectStream: function (stream) {
  325. console.log('select stream: ', stream);
  326. this.setState('selectedStream', stream);
  327. }
  328. }
  329. App.init();
  330. }