You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

RTC.js 10KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304
  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(true);
  65. if(type == "audio") {
  66. this.localAudio = localStream;
  67. } else {
  68. this.localVideo = localStream;
  69. }
  70. var eventType = StreamEventTypes.EVENT_TYPE_LOCAL_CREATED;
  71. if(change)
  72. eventType = StreamEventTypes.EVENT_TYPE_LOCAL_CHANGED;
  73. eventEmitter.emit(eventType, localStream, isMuted);
  74. return localStream;
  75. },
  76. removeLocalStream: function (stream) {
  77. for(var i = 0; i < this.localStreams.length; i++)
  78. {
  79. if(this.localStreams[i].getOriginalStream() === stream) {
  80. delete this.localStreams[i];
  81. return;
  82. }
  83. }
  84. },
  85. createRemoteStream: function (data, sid, thessrc) {
  86. var remoteStream = new MediaStream(data, sid, thessrc,
  87. RTCBrowserType.getBrowserType(), eventEmitter);
  88. var jid = data.peerjid || APP.xmpp.myJid();
  89. if(!this.remoteStreams[jid]) {
  90. this.remoteStreams[jid] = {};
  91. }
  92. this.remoteStreams[jid][remoteStream.type]= remoteStream;
  93. eventEmitter.emit(StreamEventTypes.EVENT_TYPE_REMOTE_CREATED, remoteStream);
  94. return remoteStream;
  95. },
  96. getPCConstraints: function () {
  97. return this.rtcUtils.pc_constraints;
  98. },
  99. getUserMediaWithConstraints:function(um, success_callback,
  100. failure_callback, resolution,
  101. bandwidth, fps, desktopStream)
  102. {
  103. return this.rtcUtils.getUserMediaWithConstraints(um, success_callback,
  104. failure_callback, resolution, bandwidth, fps, desktopStream);
  105. },
  106. attachMediaStream: function (elSelector, stream) {
  107. this.rtcUtils.attachMediaStream(elSelector, stream);
  108. },
  109. getStreamID: function (stream) {
  110. return this.rtcUtils.getStreamID(stream);
  111. },
  112. getVideoSrc: function (element) {
  113. return this.rtcUtils.getVideoSrc(element);
  114. },
  115. setVideoSrc: function (element, src) {
  116. this.rtcUtils.setVideoSrc(element, src);
  117. },
  118. getVideoElementName: function () {
  119. return RTCBrowserType.isTemasysPluginUsed() ? 'object' : 'video';
  120. },
  121. dispose: function() {
  122. if (this.rtcUtils) {
  123. this.rtcUtils = null;
  124. }
  125. },
  126. stop: function () {
  127. this.dispose();
  128. },
  129. start: function () {
  130. var self = this;
  131. APP.desktopsharing.addListener(
  132. function (stream, isUsingScreenStream, callback) {
  133. self.changeLocalVideo(stream, isUsingScreenStream, callback);
  134. }, DesktopSharingEventTypes.NEW_STREAM_CREATED);
  135. APP.xmpp.addListener(XMPPEvents.CALL_INCOMING, function(event) {
  136. DataChannels.init(event.peerconnection, eventEmitter);
  137. });
  138. APP.UI.addListener(UIEvents.SELECTED_ENDPOINT,
  139. DataChannels.handleSelectedEndpointEvent);
  140. APP.UI.addListener(UIEvents.PINNED_ENDPOINT,
  141. DataChannels.handlePinnedEndpointEvent);
  142. // In case of IE we continue from 'onReady' callback
  143. // passed to RTCUtils constructor. It will be invoked by Temasys plugin
  144. // once it is initialized.
  145. var onReady = function () {
  146. eventEmitter.emit(RTCEvents.RTC_READY, true);
  147. self.rtcUtils.obtainAudioAndVideoPermissions(
  148. null, null, getMediaStreamUsage());
  149. };
  150. this.rtcUtils = new RTCUtils(this, onReady);
  151. // Call onReady() if Temasys plugin is not used
  152. if (!RTCBrowserType.isTemasysPluginUsed()) {
  153. onReady();
  154. }
  155. },
  156. muteRemoteVideoStream: function (jid, value) {
  157. var stream;
  158. if(this.remoteStreams[jid] &&
  159. this.remoteStreams[jid][MediaStreamType.VIDEO_TYPE])
  160. {
  161. stream = this.remoteStreams[jid][MediaStreamType.VIDEO_TYPE];
  162. }
  163. if(!stream)
  164. return true;
  165. if (value != stream.muted) {
  166. stream.setMute(value);
  167. return true;
  168. }
  169. return false;
  170. },
  171. switchVideoStreams: function (new_stream) {
  172. this.localVideo.stream = new_stream;
  173. this.localStreams = [];
  174. //in firefox we have only one stream object
  175. if (this.localAudio.getOriginalStream() != new_stream)
  176. this.localStreams.push(this.localAudio);
  177. this.localStreams.push(this.localVideo);
  178. },
  179. changeLocalVideo: function (stream, isUsingScreenStream, callback) {
  180. var oldStream = this.localVideo.getOriginalStream();
  181. var type = (isUsingScreenStream? "screen" : "video");
  182. var localCallback = callback;
  183. if(this.localVideo.isMuted() && this.localVideo.videoType !== type)
  184. {
  185. localCallback = function() {
  186. APP.xmpp.setVideoMute(false, function(mute) {
  187. eventEmitter.emit(RTCEvents.VIDEO_MUTE, mute);
  188. });
  189. callback();
  190. };
  191. }
  192. // FIXME: Workaround for FF/IE/Safari
  193. if (stream && stream.videoStream) {
  194. stream = stream.videoStream;
  195. }
  196. var videoStream = this.rtcUtils.createStream(stream, true);
  197. this.localVideo = this.createLocalStream(videoStream, "video", true, type);
  198. // Stop the stream to trigger onended event for old stream
  199. oldStream.stop();
  200. this.switchVideoStreams(videoStream, oldStream);
  201. APP.xmpp.switchStreams(videoStream, oldStream,localCallback);
  202. },
  203. changeLocalAudio: function (stream, callback) {
  204. var oldStream = this.localAudio.getOriginalStream();
  205. var newStream = this.rtcUtils.createStream(stream);
  206. this.localAudio = this.createLocalStream(newStream, "audio", true);
  207. // Stop the stream to trigger onended event for old stream
  208. oldStream.stop();
  209. APP.xmpp.switchStreams(newStream, oldStream, callback, true);
  210. },
  211. isVideoMuted: function (jid) {
  212. if (jid === APP.xmpp.myJid()) {
  213. var localVideo = APP.RTC.localVideo;
  214. return (!localVideo || localVideo.isMuted());
  215. }
  216. else
  217. {
  218. if (!APP.RTC.remoteStreams[jid] || !APP.RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE]) {
  219. return null;
  220. }
  221. return APP.RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE].muted;
  222. }
  223. },
  224. /**
  225. * Checks if video identified by given src is desktop stream.
  226. * @param videoSrc eg.
  227. * blob:https%3A//pawel.jitsi.net/9a46e0bd-131e-4d18-9c14-a9264e8db395
  228. * @returns {boolean}
  229. */
  230. isVideoSrcDesktop: function (jid) {
  231. if(!jid)
  232. return false;
  233. var isDesktop = false;
  234. var stream = null;
  235. if (APP.xmpp.myJid() === jid) {
  236. // local video
  237. stream = this.localVideo;
  238. } else {
  239. var peerStreams = this.remoteStreams[jid];
  240. if(!peerStreams)
  241. return false;
  242. stream = peerStreams[MediaStreamType.VIDEO_TYPE];
  243. }
  244. if(stream)
  245. isDesktop = (stream.videoType === "screen");
  246. return isDesktop;
  247. },
  248. setVideoMute: function (mute, callback, options) {
  249. if (!this.localVideo)
  250. return;
  251. if (mute == APP.RTC.localVideo.isMuted())
  252. {
  253. APP.xmpp.sendVideoInfoPresence(mute);
  254. if (callback)
  255. callback(mute);
  256. }
  257. else
  258. {
  259. APP.RTC.localVideo.setMute(mute);
  260. APP.xmpp.setVideoMute(
  261. mute,
  262. callback,
  263. options);
  264. }
  265. },
  266. setDeviceAvailability: function (devices) {
  267. if(!devices)
  268. return;
  269. if(devices.audio === true || devices.audio === false)
  270. this.devices.audio = devices.audio;
  271. if(devices.video === true || devices.video === false)
  272. this.devices.video = devices.video;
  273. eventEmitter.emit(RTCEvents.AVAILABLE_DEVICES_CHANGED, this.devices);
  274. }
  275. };
  276. module.exports = RTC;