Handles audio and video streams separately. Minor fixes.
This commit is contained in:
parent
8b96d134e3
commit
9fd03e1bef
131
app.js
131
app.js
|
@ -49,11 +49,10 @@ function init() {
|
|||
if (config.useStunTurn) {
|
||||
connection.jingle.getStunAndTurnCredentials();
|
||||
}
|
||||
if (RTC.browser === 'firefox') {
|
||||
getUserMediaWithConstraints(['audio']);
|
||||
} else {
|
||||
getUserMediaWithConstraints(['audio', 'video'], config.resolution || '360');
|
||||
}
|
||||
getUserMediaWithConstraints( ['audio'], audioStreamReady,
|
||||
function(error){
|
||||
console.error('failed to obtain audio stream - stop', error);
|
||||
});
|
||||
document.getElementById('connect').disabled = true;
|
||||
} else {
|
||||
console.log('status', status);
|
||||
|
@ -61,6 +60,31 @@ function init() {
|
|||
});
|
||||
}
|
||||
|
||||
function audioStreamReady(stream) {
|
||||
|
||||
change_local_audio(stream);
|
||||
|
||||
if(RTC.browser !== 'firefox') {
|
||||
getUserMediaWithConstraints( ['video'], videoStreamReady, videoStreamFailed, config.resolution || '360' );
|
||||
} else {
|
||||
doJoin();
|
||||
}
|
||||
}
|
||||
|
||||
function videoStreamReady(stream) {
|
||||
|
||||
change_local_video(stream);
|
||||
|
||||
doJoin();
|
||||
}
|
||||
|
||||
function videoStreamFailed(error) {
|
||||
|
||||
console.warn("Failed to obtain video stream - continue anyway", error);
|
||||
|
||||
doJoin();
|
||||
}
|
||||
|
||||
function doJoin() {
|
||||
var roomnode = null;
|
||||
var path = window.location.pathname;
|
||||
|
@ -104,8 +128,17 @@ function doJoin() {
|
|||
connection.emuc.doJoin(roomjid);
|
||||
}
|
||||
|
||||
$(document).bind('mediaready.jingle', function (event, stream) {
|
||||
connection.jingle.localStream = stream;
|
||||
function change_local_audio(stream) {
|
||||
|
||||
connection.jingle.localAudio = stream;
|
||||
RTC.attachMediaStream($('#localAudio'), stream);
|
||||
document.getElementById('localAudio').autoplay = true;
|
||||
document.getElementById('localAudio').volume = 0;
|
||||
}
|
||||
|
||||
function change_local_video(stream) {
|
||||
|
||||
connection.jingle.localVideo = stream;
|
||||
RTC.attachMediaStream($('#localVideo'), stream);
|
||||
document.getElementById('localVideo').autoplay = true;
|
||||
document.getElementById('localVideo').volume = 0;
|
||||
|
@ -124,16 +157,14 @@ $(document).bind('mediaready.jingle', function (event, stream) {
|
|||
}
|
||||
});
|
||||
});
|
||||
|
||||
doJoin();
|
||||
});
|
||||
|
||||
$(document).bind('mediafailure.jingle', function () {
|
||||
// FIXME
|
||||
});
|
||||
}
|
||||
|
||||
$(document).bind('remotestreamadded.jingle', function (event, data, sid) {
|
||||
function waitForRemoteVideo(selector, sid) {
|
||||
if(selector.removed) {
|
||||
console.warn("media removed before had started", selector);
|
||||
return;
|
||||
}
|
||||
var sess = connection.jingle.sessions[sid];
|
||||
if (data.stream.id === 'mixedmslabel') return;
|
||||
videoTracks = data.stream.getVideoTracks();
|
||||
|
@ -188,8 +219,11 @@ $(document).bind('remotestreamadded.jingle', function (event, data, sid) {
|
|||
remotes.appendChild(container);
|
||||
Util.playSoundNotification('userJoined');
|
||||
}
|
||||
var vid = document.createElement('video');
|
||||
var id = 'remoteVideo_' + sid + '_' + data.stream.id;
|
||||
|
||||
var isVideo = data.stream.getVideoTracks().length > 0;
|
||||
var vid = isVideo ? document.createElement('video') : document.createElement('audio');
|
||||
var id = (isVideo ? 'remoteVideo_' : 'remoteAudio_') + sid + '_' + data.stream.id;
|
||||
|
||||
vid.id = id;
|
||||
vid.autoplay = true;
|
||||
vid.oncontextmenu = function () { return false; };
|
||||
|
@ -203,11 +237,14 @@ $(document).bind('remotestreamadded.jingle', function (event, data, sid) {
|
|||
var sel = $('#' + id);
|
||||
sel.hide();
|
||||
RTC.attachMediaStream(sel, data.stream);
|
||||
|
||||
if(isVideo) {
|
||||
waitForRemoteVideo(sel, sid);
|
||||
}
|
||||
|
||||
data.stream.onended = function () {
|
||||
console.log('stream ended', this.id);
|
||||
var src = $('#' + id).attr('src');
|
||||
if (src === $('#largeVideo').attr('src')) {
|
||||
if (sel.attr('src') === $('#largeVideo').attr('src')) {
|
||||
// this is currently displayed as large
|
||||
// pick the last visible video in the row
|
||||
// if nobody else is left, this picks the local video
|
||||
|
@ -221,9 +258,21 @@ $(document).bind('remotestreamadded.jingle', function (event, data, sid) {
|
|||
updateLargeVideo(pick.src, isLocalVideo, pick.volume);
|
||||
}
|
||||
}
|
||||
$('#' + id).parent().remove();
|
||||
// Mark video as removed to cancel waiting loop(if video is removed before has started)
|
||||
sel.removed = true;
|
||||
|
||||
var userContainer = sel.parent();
|
||||
if(userContainer.children().length === 0) {
|
||||
console.log("Remove whole user");
|
||||
// Remove whole container
|
||||
userContainer.remove();
|
||||
Util.playSoundNotification('userLeft');
|
||||
resizeThumbnails();
|
||||
} else {
|
||||
// Remove only stream holder
|
||||
sel.remove();
|
||||
console.log("Remove stream only", sel);
|
||||
}
|
||||
};
|
||||
sel.click(
|
||||
function () {
|
||||
|
@ -232,9 +281,10 @@ $(document).bind('remotestreamadded.jingle', function (event, data, sid) {
|
|||
}
|
||||
);
|
||||
// an attempt to work around https://github.com/jitsi/jitmeet/issues/32
|
||||
if (data.peerjid && sess.peerjid === data.peerjid &&
|
||||
if (isVideo
|
||||
&& data.peerjid && sess.peerjid === data.peerjid &&
|
||||
data.stream.getVideoTracks().length === 0 &&
|
||||
connection.jingle.localStream.getVideoTracks().length > 0) {
|
||||
connection.jingle.localVideo.getVideoTracks().length > 0) {
|
||||
window.setTimeout(function() {
|
||||
sendKeyframe(sess.peerconnection);
|
||||
}, 3000);
|
||||
|
@ -379,7 +429,12 @@ $(document).bind('setLocalDescription.jingle', function (event, sid) {
|
|||
var ssrc = SDPUtil.find_line(media, 'a=ssrc:').substring(7).split(' ')[0];
|
||||
newssrcs[type] = ssrc;
|
||||
|
||||
directions[type] = (SDPUtil.find_line(media, 'a=sendrecv') || SDPUtil.find_line(media, 'a=recvonly') || SDPUtil.find_line('a=sendonly') || SDPUtil.find_line('a=inactive') || 'a=sendrecv').substr(2);
|
||||
directions[type] = (
|
||||
SDPUtil.find_line(media, 'a=sendrecv')
|
||||
|| SDPUtil.find_line(media, 'a=recvonly')
|
||||
|| SDPUtil.find_line('a=sendonly')
|
||||
|| SDPUtil.find_line('a=inactive')
|
||||
|| 'a=sendrecv' ).substr(2);
|
||||
}
|
||||
});
|
||||
console.log('new ssrcs', newssrcs);
|
||||
|
@ -422,7 +477,7 @@ $(document).bind('joined.muc', function (event, jid, info) {
|
|||
|
||||
$(document).bind('entered.muc', function (event, jid, info, pres) {
|
||||
console.log('entered', jid, info);
|
||||
console.log(focus);
|
||||
console.log('is focus?' + focus ? 'true' : 'false');
|
||||
|
||||
var videoSpanId = 'participant_' + Strophe.getResourceFromJid(jid);
|
||||
var container = addRemoteVideoContainer(videoSpanId);
|
||||
|
@ -712,17 +767,22 @@ function updateLargeVideo(newSrc, localVideo, vol) {
|
|||
}
|
||||
}
|
||||
|
||||
function getConferenceHandler() {
|
||||
return focus ? focus : activecall;
|
||||
}
|
||||
|
||||
function toggleVideo() {
|
||||
if (!(connection && connection.jingle.localStream)) return;
|
||||
if (!(connection && connection.jingle.localVideo)) return;
|
||||
var ismuted = false;
|
||||
for (var idx = 0; idx < connection.jingle.localStream.getVideoTracks().length; idx++) {
|
||||
ismuted = !connection.jingle.localStream.getVideoTracks()[idx].enabled;
|
||||
var localVideo = connection.jingle.localVideo;
|
||||
for (var idx = 0; idx < localVideo.getVideoTracks().length; idx++) {
|
||||
ismuted = !localVideo.getVideoTracks()[idx].enabled;
|
||||
}
|
||||
for (var idx = 0; idx < connection.jingle.localStream.getVideoTracks().length; idx++) {
|
||||
connection.jingle.localStream.getVideoTracks()[idx].enabled = !connection.jingle.localStream.getVideoTracks()[idx].enabled;
|
||||
for (var idx = 0; idx < localVideo.getVideoTracks().length; idx++) {
|
||||
localVideo.getVideoTracks()[idx].enabled = !localVideo.getVideoTracks()[idx].enabled;
|
||||
}
|
||||
var sess = focus || activecall;
|
||||
if (!sess) {
|
||||
var sess = getConferenceHandler();
|
||||
if (sess) {
|
||||
return;
|
||||
}
|
||||
sess.peerconnection.pendingop = ismuted ? 'unmute' : 'mute';
|
||||
|
@ -730,9 +790,10 @@ function toggleVideo() {
|
|||
}
|
||||
|
||||
function toggleAudio() {
|
||||
if (!(connection && connection.jingle.localStream)) return;
|
||||
for (var idx = 0; idx < connection.jingle.localStream.getAudioTracks().length; idx++) {
|
||||
connection.jingle.localStream.getAudioTracks()[idx].enabled = !connection.jingle.localStream.getAudioTracks()[idx].enabled;
|
||||
if (!(connection && connection.jingle.localAudio)) return;
|
||||
var localAudio = connection.jingle.localAudio;
|
||||
for (var idx = 0; idx < localAudio.getAudioTracks().length; idx++) {
|
||||
localAudio.getAudioTracks()[idx].enabled = !localAudio.getAudioTracks()[idx].enabled;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -1170,6 +1231,10 @@ function showFocusIndicator() {
|
|||
var session = connection.jingle.sessions[Object.keys(connection.jingle.sessions)[0]];
|
||||
var focusId = 'participant_' + Strophe.getResourceFromJid(session.peerjid);
|
||||
var focusContainer = document.getElementById(focusId);
|
||||
if(!focusContainer) {
|
||||
console.error("No focus container!");
|
||||
return;
|
||||
}
|
||||
var indicatorSpan = $('#' + focusId + ' .focusindicator');
|
||||
|
||||
if (!indicatorSpan || indicatorSpan.length === 0) {
|
||||
|
|
|
@ -79,6 +79,7 @@
|
|||
<span id="localVideoContainer" class="videocontainer">
|
||||
<span id="localNick"></span>
|
||||
<video id="localVideo" autoplay oncontextmenu="return false;" muted></video>
|
||||
<audio id="localAudio" autoplay oncontextmenu="return false;" muted></audio>
|
||||
<span class="focusindicator"></span>
|
||||
</span>
|
||||
<audio id="userJoined" src="sounds/joined.wav" preload="auto"></audio>
|
||||
|
@ -99,6 +100,7 @@
|
|||
<textarea id="usermsg" placeholder='Enter text...' autofocus></textarea>
|
||||
</div>
|
||||
<a id="downloadlog" onclick='dump(event.target);'><i title="Download support information" class="fa fa-cloud-download"></i></a>
|
||||
<!-- Google Analytics -->
|
||||
<script>
|
||||
(function(i,s,o,g,r,a,m){i['GoogleAnalyticsObject']=r;i[r]=i[r]||function(){
|
||||
(i[r].q=i[r].q||[]).push(arguments)},i[r].l=1*new Date();a=s.createElement(o),m=s.getElementsByTagName(o)[0];a.async=1;a.src=g;m.parentNode.insertBefore(a,m)
|
||||
|
@ -106,5 +108,6 @@
|
|||
ga('create', 'UA-319188-14', 'jit.si');
|
||||
ga('send', 'pageview');
|
||||
</script>
|
||||
<!-- End Google Analytics -->
|
||||
</body>
|
||||
</html>
|
||||
|
|
|
@ -77,7 +77,12 @@ ColibriFocus.prototype.makeConference = function (peers) {
|
|||
self.channels.push([]);
|
||||
});
|
||||
|
||||
this.peerconnection.addStream(this.connection.jingle.localStream);
|
||||
if(connection.jingle.localAudio) {
|
||||
this.peerconnection.addStream(connection.jingle.localAudio);
|
||||
}
|
||||
if(connection.jingle.localVideo) {
|
||||
this.peerconnection.addStream(connection.jingle.localVideo);
|
||||
}
|
||||
this.peerconnection.oniceconnectionstatechange = function (event) {
|
||||
console.warn('ice connection state changed to', self.peerconnection.iceConnectionState);
|
||||
/*
|
||||
|
@ -146,7 +151,6 @@ ColibriFocus.prototype._makeConference = function () {
|
|||
var elem = $iq({to: this.bridgejid, type: 'get'});
|
||||
elem.c('conference', {xmlns: 'http://jitsi.org/protocol/colibri'});
|
||||
|
||||
var stream = this.connection.jingle.localStream;
|
||||
this.media.forEach(function (name) {
|
||||
elem.c('content', {name: name});
|
||||
elem.c('channel', {initiator: 'true', expire: '15'}).up();
|
||||
|
@ -443,7 +447,8 @@ ColibriFocus.prototype.initiate = function (peer, isInitiator) {
|
|||
this.connection);
|
||||
sess.initiate(peer);
|
||||
sess.colibri = this;
|
||||
sess.localStream = this.connection.jingle.localStream;
|
||||
// We do not announce our audio per conference peer, so only video is set here
|
||||
sess.localVideo = this.connection.jingle.localVideo;
|
||||
sess.media_constraints = this.connection.jingle.media_constraints;
|
||||
sess.pc_constraints = this.connection.jingle.pc_constraints;
|
||||
sess.ice_config = this.connection.jingle.ice_config;
|
||||
|
@ -610,7 +615,7 @@ ColibriFocus.prototype.sendSSRCUpdate = function (sdp, jid, isadd) {
|
|||
console.warn('got modify result');
|
||||
},
|
||||
function (err) {
|
||||
console.warn('got modify error');
|
||||
console.warn('got modify error', err);
|
||||
}
|
||||
);
|
||||
} else {
|
||||
|
@ -697,7 +702,7 @@ ColibriFocus.prototype.addIceCandidate = function (session, elem) {
|
|||
console.log('got result');
|
||||
},
|
||||
function (err) {
|
||||
console.warn('got error');
|
||||
console.error('got error', err);
|
||||
}
|
||||
);
|
||||
};
|
||||
|
@ -748,7 +753,7 @@ ColibriFocus.prototype.sendIceCandidates = function (candidates) {
|
|||
console.log('got result');
|
||||
},
|
||||
function (err) {
|
||||
console.warn('got error');
|
||||
console.error('got error', err);
|
||||
}
|
||||
);
|
||||
};
|
||||
|
@ -779,7 +784,7 @@ ColibriFocus.prototype.terminate = function (session, reason) {
|
|||
console.log('got result');
|
||||
},
|
||||
function (err) {
|
||||
console.log('got error');
|
||||
console.error('got error', err);
|
||||
}
|
||||
);
|
||||
// and remove from channels
|
||||
|
@ -811,7 +816,7 @@ ColibriFocus.prototype.hardMuteVideo = function (muted) {
|
|||
|
||||
this.peerconnection.hardMuteVideo(muted);
|
||||
|
||||
this.connection.jingle.localStream.getVideoTracks().forEach(function (track) {
|
||||
this.connection.jingle.localVideo.getVideoTracks().forEach(function (track) {
|
||||
track.enabled = !muted;
|
||||
});
|
||||
};
|
||||
|
|
|
@ -220,7 +220,6 @@ TraceablePeerConnection.prototype.addSource = function (elem) {
|
|||
});
|
||||
sdp.raw = sdp.session + sdp.media.join('');
|
||||
});
|
||||
this.modifySources();
|
||||
};
|
||||
|
||||
TraceablePeerConnection.prototype.enqueueRemoveSsrc = function(channel, ssrcLines) {
|
||||
|
@ -257,7 +256,6 @@ TraceablePeerConnection.prototype.removeSource = function (elem) {
|
|||
});
|
||||
sdp.raw = sdp.session + sdp.media.join('');
|
||||
});
|
||||
this.modifySources();
|
||||
};
|
||||
|
||||
TraceablePeerConnection.prototype.modifySources = function(successCallback) {
|
||||
|
@ -275,11 +273,11 @@ TraceablePeerConnection.prototype.modifySources = function(successCallback) {
|
|||
if (!(this.signalingState == 'stable' && this.iceConnectionState == 'connected')) {
|
||||
console.warn('modifySources not yet', this.signalingState, this.iceConnectionState);
|
||||
this.wait = true;
|
||||
window.setTimeout(function() { self.modifySources(); }, 250);
|
||||
window.setTimeout(function() { self.modifySources(successCallback); }, 250);
|
||||
return;
|
||||
}
|
||||
if (this.wait) {
|
||||
window.setTimeout(function() { self.modifySources(); }, 2500);
|
||||
window.setTimeout(function() { self.modifySources(successCallback); }, 2500);
|
||||
this.wait = false;
|
||||
return;
|
||||
}
|
||||
|
@ -325,6 +323,12 @@ TraceablePeerConnection.prototype.modifySources = function(successCallback) {
|
|||
self.pendingop = null;
|
||||
}
|
||||
|
||||
// FIXME: pushing down an answer while ice connection state
|
||||
// is still checking is bad...
|
||||
//console.log(self.peerconnection.iceConnectionState);
|
||||
|
||||
// trying to work around another chrome bug
|
||||
//modifiedAnswer.sdp = modifiedAnswer.sdp.replace(/a=setup:active/g, 'a=setup:actpass');
|
||||
self.setLocalDescription(modifiedAnswer,
|
||||
function() {
|
||||
//console.log('modified setLocalDescription ok');
|
||||
|
@ -471,7 +475,7 @@ function setupRTC() {
|
|||
return RTC;
|
||||
}
|
||||
|
||||
function getUserMediaWithConstraints(um, resolution, bandwidth, fps) {
|
||||
function getUserMediaWithConstraints(um, success_callback, failure_callback, resolution, bandwidth, fps) {
|
||||
var constraints = {audio: false, video: false};
|
||||
|
||||
if (um.indexOf('video') >= 0) {
|
||||
|
@ -553,14 +557,18 @@ function getUserMediaWithConstraints(um, resolution, bandwidth, fps) {
|
|||
RTC.getUserMedia(constraints,
|
||||
function (stream) {
|
||||
console.log('onUserMediaSuccess');
|
||||
$(document).trigger('mediaready.jingle', [stream]);
|
||||
success_callback(stream);
|
||||
},
|
||||
function (error) {
|
||||
console.warn('Failed to get access to local media. Error ', error);
|
||||
$(document).trigger('mediafailure.jingle');
|
||||
if(failure_callback) {
|
||||
failure_callback(error);
|
||||
}
|
||||
});
|
||||
} catch (e) {
|
||||
console.error('GUM failed: ', e);
|
||||
$(document).trigger('mediafailure.jingle');
|
||||
if(failure_callback) {
|
||||
failure_callback(e);
|
||||
}
|
||||
}
|
||||
}
|
|
@ -12,7 +12,8 @@ Strophe.addConnectionPlugin('jingle', {
|
|||
}
|
||||
// MozDontOfferDataChannel: true when this is firefox
|
||||
},
|
||||
localStream: null,
|
||||
localAudio: null,
|
||||
localVideo: null,
|
||||
|
||||
init: function (conn) {
|
||||
this.connection = conn;
|
||||
|
@ -38,12 +39,13 @@ Strophe.addConnectionPlugin('jingle', {
|
|||
onJingle: function (iq) {
|
||||
var sid = $(iq).find('jingle').attr('sid');
|
||||
var action = $(iq).find('jingle').attr('action');
|
||||
var fromJid = iq.getAttribute('from');
|
||||
// send ack first
|
||||
var ack = $iq({type: 'result',
|
||||
to: iq.getAttribute('from'),
|
||||
to: fromJid,
|
||||
id: iq.getAttribute('id')
|
||||
});
|
||||
console.log('on jingle ' + action);
|
||||
console.log('on jingle ' + action + ' from ' + fromJid, iq);
|
||||
var sess = this.sessions[sid];
|
||||
if ('session-initiate' != action) {
|
||||
if (sess === null) {
|
||||
|
@ -56,8 +58,8 @@ Strophe.addConnectionPlugin('jingle', {
|
|||
}
|
||||
// compare from to sess.peerjid (bare jid comparison for later compat with message-mode)
|
||||
// local jid is not checked
|
||||
if (Strophe.getBareJidFromJid(iq.getAttribute('from')) != Strophe.getBareJidFromJid(sess.peerjid)) {
|
||||
console.warn('jid mismatch for session id', sid, iq.getAttribute('from'), sess.peerjid);
|
||||
if (Strophe.getBareJidFromJid(fromJid) != Strophe.getBareJidFromJid(sess.peerjid)) {
|
||||
console.warn('jid mismatch for session id', sid, fromJid, sess.peerjid);
|
||||
ack.type = 'error';
|
||||
ack.c('error', {type: 'cancel'})
|
||||
.c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()
|
||||
|
@ -82,14 +84,17 @@ Strophe.addConnectionPlugin('jingle', {
|
|||
case 'session-initiate':
|
||||
sess = new JingleSession($(iq).attr('to'), $(iq).find('jingle').attr('sid'), this.connection);
|
||||
// configure session
|
||||
if (this.localStream) {
|
||||
sess.localStreams.push(this.localStream);
|
||||
if (this.localAudio) {
|
||||
sess.localStreams.push(this.localAudio);
|
||||
}
|
||||
if (this.localVideo) {
|
||||
sess.localStreams.push(this.localVideo);
|
||||
}
|
||||
sess.media_constraints = this.media_constraints;
|
||||
sess.pc_constraints = this.pc_constraints;
|
||||
sess.ice_config = this.ice_config;
|
||||
|
||||
sess.initiate($(iq).attr('from'), false);
|
||||
sess.initiate(fromJid, false);
|
||||
// FIXME: setRemoteDescription should only be done when this call is to be accepted
|
||||
sess.setRemoteDescription($(iq).find('>jingle'), 'offer');
|
||||
|
||||
|
@ -152,8 +157,11 @@ Strophe.addConnectionPlugin('jingle', {
|
|||
Math.random().toString(36).substr(2, 12), // random string
|
||||
this.connection);
|
||||
// configure session
|
||||
if (this.localStream) {
|
||||
sess.localStreams.push(this.localStream);
|
||||
if (this.localAudio) {
|
||||
sess.localStreams.push(this.localAudio);
|
||||
}
|
||||
if (this.localVideo) {
|
||||
sess.localStreams.push(this.localVideo);
|
||||
}
|
||||
sess.media_constraints = this.media_constraints;
|
||||
sess.pc_constraints = this.pc_constraints;
|
||||
|
|
|
@ -76,6 +76,7 @@ JingleSession.prototype.initiate = function (peerjid, isInitiator) {
|
|||
this.peerconnection.onremovestream = function (event) {
|
||||
self.remoteStream = null;
|
||||
// FIXME: remove from this.remoteStreams
|
||||
// FIXME: remotestreamremoved.jingle not defined anywhere(unused)
|
||||
$(document).trigger('remotestreamremoved.jingle', [event, self.sid]);
|
||||
};
|
||||
this.peerconnection.onsignalingstatechange = function (event) {
|
||||
|
@ -635,11 +636,15 @@ JingleSession.prototype.sendTerminate = function (reason, text) {
|
|||
JingleSession.prototype.addSource = function (elem) {
|
||||
|
||||
this.peerconnection.addSource(elem);
|
||||
|
||||
this.modifySources();
|
||||
};
|
||||
|
||||
JingleSession.prototype.removeSource = function (elem) {
|
||||
|
||||
this.peerconnection.removeSource(elem);
|
||||
|
||||
this.modifySources();
|
||||
};
|
||||
|
||||
JingleSession.prototype.modifySources = function() {
|
||||
|
@ -655,7 +660,7 @@ JingleSession.prototype.hardMuteVideo = function (muted) {
|
|||
|
||||
this.peerconnection.hardMuteVideo(muted);
|
||||
|
||||
this.connection.jingle.localStream.getVideoTracks().forEach(function (track) {
|
||||
this.connection.jingle.localVideo.getVideoTracks().forEach(function (track) {
|
||||
track.enabled = !muted;
|
||||
});
|
||||
};
|
||||
|
|
Loading…
Reference in New Issue