index.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340
  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. this.renderRoomId();
  161. const token = UCloudRTC.generateToken(AppId, AppKey, RoomId, UserId);
  162. this.client = new UCloudRTC.Client(AppId, token);
  163. // 监听 publish 成功的事件
  164. this.client.on('stream-published', (localStream) => {
  165. console.info('stream-published: ', localStream);
  166. const { localStreams } = this.state;
  167. localStreams.push(localStream);
  168. this.setState('localStream-add', localStream);
  169. });
  170. this.client.on('stream-added', (remoteStream) => {
  171. console.info('stream-added: ', remoteStream);
  172. const { remoteStreams } = this.state;
  173. remoteStreams.push(remoteStream);
  174. // 自动订阅
  175. this.client.subscribe(remoteStream.sid, (err) => {
  176. console.error('自动订阅失败:', err);
  177. });
  178. this.setState('remoteStream-add', remoteStream);
  179. });
  180. this.client.on('stream-subscribed', (remoteStream) => {
  181. console.info('stream-subscribed: ', remoteStream);
  182. const { remoteStreams } = this.state;
  183. const idx = remoteStreams.findIndex(item => item.sid === remoteStream.sid);
  184. if (idx >= 0) {
  185. remoteStreams.splice(idx, 1, remoteStream);
  186. }
  187. this.setState('remoteStream-update', remoteStream);
  188. });
  189. this.client.on('stream-removed', (remoteStream) => {
  190. console.info('stream-removed: ', remoteStream);
  191. const { remoteStreams } = this.state;
  192. const idx = remoteStreams.findIndex(item => item.sid === remoteStream.sid);
  193. if (idx >= 0) {
  194. const p = remoteStreams.splice(idx, 1)[0];
  195. this.setState('remoteStream-remove', p);
  196. }
  197. });
  198. document.querySelector('#joinRoomBtn').addEventListener('click', this.handleJoinRoom.bind(this));
  199. document.querySelector('#publishBtn').addEventListener('click', this.handlePublish.bind(this));
  200. document.querySelector('#publishScreenBtn').addEventListener('click', this.handlePublishScreen.bind(this));
  201. document.querySelector('#unPublishBtn').addEventListener('click', this.handleUnpublish.bind(this));
  202. document.querySelector('#subscribeBtn').addEventListener('click', this.handleSubscribe.bind(this));
  203. document.querySelector('#unSubscribeBtn').addEventListener('click', this.handleUnsubscribe.bind(this));
  204. document.querySelector('#leaveRoomBtn').addEventListener('click', this.handleLeaveRoom.bind(this));
  205. window.addEventListener('beforeunload', this.handleLeaveRoom.bind(this));
  206. },
  207. // 操作
  208. handleJoinRoom: function () {
  209. const {
  210. roomId,
  211. userId,
  212. isJoinedRoom
  213. } = this.state;
  214. if (isJoinedRoom) {
  215. alert('已经加入了房间');
  216. return;
  217. }
  218. if (!roomId) {
  219. alert('请先填写房间号');
  220. return;
  221. }
  222. this.client.joinRoom(roomId, userId, () => {
  223. console.info('加入房间成功');
  224. this.setState('isJoinedRoom', true);
  225. }, (err) => {
  226. console.error('加入房间失败: ', err);
  227. });
  228. },
  229. handlePublish: function () {
  230. this.client.publish(err => {
  231. console.error('发布失败:', err);
  232. });
  233. },
  234. handlePublishScreen: function () {
  235. this.client.publish({
  236. audio: true,
  237. video: false,
  238. screen: true
  239. }, err => {
  240. console.error('发布失败:', err);
  241. });
  242. },
  243. handleUnpublish: function () {
  244. const {
  245. selectedStream
  246. } = this.state;
  247. if (!selectedStream) {
  248. alert('未选择需要取消发布的本地流');
  249. return;
  250. }
  251. this.client.unpublish(selectedStream.sid, (stream) => {
  252. console.info('取消发布本地流成功:', stream);
  253. const {
  254. localStreams
  255. } = this.state;
  256. const idx = localStreams.findIndex(item => item.sid === stream.sid);
  257. if (idx >= 0) {
  258. const p = localStreams.splice(idx, 1)[0];
  259. this.setState('selectedStream', null);
  260. this.setState('localStream-remove', p);
  261. }
  262. }, (err) => {
  263. console.error('取消发布本地流失败:', err);
  264. })
  265. },
  266. handleSubscribe: function () {
  267. const {
  268. selectedStream
  269. } = this.state;
  270. if (!selectedStream) {
  271. alert('未选择需要订阅的远端流');
  272. return;
  273. }
  274. this.client.subscribe(selectedStream.sid, (err) => {
  275. console.error('订阅失败:', err);
  276. });
  277. },
  278. handleUnsubscribe: function () {
  279. const {
  280. selectedStream
  281. } = this.state;
  282. if (!selectedStream) {
  283. alert('未选择需要取消订阅的远端流');
  284. return;
  285. }
  286. this.client.unsubscribe(selectedStream.sid, (stream) => {
  287. console.info('取消订阅成功:', stream);
  288. const {
  289. remoteStreams
  290. } = this.state;
  291. const idx = remoteStreams.findIndex(item => item.sid === stream.sid);
  292. if (idx >= 0) {
  293. remoteStreams.splice(idx, 1, stream);
  294. this.setState('remoteStream-update', stream);
  295. }
  296. }, (err) => {
  297. console.error('订阅失败:', err);
  298. });
  299. },
  300. handleLeaveRoom: function () {
  301. const {
  302. isJoinedRoom
  303. } = this.state;
  304. if (!isJoinedRoom) {
  305. return;
  306. }
  307. this.client.leaveRoom(() => {
  308. console.info('离开房间成功');
  309. this.setState('selectedStream', null);
  310. const {
  311. localStreams,
  312. remoteStreams
  313. } = this.state;
  314. localStreams.forEach(item => {
  315. this.setState('localStream-remove', item);
  316. });
  317. remoteStreams.forEach(item => {
  318. this.setState('remoteStream-remove', item);
  319. });
  320. this.setState('isJoinedRoom', false);
  321. }, (err) => {
  322. console.error('离开房间失败:', err);
  323. });
  324. },
  325. handleSelectStream: function (stream) {
  326. console.log('select stream: ', stream);
  327. this.setState('selectedStream', stream);
  328. }
  329. }
  330. App.init();
  331. }