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 9.7KB

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