index.js 11 KB

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