您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318
  1. var EventEmitter = require("events");
  2. var RTCBrowserType = require("./RTCBrowserType");
  3. var RTCUtils = require("./RTCUtils.js");
  4. var LocalStream = require("./LocalStream.js");
  5. var DataChannels = require("./DataChannels");
  6. var MediaStream = require("./MediaStream.js");
  7. var DesktopSharingEventTypes
  8. = require("../../service/desktopsharing/DesktopSharingEventTypes");
  9. var MediaStreamType = require("../../service/RTC/MediaStreamTypes");
  10. var StreamEventTypes = require("../../service/RTC/StreamEventTypes.js");
  11. var RTCEvents = require("../../service/RTC/RTCEvents.js");
  12. var XMPPEvents = require("../../service/xmpp/XMPPEvents");
  13. var UIEvents = require("../../service/UI/UIEvents");
  14. var eventEmitter = new EventEmitter();
  15. function getMediaStreamUsage()
  16. {
  17. var result = {
  18. audio: true,
  19. video: true
  20. };
  21. /** There are some issues with the desktop sharing
  22. * when this property is enabled.
  23. * WARNING: We must change the implementation to start video/audio if we
  24. * receive from the focus that the peer is not muted.
  25. var isSecureConnection = window.location.protocol == "https:";
  26. if(config.disableEarlyMediaPermissionRequests || !isSecureConnection)
  27. {
  28. result = {
  29. audio: false,
  30. video: false
  31. };
  32. }
  33. **/
  34. return result;
  35. }
  36. var RTC = {
  37. rtcUtils: null,
  38. devices: {
  39. audio: true,
  40. video: true
  41. },
  42. localStreams: [],
  43. remoteStreams: {},
  44. localAudio: null,
  45. localVideo: null,
  46. addStreamListener: function (listener, eventType) {
  47. eventEmitter.on(eventType, listener);
  48. },
  49. addListener: function (type, listener) {
  50. eventEmitter.on(type, listener);
  51. },
  52. removeStreamListener: function (listener, eventType) {
  53. if(!(eventType instanceof StreamEventTypes))
  54. throw "Illegal argument";
  55. eventEmitter.removeListener(eventType, listener);
  56. },
  57. createLocalStream: function (stream, type, change, videoType, isMuted, isGUMStream) {
  58. var localStream = new LocalStream(stream, type, eventEmitter, videoType, isGUMStream);
  59. //in firefox we have only one stream object
  60. if(this.localStreams.length == 0 ||
  61. this.localStreams[0].getOriginalStream() != stream)
  62. this.localStreams.push(localStream);
  63. if(isMuted === true)
  64. localStream.setMute(false);
  65. if(type == "audio")
  66. {
  67. this.localAudio = localStream;
  68. }
  69. else
  70. {
  71. this.localVideo = localStream;
  72. }
  73. var eventType = StreamEventTypes.EVENT_TYPE_LOCAL_CREATED;
  74. if(change)
  75. eventType = StreamEventTypes.EVENT_TYPE_LOCAL_CHANGED;
  76. eventEmitter.emit(eventType, localStream, isMuted);
  77. return localStream;
  78. },
  79. removeLocalStream: function (stream) {
  80. for(var i = 0; i < this.localStreams.length; i++)
  81. {
  82. if(this.localStreams[i].getOriginalStream() === stream) {
  83. delete this.localStreams[i];
  84. return;
  85. }
  86. }
  87. },
  88. createRemoteStream: function (data, sid, thessrc) {
  89. var remoteStream = new MediaStream(data, sid, thessrc,
  90. RTCBrowserType.getBrowserType(), eventEmitter);
  91. var jid = data.peerjid || APP.xmpp.myJid();
  92. if(!this.remoteStreams[jid]) {
  93. this.remoteStreams[jid] = {};
  94. }
  95. this.remoteStreams[jid][remoteStream.type]= remoteStream;
  96. eventEmitter.emit(StreamEventTypes.EVENT_TYPE_REMOTE_CREATED, remoteStream);
  97. return remoteStream;
  98. },
  99. getPCConstraints: function () {
  100. return this.rtcUtils.pc_constraints;
  101. },
  102. getUserMediaWithConstraints:function(um, success_callback,
  103. failure_callback, resolution,
  104. bandwidth, fps, desktopStream)
  105. {
  106. return this.rtcUtils.getUserMediaWithConstraints(um, success_callback,
  107. failure_callback, resolution, bandwidth, fps, desktopStream);
  108. },
  109. attachMediaStream: function (element, stream) {
  110. this.rtcUtils.attachMediaStream(element, stream);
  111. },
  112. getStreamID: function (stream) {
  113. return this.rtcUtils.getStreamID(stream);
  114. },
  115. getVideoSrc: function (element) {
  116. return this.rtcUtils.getVideoSrc(element);
  117. },
  118. setVideoSrc: function (element, src) {
  119. this.rtcUtils.setVideoSrc(element, src);
  120. },
  121. getVideoElementName: function () {
  122. return RTCBrowserType.isTemasysPluginUsed() ? 'object' : 'video';
  123. },
  124. dispose: function() {
  125. if (this.rtcUtils) {
  126. this.rtcUtils = null;
  127. }
  128. },
  129. stop: function () {
  130. this.dispose();
  131. },
  132. start: function () {
  133. var self = this;
  134. APP.desktopsharing.addListener(
  135. function (stream, isUsingScreenStream, callback) {
  136. self.changeLocalVideo(stream, isUsingScreenStream, callback);
  137. }, DesktopSharingEventTypes.NEW_STREAM_CREATED);
  138. APP.xmpp.addListener(XMPPEvents.STREAMS_CHANGED, function (jid, changedStreams) {
  139. for(var i = 0; i < changedStreams.length; i++) {
  140. var type = changedStreams[i].type;
  141. if (type != "audio") {
  142. var peerStreams = self.remoteStreams[jid];
  143. if(!peerStreams)
  144. continue;
  145. var videoStream = peerStreams[MediaStreamType.VIDEO_TYPE];
  146. if(!videoStream)
  147. continue;
  148. videoStream.setVideoType(changedStreams[i].type);
  149. }
  150. }
  151. });
  152. APP.xmpp.addListener(XMPPEvents.CALL_INCOMING, function(event) {
  153. DataChannels.init(event.peerconnection, eventEmitter);
  154. });
  155. APP.UI.addListener(UIEvents.SELECTED_ENDPOINT,
  156. DataChannels.handleSelectedEndpointEvent);
  157. APP.UI.addListener(UIEvents.PINNED_ENDPOINT,
  158. DataChannels.handlePinnedEndpointEvent);
  159. // In case of IE we continue from 'onReady' callback
  160. // passed to RTCUtils constructor. It will be invoked by Temasys plugin
  161. // once it is initialized.
  162. var onReady = function () {
  163. eventEmitter.emit(RTCEvents.RTC_READY, true);
  164. self.rtcUtils.obtainAudioAndVideoPermissions(
  165. null, null, getMediaStreamUsage());
  166. };
  167. this.rtcUtils = new RTCUtils(this, onReady);
  168. // Call onReady() if Temasys plugin is not used
  169. if (!RTCBrowserType.isTemasysPluginUsed()) {
  170. onReady();
  171. }
  172. },
  173. muteRemoteVideoStream: function (jid, value) {
  174. var stream;
  175. if(this.remoteStreams[jid] &&
  176. this.remoteStreams[jid][MediaStreamType.VIDEO_TYPE])
  177. {
  178. stream = this.remoteStreams[jid][MediaStreamType.VIDEO_TYPE];
  179. }
  180. if(!stream)
  181. return true;
  182. if (value != stream.muted) {
  183. stream.setMute(value);
  184. return true;
  185. }
  186. return false;
  187. },
  188. switchVideoStreams: function (new_stream) {
  189. this.localVideo.stream = new_stream;
  190. this.localStreams = [];
  191. //in firefox we have only one stream object
  192. if (this.localAudio.getOriginalStream() != new_stream)
  193. this.localStreams.push(this.localAudio);
  194. this.localStreams.push(this.localVideo);
  195. },
  196. changeLocalVideo: function (stream, isUsingScreenStream, callback) {
  197. var oldStream = this.localVideo.getOriginalStream();
  198. var type = (isUsingScreenStream? "screen" : "video");
  199. var localCallback = callback;
  200. if(this.localVideo.isMuted() && this.localVideo.videoType !== type)
  201. {
  202. localCallback = function() {
  203. APP.xmpp.setVideoMute(false, APP.UI.setVideoMuteButtonsState);
  204. callback();
  205. };
  206. }
  207. // FIXME: Workaround for FF/IE/Safari
  208. if (stream && stream.videoStream) {
  209. stream = stream.videoStream;
  210. }
  211. var videoStream = this.rtcUtils.createStream(stream, true);
  212. this.localVideo = this.createLocalStream(videoStream, "video", true, type);
  213. // Stop the stream to trigger onended event for old stream
  214. oldStream.stop();
  215. this.switchVideoStreams(videoStream, oldStream);
  216. APP.xmpp.switchStreams(videoStream, oldStream,localCallback);
  217. },
  218. changeLocalAudio: function (stream, callback) {
  219. var oldStream = this.localAudio.getOriginalStream();
  220. var newStream = this.rtcUtils.createStream(stream);
  221. this.localAudio = this.createLocalStream(newStream, "audio", true);
  222. // Stop the stream to trigger onended event for old stream
  223. oldStream.stop();
  224. APP.xmpp.switchStreams(newStream, oldStream, callback, true);
  225. },
  226. isVideoMuted: function (jid) {
  227. if (jid === APP.xmpp.myJid()) {
  228. var localVideo = APP.RTC.localVideo;
  229. return (!localVideo || localVideo.isMuted());
  230. }
  231. else
  232. {
  233. if (!APP.RTC.remoteStreams[jid] || !APP.RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE]) {
  234. return null;
  235. }
  236. return APP.RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE].muted;
  237. }
  238. },
  239. /**
  240. * Checks if video identified by given src is desktop stream.
  241. * @param videoSrc eg.
  242. * blob:https%3A//pawel.jitsi.net/9a46e0bd-131e-4d18-9c14-a9264e8db395
  243. * @returns {boolean}
  244. */
  245. isVideoSrcDesktop: function (jid) {
  246. if(!jid)
  247. return false;
  248. var isDesktop = false;
  249. var stream = null;
  250. if (APP.xmpp.myJid() === jid) {
  251. // local video
  252. stream = this.localVideo;
  253. } else {
  254. var peerStreams = this.remoteStreams[jid];
  255. if(!peerStreams)
  256. return false;
  257. stream = peerStreams[MediaStreamType.VIDEO_TYPE];
  258. }
  259. if(stream)
  260. isDesktop = (stream.videoType === "screen");
  261. return isDesktop;
  262. },
  263. setVideoMute: function(mute, callback, options) {
  264. if(!this.localVideo)
  265. return;
  266. if (mute == APP.RTC.localVideo.isMuted())
  267. {
  268. APP.xmpp.sendVideoInfoPresence(mute);
  269. if(callback)
  270. callback(mute);
  271. }
  272. else
  273. {
  274. APP.RTC.localVideo.setMute(!mute);
  275. APP.xmpp.setVideoMute(
  276. mute,
  277. callback,
  278. options);
  279. }
  280. },
  281. setDeviceAvailability: function (devices) {
  282. if(!devices)
  283. return;
  284. if(devices.audio === true || devices.audio === false)
  285. this.devices.audio = devices.audio;
  286. if(devices.video === true || devices.video === false)
  287. this.devices.video = devices.video;
  288. eventEmitter.emit(RTCEvents.AVAILABLE_DEVICES_CHANGED, this.devices);
  289. }
  290. };
  291. module.exports = RTC;