2014-12-19 13:59:08 +00:00
|
|
|
var EventEmitter = require("events");
|
2015-07-10 09:57:20 +00:00
|
|
|
var RTCBrowserType = require("./RTCBrowserType");
|
2014-12-19 13:59:08 +00:00
|
|
|
var RTCUtils = require("./RTCUtils.js");
|
2014-12-22 09:03:21 +00:00
|
|
|
var LocalStream = require("./LocalStream.js");
|
|
|
|
var DataChannels = require("./DataChannels");
|
|
|
|
var MediaStream = require("./MediaStream.js");
|
2015-01-28 14:35:22 +00:00
|
|
|
var DesktopSharingEventTypes
|
|
|
|
= require("../../service/desktopsharing/DesktopSharingEventTypes");
|
|
|
|
var MediaStreamType = require("../../service/RTC/MediaStreamTypes");
|
|
|
|
var StreamEventTypes = require("../../service/RTC/StreamEventTypes.js");
|
2015-03-27 09:36:39 +00:00
|
|
|
var RTCEvents = require("../../service/RTC/RTCEvents.js");
|
2015-01-28 14:35:22 +00:00
|
|
|
var XMPPEvents = require("../../service/xmpp/XMPPEvents");
|
|
|
|
var UIEvents = require("../../service/UI/UIEvents");
|
2014-12-19 13:59:08 +00:00
|
|
|
|
|
|
|
var eventEmitter = new EventEmitter();
|
|
|
|
|
2015-04-22 09:31:08 +00:00
|
|
|
|
|
|
|
function getMediaStreamUsage()
|
|
|
|
{
|
|
|
|
var result = {
|
2015-05-19 15:03:01 +00:00
|
|
|
audio: true,
|
|
|
|
video: true
|
2015-04-22 09:31:08 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/** There are some issues with the desktop sharing
|
|
|
|
* when this property is enabled.
|
2015-05-19 15:03:01 +00:00
|
|
|
* WARNING: We must change the implementation to start video/audio if we
|
|
|
|
* receive from the focus that the peer is not muted.
|
2015-04-22 09:31:08 +00:00
|
|
|
|
|
|
|
var isSecureConnection = window.location.protocol == "https:";
|
|
|
|
|
|
|
|
if(config.disableEarlyMediaPermissionRequests || !isSecureConnection)
|
|
|
|
{
|
2015-05-19 15:03:01 +00:00
|
|
|
result = {
|
|
|
|
audio: false,
|
|
|
|
video: false
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
**/
|
2015-04-22 09:31:08 +00:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2014-12-19 13:59:08 +00:00
|
|
|
var RTC = {
|
|
|
|
rtcUtils: null,
|
2015-03-27 09:36:39 +00:00
|
|
|
devices: {
|
2015-04-22 09:31:08 +00:00
|
|
|
audio: true,
|
|
|
|
video: true
|
2015-03-27 09:36:39 +00:00
|
|
|
},
|
2014-12-19 13:59:08 +00:00
|
|
|
localStreams: [],
|
|
|
|
remoteStreams: {},
|
|
|
|
localAudio: null,
|
|
|
|
localVideo: null,
|
|
|
|
addStreamListener: function (listener, eventType) {
|
|
|
|
eventEmitter.on(eventType, listener);
|
|
|
|
},
|
2015-01-27 09:56:22 +00:00
|
|
|
addListener: function (type, listener) {
|
|
|
|
eventEmitter.on(type, listener);
|
|
|
|
},
|
2014-12-19 13:59:08 +00:00
|
|
|
removeStreamListener: function (listener, eventType) {
|
|
|
|
if(!(eventType instanceof StreamEventTypes))
|
|
|
|
throw "Illegal argument";
|
|
|
|
|
|
|
|
eventEmitter.removeListener(eventType, listener);
|
|
|
|
},
|
2015-04-22 09:31:08 +00:00
|
|
|
createLocalStream: function (stream, type, change, videoType, isMuted, isGUMStream) {
|
2014-12-22 09:03:21 +00:00
|
|
|
|
2015-04-22 09:31:08 +00:00
|
|
|
var localStream = new LocalStream(stream, type, eventEmitter, videoType, isGUMStream);
|
2015-01-12 13:23:29 +00:00
|
|
|
//in firefox we have only one stream object
|
|
|
|
if(this.localStreams.length == 0 ||
|
|
|
|
this.localStreams[0].getOriginalStream() != stream)
|
|
|
|
this.localStreams.push(localStream);
|
2015-04-22 09:31:08 +00:00
|
|
|
if(isMuted === true)
|
|
|
|
localStream.setMute(false);
|
2015-03-27 09:36:39 +00:00
|
|
|
|
2014-12-19 13:59:08 +00:00
|
|
|
if(type == "audio")
|
|
|
|
{
|
|
|
|
this.localAudio = localStream;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this.localVideo = localStream;
|
|
|
|
}
|
2015-01-13 13:11:05 +00:00
|
|
|
var eventType = StreamEventTypes.EVENT_TYPE_LOCAL_CREATED;
|
|
|
|
if(change)
|
|
|
|
eventType = StreamEventTypes.EVENT_TYPE_LOCAL_CHANGED;
|
|
|
|
|
2015-04-22 09:31:08 +00:00
|
|
|
eventEmitter.emit(eventType, localStream, isMuted);
|
2014-12-19 13:59:08 +00:00
|
|
|
return localStream;
|
|
|
|
},
|
|
|
|
removeLocalStream: function (stream) {
|
|
|
|
for(var i = 0; i < this.localStreams.length; i++)
|
|
|
|
{
|
|
|
|
if(this.localStreams[i].getOriginalStream() === stream) {
|
|
|
|
delete this.localStreams[i];
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
createRemoteStream: function (data, sid, thessrc) {
|
2015-01-19 16:54:41 +00:00
|
|
|
var remoteStream = new MediaStream(data, sid, thessrc,
|
2015-07-10 09:57:20 +00:00
|
|
|
RTCBrowserType.getBrowserType(), eventEmitter);
|
2015-01-28 14:35:22 +00:00
|
|
|
var jid = data.peerjid || APP.xmpp.myJid();
|
2014-12-19 13:59:08 +00:00
|
|
|
if(!this.remoteStreams[jid]) {
|
|
|
|
this.remoteStreams[jid] = {};
|
|
|
|
}
|
|
|
|
this.remoteStreams[jid][remoteStream.type]= remoteStream;
|
2015-01-19 16:54:41 +00:00
|
|
|
eventEmitter.emit(StreamEventTypes.EVENT_TYPE_REMOTE_CREATED, remoteStream);
|
2014-12-19 13:59:08 +00:00
|
|
|
return remoteStream;
|
|
|
|
},
|
|
|
|
getPCConstraints: function () {
|
|
|
|
return this.rtcUtils.pc_constraints;
|
|
|
|
},
|
|
|
|
getUserMediaWithConstraints:function(um, success_callback,
|
|
|
|
failure_callback, resolution,
|
|
|
|
bandwidth, fps, desktopStream)
|
|
|
|
{
|
|
|
|
return this.rtcUtils.getUserMediaWithConstraints(um, success_callback,
|
|
|
|
failure_callback, resolution, bandwidth, fps, desktopStream);
|
|
|
|
},
|
2015-07-15 08:45:51 +00:00
|
|
|
attachMediaStream: function (elSelector, stream) {
|
|
|
|
this.rtcUtils.attachMediaStream(elSelector, stream);
|
2014-12-19 13:59:08 +00:00
|
|
|
},
|
|
|
|
getStreamID: function (stream) {
|
|
|
|
return this.rtcUtils.getStreamID(stream);
|
|
|
|
},
|
|
|
|
getVideoSrc: function (element) {
|
|
|
|
return this.rtcUtils.getVideoSrc(element);
|
|
|
|
},
|
|
|
|
setVideoSrc: function (element, src) {
|
|
|
|
this.rtcUtils.setVideoSrc(element, src);
|
|
|
|
},
|
2015-07-10 09:57:20 +00:00
|
|
|
getVideoElementName: function () {
|
|
|
|
return RTCBrowserType.isTemasysPluginUsed() ? 'object' : 'video';
|
|
|
|
},
|
2014-12-19 13:59:08 +00:00
|
|
|
dispose: function() {
|
|
|
|
if (this.rtcUtils) {
|
|
|
|
this.rtcUtils = null;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
stop: function () {
|
|
|
|
this.dispose();
|
|
|
|
},
|
|
|
|
start: function () {
|
2015-01-13 13:11:05 +00:00
|
|
|
var self = this;
|
2015-01-28 14:35:22 +00:00
|
|
|
APP.desktopsharing.addListener(
|
2015-01-13 13:11:05 +00:00
|
|
|
function (stream, isUsingScreenStream, callback) {
|
|
|
|
self.changeLocalVideo(stream, isUsingScreenStream, callback);
|
|
|
|
}, DesktopSharingEventTypes.NEW_STREAM_CREATED);
|
2015-01-28 14:35:22 +00:00
|
|
|
APP.xmpp.addListener(XMPPEvents.CALL_INCOMING, function(event) {
|
2015-01-27 09:56:22 +00:00
|
|
|
DataChannels.init(event.peerconnection, eventEmitter);
|
2015-01-24 14:28:02 +00:00
|
|
|
});
|
2015-01-28 14:35:22 +00:00
|
|
|
APP.UI.addListener(UIEvents.SELECTED_ENDPOINT,
|
2015-01-27 12:03:26 +00:00
|
|
|
DataChannels.handleSelectedEndpointEvent);
|
2015-01-28 14:35:22 +00:00
|
|
|
APP.UI.addListener(UIEvents.PINNED_ENDPOINT,
|
2015-01-27 12:03:26 +00:00
|
|
|
DataChannels.handlePinnedEndpointEvent);
|
2015-07-10 09:57:20 +00:00
|
|
|
|
|
|
|
// In case of IE we continue from 'onReady' callback
|
|
|
|
// passed to RTCUtils constructor. It will be invoked by Temasys plugin
|
|
|
|
// once it is initialized.
|
|
|
|
var onReady = function () {
|
|
|
|
eventEmitter.emit(RTCEvents.RTC_READY, true);
|
|
|
|
self.rtcUtils.obtainAudioAndVideoPermissions(
|
|
|
|
null, null, getMediaStreamUsage());
|
|
|
|
};
|
|
|
|
|
|
|
|
this.rtcUtils = new RTCUtils(this, onReady);
|
|
|
|
|
|
|
|
// Call onReady() if Temasys plugin is not used
|
|
|
|
if (!RTCBrowserType.isTemasysPluginUsed()) {
|
|
|
|
onReady();
|
|
|
|
}
|
2014-12-19 13:59:08 +00:00
|
|
|
},
|
|
|
|
muteRemoteVideoStream: function (jid, value) {
|
|
|
|
var stream;
|
|
|
|
|
|
|
|
if(this.remoteStreams[jid] &&
|
|
|
|
this.remoteStreams[jid][MediaStreamType.VIDEO_TYPE])
|
|
|
|
{
|
|
|
|
stream = this.remoteStreams[jid][MediaStreamType.VIDEO_TYPE];
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!stream)
|
2015-03-30 14:19:35 +00:00
|
|
|
return true;
|
2014-12-19 13:59:08 +00:00
|
|
|
|
2015-01-13 09:33:45 +00:00
|
|
|
if (value != stream.muted) {
|
|
|
|
stream.setMute(value);
|
2014-12-19 13:59:08 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2015-01-12 13:23:29 +00:00
|
|
|
},
|
|
|
|
switchVideoStreams: function (new_stream) {
|
|
|
|
this.localVideo.stream = new_stream;
|
|
|
|
|
|
|
|
this.localStreams = [];
|
|
|
|
|
|
|
|
//in firefox we have only one stream object
|
2015-01-13 13:11:05 +00:00
|
|
|
if (this.localAudio.getOriginalStream() != new_stream)
|
2015-01-12 13:23:29 +00:00
|
|
|
this.localStreams.push(this.localAudio);
|
|
|
|
this.localStreams.push(this.localVideo);
|
2015-01-13 13:11:05 +00:00
|
|
|
},
|
|
|
|
changeLocalVideo: function (stream, isUsingScreenStream, callback) {
|
|
|
|
var oldStream = this.localVideo.getOriginalStream();
|
2015-01-20 15:56:00 +00:00
|
|
|
var type = (isUsingScreenStream? "screen" : "video");
|
2015-03-23 16:12:24 +00:00
|
|
|
var localCallback = callback;
|
|
|
|
if(this.localVideo.isMuted() && this.localVideo.videoType !== type)
|
|
|
|
{
|
|
|
|
localCallback = function() {
|
|
|
|
APP.xmpp.setVideoMute(false, APP.UI.setVideoMuteButtonsState);
|
|
|
|
callback();
|
|
|
|
};
|
|
|
|
}
|
2015-07-10 09:57:20 +00:00
|
|
|
// FIXME: Workaround for FF/IE/Safari
|
|
|
|
if (stream && stream.videoStream) {
|
|
|
|
stream = stream.videoStream;
|
|
|
|
}
|
2015-04-22 09:31:08 +00:00
|
|
|
var videoStream = this.rtcUtils.createStream(stream, true);
|
2015-03-24 15:43:33 +00:00
|
|
|
this.localVideo = this.createLocalStream(videoStream, "video", true, type);
|
2015-01-13 13:11:05 +00:00
|
|
|
// Stop the stream to trigger onended event for old stream
|
|
|
|
oldStream.stop();
|
2015-07-09 13:04:08 +00:00
|
|
|
|
|
|
|
this.switchVideoStreams(videoStream, oldStream);
|
|
|
|
|
2015-03-24 15:43:33 +00:00
|
|
|
APP.xmpp.switchStreams(videoStream, oldStream,localCallback);
|
2015-01-20 15:56:00 +00:00
|
|
|
},
|
2015-04-22 09:31:08 +00:00
|
|
|
changeLocalAudio: function (stream, callback) {
|
|
|
|
var oldStream = this.localAudio.getOriginalStream();
|
|
|
|
var newStream = this.rtcUtils.createStream(stream);
|
|
|
|
this.localAudio = this.createLocalStream(newStream, "audio", true);
|
|
|
|
// Stop the stream to trigger onended event for old stream
|
|
|
|
oldStream.stop();
|
|
|
|
APP.xmpp.switchStreams(newStream, oldStream, callback, true);
|
|
|
|
},
|
2015-06-29 14:24:21 +00:00
|
|
|
isVideoMuted: function (jid) {
|
|
|
|
if (jid === APP.xmpp.myJid()) {
|
|
|
|
var localVideo = APP.RTC.localVideo;
|
|
|
|
return (!localVideo || localVideo.isMuted());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!APP.RTC.remoteStreams[jid] || !APP.RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE]) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return APP.RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE].muted;
|
|
|
|
}
|
|
|
|
},
|
2015-01-20 15:56:00 +00:00
|
|
|
/**
|
|
|
|
* Checks if video identified by given src is desktop stream.
|
|
|
|
* @param videoSrc eg.
|
|
|
|
* blob:https%3A//pawel.jitsi.net/9a46e0bd-131e-4d18-9c14-a9264e8db395
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
isVideoSrcDesktop: function (jid) {
|
|
|
|
if(!jid)
|
|
|
|
return false;
|
|
|
|
var isDesktop = false;
|
|
|
|
var stream = null;
|
2015-03-17 16:46:08 +00:00
|
|
|
if (APP.xmpp.myJid() === jid) {
|
2015-01-20 15:56:00 +00:00
|
|
|
// local video
|
|
|
|
stream = this.localVideo;
|
|
|
|
} else {
|
|
|
|
var peerStreams = this.remoteStreams[jid];
|
|
|
|
if(!peerStreams)
|
|
|
|
return false;
|
|
|
|
stream = peerStreams[MediaStreamType.VIDEO_TYPE];
|
|
|
|
}
|
|
|
|
|
|
|
|
if(stream)
|
|
|
|
isDesktop = (stream.videoType === "screen");
|
2014-12-19 13:59:08 +00:00
|
|
|
|
2015-01-20 15:56:00 +00:00
|
|
|
return isDesktop;
|
2015-03-23 16:12:24 +00:00
|
|
|
},
|
|
|
|
setVideoMute: function(mute, callback, options) {
|
|
|
|
if(!this.localVideo)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (mute == APP.RTC.localVideo.isMuted())
|
|
|
|
{
|
|
|
|
APP.xmpp.sendVideoInfoPresence(mute);
|
|
|
|
if(callback)
|
2015-04-22 09:31:08 +00:00
|
|
|
callback(mute);
|
2015-03-23 16:12:24 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
APP.RTC.localVideo.setMute(!mute);
|
|
|
|
APP.xmpp.setVideoMute(
|
|
|
|
mute,
|
|
|
|
callback,
|
|
|
|
options);
|
|
|
|
}
|
2015-03-27 09:36:39 +00:00
|
|
|
},
|
|
|
|
setDeviceAvailability: function (devices) {
|
2015-03-27 13:56:17 +00:00
|
|
|
if(!devices)
|
|
|
|
return;
|
|
|
|
if(devices.audio === true || devices.audio === false)
|
|
|
|
this.devices.audio = devices.audio;
|
|
|
|
if(devices.video === true || devices.video === false)
|
|
|
|
this.devices.video = devices.video;
|
2015-03-27 09:36:39 +00:00
|
|
|
eventEmitter.emit(RTCEvents.AVAILABLE_DEVICES_CHANGED, this.devices);
|
2015-01-20 15:56:00 +00:00
|
|
|
}
|
2014-12-19 13:59:08 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = RTC;
|