diff --git a/app.js b/app.js
index 93c88eb45..5cee715dc 100644
--- a/app.js
+++ b/app.js
@@ -1,17 +1,8 @@
/* jshint -W117 */
/* application specific logic */
-var connection = null;
-var authenticatedUser = false;
-/* Initial "authentication required" dialog */
-var authDialog = null;
-/* Loop retry ID that wits for other user to create the room */
-var authRetryId = null;
-var activecall = null;
var nickname = null;
var focusMucJid = null;
-var roomName = null;
var ssrc2jid = {};
-var bridgeIsDown = false;
//TODO: this array must be removed when firefox implement multistream support
var notReceivedSSRCs = [];
@@ -27,674 +18,12 @@ var ssrc2videoType = {};
* @type {String}
*/
var focusedVideoInfo = null;
-var mutedAudios = {};
-/**
- * Remembers if we were muted by the focus.
- * @type {boolean}
- */
-var forceMuted = false;
-/**
- * Indicates if we have muted our audio before the conference has started.
- * @type {boolean}
- */
-var preMuted = false;
-
-var localVideoSrc = null;
-var flipXLocalVideo = true;
-var isFullScreen = false;
-var currentVideoWidth = null;
-var currentVideoHeight = null;
-
-var sessionTerminated = false;
function init() {
-
- RTC.addStreamListener(maybeDoJoin, StreamEventTypes.EVENT_TYPE_LOCAL_CREATED);
RTC.start();
+ xmpp.start(UI.getCreadentials);
- var jid = document.getElementById('jid').value || config.hosts.anonymousdomain || config.hosts.domain || window.location.hostname;
- connect(jid);
-}
-
-function connect(jid, password) {
- connection = new Strophe.Connection(document.getElementById('boshURL').value || config.bosh || '/http-bind');
-
- var settings = UI.getSettings();
- var email = settings.email;
- var displayName = settings.displayName;
- if(email) {
- connection.emuc.addEmailToPresence(email);
- } else {
- connection.emuc.addUserIdToPresence(settings.uid);
- }
- if(displayName) {
- connection.emuc.addDisplayNameToPresence(displayName);
- }
-
- if (connection.disco) {
- // for chrome, add multistream cap
- }
- connection.jingle.pc_constraints = RTC.getPCConstraints();
- if (config.useIPv6) {
- // https://code.google.com/p/webrtc/issues/detail?id=2828
- if (!connection.jingle.pc_constraints.optional) connection.jingle.pc_constraints.optional = [];
- connection.jingle.pc_constraints.optional.push({googIPv6: true});
- }
-
- if(!password)
- password = document.getElementById('password').value;
-
- var anonymousConnectionFailed = false;
- connection.connect(jid, password, function (status, msg) {
- console.log('Strophe status changed to', Strophe.getStatusString(status));
- if (status === Strophe.Status.CONNECTED) {
- if (config.useStunTurn) {
- connection.jingle.getStunAndTurnCredentials();
- }
- document.getElementById('connect').disabled = true;
-
- console.info("My Jabber ID: " + connection.jid);
-
- if(password)
- authenticatedUser = true;
- maybeDoJoin();
- } else if (status === Strophe.Status.CONNFAIL) {
- if(msg === 'x-strophe-bad-non-anon-jid') {
- anonymousConnectionFailed = true;
- }
- } else if (status === Strophe.Status.DISCONNECTED) {
- if(anonymousConnectionFailed) {
- // prompt user for username and password
- $(document).trigger('passwordrequired.main');
- }
- } else if (status === Strophe.Status.AUTHFAIL) {
- // wrong password or username, prompt user
- $(document).trigger('passwordrequired.main');
-
- }
- });
-}
-
-
-
-function maybeDoJoin() {
- if (connection && connection.connected && Strophe.getResourceFromJid(connection.jid) // .connected is true while connecting?
- && (RTC.localAudio || RTC.localVideo)) {
- doJoin();
- }
-}
-
-function doJoin() {
- if (!roomName) {
- UI.generateRoomName();
- }
-
- Moderator.allocateConferenceFocus(
- roomName, doJoinAfterFocus);
-}
-
-function doJoinAfterFocus() {
-
- // Close authentication dialog if opened
- if (authDialog) {
- UI.messageHandler.closeDialog();
- authDialog = null;
- }
- // Clear retry interval, so that we don't call 'doJoinAfterFocus' twice
- if (authRetryId) {
- window.clearTimeout(authRetryId);
- authRetryId = null;
- }
-
- var roomjid;
- roomjid = roomName;
-
- if (config.useNicks) {
- var nick = window.prompt('Your nickname (optional)');
- if (nick) {
- roomjid += '/' + nick;
- } else {
- roomjid += '/' + Strophe.getNodeFromJid(connection.jid);
- }
- } else {
-
- var tmpJid = Strophe.getNodeFromJid(connection.jid);
-
- if(!authenticatedUser)
- tmpJid = tmpJid.substr(0, 8);
-
- roomjid += '/' + tmpJid;
- }
- connection.emuc.doJoin(roomjid);
-}
-
-function waitForRemoteVideo(selector, ssrc, stream, jid) {
- // XXX(gp) so, every call to this function is *always* preceded by a call
- // to the RTC.attachMediaStream() function but that call is *not* followed
- // by an update to the videoSrcToSsrc map!
- //
- // The above way of doing things results in video SRCs that don't correspond
- // to any SSRC for a short period of time (to be more precise, for as long
- // the waitForRemoteVideo takes to complete). This causes problems (see
- // bellow).
- //
- // I'm wondering why we need to do that; i.e. why call RTC.attachMediaStream()
- // a second time in here and only then update the videoSrcToSsrc map? Why
- // not simply update the videoSrcToSsrc map when the RTC.attachMediaStream()
- // is called the first time? I actually do that in the lastN changed event
- // handler because the "orphan" video SRC is causing troubles there. The
- // purpose of this method would then be to fire the "videoactive.jingle".
- //
- // Food for though I guess :-)
-
- if (selector.removed || !selector.parent().is(":visible")) {
- console.warn("Media removed before had started", selector);
- return;
- }
-
- if (stream.id === 'mixedmslabel') return;
-
- if (selector[0].currentTime > 0) {
- var videoStream = simulcast.getReceivingVideoStream(stream);
- RTC.attachMediaStream(selector, videoStream); // FIXME: why do i have to do this for FF?
-
- // FIXME: add a class that will associate peer Jid, video.src, it's ssrc and video type
- // in order to get rid of too many maps
- if (ssrc && jid) {
- jid2Ssrc[Strophe.getResourceFromJid(jid)] = ssrc;
- } else {
- console.warn("No ssrc given for jid", jid);
- }
-
- $(document).trigger('videoactive.jingle', [selector]);
- } else {
- setTimeout(function () {
- waitForRemoteVideo(selector, ssrc, stream, jid);
- }, 250);
- }
-}
-
-$(document).bind('remotestreamadded.jingle', function (event, data, sid) {
- waitForPresence(data, sid);
-});
-
-function waitForPresence(data, sid) {
- var sess = connection.jingle.sessions[sid];
-
- var thessrc;
-
- // look up an associated JID for a stream id
- if (data.stream.id && data.stream.id.indexOf('mixedmslabel') === -1) {
- // look only at a=ssrc: and _not_ at a=ssrc-group: lines
-
- var ssrclines
- = SDPUtil.find_lines(sess.peerconnection.remoteDescription.sdp, 'a=ssrc:');
- ssrclines = ssrclines.filter(function (line) {
- // NOTE(gp) previously we filtered on the mslabel, but that property
- // is not always present.
- // return line.indexOf('mslabel:' + data.stream.label) !== -1;
-
- return ((line.indexOf('msid:' + data.stream.id) !== -1));
- });
- if (ssrclines.length) {
- thessrc = ssrclines[0].substring(7).split(' ')[0];
-
- // We signal our streams (through Jingle to the focus) before we set
- // our presence (through which peers associate remote streams to
- // jids). So, it might arrive that a remote stream is added but
- // ssrc2jid is not yet updated and thus data.peerjid cannot be
- // successfully set. Here we wait for up to a second for the
- // presence to arrive.
-
- if (!ssrc2jid[thessrc]) {
- // TODO(gp) limit wait duration to 1 sec.
- setTimeout(function(d, s) {
- return function() {
- waitForPresence(d, s);
- }
- }(data, sid), 250);
- return;
- }
-
- // ok to overwrite the one from focus? might save work in colibri.js
- console.log('associated jid', ssrc2jid[thessrc], data.peerjid);
- if (ssrc2jid[thessrc]) {
- data.peerjid = ssrc2jid[thessrc];
- }
- }
- }
-
- //TODO: this code should be removed when firefox implement multistream support
- if(RTC.getBrowserType() == RTCBrowserType.RTC_BROWSER_FIREFOX)
- {
- if((notReceivedSSRCs.length == 0) ||
- !ssrc2jid[notReceivedSSRCs[notReceivedSSRCs.length - 1]])
- {
- // TODO(gp) limit wait duration to 1 sec.
- setTimeout(function(d, s) {
- return function() {
- waitForPresence(d, s);
- }
- }(data, sid), 250);
- return;
- }
-
- thessrc = notReceivedSSRCs.pop();
- if (ssrc2jid[thessrc]) {
- data.peerjid = ssrc2jid[thessrc];
- }
- }
-
- RTC.createRemoteStream(data, sid, thessrc);
-
- var isVideo = data.stream.getVideoTracks().length > 0;
- // an attempt to work around https://github.com/jitsi/jitmeet/issues/32
- if (isVideo &&
- data.peerjid && sess.peerjid === data.peerjid &&
- data.stream.getVideoTracks().length === 0 &&
- RTC.localVideo.getTracks().length > 0) {
- //
- window.setTimeout(function () {
- sendKeyframe(sess.peerconnection);
- }, 3000);
- }
-}
-
-// an attempt to work around https://github.com/jitsi/jitmeet/issues/32
-function sendKeyframe(pc) {
- console.log('sendkeyframe', pc.iceConnectionState);
- if (pc.iceConnectionState !== 'connected') return; // safe...
- pc.setRemoteDescription(
- pc.remoteDescription,
- function () {
- pc.createAnswer(
- function (modifiedAnswer) {
- pc.setLocalDescription(
- modifiedAnswer,
- function () {
- // noop
- },
- function (error) {
- console.log('triggerKeyframe setLocalDescription failed', error);
- UI.messageHandler.showError();
- }
- );
- },
- function (error) {
- console.log('triggerKeyframe createAnswer failed', error);
- UI.messageHandler.showError();
- }
- );
- },
- function (error) {
- console.log('triggerKeyframe setRemoteDescription failed', error);
- UI.messageHandler.showError();
- }
- );
-}
-
-// Really mute video, i.e. dont even send black frames
-function muteVideo(pc, unmute) {
- // FIXME: this probably needs another of those lovely state safeguards...
- // which checks for iceconn == connected and sigstate == stable
- pc.setRemoteDescription(pc.remoteDescription,
- function () {
- pc.createAnswer(
- function (answer) {
- var sdp = new SDP(answer.sdp);
- if (sdp.media.length > 1) {
- if (unmute)
- sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');
- else
- sdp.media[1] = sdp.media[1].replace('a=sendrecv', 'a=recvonly');
- sdp.raw = sdp.session + sdp.media.join('');
- answer.sdp = sdp.raw;
- }
- pc.setLocalDescription(answer,
- function () {
- console.log('mute SLD ok');
- },
- function (error) {
- console.log('mute SLD error');
- UI.messageHandler.showError('Error',
- 'Oops! Something went wrong and we failed to ' +
- 'mute! (SLD Failure)');
- }
- );
- },
- function (error) {
- console.log(error);
- UI.messageHandler.showError();
- }
- );
- },
- function (error) {
- console.log('muteVideo SRD error');
- UI.messageHandler.showError('Error',
- 'Oops! Something went wrong and we failed to stop video!' +
- '(SRD Failure)');
-
- }
- );
-}
-
-$(document).bind('setLocalDescription.jingle', function (event, sid) {
- // put our ssrcs into presence so other clients can identify our stream
- var sess = connection.jingle.sessions[sid];
- var newssrcs = [];
- var media = simulcast.parseMedia(sess.peerconnection.localDescription);
- media.forEach(function (media) {
-
- if(Object.keys(media.sources).length > 0) {
- // TODO(gp) maybe exclude FID streams?
- Object.keys(media.sources).forEach(function (ssrc) {
- newssrcs.push({
- 'ssrc': ssrc,
- 'type': media.type,
- 'direction': media.direction
- });
- });
- }
- else if(sess.localStreamsSSRC && sess.localStreamsSSRC[media.type])
- {
- newssrcs.push({
- 'ssrc': sess.localStreamsSSRC[media.type],
- 'type': media.type,
- 'direction': media.direction
- });
- }
-
- });
-
- console.log('new ssrcs', newssrcs);
-
- // Have to clear presence map to get rid of removed streams
- connection.emuc.clearPresenceMedia();
-
- if (newssrcs.length > 0) {
- for (var i = 1; i <= newssrcs.length; i ++) {
- // Change video type to screen
- if (newssrcs[i-1].type === 'video' && desktopsharing.isUsingScreenStream()) {
- newssrcs[i-1].type = 'screen';
- }
- connection.emuc.addMediaToPresence(i,
- newssrcs[i-1].type, newssrcs[i-1].ssrc, newssrcs[i-1].direction);
- }
-
- connection.emuc.sendPresence();
- }
-});
-
-$(document).bind('iceconnectionstatechange.jingle', function (event, sid, session) {
- switch (session.peerconnection.iceConnectionState) {
- case 'checking':
- session.timeChecking = (new Date()).getTime();
- session.firstconnect = true;
- break;
- case 'completed': // on caller side
- case 'connected':
- if (session.firstconnect) {
- session.firstconnect = false;
- var metadata = {};
- metadata.setupTime = (new Date()).getTime() - session.timeChecking;
- session.peerconnection.getStats(function (res) {
- if(res && res.result) {
- res.result().forEach(function (report) {
- if (report.type == 'googCandidatePair' && report.stat('googActiveConnection') == 'true') {
- metadata.localCandidateType = report.stat('googLocalCandidateType');
- metadata.remoteCandidateType = report.stat('googRemoteCandidateType');
-
- // log pair as well so we can get nice pie charts
- metadata.candidatePair = report.stat('googLocalCandidateType') + ';' + report.stat('googRemoteCandidateType');
-
- if (report.stat('googRemoteAddress').indexOf('[') === 0) {
- metadata.ipv6 = true;
- }
- }
- });
- }
- });
- }
- break;
- }
-});
-
-$(document).bind('presence.muc', function (event, jid, info, pres) {
-
- //check if the video bridge is available
- if($(pres).find(">bridgeIsDown").length > 0 && !bridgeIsDown) {
- bridgeIsDown = true;
- UI.messageHandler.showError("Error",
- "Jitsi Videobridge is currently unavailable. Please try again later!");
- }
-
- if (info.isFocus)
- {
- return;
- }
-
- // Remove old ssrcs coming from the jid
- Object.keys(ssrc2jid).forEach(function (ssrc) {
- if (ssrc2jid[ssrc] == jid) {
- delete ssrc2jid[ssrc];
- delete ssrc2videoType[ssrc];
- }
- });
-
- $(pres).find('>media[xmlns="http://estos.de/ns/mjs"]>source').each(function (idx, ssrc) {
- //console.log(jid, 'assoc ssrc', ssrc.getAttribute('type'), ssrc.getAttribute('ssrc'));
- var ssrcV = ssrc.getAttribute('ssrc');
- ssrc2jid[ssrcV] = jid;
- notReceivedSSRCs.push(ssrcV);
-
- var type = ssrc.getAttribute('type');
- ssrc2videoType[ssrcV] = type;
-
- // might need to update the direction if participant just went from sendrecv to recvonly
- if (type === 'video' || type === 'screen') {
- var el = $('#participant_' + Strophe.getResourceFromJid(jid) + '>video');
- switch (ssrc.getAttribute('direction')) {
- case 'sendrecv':
- el.show();
- break;
- case 'recvonly':
- el.hide();
- // FIXME: Check if we have to change large video
- //VideoLayout.updateLargeVideo(el);
- break;
- }
- }
- });
-
- var displayName = !config.displayJids
- ? info.displayName : Strophe.getResourceFromJid(jid);
-
- if (displayName && displayName.length > 0)
- $(document).trigger('displaynamechanged',
- [jid, displayName]);
- /*if (focus !== null && info.displayName !== null) {
- focus.setEndpointDisplayName(jid, info.displayName);
- }*/
-
- //check if the video bridge is available
- if($(pres).find(">bridgeIsDown").length > 0 && !bridgeIsDown) {
- bridgeIsDown = true;
- UI.messageHandler.showError("Error",
- "Jitsi Videobridge is currently unavailable. Please try again later!");
- }
-
- var id = $(pres).find('>userID').text();
- var email = $(pres).find('>email');
- if(email.length > 0) {
- id = email.text();
- }
- UI.setUserAvatar(jid, id);
-
-});
-
-$(document).bind('kicked.muc', function (event, jid) {
- console.info(jid + " has been kicked from MUC!");
- if (connection.emuc.myroomjid === jid) {
- sessionTerminated = true;
- disposeConference(false);
- connection.emuc.doLeave();
- UI.messageHandler.openMessageDialog("Session Terminated",
- "Ouch! You have been kicked out of the meet!");
- }
-});
-
-$(document).bind('passwordrequired.main', function (event) {
- console.log('password is required');
-
- UI.messageHandler.openTwoButtonDialog(null,
- '
Password required
' +
- '' +
- '',
- true,
- "Ok",
- function (e, v, m, f) {
- if (v) {
- var username = document.getElementById('passwordrequired.username');
- var password = document.getElementById('passwordrequired.password');
-
- if (username.value !== null && password.value != null) {
- connect(username.value, password.value);
- }
- }
- },
- function (event) {
- document.getElementById('passwordrequired.username').focus();
- }
- );
-});
-
-/**
- * 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}
- */
-function isVideoSrcDesktop(jid) {
- // FIXME: fix this mapping mess...
- // figure out if large video is desktop stream or just a camera
-
- if(!jid)
- return false;
- var isDesktop = false;
- if (connection.emuc.myroomjid &&
- Strophe.getResourceFromJid(connection.emuc.myroomjid) === jid) {
- // local video
- isDesktop = desktopsharing.isUsingScreenStream();
- } else {
- // Do we have associations...
- var videoSsrc = jid2Ssrc[jid];
- if (videoSsrc) {
- var videoType = ssrc2videoType[videoSsrc];
- if (videoType) {
- // Finally there...
- isDesktop = videoType === 'screen';
- } else {
- console.error("No video type for ssrc: " + videoSsrc);
- }
- } else {
- console.error("No ssrc for jid: " + jid);
- }
- }
- return isDesktop;
-}
-
-/**
- * Mutes/unmutes the local video.
- *
- * @param mute true to mute the local video; otherwise, false
- * @param options an object which specifies optional arguments such as the
- * boolean key byUser with default value true which
- * specifies whether the method was initiated in response to a user command (in
- * contrast to an automatic decision taken by the application logic)
- */
-function setVideoMute(mute, options) {
- if (connection && RTC.localVideo) {
- if (activecall) {
- activecall.setVideoMute(
- mute,
- function (mute) {
- var video = $('#video');
- var communicativeClass = "icon-camera";
- var muteClass = "icon-camera icon-camera-disabled";
-
- if (mute) {
- video.removeClass(communicativeClass);
- video.addClass(muteClass);
- } else {
- video.removeClass(muteClass);
- video.addClass(communicativeClass);
- }
- connection.emuc.addVideoInfoToPresence(mute);
- connection.emuc.sendPresence();
- },
- options);
- }
- }
-}
-
-$(document).on('inlastnchanged', function (event, oldValue, newValue) {
- if (config.muteLocalVideoIfNotInLastN) {
- setVideoMute(!newValue, { 'byUser': false });
- }
-});
-
-/**
- * Mutes/unmutes the local video.
- */
-function toggleVideo() {
- buttonClick("#video", "icon-camera icon-camera-disabled");
-
- if (connection && activecall && RTC.localVideo ) {
- setVideoMute(!RTC.localVideo.isMuted());
- }
-}
-
-/**
- * Mutes / unmutes audio for the local participant.
- */
-function toggleAudio() {
- setAudioMuted(!RTC.localAudio.isMuted());
-}
-
-/**
- * Sets muted audio state for the local participant.
- */
-function setAudioMuted(mute) {
- if (!(connection && RTC.localAudio)) {
- preMuted = mute;
- // We still click the button.
- buttonClick("#mute", "icon-microphone icon-mic-disabled");
- return;
- }
-
- if (forceMuted && !mute) {
- console.info("Asking focus for unmute");
- connection.moderate.setMute(connection.emuc.myroomjid, mute);
- // FIXME: wait for result before resetting muted status
- forceMuted = false;
- }
-
- if (mute == RTC.localAudio.isMuted()) {
- // Nothing to do
- return;
- }
-
- // It is not clear what is the right way to handle multiple tracks.
- // So at least make sure that they are all muted or all unmuted and
- // that we send presence just once.
- RTC.localAudio.mute();
- // isMuted is the opposite of audioEnabled
- connection.emuc.addAudioInfoToPresence(mute);
- connection.emuc.sendPresence();
- UI.showLocalAudioIndicator(mute);
-
- buttonClick("#mute", "icon-microphone icon-mic-disabled");
}
@@ -706,60 +35,12 @@ $(document).ready(function () {
UI.start();
statistics.start();
- Moderator.init();
-
// Set default desktop sharing method
desktopsharing.init();
});
$(window).bind('beforeunload', function () {
- if (connection && connection.connected) {
- // ensure signout
- $.ajax({
- type: 'POST',
- url: config.bosh,
- async: false,
- cache: false,
- contentType: 'application/xml',
- data: "",
- success: function (data) {
- console.log('signed out');
- console.log(data);
- },
- error: function (XMLHttpRequest, textStatus, errorThrown) {
- console.log('signout error', textStatus + ' (' + errorThrown + ')');
- }
- });
- }
- disposeConference(true);
if(API.isEnabled())
API.dispose();
});
-function disposeConference(onUnload) {
- UI.onDisposeConference(onUnload);
- var handler = activecall;
- if (handler && handler.peerconnection) {
- // FIXME: probably removing streams is not required and close() should
- // be enough
- if (RTC.localAudio) {
- handler.peerconnection.removeStream(RTC.localAudio.getOriginalStream(), onUnload);
- }
- if (RTC.localVideo) {
- handler.peerconnection.removeStream(RTC.localVideo.getOriginalStream(), onUnload);
- }
- handler.peerconnection.close();
- }
- statistics.onDisposeConference(onUnload);
- activecall = null;
-}
-
-/**
- * Changes the style class of the element given by id.
- */
-function buttonClick(id, classname) {
- $(id).toggleClass(classname); // add the class to the clicked element
-}
diff --git a/estos_log.js b/estos_log.js
deleted file mode 100644
index f822fa7a1..000000000
--- a/estos_log.js
+++ /dev/null
@@ -1,17 +0,0 @@
-/* global Strophe */
-Strophe.addConnectionPlugin('logger', {
- // logs raw stanzas and makes them available for download as JSON
- connection: null,
- log: [],
- init: function (conn) {
- this.connection = conn;
- this.connection.rawInput = this.log_incoming.bind(this);
- this.connection.rawOutput = this.log_outgoing.bind(this);
- },
- log_incoming: function (stanza) {
- this.log.push([new Date().getTime(), 'incoming', stanza]);
- },
- log_outgoing: function (stanza) {
- this.log.push([new Date().getTime(), 'outgoing', stanza]);
- },
-});
diff --git a/index.html b/index.html
index a50ef1b2c..2c4520ac1 100644
--- a/index.html
+++ b/index.html
@@ -11,16 +11,10 @@
-
-
-
-
-
-
@@ -29,22 +23,20 @@
+
-
-
+
+
-
-
+
-
-
diff --git a/keyboard_shortcut.js b/keyboard_shortcut.js
index 64be13994..b74d52cc3 100644
--- a/keyboard_shortcut.js
+++ b/keyboard_shortcut.js
@@ -14,20 +14,20 @@ var KeyboardShortcut = (function(my) {
77: {
character: "M",
id: "mutePopover",
- function: toggleAudio
+ function: UI.toggleAudio
},
84: {
character: "T",
function: function() {
if(!RTC.localAudio.isMuted()) {
- toggleAudio();
+ UI.toggleAudio();
}
}
},
86: {
character: "V",
id: "toggleVideoPopover",
- function: toggleVideo
+ function: UI.toggleVideo
}
};
@@ -53,7 +53,7 @@ var KeyboardShortcut = (function(my) {
if(!($(":focus").is("input[type=text]") || $(":focus").is("input[type=password]") || $(":focus").is("textarea"))) {
if(e.which === "T".charCodeAt(0)) {
if(RTC.localAudio.isMuted()) {
- toggleAudio();
+ UI.toggleAudio();
}
}
}
diff --git a/libs/modules/API.bundle.js b/libs/modules/API.bundle.js
index ca7317bc4..3c10be005 100644
--- a/libs/modules/API.bundle.js
+++ b/libs/modules/API.bundle.js
@@ -19,8 +19,8 @@
var commands =
{
displayName: UI.inputDisplayNameHandler,
- muteAudio: toggleAudio,
- muteVideo: toggleVideo,
+ muteAudio: UI.toggleAudio,
+ muteVideo: UI.toggleVideo,
toggleFilmStrip: UI.toggleFilmStrip,
toggleChat: UI.toggleChat,
toggleContactList: UI.toggleContactList
@@ -204,4 +204,5 @@ var API = {
module.exports = API;
},{}]},{},[1])(1)
-});
\ No newline at end of file
+});
+//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/usr/local/lib/node_modules/browserify/node_modules/browser-pack/_prelude.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/API/API.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","/**\n * Implements API class that communicates with external api class\n * and provides interface to access Jitsi Meet features by external\n * applications that embed Jitsi Meet\n */\n\n\n\n/**\n * List of the available commands.\n * @type {{\n *              displayName: inputDisplayNameHandler,\n *              muteAudio: toggleAudio,\n *              muteVideo: toggleVideo,\n *              filmStrip: toggleFilmStrip\n *          }}\n */\nvar commands =\n{\n    displayName: UI.inputDisplayNameHandler,\n    muteAudio: UI.toggleAudio,\n    muteVideo: UI.toggleVideo,\n    toggleFilmStrip: UI.toggleFilmStrip,\n    toggleChat: UI.toggleChat,\n    toggleContactList: UI.toggleContactList\n};\n\n\n/**\n * Maps the supported events and their status\n * (true it the event is enabled and false if it is disabled)\n * @type {{\n *              incomingMessage: boolean,\n *              outgoingMessage: boolean,\n *              displayNameChange: boolean,\n *              participantJoined: boolean,\n *              participantLeft: boolean\n *      }}\n */\nvar events =\n{\n    incomingMessage: false,\n    outgoingMessage:false,\n    displayNameChange: false,\n    participantJoined: false,\n    participantLeft: false\n};\n\n/**\n * Processes commands from external applicaiton.\n * @param message the object with the command\n */\nfunction processCommand(message)\n{\n    if(message.action != \"execute\")\n    {\n        console.error(\"Unknown action of the message\");\n        return;\n    }\n    for(var key in message)\n    {\n        if(commands[key])\n            commands[key].apply(null, message[key]);\n    }\n}\n\n/**\n * Processes events objects from external applications\n * @param event the event\n */\nfunction processEvent(event) {\n    if(!event.action)\n    {\n        console.error(\"Event with no action is received.\");\n        return;\n    }\n\n    var i = 0;\n    switch(event.action)\n    {\n        case \"add\":\n            for(; i < event.events.length; i++)\n            {\n                events[event.events[i]] = true;\n            }\n            break;\n        case \"remove\":\n            for(; i < event.events.length; i++)\n            {\n                events[event.events[i]] = false;\n            }\n            break;\n        default:\n            console.error(\"Unknown action for event.\");\n    }\n\n}\n\n/**\n * Sends message to the external application.\n * @param object\n */\nfunction sendMessage(object) {\n    window.parent.postMessage(JSON.stringify(object), \"*\");\n}\n\n/**\n * Processes a message event from the external application\n * @param event the message event\n */\nfunction processMessage(event)\n{\n    var message;\n    try {\n        message = JSON.parse(event.data);\n    } catch (e) {}\n\n    if(!message.type)\n        return;\n    switch (message.type)\n    {\n        case \"command\":\n            processCommand(message);\n            break;\n        case \"event\":\n            processEvent(message);\n            break;\n        default:\n            console.error(\"Unknown type of the message\");\n            return;\n    }\n\n}\n\nvar API = {\n    /**\n     * Check whether the API should be enabled or not.\n     * @returns {boolean}\n     */\n    isEnabled: function () {\n        var hash = location.hash;\n        if(hash && hash.indexOf(\"external\") > -1 && window.postMessage)\n            return true;\n        return false;\n    },\n    /**\n     * Initializes the APIConnector. Setups message event listeners that will\n     * receive information from external applications that embed Jitsi Meet.\n     * It also sends a message to the external application that APIConnector\n     * is initialized.\n     */\n    init: function () {\n        if (window.addEventListener)\n        {\n            window.addEventListener('message',\n                processMessage, false);\n        }\n        else\n        {\n            window.attachEvent('onmessage', processMessage);\n        }\n        sendMessage({type: \"system\", loaded: true});\n    },\n    /**\n     * Checks whether the event is enabled ot not.\n     * @param name the name of the event.\n     * @returns {*}\n     */\n    isEventEnabled: function (name) {\n        return events[name];\n    },\n\n    /**\n     * Sends event object to the external application that has been subscribed\n     * for that event.\n     * @param name the name event\n     * @param object data associated with the event\n     */\n    triggerEvent: function (name, object) {\n        if(this.isEnabled() && this.isEventEnabled(name))\n            sendMessage({\n                type: \"event\", action: \"result\", event: name, result: object});\n    },\n\n    /**\n     * Removes the listeners.\n     */\n    dispose: function () {\n        if(window.removeEventListener)\n        {\n            window.removeEventListener(\"message\",\n                processMessage, false);\n        }\n        else\n        {\n            window.detachEvent('onmessage', processMessage);\n        }\n\n    }\n\n\n};\n\nmodule.exports = API;"]}
diff --git a/libs/modules/RTC.bundle.js b/libs/modules/RTC.bundle.js
index cf4ab3e03..bf8405b4f 100644
--- a/libs/modules/RTC.bundle.js
+++ b/libs/modules/RTC.bundle.js
@@ -1,5 +1,5 @@
!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var f;"undefined"!=typeof window?f=window:"undefined"!=typeof global?f=global:"undefined"!=typeof self&&(f=self),f.RTC=e()}}(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;ovideo');
+ switch (stream.direction) {
+ case 'sendrecv':
+ el.show();
+ break;
+ case 'recvonly':
+ el.hide();
+ // FIXME: Check if we have to change large video
+ //VideoLayout.updateLargeVideo(el);
+ break;
+ }
+ }
+ }
-
+ });
+ xmpp.addListener(XMPPEvents.DISPLAY_NAME_CHANGED, onDisplayNameChanged);
+ xmpp.addListener(XMPPEvents.MUC_JOINED, onMucJoined);
}
function bindEvents()
@@ -118,10 +153,6 @@ function bindEvents()
function () {
VideoLayout.resizeLargeVideoContainer();
VideoLayout.positionLarge();
- isFullScreen = document.fullScreen ||
- document.mozFullScreen ||
- document.webkitIsFullScreen;
-
}
);
@@ -256,11 +287,6 @@ UI.start = function () {
};
-
-UI.setUserAvatar = function (jid, id) {
- Avatar.setUserAvatar(jid, id);
-};
-
UI.toggleSmileys = function () {
Chat.toggleSmileys();
};
@@ -279,7 +305,7 @@ UI.updateChatConversation = function (from, displayName, message) {
return Chat.updateChatConversation(from, displayName, message);
};
-UI.onMucJoined = function (jid, info) {
+function onMucJoined(jid, info) {
Toolbar.updateRoomUrl(window.location.href);
document.getElementById('localNick').appendChild(
document.createTextNode(Strophe.getResourceFromJid(jid) + ' (me)')
@@ -294,15 +320,14 @@ UI.onMucJoined = function (jid, info) {
// Show authenticate button if needed
Toolbar.showAuthenticateButton(
- Moderator.isExternalAuthEnabled() && !Moderator.isModerator());
+ xmpp.isExternalAuthEnabled() && !xmpp.isModerator());
var displayName = !config.displayJids
? info.displayName : Strophe.getResourceFromJid(jid);
if (displayName)
- $(document).trigger('displaynamechanged',
- ['localVideoContainer', displayName + ' (me)']);
-};
+ onDisplayNameChanged('localVideoContainer', displayName + ' (me)');
+}
UI.initEtherpad = function (name) {
Etherpad.init(name);
@@ -358,23 +383,19 @@ UI.toggleContactList = function () {
UI.onLocalRoleChange = function (jid, info, pres) {
console.info("My role changed, new role: " + info.role);
- var isModerator = Moderator.isModerator();
+ var isModerator = xmpp.isModerator();
VideoLayout.showModeratorIndicator();
Toolbar.showAuthenticateButton(
- Moderator.isExternalAuthEnabled() && !isModerator);
+ xmpp.isExternalAuthEnabled() && !isModerator);
if (isModerator) {
- Toolbar.closeAuthenticationWindow();
+ Authentication.closeAuthenticationWindow();
messageHandler.notify(
'Me', 'connected', 'Moderator rights granted !');
}
};
-UI.onDisposeConference = function (unload) {
- Toolbar.showAuthenticateButton(false);
-};
-
UI.onModeratorStatusChanged = function (isModerator) {
Toolbar.showSipCallButton(isModerator);
@@ -415,40 +436,11 @@ UI.onPasswordReqiured = function (callback) {
);
};
-UI.onAuthenticationRequired = function () {
- // This is the loop that will wait for the room to be created by
- // someone else. 'auth_required.moderator' will bring us back here.
- authRetryId = window.setTimeout(
- function () {
- Moderator.allocateConferenceFocus(roomName, doJoinAfterFocus);
- }, 5000);
- // Show prompt only if it's not open
- if (authDialog !== null) {
- return;
- }
- // extract room name from 'room@muc.server.net'
- var room = roomName.substr(0, roomName.indexOf('@'));
-
- authDialog = messageHandler.openDialog(
- 'Stop',
- 'Authentication is required to create room: ' + room +
- ' You can either authenticate to create the room or ' +
- 'just wait for someone else to do so.',
- true,
- {
- Authenticate: 'authNow'
- },
- function (onSubmitEvent, submitValue) {
-
- // Do not close the dialog yet
- onSubmitEvent.preventDefault();
-
- // Open login popup
- if (submitValue === 'authNow') {
- Toolbar.authenticateClicked();
- }
- }
- );
+UI.onAuthenticationRequired = function (intervalCallback) {
+ Authentication.openAuthenticationDialog(
+ roomName, intervalCallback, function () {
+ Toolbar.authenticateClicked();
+ });
};
UI.setRecordingButtonState = function (state) {
@@ -512,6 +504,8 @@ UI.showLocalAudioIndicator = function (mute) {
};
UI.generateRoomName = function() {
+ if(roomName)
+ return roomName;
var roomnode = null;
var path = window.location.pathname;
@@ -541,6 +535,7 @@ UI.generateRoomName = function() {
}
roomName = roomnode + '@' + config.hosts.muc;
+ return roomName;
};
@@ -557,30 +552,151 @@ UI.dockToolbar = function (isDock) {
return ToolbarToggler.dockToolbar(isDock);
};
+UI.getCreadentials = function () {
+ return {
+ bosh: document.getElementById('boshURL').value,
+ password: document.getElementById('password').value,
+ jid: document.getElementById('jid').value
+ };
+};
+
+UI.disableConnect = function () {
+ document.getElementById('connect').disabled = true;
+};
+
+UI.showLoginPopup = function(callback)
+{
+ console.log('password is required');
+
+ UI.messageHandler.openTwoButtonDialog(null,
+ '
Password required
' +
+ '' +
+ '',
+ true,
+ "Ok",
+ function (e, v, m, f) {
+ if (v) {
+ var username = document.getElementById('passwordrequired.username');
+ var password = document.getElementById('passwordrequired.password');
+
+ if (username.value !== null && password.value != null) {
+ callback(username.value, password.value);
+ }
+ }
+ },
+ function (event) {
+ document.getElementById('passwordrequired.username').focus();
+ }
+ );
+}
+
+UI.checkForNicknameAndJoin = function () {
+
+ Authentication.closeAuthenticationDialog();
+ Authentication.stopInterval();
+
+ var nick = null;
+ if (config.useNicks) {
+ nick = window.prompt('Your nickname (optional)');
+ }
+ xmpp.joinRooom(roomName, config.useNicks, nick);
+}
+
+
function dump(elem, filename) {
elem = elem.parentNode;
elem.download = filename || 'meetlog.json';
elem.href = 'data:application/json;charset=utf-8,\n';
- var data = {};
- if (connection.jingle) {
- data = connection.jingle.populateData();
- }
+ var data = xmpp.populateData();
var metadata = {};
metadata.time = new Date();
metadata.url = window.location.href;
metadata.ua = navigator.userAgent;
- if (connection.logger) {
- metadata.xmpp = connection.logger.log;
+ var log = xmpp.getLogger();
+ if (log) {
+ metadata.xmpp = log;
}
data.metadata = metadata;
elem.href += encodeURIComponent(JSON.stringify(data, null, ' '));
return false;
}
+UI.getRoomName = function () {
+ return roomName;
+}
+
+/**
+ * Mutes/unmutes the local video.
+ *
+ * @param mute true to mute the local video; otherwise, false
+ * @param options an object which specifies optional arguments such as the
+ * boolean key byUser with default value true which
+ * specifies whether the method was initiated in response to a user command (in
+ * contrast to an automatic decision taken by the application logic)
+ */
+function setVideoMute(mute, options) {
+ xmpp.setVideoMute(
+ mute,
+ function (mute) {
+ var video = $('#video');
+ var communicativeClass = "icon-camera";
+ var muteClass = "icon-camera icon-camera-disabled";
+
+ if (mute) {
+ video.removeClass(communicativeClass);
+ video.addClass(muteClass);
+ } else {
+ video.removeClass(muteClass);
+ video.addClass(communicativeClass);
+ }
+ },
+ options);
+}
+
+/**
+ * Mutes/unmutes the local video.
+ */
+UI.toggleVideo = function () {
+ UIUtil.buttonClick("#video", "icon-camera icon-camera-disabled");
+
+ setVideoMute(!RTC.localVideo.isMuted());
+};
+
+/**
+ * Mutes / unmutes audio for the local participant.
+ */
+UI.toggleAudio = function() {
+ UI.setAudioMuted(!RTC.localAudio.isMuted());
+};
+
+/**
+ * Sets muted audio state for the local participant.
+ */
+UI.setAudioMuted = function (mute) {
+
+ if(!xmpp.setAudioMute(mute, function () {
+ UI.showLocalAudioIndicator(mute);
+
+ UIUtil.buttonClick("#mute", "icon-microphone icon-mic-disabled");
+ }))
+ {
+ // We still click the button.
+ UIUtil.buttonClick("#mute", "icon-microphone icon-mic-disabled");
+ return;
+ }
+
+}
+
+UI.onLastNChanged = function (oldValue, newValue) {
+ if (config.muteLocalVideoIfNotInLastN) {
+ setVideoMute(!newValue, { 'byUser': false });
+ }
+}
+
module.exports = UI;
-},{"./audio_levels/AudioLevels.js":2,"./avatar/Avatar":4,"./etherpad/Etherpad.js":5,"./prezi/Prezi.js":6,"./side_pannels/SidePanelToggler":7,"./side_pannels/chat/Chat.js":8,"./side_pannels/contactlist/ContactList":12,"./side_pannels/settings/Settings":13,"./side_pannels/settings/SettingsMenu":14,"./toolbars/BottomToolbar":15,"./toolbars/toolbar":17,"./toolbars/toolbartoggler":18,"./util/MessageHandler":20,"./videolayout/VideoLayout.js":23,"./welcome_page/RoomnameGenerator":24,"./welcome_page/WelcomePage":25}],2:[function(require,module,exports){
+},{"./audio_levels/AudioLevels.js":2,"./authentication/Authentication":4,"./avatar/Avatar":5,"./etherpad/Etherpad.js":6,"./prezi/Prezi.js":7,"./side_pannels/SidePanelToggler":8,"./side_pannels/chat/Chat.js":9,"./side_pannels/contactlist/ContactList":13,"./side_pannels/settings/Settings":14,"./side_pannels/settings/SettingsMenu":15,"./toolbars/BottomToolbar":16,"./toolbars/toolbar":18,"./toolbars/toolbartoggler":19,"./util/MessageHandler":21,"./util/UIUtil":22,"./videolayout/VideoLayout.js":24,"./welcome_page/RoomnameGenerator":25,"./welcome_page/WelcomePage":26}],2:[function(require,module,exports){
var CanvasUtil = require("./CanvasUtils");
/**
@@ -670,10 +786,10 @@ var AudioLevels = (function(my) {
drawContext.drawImage(canvasCache, 0, 0);
if(resourceJid === AudioLevels.LOCAL_LEVEL) {
- if(!connection.emuc.myroomjid) {
+ if(!xmpp.myJid()) {
return;
}
- resourceJid = Strophe.getResourceFromJid(connection.emuc.myroomjid);
+ resourceJid = xmpp.myResource();
}
if(resourceJid === largeVideoResourceJid) {
@@ -804,8 +920,8 @@ var AudioLevels = (function(my) {
function getVideoSpanId(resourceJid) {
var videoSpanId = null;
if (resourceJid === AudioLevels.LOCAL_LEVEL
- || (connection.emuc.myroomjid && resourceJid
- === Strophe.getResourceFromJid(connection.emuc.myroomjid)))
+ || (xmpp.myResource() && resourceJid
+ === xmpp.myResource()))
videoSpanId = 'localVideoContainer';
else
videoSpanId = 'participant_' + resourceJid;
@@ -958,6 +1074,91 @@ var CanvasUtil = (function(my) {
module.exports = CanvasUtil;
},{}],4:[function(require,module,exports){
+/* Initial "authentication required" dialog */
+var authDialog = null;
+/* Loop retry ID that wits for other user to create the room */
+var authRetryId = null;
+var authenticationWindow = null;
+
+var Authentication = {
+ openAuthenticationDialog: function (roomName, intervalCallback, callback) {
+ // This is the loop that will wait for the room to be created by
+ // someone else. 'auth_required.moderator' will bring us back here.
+ authRetryId = window.setTimeout(intervalCallback , 5000);
+ // Show prompt only if it's not open
+ if (authDialog !== null) {
+ return;
+ }
+ // extract room name from 'room@muc.server.net'
+ var room = roomName.substr(0, roomName.indexOf('@'));
+
+ authDialog = messageHandler.openDialog(
+ 'Stop',
+ 'Authentication is required to create room: ' + room +
+ ' You can either authenticate to create the room or ' +
+ 'just wait for someone else to do so.',
+ true,
+ {
+ Authenticate: 'authNow'
+ },
+ function (onSubmitEvent, submitValue) {
+
+ // Do not close the dialog yet
+ onSubmitEvent.preventDefault();
+
+ // Open login popup
+ if (submitValue === 'authNow') {
+ callback();
+ }
+ }
+ );
+ },
+ closeAuthenticationWindow:function () {
+ if (authenticationWindow) {
+ authenticationWindow.close();
+ authenticationWindow = null;
+ }
+ },
+ focusAuthenticationWindow: function () {
+ // If auth window exists just bring it to the front
+ if (authenticationWindow) {
+ authenticationWindow.focus();
+ return;
+ }
+ },
+ closeAuthenticationDialog: function () {
+ // Close authentication dialog if opened
+ if (authDialog) {
+ UI.messageHandler.closeDialog();
+ authDialog = null;
+ }
+ },
+ createAuthenticationWindow: function (callback, url) {
+ authenticationWindow = messageHandler.openCenteredPopup(
+ url, 910, 660,
+ // On closed
+ function () {
+ // Close authentication dialog if opened
+ if (authDialog) {
+ messageHandler.closeDialog();
+ authDialog = null;
+ }
+ callback();
+ authenticationWindow = null;
+ });
+ return authenticationWindow;
+ },
+ stopInterval: function () {
+ // Clear retry interval, so that we don't call 'doJoinAfterFocus' twice
+ if (authRetryId) {
+ window.clearTimeout(authRetryId);
+ authRetryId = null;
+ }
+ }
+};
+
+module.exports = Authentication;
+},{}],5:[function(require,module,exports){
var Settings = require("../side_pannels/settings/Settings");
var users = {};
@@ -972,7 +1173,7 @@ function setVisibility(selector, show) {
function isUserMuted(jid) {
// XXX(gp) we may want to rename this method to something like
// isUserStreaming, for example.
- if (jid && jid != connection.emuc.myroomjid) {
+ if (jid && jid != xmpp.myJid()) {
var resource = Strophe.getResourceFromJid(jid);
if (!require("../videolayout/VideoLayout").isInLastN(resource)) {
return true;
@@ -986,7 +1187,7 @@ function isUserMuted(jid) {
}
function getGravatarUrl(id, size) {
- if(id === connection.emuc.myroomjid || !id) {
+ if(id === xmpp.myJid() || !id) {
id = Settings.getSettings().uid;
}
return 'https://www.gravatar.com/avatar/' +
@@ -1017,7 +1218,7 @@ var Avatar = {
// set the avatar in the settings menu if it is local user and get the
// local video container
- if (jid === connection.emuc.myroomjid) {
+ if (jid === xmpp.myJid()) {
$('#avatar').get(0).src = thumbUrl;
thumbnail = $('#localVideoContainer');
}
@@ -1060,7 +1261,7 @@ var Avatar = {
var video = $('#participant_' + resourceJid + '>video');
var avatar = $('#avatar_' + resourceJid);
- if (jid === connection.emuc.myroomjid) {
+ if (jid === xmpp.myJid()) {
video = $('#localVideoWrapper>video');
}
if (show === undefined || show === null) {
@@ -1090,7 +1291,7 @@ var Avatar = {
*/
updateActiveSpeakerAvatarSrc: function (jid) {
if (!jid) {
- jid = connection.emuc.findJidFromResource(
+ jid = xmpp.findJidFromResource(
require("../videolayout/VideoLayout").getLargeVideoState().userResourceJid);
}
var avatar = $("#activeSpeakerAvatar")[0];
@@ -1112,8 +1313,8 @@ var Avatar = {
module.exports = Avatar;
-},{"../side_pannels/settings/Settings":13,"../videolayout/VideoLayout":23}],5:[function(require,module,exports){
-/* global $, config, connection, dockToolbar, Moderator,
+},{"../side_pannels/settings/Settings":14,"../videolayout/VideoLayout":24}],6:[function(require,module,exports){
+/* global $, config, dockToolbar,
setLargeVideoVisible, Util */
var VideoLayout = require("../videolayout/VideoLayout");
@@ -1145,8 +1346,7 @@ function resize() {
* Shares the Etherpad name with other participants.
*/
function shareEtherpad() {
- connection.emuc.addEtherpadToPresence(etherpadName);
- connection.emuc.sendPresence();
+ xmpp.addToPresence("etherpad", etherpadName);
}
/**
@@ -1309,7 +1509,7 @@ var Etherpad = {
module.exports = Etherpad;
-},{"../prezi/Prezi":6,"../util/UIUtil":21,"../videolayout/VideoLayout":23}],6:[function(require,module,exports){
+},{"../prezi/Prezi":7,"../util/UIUtil":22,"../videolayout/VideoLayout":24}],7:[function(require,module,exports){
var ToolbarToggler = require("../toolbars/ToolbarToggler");
var UIUtil = require("../util/UIUtil");
var VideoLayout = require("../videolayout/VideoLayout");
@@ -1342,7 +1542,7 @@ var Prezi = {
* to load.
*/
openPreziDialog: function() {
- var myprezi = connection.emuc.getPrezi(connection.emuc.myroomjid);
+ var myprezi = xmpp.getPrezi();
if (myprezi) {
messageHandler.openTwoButtonDialog("Remove Prezi",
"Are you sure you would like to remove your Prezi?",
@@ -1350,8 +1550,7 @@ var Prezi = {
"Remove",
function(e,v,m,f) {
if(v) {
- connection.emuc.removePreziFromPresence();
- connection.emuc.sendPresence();
+ xmpp.removePreziFromPresence();
}
}
);
@@ -1403,9 +1602,7 @@ var Prezi = {
return false;
}
else {
- connection.emuc
- .addPreziToPresence(urlValue, 0);
- connection.emuc.sendPresence();
+ xmpp.addToPresence("prezi", urlValue);
$.prompt.close();
}
}
@@ -1463,7 +1660,7 @@ function presentationAdded(event, jid, presUrl, currentSlide) {
VideoLayout.resizeThumbnails();
var controlsEnabled = false;
- if (jid === connection.emuc.myroomjid)
+ if (jid === xmpp.myJid())
controlsEnabled = true;
setPresentationVisible(true);
@@ -1503,15 +1700,14 @@ function presentationAdded(event, jid, presUrl, currentSlide) {
preziPlayer.on(PreziPlayer.EVENT_STATUS, function(event) {
console.log("prezi status", event.value);
if (event.value == PreziPlayer.STATUS_CONTENT_READY) {
- if (jid != connection.emuc.myroomjid)
+ if (jid != xmpp.myJid())
preziPlayer.flyToStep(currentSlide);
}
});
preziPlayer.on(PreziPlayer.EVENT_CURRENT_STEP, function(event) {
console.log("event value", event.value);
- connection.emuc.addCurrentSlideToPresence(event.value);
- connection.emuc.sendPresence();
+ xmpp.addToPresence("preziSlide", event.value);
});
$("#" + elementId).css( 'background-image',
@@ -1665,13 +1861,14 @@ $(window).resize(function () {
module.exports = Prezi;
-},{"../toolbars/ToolbarToggler":16,"../util/MessageHandler":20,"../util/UIUtil":21,"../videolayout/VideoLayout":23}],7:[function(require,module,exports){
+},{"../toolbars/ToolbarToggler":17,"../util/MessageHandler":21,"../util/UIUtil":22,"../videolayout/VideoLayout":24}],8:[function(require,module,exports){
var Chat = require("./chat/Chat");
var ContactList = require("./contactlist/ContactList");
var Settings = require("./settings/Settings");
var SettingsMenu = require("./settings/SettingsMenu");
var VideoLayout = require("../videolayout/VideoLayout");
var ToolbarToggler = require("../toolbars/ToolbarToggler");
+var UIUtil = require("../util/UIUtil");
/**
* Toggler for the chat, contact list, settings menu, etc..
@@ -1778,7 +1975,7 @@ var PanelToggler = (function(my) {
* @param onClose function to be called if the window is going to be closed
*/
var toggle = function(object, selector, onOpenComplete, onOpen, onClose) {
- buttonClick(buttons[selector], "active");
+ UIUtil.buttonClick(buttons[selector], "active");
if (object.isVisible()) {
$("#toast-container").animate({
@@ -1808,7 +2005,7 @@ var PanelToggler = (function(my) {
if(currentlyOpen) {
var current = $(currentlyOpen);
- buttonClick(buttons[currentlyOpen], "active");
+ UIUtil.buttonClick(buttons[currentlyOpen], "active");
current.css('z-index', 4);
setTimeout(function () {
current.css('display', 'none');
@@ -1921,8 +2118,8 @@ var PanelToggler = (function(my) {
}(PanelToggler || {}));
module.exports = PanelToggler;
-},{"../toolbars/ToolbarToggler":16,"../videolayout/VideoLayout":23,"./chat/Chat":8,"./contactlist/ContactList":12,"./settings/Settings":13,"./settings/SettingsMenu":14}],8:[function(require,module,exports){
-/* global $, Util, connection, nickname:true, showToolbar */
+},{"../toolbars/ToolbarToggler":17,"../util/UIUtil":22,"../videolayout/VideoLayout":24,"./chat/Chat":9,"./contactlist/ContactList":13,"./settings/Settings":14,"./settings/SettingsMenu":15}],9:[function(require,module,exports){
+/* global $, Util, nickname:true, showToolbar */
var Replacement = require("./Replacement");
var CommandsProcessor = require("./Commands");
var ToolbarToggler = require("../../toolbars/ToolbarToggler");
@@ -2108,8 +2305,7 @@ var Chat = (function (my) {
nickname = val;
window.localStorage.displayname = nickname;
- connection.emuc.addDisplayNameToPresence(nickname);
- connection.emuc.sendPresence();
+ xmpp.addToPresence("displayName", nickname);
Chat.setChatConversationMode(true);
@@ -2132,7 +2328,7 @@ var Chat = (function (my) {
else
{
var message = Util.escapeHtml(value);
- connection.emuc.sendMessage(message, nickname);
+ xmpp.sendChatMessage(message, nickname);
}
}
});
@@ -2158,7 +2354,7 @@ var Chat = (function (my) {
my.updateChatConversation = function (from, displayName, message) {
var divClassName = '';
- if (connection.emuc.myroomjid === from) {
+ if (xmpp.myJid() === from) {
divClassName = "localuser";
}
else {
@@ -2282,7 +2478,7 @@ var Chat = (function (my) {
return my;
}(Chat || {}));
module.exports = Chat;
-},{"../../toolbars/ToolbarToggler":16,"../SidePanelToggler":7,"./Commands":9,"./Replacement":10,"./smileys.json":11}],9:[function(require,module,exports){
+},{"../../toolbars/ToolbarToggler":17,"../SidePanelToggler":8,"./Commands":10,"./Replacement":11,"./smileys.json":12}],10:[function(require,module,exports){
/**
* List with supported commands. The keys are the names of the commands and
* the value is the function that processes the message.
@@ -2317,7 +2513,7 @@ function getCommand(message)
function processTopic(commandArguments)
{
var topic = Util.escapeHtml(commandArguments);
- connection.emuc.setSubject(topic);
+ xmpp.setSubject(topic);
}
/**
@@ -2378,7 +2574,7 @@ CommandsProcessor.prototype.processCommand = function()
};
module.exports = CommandsProcessor;
-},{}],10:[function(require,module,exports){
+},{}],11:[function(require,module,exports){
var Smileys = require("./smileys.json");
/**
* Processes links and smileys in "body"
@@ -2442,7 +2638,7 @@ module.exports = {
linkify: linkify
};
-},{"./smileys.json":11}],11:[function(require,module,exports){
+},{"./smileys.json":12}],12:[function(require,module,exports){
module.exports={
"smileys": {
"smiley1": ":)",
@@ -2492,7 +2688,7 @@ module.exports={
}
}
-},{}],12:[function(require,module,exports){
+},{}],13:[function(require,module,exports){
var numberOfContacts = 0;
var notificationInterval;
@@ -2541,23 +2737,6 @@ function createDisplayNameParagraph(displayName) {
}
-/**
- * Indicates that the display name has changed.
- */
-$(document).bind( 'displaynamechanged',
- function (event, peerJid, displayName) {
- if (peerJid === 'localVideoContainer')
- peerJid = connection.emuc.myroomjid;
-
- var resourceJid = Strophe.getResourceFromJid(peerJid);
-
- var contactName = $('#contactlist #' + resourceJid + '>p');
-
- if (contactName && displayName && displayName.length > 0)
- contactName.html(displayName);
- });
-
-
function stopGlowing(glower) {
window.clearInterval(notificationInterval);
notificationInterval = false;
@@ -2622,7 +2801,7 @@ var ContactList = {
var clElement = contactlist.get(0);
- if (resourceJid === Strophe.getResourceFromJid(connection.emuc.myroomjid)
+ if (resourceJid === xmpp.myResource()
&& $('#contactlist>ul .title')[0].nextSibling.nextSibling) {
clElement.insertBefore(newContact,
$('#contactlist>ul .title')[0].nextSibling.nextSibling);
@@ -2677,11 +2856,23 @@ var ContactList = {
} else {
contact.removeClass('clickable');
}
+ },
+
+ onDisplayNameChange: function (peerJid, displayName) {
+ if (peerJid === 'localVideoContainer')
+ peerJid = xmpp.myJid();
+
+ var resourceJid = Strophe.getResourceFromJid(peerJid);
+
+ var contactName = $('#contactlist #' + resourceJid + '>p');
+
+ if (contactName && displayName && displayName.length > 0)
+ contactName.html(displayName);
}
};
module.exports = ContactList;
-},{}],13:[function(require,module,exports){
+},{}],14:[function(require,module,exports){
var email = '';
var displayName = '';
var userId;
@@ -2741,7 +2932,7 @@ var Settings =
module.exports = Settings;
-},{}],14:[function(require,module,exports){
+},{}],15:[function(require,module,exports){
var Avatar = require("../../avatar/Avatar");
var Settings = require("./Settings");
@@ -2754,16 +2945,15 @@ var SettingsMenu = {
if(newDisplayName) {
var displayName = Settings.setDisplayName(newDisplayName);
- connection.emuc.addDisplayNameToPresence(displayName);
+ xmpp.addToPresence("displayName", displayName, true);
}
- connection.emuc.addEmailToPresence(newEmail);
+ xmpp.addToPresence("email", newEmail);
var email = Settings.setEmail(newEmail);
- connection.emuc.sendPresence();
- Avatar.setUserAvatar(connection.emuc.myroomjid, email);
+ Avatar.setUserAvatar(xmpp.myJid(), email);
},
isVisible: function() {
@@ -2773,18 +2963,19 @@ var SettingsMenu = {
setDisplayName: function(newDisplayName) {
var displayName = Settings.setDisplayName(newDisplayName);
$('#setDisplayName').get(0).value = displayName;
+ },
+
+ onDisplayNameChange: function(peerJid, newDisplayName) {
+ if(peerJid === 'localVideoContainer' ||
+ peerJid === xmpp.myJid()) {
+ this.setDisplayName(newDisplayName);
+ }
}
};
-$(document).bind('displaynamechanged', function(event, peerJid, newDisplayName) {
- if(peerJid === 'localVideoContainer' ||
- peerJid === connection.emuc.myroomjid) {
- SettingsMenu.setDisplayName(newDisplayName);
- }
-});
module.exports = SettingsMenu;
-},{"../../avatar/Avatar":4,"./Settings":13}],15:[function(require,module,exports){
+},{"../../avatar/Avatar":5,"./Settings":14}],16:[function(require,module,exports){
var PanelToggler = require("../side_pannels/SidePanelToggler");
var buttonHandlers = {
@@ -2829,7 +3020,7 @@ var BottomToolbar = (function (my) {
module.exports = BottomToolbar;
-},{"../side_pannels/SidePanelToggler":7}],16:[function(require,module,exports){
+},{"../side_pannels/SidePanelToggler":8}],17:[function(require,module,exports){
/* global $, interfaceConfig, Moderator, DesktopStreaming.showDesktopSharingButton */
var toolbarTimeoutObject,
@@ -2899,7 +3090,7 @@ var ToolbarToggler = {
toolbarTimeout = interfaceConfig.TOOLBAR_TIMEOUT;
}
- if (Moderator.isModerator())
+ if (xmpp.isModerator())
{
// TODO: Enable settings functionality.
// Need to uncomment the settings button in index.html.
@@ -2944,26 +3135,28 @@ var ToolbarToggler = {
};
module.exports = ToolbarToggler;
-},{}],17:[function(require,module,exports){
-/* global $, buttonClick, config, lockRoom, Moderator, roomName,
- setSharedKey, sharedKey, Util */
+},{}],18:[function(require,module,exports){
+/* global $, buttonClick, config, lockRoom,
+ setSharedKey, Util */
var messageHandler = require("../util/MessageHandler");
var BottomToolbar = require("./BottomToolbar");
var Prezi = require("../prezi/Prezi");
var Etherpad = require("../etherpad/Etherpad");
var PanelToggler = require("../side_pannels/SidePanelToggler");
+var Authentication = require("../authentication/Authentication");
+var UIUtil = require("../util/UIUtil");
var roomUrl = null;
var sharedKey = '';
-var authenticationWindow = null;
+var UI = null;
var buttonHandlers =
{
"toolbar_button_mute": function () {
- return toggleAudio();
+ return UI.toggleAudio();
},
"toolbar_button_camera": function () {
- return toggleVideo();
+ return UI.toggleVideo();
},
"toolbar_button_authentication": function () {
return Toolbar.authenticateClicked();
@@ -2991,7 +3184,7 @@ var buttonHandlers =
},
"toolbar_button_fullScreen": function()
{
- buttonClick("#fullScreen", "icon-full-screen icon-exit-full-screen");
+ UIUtil.buttonClick("#fullScreen", "icon-full-screen icon-exit-full-screen");
return Toolbar.toggleFullScreen();
},
"toolbar_button_sip": function () {
@@ -3006,9 +3199,7 @@ var buttonHandlers =
};
function hangup() {
- disposeConference();
- sessionTerminated = true;
- connection.emuc.doLeave();
+ xmpp.disposeConference();
if(config.enableWelcomePage)
{
setTimeout(function()
@@ -3037,7 +3228,29 @@ function hangup() {
*/
function toggleRecording() {
- Recording.toggleRecording();
+ xmpp.toggleRecording(function (callback) {
+ UI.messageHandler.openTwoButtonDialog(null,
+ '
Enter recording token
' +
+ '',
+ false,
+ "Save",
+ function (e, v, m, f) {
+ if (v) {
+ var token = document.getElementById('recordingToken');
+
+ if (token.value) {
+ callback(Util.escapeHtml(token.value));
+ }
+ }
+ },
+ function (event) {
+ document.getElementById('recordingToken').focus();
+ },
+ function () {
+ }
+ );
+ }, Toolbar.setRecordingButtonState, Toolbar.setRecordingButtonState);
}
/**
@@ -3048,7 +3261,7 @@ function lockRoom(lock) {
if (lock)
currentSharedKey = sharedKey;
- connection.emuc.lockRoom(currentSharedKey, function (res) {
+ xmpp.lockRoom(currentSharedKey, function (res) {
// password is required
if (sharedKey)
{
@@ -3130,9 +3343,8 @@ function callSipButtonClicked()
if (v) {
var numberInput = document.getElementById('sipNumber');
if (numberInput.value) {
- connection.rayo.dial(
- numberInput.value, 'fromnumber',
- roomName, sharedKey);
+ xmpp.dial(numberInput.value, 'fromnumber',
+ UI.getRoomName(), sharedKey);
}
}
},
@@ -3144,9 +3356,10 @@ function callSipButtonClicked()
var Toolbar = (function (my) {
- my.init = function () {
+ my.init = function (ui) {
for(var k in buttonHandlers)
$("#" + k).click(buttonHandlers[k]);
+ UI = ui;
}
/**
@@ -3157,35 +3370,15 @@ var Toolbar = (function (my) {
sharedKey = sKey;
};
- my.closeAuthenticationWindow = function () {
- if (authenticationWindow) {
- authenticationWindow.close();
- authenticationWindow = null;
- }
- }
-
my.authenticateClicked = function () {
- // If auth window exists just bring it to the front
- if (authenticationWindow) {
- authenticationWindow.focus();
- return;
- }
+ Authentication.focusAuthenticationWindow();
// Get authentication URL
- Moderator.getAuthUrl(function (url) {
+ xmpp.getAuthUrl(UI.getRoomName(), function (url) {
// Open popup with authentication URL
- authenticationWindow = messageHandler.openCenteredPopup(
- url, 910, 660,
- // On closed
- function () {
- // Close authentication dialog if opened
- if (authDialog) {
- messageHandler.closeDialog();
- authDialog = null;
- }
- // On popup closed - retry room allocation
- Moderator.allocateConferenceFocus(roomName, doJoinAfterFocus);
- authenticationWindow = null;
- });
+ var authenticationWindow = Authentication.createAuthenticationWindow(function () {
+ // On popup closed - retry room allocation
+ xmpp.allocateConferenceFocus(UI.getRoomName(), UI.checkForNicknameAndJoin);
+ }, url);
if (!authenticationWindow) {
Toolbar.showAuthenticateButton(true);
messageHandler.openMessageDialog(
@@ -3226,7 +3419,7 @@ var Toolbar = (function (my) {
*/
my.openLockDialog = function () {
// Only the focus is able to set a shared key.
- if (!Moderator.isModerator()) {
+ if (!xmpp.isModerator()) {
if (sharedKey) {
messageHandler.openMessageDialog(null,
"This conversation is currently protected by" +
@@ -3383,14 +3576,14 @@ var Toolbar = (function (my) {
*/
my.unlockLockButton = function () {
if ($("#lockIcon").hasClass("icon-security-locked"))
- buttonClick("#lockIcon", "icon-security icon-security-locked");
+ UIUtil.buttonClick("#lockIcon", "icon-security icon-security-locked");
};
/**
* Updates the lock button state to locked.
*/
my.lockLockButton = function () {
if ($("#lockIcon").hasClass("icon-security"))
- buttonClick("#lockIcon", "icon-security icon-security-locked");
+ UIUtil.buttonClick("#lockIcon", "icon-security icon-security-locked");
};
/**
@@ -3433,7 +3626,7 @@ var Toolbar = (function (my) {
// Shows or hides SIP calls button
my.showSipCallButton = function (show) {
- if (Moderator.isSipGatewayEnabled() && show) {
+ if (xmpp.isSipGatewayEnabled() && show) {
$('#sipCallButton').css({display: "inline"});
} else {
$('#sipCallButton').css({display: "none"});
@@ -3461,9 +3654,9 @@ var Toolbar = (function (my) {
}(Toolbar || {}));
module.exports = Toolbar;
-},{"../etherpad/Etherpad":5,"../prezi/Prezi":6,"../side_pannels/SidePanelToggler":7,"../util/MessageHandler":20,"./BottomToolbar":15}],18:[function(require,module,exports){
-module.exports=require(16)
-},{}],19:[function(require,module,exports){
+},{"../authentication/Authentication":4,"../etherpad/Etherpad":6,"../prezi/Prezi":7,"../side_pannels/SidePanelToggler":8,"../util/MessageHandler":21,"../util/UIUtil":22,"./BottomToolbar":16}],19:[function(require,module,exports){
+module.exports=require(17)
+},{}],20:[function(require,module,exports){
var JitsiPopover = (function () {
/**
* Constructs new JitsiPopover and attaches it to the element
@@ -3587,7 +3780,7 @@ var JitsiPopover = (function () {
})();
module.exports = JitsiPopover;
-},{}],20:[function(require,module,exports){
+},{}],21:[function(require,module,exports){
/* global $, jQuery */
var messageHandler = (function(my) {
@@ -3756,7 +3949,7 @@ module.exports = messageHandler;
-},{}],21:[function(require,module,exports){
+},{}],22:[function(require,module,exports){
/**
* Created by hristo on 12/22/14.
*/
@@ -3770,10 +3963,17 @@ module.exports = {
= PanelToggler.isVisible() ? PanelToggler.getPanelSize()[0] : 0;
return window.innerWidth - rightPanelWidth;
+ },
+ /**
+ * Changes the style class of the element given by id.
+ */
+ buttonClick: function(id, classname) {
+ $(id).toggleClass(classname); // add the class to the clicked element
}
+
};
-},{"../side_pannels/SidePanelToggler":7}],22:[function(require,module,exports){
+},{"../side_pannels/SidePanelToggler":8}],23:[function(require,module,exports){
var JitsiPopover = require("../util/JitsiPopover");
/**
@@ -4184,7 +4384,7 @@ ConnectionIndicator.prototype.hideIndicator = function () {
};
module.exports = ConnectionIndicator;
-},{"../util/JitsiPopover":19}],23:[function(require,module,exports){
+},{"../util/JitsiPopover":20}],24:[function(require,module,exports){
var AudioLevels = require("../audio_levels/AudioLevels");
var Avatar = require("../avatar/Avatar");
var Chat = require("../side_pannels/chat/Chat");
@@ -4203,8 +4403,99 @@ var largeVideoState = {
newSrc: ''
};
+/**
+ * Indicates if we have muted our audio before the conference has started.
+ * @type {boolean}
+ */
+var preMuted = false;
+
+var mutedAudios = {};
+
+var flipXLocalVideo = true;
+var currentVideoWidth = null;
+var currentVideoHeight = null;
+
+var localVideoSrc = null;
+
var defaultLocalDisplayName = "Me";
+function videoactive( videoelem) {
+ if (videoelem.attr('id').indexOf('mixedmslabel') === -1) {
+ // ignore mixedmslabela0 and v0
+
+ videoelem.show();
+ VideoLayout.resizeThumbnails();
+
+ var videoParent = videoelem.parent();
+ var parentResourceJid = null;
+ if (videoParent)
+ parentResourceJid
+ = VideoLayout.getPeerContainerResourceJid(videoParent[0]);
+
+ // Update the large video to the last added video only if there's no
+ // current dominant, focused speaker or prezi playing or update it to
+ // the current dominant speaker.
+ if ((!focusedVideoInfo &&
+ !VideoLayout.getDominantSpeakerResourceJid() &&
+ !require("../prezi/Prezi").isPresentationVisible()) ||
+ (parentResourceJid &&
+ VideoLayout.getDominantSpeakerResourceJid() === parentResourceJid)) {
+ VideoLayout.updateLargeVideo(
+ RTC.getVideoSrc(videoelem[0]),
+ 1,
+ parentResourceJid);
+ }
+
+ VideoLayout.showModeratorIndicator();
+ }
+}
+
+function waitForRemoteVideo(selector, ssrc, stream, jid) {
+ // XXX(gp) so, every call to this function is *always* preceded by a call
+ // to the RTC.attachMediaStream() function but that call is *not* followed
+ // by an update to the videoSrcToSsrc map!
+ //
+ // The above way of doing things results in video SRCs that don't correspond
+ // to any SSRC for a short period of time (to be more precise, for as long
+ // the waitForRemoteVideo takes to complete). This causes problems (see
+ // bellow).
+ //
+ // I'm wondering why we need to do that; i.e. why call RTC.attachMediaStream()
+ // a second time in here and only then update the videoSrcToSsrc map? Why
+ // not simply update the videoSrcToSsrc map when the RTC.attachMediaStream()
+ // is called the first time? I actually do that in the lastN changed event
+ // handler because the "orphan" video SRC is causing troubles there. The
+ // purpose of this method would then be to fire the "videoactive.jingle".
+ //
+ // Food for though I guess :-)
+
+ if (selector.removed || !selector.parent().is(":visible")) {
+ console.warn("Media removed before had started", selector);
+ return;
+ }
+
+ if (stream.id === 'mixedmslabel') return;
+
+ if (selector[0].currentTime > 0) {
+ var videoStream = simulcast.getReceivingVideoStream(stream);
+ RTC.attachMediaStream(selector, videoStream); // FIXME: why do i have to do this for FF?
+
+ // FIXME: add a class that will associate peer Jid, video.src, it's ssrc and video type
+ // in order to get rid of too many maps
+ if (ssrc && jid) {
+ jid2Ssrc[Strophe.getResourceFromJid(jid)] = ssrc;
+ } else {
+ console.warn("No ssrc given for jid", jid);
+ }
+
+ videoactive(selector);
+ } else {
+ setTimeout(function () {
+ waitForRemoteVideo(selector, ssrc, stream, jid);
+ }, 250);
+ }
+}
+
/**
* Returns an array of the video horizontal and vertical indents,
* so that if fits its parent.
@@ -4381,7 +4672,7 @@ function getParticipantContainer(resourceJid)
if (!resourceJid)
return null;
- if (resourceJid === Strophe.getResourceFromJid(connection.emuc.myroomjid))
+ if (resourceJid === xmpp.myResource())
return $("#localVideoContainer");
else
return $("#participant_" + resourceJid);
@@ -4457,7 +4748,8 @@ function addRemoteVideoMenu(jid, parentElement) {
event.preventDefault();
}
var isMute = mutedAudios[jid] == true;
- connection.moderate.setMute(jid, !isMute);
+ xmpp.setMute(jid, !isMute);
+
popupmenuElement.setAttribute('style', 'display:none;');
if (isMute) {
@@ -4479,7 +4771,7 @@ function addRemoteVideoMenu(jid, parentElement) {
var ejectLinkItem = document.createElement('a');
ejectLinkItem.innerHTML = ejectIndicator + ' Kick out';
ejectLinkItem.onclick = function(){
- connection.moderate.eject(jid);
+ xmpp.eject(jid);
popupmenuElement.setAttribute('style', 'display:none;');
};
@@ -4587,6 +4879,43 @@ function createModeratorIndicatorElement(parentElement) {
}
+/**
+ * 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}
+ */
+function isVideoSrcDesktop(jid) {
+ // FIXME: fix this mapping mess...
+ // figure out if large video is desktop stream or just a camera
+
+ if(!jid)
+ return false;
+ var isDesktop = false;
+ if (xmpp.myJid() &&
+ xmpp.myResource() === jid) {
+ // local video
+ isDesktop = desktopsharing.isUsingScreenStream();
+ } else {
+ // Do we have associations...
+ var videoSsrc = jid2Ssrc[jid];
+ if (videoSsrc) {
+ var videoType = ssrc2videoType[videoSsrc];
+ if (videoType) {
+ // Finally there...
+ isDesktop = videoType === 'screen';
+ } else {
+ console.error("No video type for ssrc: " + videoSsrc);
+ }
+ } else {
+ console.error("No ssrc for jid: " + jid);
+ }
+ }
+ return isDesktop;
+}
+
+
+
var VideoLayout = (function (my) {
my.connectionIndicators = {};
@@ -4594,6 +4923,16 @@ var VideoLayout = (function (my) {
my.getVideoSize = getCameraVideoSize;
my.getVideoPosition = getCameraVideoPosition;
+ my.init = function () {
+ // Listen for large video size updates
+ document.getElementById('largeVideo')
+ .addEventListener('loadedmetadata', function (e) {
+ currentVideoWidth = this.videoWidth;
+ currentVideoHeight = this.videoHeight;
+ VideoLayout.positionLarge(currentVideoWidth, currentVideoHeight);
+ });
+ };
+
my.isInLastN = function(resource) {
return lastNCount < 0 // lastN is disabled, return true
|| (lastNCount > 0 && lastNEndpointsCache.length == 0) // lastNEndpoints cache not built yet, return true
@@ -4609,7 +4948,10 @@ var VideoLayout = (function (my) {
document.getElementById('localAudio').autoplay = true;
document.getElementById('localAudio').volume = 0;
if (preMuted) {
- setAudioMuted(true);
+ if(!UI.setAudioMuted(true))
+ {
+ preMuted = mute;
+ }
preMuted = false;
}
};
@@ -4646,14 +4988,14 @@ var VideoLayout = (function (my) {
VideoLayout.handleVideoThumbClicked(
RTC.getVideoSrc(localVideo),
false,
- Strophe.getResourceFromJid(connection.emuc.myroomjid));
+ xmpp.myResource());
});
$('#localVideoContainer').click(function (event) {
event.stopPropagation();
VideoLayout.handleVideoThumbClicked(
RTC.getVideoSrc(localVideo),
false,
- Strophe.getResourceFromJid(connection.emuc.myroomjid));
+ xmpp.myResource());
});
// Add hover handler
@@ -4683,11 +5025,8 @@ var VideoLayout = (function (my) {
localVideoSrc = RTC.getVideoSrc(localVideo);
- var myResourceJid = null;
- if(connection.emuc.myroomjid)
- {
- myResourceJid = Strophe.getResourceFromJid(connection.emuc.myroomjid);
- }
+ var myResourceJid = xmpp.myResource();
+
VideoLayout.updateLargeVideo(localVideoSrc, 0,
myResourceJid);
@@ -4726,7 +5065,7 @@ var VideoLayout = (function (my) {
{
if(container.id == "localVideoWrapper")
{
- jid = Strophe.getResourceFromJid(connection.emuc.myroomjid);
+ jid = xmpp.myResource();
}
else
{
@@ -4804,9 +5143,9 @@ var VideoLayout = (function (my) {
largeVideoState.isVisible = $('#largeVideo').is(':visible');
largeVideoState.isDesktop = isVideoSrcDesktop(resourceJid);
if(jid2Ssrc[largeVideoState.userResourceJid] ||
- (connection && connection.emuc.myroomjid &&
+ (xmpp.myResource() &&
largeVideoState.userResourceJid ===
- Strophe.getResourceFromJid(connection.emuc.myroomjid))) {
+ xmpp.myResource())) {
largeVideoState.oldResourceJid = largeVideoState.userResourceJid;
} else {
largeVideoState.oldResourceJid = null;
@@ -4830,7 +5169,7 @@ var VideoLayout = (function (my) {
var doUpdate = function () {
Avatar.updateActiveSpeakerAvatarSrc(
- connection.emuc.findJidFromResource(
+ xmpp.findJidFromResource(
largeVideoState.userResourceJid));
if (!userChanged && largeVideoState.preload &&
@@ -4910,7 +5249,7 @@ var VideoLayout = (function (my) {
if(userChanged) {
Avatar.showUserAvatar(
- connection.emuc.findJidFromResource(
+ xmpp.findJidFromResource(
largeVideoState.oldResourceJid));
}
@@ -4925,7 +5264,7 @@ var VideoLayout = (function (my) {
}
} else {
Avatar.showUserAvatar(
- connection.emuc.findJidFromResource(
+ xmpp.findJidFromResource(
largeVideoState.userResourceJid));
}
@@ -5064,7 +5403,7 @@ var VideoLayout = (function (my) {
focusedVideoInfo = null;
if(focusResourceJid) {
Avatar.showUserAvatar(
- connection.emuc.findJidFromResource(focusResourceJid));
+ xmpp.findJidFromResource(focusResourceJid));
}
}
}
@@ -5136,7 +5475,7 @@ var VideoLayout = (function (my) {
// If the peerJid is null then this video span couldn't be directly
// associated with a participant (this could happen in the case of prezi).
- if (Moderator.isModerator() && peerJid !== null)
+ if (xmpp.isModerator() && peerJid !== null)
addRemoteVideoMenu(peerJid, container);
remotes.appendChild(container);
@@ -5321,13 +5660,13 @@ var VideoLayout = (function (my) {
if (state == 'show')
{
// peerContainer.css('-webkit-filter', '');
- var jid = connection.emuc.findJidFromResource(resourceJid);
+ var jid = xmpp.findJidFromResource(resourceJid);
Avatar.showUserAvatar(jid, false);
}
else // if (state == 'avatar')
{
// peerContainer.css('-webkit-filter', 'grayscale(100%)');
- var jid = connection.emuc.findJidFromResource(resourceJid);
+ var jid = xmpp.findJidFromResource(resourceJid);
Avatar.showUserAvatar(jid, true);
}
}
@@ -5353,8 +5692,7 @@ var VideoLayout = (function (my) {
if (name && nickname !== name) {
nickname = name;
window.localStorage.displayname = nickname;
- connection.emuc.addDisplayNameToPresence(nickname);
- connection.emuc.sendPresence();
+ xmpp.addToPresence("displayName", nickname);
Chat.setChatConversationMode(true);
}
@@ -5425,7 +5763,7 @@ var VideoLayout = (function (my) {
*/
my.showModeratorIndicator = function () {
- var isModerator = Moderator.isModerator();
+ var isModerator = xmpp.isModerator();
if (isModerator) {
var indicatorSpan = $('#localVideoContainer .focusindicator');
@@ -5434,7 +5772,10 @@ var VideoLayout = (function (my) {
createModeratorIndicatorElement(indicatorSpan[0]);
}
}
- Object.keys(connection.emuc.members).forEach(function (jid) {
+
+ var members = xmpp.getMembers();
+
+ Object.keys(members).forEach(function (jid) {
if (Strophe.getResourceFromJid(jid) === 'focus') {
// Skip server side focus
@@ -5450,7 +5791,7 @@ var VideoLayout = (function (my) {
return;
}
- var member = connection.emuc.members[jid];
+ var member = members[jid];
if (member.role === 'moderator') {
// Remove menu if peer is moderator
@@ -5622,7 +5963,7 @@ var VideoLayout = (function (my) {
var videoSpanId = null;
var videoContainerId = null;
if (resourceJid
- === Strophe.getResourceFromJid(connection.emuc.myroomjid)) {
+ === xmpp.myResource()) {
videoSpanId = 'localVideoWrapper';
videoContainerId = 'localVideoContainer';
}
@@ -5665,7 +6006,7 @@ var VideoLayout = (function (my) {
}
Avatar.showUserAvatar(
- connection.emuc.findJidFromResource(resourceJid));
+ xmpp.findJidFromResource(resourceJid));
}
};
@@ -5790,7 +6131,7 @@ var VideoLayout = (function (my) {
lastNPickupJid = jid;
$(document).trigger("pinnedendpointchanged", [jid]);
}
- } else if (jid == connection.emuc.myroomjid) {
+ } else if (jid == xmpp.myJid()) {
$("#localVideoContainer").click();
}
}
@@ -5802,13 +6143,13 @@ var VideoLayout = (function (my) {
$(document).bind('audiomuted.muc', function (event, jid, isMuted) {
/*
// FIXME: but focus can not mute in this case ? - check
- if (jid === connection.emuc.myroomjid) {
+ if (jid === xmpp.myJid()) {
// The local mute indicator is controlled locally
return;
}*/
var videoSpanId = null;
- if (jid === connection.emuc.myroomjid) {
+ if (jid === xmpp.myJid()) {
videoSpanId = 'localVideoContainer';
} else {
VideoLayout.ensurePeerContainerExists(jid);
@@ -5817,7 +6158,7 @@ var VideoLayout = (function (my) {
mutedAudios[jid] = isMuted;
- if (Moderator.isModerator()) {
+ if (xmpp.isModerator()) {
VideoLayout.updateRemoteVideoMenu(jid, isMuted);
}
@@ -5835,7 +6176,7 @@ var VideoLayout = (function (my) {
Avatar.showUserAvatar(jid, isMuted);
var videoSpanId = null;
- if (jid === connection.emuc.myroomjid) {
+ if (jid === xmpp.myJid()) {
videoSpanId = 'localVideoContainer';
} else {
VideoLayout.ensurePeerContainerExists(jid);
@@ -5849,11 +6190,11 @@ var VideoLayout = (function (my) {
/**
* Display name changed.
*/
- $(document).bind('displaynamechanged',
- function (event, jid, displayName, status) {
+ my.onDisplayNameChanged =
+ function (jid, displayName, status) {
var name = null;
if (jid === 'localVideoContainer'
- || jid === connection.emuc.myroomjid) {
+ || jid === xmpp.myJid()) {
name = nickname;
setDisplayName('localVideoContainer',
displayName);
@@ -5867,10 +6208,10 @@ var VideoLayout = (function (my) {
}
if(jid === 'localVideoContainer')
- jid = connection.emuc.myroomjid;
+ jid = xmpp.myJid();
if(!name || name != displayName)
API.triggerEvent("displayNameChange",{jid: jid, displayname: displayName});
- });
+ };
/**
* On dominant speaker changed event.
@@ -5878,7 +6219,7 @@ var VideoLayout = (function (my) {
$(document).bind('dominantspeakerchanged', function (event, resourceJid) {
// We ignore local user events.
if (resourceJid
- === Strophe.getResourceFromJid(connection.emuc.myroomjid))
+ === xmpp.myResource())
return;
// Update the current dominant speaker.
@@ -6009,7 +6350,7 @@ var VideoLayout = (function (my) {
if (!isVisible) {
console.log("Add to last N", resourceJid);
- var jid = connection.emuc.findJidFromResource(resourceJid);
+ var jid = xmpp.findJidFromResource(resourceJid);
var mediaStream = RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE];
var sel = $('#participant_' + resourceJid + '>video');
@@ -6042,7 +6383,7 @@ var VideoLayout = (function (my) {
var resource, container, src;
var myResource
- = Strophe.getResourceFromJid(connection.emuc.myroomjid);
+ = xmpp.myResource();
// Find out which endpoint to show in the large video.
for (var i = 0; i < lastNEndpoints.length; i++) {
@@ -6066,37 +6407,6 @@ var VideoLayout = (function (my) {
}
});
- $(document).bind('videoactive.jingle', function (event, videoelem) {
- if (videoelem.attr('id').indexOf('mixedmslabel') === -1) {
- // ignore mixedmslabela0 and v0
-
- videoelem.show();
- VideoLayout.resizeThumbnails();
-
- var videoParent = videoelem.parent();
- var parentResourceJid = null;
- if (videoParent)
- parentResourceJid
- = VideoLayout.getPeerContainerResourceJid(videoParent[0]);
-
- // Update the large video to the last added video only if there's no
- // current dominant, focused speaker or prezi playing or update it to
- // the current dominant speaker.
- if ((!focusedVideoInfo &&
- !VideoLayout.getDominantSpeakerResourceJid() &&
- !require("../prezi/Prezi").isPresentationVisible()) ||
- (parentResourceJid &&
- VideoLayout.getDominantSpeakerResourceJid() === parentResourceJid)) {
- VideoLayout.updateLargeVideo(
- RTC.getVideoSrc(videoelem[0]),
- 1,
- parentResourceJid);
- }
-
- VideoLayout.showModeratorIndicator();
- }
- });
-
$(document).bind('simulcastlayerschanging', function (event, endpointSimulcastLayers) {
endpointSimulcastLayers.forEach(function (esl) {
@@ -6117,13 +6427,13 @@ var VideoLayout = (function (my) {
// Get session and stream from primary ssrc.
var res = simulcast.getReceivingVideoStreamBySSRC(primarySSRC);
- var session = res.session;
+ var sid = res.sid;
var electedStream = res.stream;
- if (session && electedStream) {
+ if (sid && electedStream) {
var msid = simulcast.getRemoteVideoStreamIdBySSRC(primarySSRC);
- console.info([esl, primarySSRC, msid, session, electedStream]);
+ console.info([esl, primarySSRC, msid, sid, electedStream]);
var msidParts = msid.split(' ');
@@ -6143,7 +6453,7 @@ var VideoLayout = (function (my) {
}
} else {
- console.error('Could not find a stream or a session.', session, electedStream);
+ console.error('Could not find a stream or a session.', sid, electedStream);
}
});
});
@@ -6175,17 +6485,17 @@ var VideoLayout = (function (my) {
// Get session and stream from primary ssrc.
var res = simulcast.getReceivingVideoStreamBySSRC(primarySSRC);
- var session = res.session;
+ var sid = res.sid;
var electedStream = res.stream;
- if (session && electedStream) {
+ if (sid && electedStream) {
var msid = simulcast.getRemoteVideoStreamIdBySSRC(primarySSRC);
console.info('Switching simulcast substream.');
- console.info([esl, primarySSRC, msid, session, electedStream]);
+ console.info([esl, primarySSRC, msid, sid, electedStream]);
var msidParts = msid.split(' ');
- var selRemoteVideo = $(['#', 'remoteVideo_', session.sid, '_', msidParts[0]].join(''));
+ var selRemoteVideo = $(['#', 'remoteVideo_', sid, '_', msidParts[0]].join(''));
var updateLargeVideo = (Strophe.getResourceFromJid(ssrc2jid[primarySSRC])
== largeVideoState.userResourceJid);
@@ -6222,7 +6532,7 @@ var VideoLayout = (function (my) {
}
var videoId;
- if(resource == Strophe.getResourceFromJid(connection.emuc.myroomjid))
+ if(resource == xmpp.myResource())
{
videoId = "localVideoContainer";
}
@@ -6235,7 +6545,7 @@ var VideoLayout = (function (my) {
connectionIndicator.updatePopoverData();
} else {
- console.error('Could not find a stream or a session.', session, electedStream);
+ console.error('Could not find a stream or a sid.', sid, electedStream);
}
});
});
@@ -6250,8 +6560,8 @@ var VideoLayout = (function (my) {
if(object.resolution !== null)
{
resolution = object.resolution;
- object.resolution = resolution[connection.emuc.myroomjid];
- delete resolution[connection.emuc.myroomjid];
+ object.resolution = resolution[xmpp.myJid()];
+ delete resolution[xmpp.myJid()];
}
updateStatsIndicator("localVideoContainer", percent, object);
for(var jid in resolution)
@@ -6312,7 +6622,7 @@ var VideoLayout = (function (my) {
}(VideoLayout || {}));
module.exports = VideoLayout;
-},{"../audio_levels/AudioLevels":2,"../avatar/Avatar":4,"../etherpad/Etherpad":5,"../prezi/Prezi":6,"../side_pannels/chat/Chat":8,"../side_pannels/contactlist/ContactList":12,"../util/UIUtil":21,"./ConnectionIndicator":22}],24:[function(require,module,exports){
+},{"../audio_levels/AudioLevels":2,"../avatar/Avatar":5,"../etherpad/Etherpad":6,"../prezi/Prezi":7,"../side_pannels/chat/Chat":9,"../side_pannels/contactlist/ContactList":13,"../util/UIUtil":22,"./ConnectionIndicator":23}],25:[function(require,module,exports){
//var nouns = [
//];
var pluralNouns = [
@@ -6493,7 +6803,7 @@ var RoomNameGenerator = {
module.exports = RoomNameGenerator;
-},{}],25:[function(require,module,exports){
+},{}],26:[function(require,module,exports){
var animateTimeout, updateTimeout;
var RoomNameGenerator = require("./RoomnameGenerator");
@@ -6597,5 +6907,6 @@ function setupWelcomePage()
}
module.exports = setupWelcomePage;
-},{"./RoomnameGenerator":24}]},{},[1])(1)
-});
\ No newline at end of file
+},{"./RoomnameGenerator":25}]},{},[1])(1)
+});
+//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/usr/local/lib/node_modules/browserify/node_modules/browser-pack/_prelude.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/UI.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/audio_levels/AudioLevels.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/audio_levels/CanvasUtils.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/authentication/Authentication.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/avatar/Avatar.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/etherpad/Etherpad.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/prezi/Prezi.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/side_pannels/SidePanelToggler.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/side_pannels/chat/Chat.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/side_pannels/chat/Commands.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/side_pannels/chat/Replacement.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/side_pannels/chat/smileys.json","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/side_pannels/contactlist/ContactList.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/side_pannels/settings/Settings.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/side_pannels/settings/SettingsMenu.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/toolbars/BottomToolbar.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/toolbars/ToolbarToggler.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/toolbars/toolbar.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/util/JitsiPopover.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/util/MessageHandler.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/util/UIUtil.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/videolayout/ConnectionIndicator.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/videolayout/VideoLayout.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/welcome_page/RoomnameGenerator.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/UI/welcome_page/WelcomePage.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxrBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9VA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/PA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtWA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACrgBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1HA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5rEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","var UI = {};\n\nvar VideoLayout = require(\"./videolayout/VideoLayout.js\");\nvar AudioLevels = require(\"./audio_levels/AudioLevels.js\");\nvar Prezi = require(\"./prezi/Prezi.js\");\nvar Etherpad = require(\"./etherpad/Etherpad.js\");\nvar Chat = require(\"./side_pannels/chat/Chat.js\");\nvar Toolbar = require(\"./toolbars/toolbar\");\nvar ToolbarToggler = require(\"./toolbars/toolbartoggler\");\nvar BottomToolbar = require(\"./toolbars/BottomToolbar\");\nvar ContactList = require(\"./side_pannels/contactlist/ContactList\");\nvar Avatar = require(\"./avatar/Avatar\");\n//var EventEmitter = require(\"events\");\nvar SettingsMenu = require(\"./side_pannels/settings/SettingsMenu\");\nvar Settings = require(\"./side_pannels/settings/Settings\");\nvar PanelToggler = require(\"./side_pannels/SidePanelToggler\");\nvar RoomNameGenerator = require(\"./welcome_page/RoomnameGenerator\");\nUI.messageHandler = require(\"./util/MessageHandler\");\nvar messageHandler = UI.messageHandler;\nvar Authentication  = require(\"./authentication/Authentication\");\nvar UIUtil = require(\"./util/UIUtil\");\n\n//var eventEmitter = new EventEmitter();\nvar roomName = null;\n\n\nfunction setupPrezi()\n{\n    $(\"#reloadPresentationLink\").click(function()\n    {\n        Prezi.reloadPresentation();\n    });\n}\n\nfunction setupChat()\n{\n    Chat.init();\n    $(\"#toggle_smileys\").click(function() {\n        Chat.toggleSmileys();\n    });\n}\n\nfunction setupToolbars() {\n    Toolbar.init(UI);\n    Toolbar.setupButtonsFromConfig();\n    BottomToolbar.init();\n}\n\nfunction streamHandler(stream) {\n    switch (stream.type)\n    {\n        case \"audio\":\n            VideoLayout.changeLocalAudio(stream);\n            break;\n        case \"video\":\n            VideoLayout.changeLocalVideo(stream);\n            break;\n        case \"stream\":\n            VideoLayout.changeLocalStream(stream);\n            break;\n        case \"desktop\":\n            VideoLayout.changeLocalVideo(stream);\n            break;\n    }\n}\n\nfunction onDisposeConference(unload) {\n    Toolbar.showAuthenticateButton(false);\n};\n\nfunction onDisplayNameChanged(jid, displayName) {\n    ContactList.onDisplayNameChange(jid, displayName);\n    SettingsMenu.onDisplayNameChange(jid, displayName);\n    VideoLayout.onDisplayNameChanged(jid, displayName);\n}\n\nfunction registerListeners() {\n    RTC.addStreamListener(streamHandler, StreamEventTypes.EVENT_TYPE_LOCAL_CREATED);\n\n    RTC.addStreamListener(streamHandler, StreamEventTypes.EVENT_TYPE_LOCAL_CHANGED);\n    RTC.addStreamListener(function (stream) {\n        VideoLayout.onRemoteStreamAdded(stream);\n    }, StreamEventTypes.EVENT_TYPE_REMOTE_CREATED);\n\n    VideoLayout.init();\n\n    statistics.addAudioLevelListener(function(jid, audioLevel)\n    {\n        var resourceJid;\n        if(jid === statistics.LOCAL_JID)\n        {\n            resourceJid = AudioLevels.LOCAL_LEVEL;\n            if(RTC.localAudio.isMuted())\n            {\n                audioLevel = 0;\n            }\n        }\n        else\n        {\n            resourceJid = Strophe.getResourceFromJid(jid);\n        }\n\n        AudioLevels.updateAudioLevel(resourceJid, audioLevel,\n            UI.getLargeVideoState().userResourceJid);\n    });\n    desktopsharing.addListener(function () {\n        ToolbarToggler.showDesktopSharingButton();\n    }, DesktopSharingEventTypes.INIT);\n    desktopsharing.addListener(\n        Toolbar.changeDesktopSharingButtonState,\n        DesktopSharingEventTypes.SWITCHING_DONE);\n    xmpp.addListener(XMPPEvents.DISPOSE_CONFERENCE, onDisposeConference);\n    xmpp.addListener(XMPPEvents.KICKED, function () {\n        messageHandler.openMessageDialog(\"Session Terminated\",\n            \"Ouch! You have been kicked out of the meet!\");\n    });\n    xmpp.addListener(XMPPEvents.BRIDGE_DOWN, function () {\n        messageHandler.showError(\"Error\",\n            \"Jitsi Videobridge is currently unavailable. Please try again later!\");\n    });\n    xmpp.addListener(XMPPEvents.USER_ID_CHANGED, Avatar.setUserAvatar);\n    xmpp.addListener(XMPPEvents.CHANGED_STREAMS, function (jid, changedStreams) {\n        for(stream in changedStreams)\n        {\n            // might need to update the direction if participant just went from sendrecv to recvonly\n            if (stream.type === 'video' || stream.type === 'screen') {\n                var el = $('#participant_'  + Strophe.getResourceFromJid(jid) + '>video');\n                switch (stream.direction) {\n                    case 'sendrecv':\n                        el.show();\n                        break;\n                    case 'recvonly':\n                        el.hide();\n                        // FIXME: Check if we have to change large video\n                        //VideoLayout.updateLargeVideo(el);\n                        break;\n                }\n            }\n        }\n\n    });\n    xmpp.addListener(XMPPEvents.DISPLAY_NAME_CHANGED, onDisplayNameChanged);\n    xmpp.addListener(XMPPEvents.MUC_JOINED, onMucJoined);\n}\n\nfunction bindEvents()\n{\n    /**\n     * Resizes and repositions videos in full screen mode.\n     */\n    $(document).on('webkitfullscreenchange mozfullscreenchange fullscreenchange',\n        function () {\n            VideoLayout.resizeLargeVideoContainer();\n            VideoLayout.positionLarge();\n        }\n    );\n\n    $(window).resize(function () {\n        VideoLayout.resizeLargeVideoContainer();\n        VideoLayout.positionLarge();\n    });\n}\n\nUI.start = function () {\n    document.title = interfaceConfig.APP_NAME;\n    if(config.enableWelcomePage && window.location.pathname == \"/\" &&\n        (!window.localStorage.welcomePageDisabled || window.localStorage.welcomePageDisabled == \"false\"))\n    {\n        $(\"#videoconference_page\").hide();\n        var setupWelcomePage = require(\"./welcome_page/WelcomePage\");\n        setupWelcomePage();\n\n        return;\n    }\n\n    if (interfaceConfig.SHOW_JITSI_WATERMARK) {\n        var leftWatermarkDiv\n            = $(\"#largeVideoContainer div[class='watermark leftwatermark']\");\n\n        leftWatermarkDiv.css({display: 'block'});\n        leftWatermarkDiv.parent().get(0).href\n            = interfaceConfig.JITSI_WATERMARK_LINK;\n    }\n\n    if (interfaceConfig.SHOW_BRAND_WATERMARK) {\n        var rightWatermarkDiv\n            = $(\"#largeVideoContainer div[class='watermark rightwatermark']\");\n\n        rightWatermarkDiv.css({display: 'block'});\n        rightWatermarkDiv.parent().get(0).href\n            = interfaceConfig.BRAND_WATERMARK_LINK;\n        rightWatermarkDiv.get(0).style.backgroundImage\n            = \"url(images/rightwatermark.png)\";\n    }\n\n    if (interfaceConfig.SHOW_POWERED_BY) {\n        $(\"#largeVideoContainer>a[class='poweredby']\").css({display: 'block'});\n    }\n\n    $(\"#welcome_page\").hide();\n\n    $('body').popover({ selector: '[data-toggle=popover]',\n        trigger: 'click hover',\n        content: function() {\n            return this.getAttribute(\"content\") +\n                KeyboardShortcut.getShortcut(this.getAttribute(\"shortcut\"));\n        }\n    });\n    VideoLayout.resizeLargeVideoContainer();\n    $(\"#videospace\").mousemove(function () {\n        return ToolbarToggler.showToolbar();\n    });\n    // Set the defaults for prompt dialogs.\n    jQuery.prompt.setDefaults({persistent: false});\n\n//    KeyboardShortcut.init();\n    registerListeners();\n    bindEvents();\n    setupPrezi();\n    setupToolbars();\n    setupChat();\n\n    document.title = interfaceConfig.APP_NAME;\n\n    $(\"#downloadlog\").click(function (event) {\n        dump(event.target);\n    });\n\n    if(config.enableWelcomePage && window.location.pathname == \"/\" &&\n        (!window.localStorage.welcomePageDisabled || window.localStorage.welcomePageDisabled == \"false\"))\n    {\n        $(\"#videoconference_page\").hide();\n        var setupWelcomePage = require(\"./welcome_page/WelcomePage\");\n        setupWelcomePage();\n\n        return;\n    }\n\n    $(\"#welcome_page\").hide();\n\n    document.getElementById('largeVideo').volume = 0;\n\n    if (!$('#settings').is(':visible')) {\n        console.log('init');\n        init();\n    } else {\n        loginInfo.onsubmit = function (e) {\n            if (e.preventDefault) e.preventDefault();\n            $('#settings').hide();\n            init();\n        };\n    }\n\n    toastr.options = {\n        \"closeButton\": true,\n        \"debug\": false,\n        \"positionClass\": \"notification-bottom-right\",\n        \"onclick\": null,\n        \"showDuration\": \"300\",\n        \"hideDuration\": \"1000\",\n        \"timeOut\": \"2000\",\n        \"extendedTimeOut\": \"1000\",\n        \"showEasing\": \"swing\",\n        \"hideEasing\": \"linear\",\n        \"showMethod\": \"fadeIn\",\n        \"hideMethod\": \"fadeOut\",\n        \"reposition\": function() {\n            if(PanelToggler.isVisible()) {\n                $(\"#toast-container\").addClass(\"notification-bottom-right-center\");\n            } else {\n                $(\"#toast-container\").removeClass(\"notification-bottom-right-center\");\n            }\n        },\n        \"newestOnTop\": false\n    };\n\n    $('#settingsmenu>input').keyup(function(event){\n        if(event.keyCode === 13) {//enter\n            SettingsMenu.update();\n        }\n    });\n\n    $(\"#updateSettings\").click(function () {\n        SettingsMenu.update();\n    });\n\n};\n\nUI.toggleSmileys = function () {\n    Chat.toggleSmileys();\n};\n\nUI.chatAddError = function(errorMessage, originalText)\n{\n    return Chat.chatAddError(errorMessage, originalText);\n};\n\nUI.chatSetSubject = function(text)\n{\n    return Chat.chatSetSubject(text);\n};\n\nUI.updateChatConversation = function (from, displayName, message) {\n    return Chat.updateChatConversation(from, displayName, message);\n};\n\nfunction onMucJoined(jid, info) {\n    Toolbar.updateRoomUrl(window.location.href);\n    document.getElementById('localNick').appendChild(\n        document.createTextNode(Strophe.getResourceFromJid(jid) + ' (me)')\n    );\n\n    var settings = Settings.getSettings();\n    // Add myself to the contact list.\n    ContactList.addContact(jid, settings.email || settings.uid);\n\n    // Once we've joined the muc show the toolbar\n    ToolbarToggler.showToolbar();\n\n    // Show authenticate button if needed\n    Toolbar.showAuthenticateButton(\n            xmpp.isExternalAuthEnabled() && !xmpp.isModerator());\n\n    var displayName = !config.displayJids\n        ? info.displayName : Strophe.getResourceFromJid(jid);\n\n    if (displayName)\n        onDisplayNameChanged('localVideoContainer', displayName + ' (me)');\n}\n\nUI.initEtherpad = function (name) {\n    Etherpad.init(name);\n};\n\nUI.onMucLeft = function (jid) {\n    console.log('left.muc', jid);\n    var displayName = $('#participant_' + Strophe.getResourceFromJid(jid) +\n        '>.displayname').html();\n    messageHandler.notify(displayName || 'Somebody',\n        'disconnected',\n        'disconnected');\n    // Need to call this with a slight delay, otherwise the element couldn't be\n    // found for some reason.\n    // XXX(gp) it works fine without the timeout for me (with Chrome 38).\n    window.setTimeout(function () {\n        var container = document.getElementById(\n                'participant_' + Strophe.getResourceFromJid(jid));\n        if (container) {\n            ContactList.removeContact(jid);\n            VideoLayout.removeConnectionIndicator(jid);\n            // hide here, wait for video to close before removing\n            $(container).hide();\n            VideoLayout.resizeThumbnails();\n        }\n    }, 10);\n\n    // Unlock large video\n    if (focusedVideoInfo && focusedVideoInfo.jid === jid)\n    {\n        console.info(\"Focused video owner has left the conference\");\n        focusedVideoInfo = null;\n    }\n\n};\n\nUI.getSettings = function () {\n    return Settings.getSettings();\n};\n\nUI.toggleFilmStrip = function () {\n    return BottomToolbar.toggleFilmStrip();\n};\n\nUI.toggleChat = function () {\n    return BottomToolbar.toggleChat();\n};\n\nUI.toggleContactList = function () {\n    return BottomToolbar.toggleContactList();\n};\n\nUI.onLocalRoleChange = function (jid, info, pres) {\n\n    console.info(\"My role changed, new role: \" + info.role);\n    var isModerator = xmpp.isModerator();\n\n    VideoLayout.showModeratorIndicator();\n    Toolbar.showAuthenticateButton(\n            xmpp.isExternalAuthEnabled() && !isModerator);\n\n    if (isModerator) {\n        Authentication.closeAuthenticationWindow();\n        messageHandler.notify(\n            'Me', 'connected', 'Moderator rights granted !');\n    }\n};\n\nUI.onModeratorStatusChanged = function (isModerator) {\n\n    Toolbar.showSipCallButton(isModerator);\n    Toolbar.showRecordingButton(\n        isModerator); //&&\n    // FIXME:\n    // Recording visible if\n    // there are at least 2(+ 1 focus) participants\n    //Object.keys(connection.emuc.members).length >= 3);\n\n    if (isModerator && config.etherpad_base) {\n        Etherpad.init();\n    }\n};\n\nUI.onPasswordReqiured = function (callback) {\n    // password is required\n    Toolbar.lockLockButton();\n\n    messageHandler.openTwoButtonDialog(null,\n            '<h2>Password required</h2>' +\n            '<input id=\"lockKey\" type=\"text\" placeholder=\"password\" autofocus>',\n        true,\n        \"Ok\",\n        function (e, v, m, f) {},\n        function (event) {\n            document.getElementById('lockKey').focus();\n        },\n        function (e, v, m, f) {\n            if (v) {\n                var lockKey = document.getElementById('lockKey');\n                if (lockKey.value !== null) {\n                    Toolbar.setSharedKey(lockKey.value);\n                    callback(lockKey.value);\n                }\n            }\n        }\n    );\n};\n\nUI.onAuthenticationRequired = function (intervalCallback) {\n    Authentication.openAuthenticationDialog(\n        roomName, intervalCallback, function () {\n            Toolbar.authenticateClicked();\n        });\n};\n\nUI.setRecordingButtonState = function (state) {\n    Toolbar.setRecordingButtonState(state);\n};\n\nUI.inputDisplayNameHandler = function (value) {\n    VideoLayout.inputDisplayNameHandler(value);\n};\n\nUI.onMucEntered = function (jid, id, displayName) {\n    messageHandler.notify(displayName || 'Somebody',\n        'connected',\n        'connected');\n\n    // Add Peer's container\n    VideoLayout.ensurePeerContainerExists(jid,id);\n};\n\nUI.onMucPresenceStatus = function ( jid, info) {\n    VideoLayout.setPresenceStatus(\n            'participant_' + Strophe.getResourceFromJid(jid), info.status);\n};\n\nUI.onMucRoleChanged = function (role, displayName) {\n    VideoLayout.showModeratorIndicator();\n\n    if (role === 'moderator') {\n        var displayName = displayName;\n        if (!displayName) {\n            displayName = 'Somebody';\n        }\n        messageHandler.notify(\n            displayName,\n            'connected',\n                'Moderator rights granted to ' + displayName + '!');\n    }\n};\n\nUI.updateLocalConnectionStats = function(percent, stats)\n{\n    VideoLayout.updateLocalConnectionStats(percent, stats);\n};\n\nUI.updateConnectionStats = function(jid, percent, stats)\n{\n    VideoLayout.updateConnectionStats(jid, percent, stats);\n};\n\nUI.onStatsStop = function () {\n    VideoLayout.onStatsStop();\n};\n\nUI.getLargeVideoState = function()\n{\n    return VideoLayout.getLargeVideoState();\n};\n\nUI.showLocalAudioIndicator = function (mute) {\n    VideoLayout.showLocalAudioIndicator(mute);\n};\n\nUI.generateRoomName = function() {\n    if(roomName)\n        return roomName;\n    var roomnode = null;\n    var path = window.location.pathname;\n\n    // determinde the room node from the url\n    // TODO: just the roomnode or the whole bare jid?\n    if (config.getroomnode && typeof config.getroomnode === 'function') {\n        // custom function might be responsible for doing the pushstate\n        roomnode = config.getroomnode(path);\n    } else {\n        /* fall back to default strategy\n         * this is making assumptions about how the URL->room mapping happens.\n         * It currently assumes deployment at root, with a rewrite like the\n         * following one (for nginx):\n         location ~ ^/([a-zA-Z0-9]+)$ {\n         rewrite ^/(.*)$ / break;\n         }\n         */\n        if (path.length > 1) {\n            roomnode = path.substr(1).toLowerCase();\n        } else {\n            var word = RoomNameGenerator.generateRoomWithoutSeparator();\n            roomnode = word.toLowerCase();\n\n            window.history.pushState('VideoChat',\n                    'Room: ' + word, window.location.pathname + word);\n        }\n    }\n\n    roomName = roomnode + '@' + config.hosts.muc;\n    return roomName;\n};\n\n\nUI.connectionIndicatorShowMore = function(id)\n{\n    return VideoLayout.connectionIndicators[id].showMore();\n};\n\nUI.showToolbar = function () {\n    return ToolbarToggler.showToolbar();\n};\n\nUI.dockToolbar = function (isDock) {\n    return ToolbarToggler.dockToolbar(isDock);\n};\n\nUI.getCreadentials = function () {\n    return {\n        bosh: document.getElementById('boshURL').value,\n        password: document.getElementById('password').value,\n        jid: document.getElementById('jid').value\n    };\n};\n\nUI.disableConnect = function () {\n    document.getElementById('connect').disabled = true;\n};\n\nUI.showLoginPopup = function(callback)\n{\n    console.log('password is required');\n\n    UI.messageHandler.openTwoButtonDialog(null,\n            '<h2>Password required</h2>' +\n            '<input id=\"passwordrequired.username\" type=\"text\" placeholder=\"user@domain.net\" autofocus>' +\n            '<input id=\"passwordrequired.password\" type=\"password\" placeholder=\"user password\">',\n        true,\n        \"Ok\",\n        function (e, v, m, f) {\n            if (v) {\n                var username = document.getElementById('passwordrequired.username');\n                var password = document.getElementById('passwordrequired.password');\n\n                if (username.value !== null && password.value != null) {\n                    callback(username.value, password.value);\n                }\n            }\n        },\n        function (event) {\n            document.getElementById('passwordrequired.username').focus();\n        }\n    );\n}\n\nUI.checkForNicknameAndJoin = function () {\n\n    Authentication.closeAuthenticationDialog();\n    Authentication.stopInterval();\n\n    var nick = null;\n    if (config.useNicks) {\n        nick = window.prompt('Your nickname (optional)');\n    }\n    xmpp.joinRooom(roomName, config.useNicks, nick);\n}\n\n\nfunction dump(elem, filename) {\n    elem = elem.parentNode;\n    elem.download = filename || 'meetlog.json';\n    elem.href = 'data:application/json;charset=utf-8,\\n';\n    var data = xmpp.populateData();\n    var metadata = {};\n    metadata.time = new Date();\n    metadata.url = window.location.href;\n    metadata.ua = navigator.userAgent;\n    var log = xmpp.getLogger();\n    if (log) {\n        metadata.xmpp = log;\n    }\n    data.metadata = metadata;\n    elem.href += encodeURIComponent(JSON.stringify(data, null, '  '));\n    return false;\n}\n\nUI.getRoomName = function () {\n    return roomName;\n}\n\n/**\n * Mutes/unmutes the local video.\n *\n * @param mute <tt>true</tt> to mute the local video; otherwise, <tt>false</tt>\n * @param options an object which specifies optional arguments such as the\n * <tt>boolean</tt> key <tt>byUser</tt> with default value <tt>true</tt> which\n * specifies whether the method was initiated in response to a user command (in\n * contrast to an automatic decision taken by the application logic)\n */\nfunction setVideoMute(mute, options) {\n    xmpp.setVideoMute(\n        mute,\n        function (mute) {\n            var video = $('#video');\n            var communicativeClass = \"icon-camera\";\n            var muteClass = \"icon-camera icon-camera-disabled\";\n\n            if (mute) {\n                video.removeClass(communicativeClass);\n                video.addClass(muteClass);\n            } else {\n                video.removeClass(muteClass);\n                video.addClass(communicativeClass);\n            }\n        },\n        options);\n}\n\n/**\n * Mutes/unmutes the local video.\n */\nUI.toggleVideo = function () {\n    UIUtil.buttonClick(\"#video\", \"icon-camera icon-camera-disabled\");\n\n    setVideoMute(!RTC.localVideo.isMuted());\n};\n\n/**\n * Mutes / unmutes audio for the local participant.\n */\nUI.toggleAudio = function() {\n    UI.setAudioMuted(!RTC.localAudio.isMuted());\n};\n\n/**\n * Sets muted audio state for the local participant.\n */\nUI.setAudioMuted = function (mute) {\n\n    if(!xmpp.setAudioMute(mute, function () {\n        UI.showLocalAudioIndicator(mute);\n\n        UIUtil.buttonClick(\"#mute\", \"icon-microphone icon-mic-disabled\");\n    }))\n    {\n        // We still click the button.\n        UIUtil.buttonClick(\"#mute\", \"icon-microphone icon-mic-disabled\");\n        return;\n    }\n\n}\n\nUI.onLastNChanged = function (oldValue, newValue) {\n    if (config.muteLocalVideoIfNotInLastN) {\n        setVideoMute(!newValue, { 'byUser': false });\n    }\n}\n\nmodule.exports = UI;\n\n","var CanvasUtil = require(\"./CanvasUtils\");\n\n/**\n * The audio Levels plugin.\n */\nvar AudioLevels = (function(my) {\n    var audioLevelCanvasCache = {};\n\n    my.LOCAL_LEVEL = 'local';\n\n    /**\n     * Updates the audio level canvas for the given peerJid. If the canvas\n     * didn't exist we create it.\n     */\n    my.updateAudioLevelCanvas = function (peerJid, VideoLayout) {\n        var resourceJid = null;\n        var videoSpanId = null;\n        if (!peerJid)\n            videoSpanId = 'localVideoContainer';\n        else {\n            resourceJid = Strophe.getResourceFromJid(peerJid);\n\n            videoSpanId = 'participant_' + resourceJid;\n        }\n\n        var videoSpan = document.getElementById(videoSpanId);\n\n        if (!videoSpan) {\n            if (resourceJid)\n                console.error(\"No video element for jid\", resourceJid);\n            else\n                console.error(\"No video element for local video.\");\n\n            return;\n        }\n\n        var audioLevelCanvas = $('#' + videoSpanId + '>canvas');\n\n        var videoSpaceWidth = $('#remoteVideos').width();\n        var thumbnailSize = VideoLayout.calculateThumbnailSize(videoSpaceWidth);\n        var thumbnailWidth = thumbnailSize[0];\n        var thumbnailHeight = thumbnailSize[1];\n\n        if (!audioLevelCanvas || audioLevelCanvas.length === 0) {\n\n            audioLevelCanvas = document.createElement('canvas');\n            audioLevelCanvas.className = \"audiolevel\";\n            audioLevelCanvas.style.bottom = \"-\" + interfaceConfig.CANVAS_EXTRA/2 + \"px\";\n            audioLevelCanvas.style.left = \"-\" + interfaceConfig.CANVAS_EXTRA/2 + \"px\";\n            resizeAudioLevelCanvas( audioLevelCanvas,\n                    thumbnailWidth,\n                    thumbnailHeight);\n\n            videoSpan.appendChild(audioLevelCanvas);\n        } else {\n            audioLevelCanvas = audioLevelCanvas.get(0);\n\n            resizeAudioLevelCanvas( audioLevelCanvas,\n                    thumbnailWidth,\n                    thumbnailHeight);\n        }\n    };\n\n    /**\n     * Updates the audio level UI for the given resourceJid.\n     *\n     * @param resourceJid the resource jid indicating the video element for\n     * which we draw the audio level\n     * @param audioLevel the newAudio level to render\n     */\n    my.updateAudioLevel = function (resourceJid, audioLevel, largeVideoResourceJid) {\n        drawAudioLevelCanvas(resourceJid, audioLevel);\n\n        var videoSpanId = getVideoSpanId(resourceJid);\n\n        var audioLevelCanvas = $('#' + videoSpanId + '>canvas').get(0);\n\n        if (!audioLevelCanvas)\n            return;\n\n        var drawContext = audioLevelCanvas.getContext('2d');\n\n        var canvasCache = audioLevelCanvasCache[resourceJid];\n\n        drawContext.clearRect (0, 0,\n                audioLevelCanvas.width, audioLevelCanvas.height);\n        drawContext.drawImage(canvasCache, 0, 0);\n\n        if(resourceJid === AudioLevels.LOCAL_LEVEL) {\n            if(!xmpp.myJid()) {\n                return;\n            }\n            resourceJid = xmpp.myResource();\n        }\n\n        if(resourceJid  === largeVideoResourceJid) {\n            AudioLevels.updateActiveSpeakerAudioLevel(audioLevel);\n        }\n    };\n\n    my.updateActiveSpeakerAudioLevel = function(audioLevel) {\n        var drawContext = $('#activeSpeakerAudioLevel')[0].getContext('2d');\n        var r = interfaceConfig.ACTIVE_SPEAKER_AVATAR_SIZE / 2;\n        var center = (interfaceConfig.ACTIVE_SPEAKER_AVATAR_SIZE + r) / 2;\n\n        // Save the previous state of the context.\n        drawContext.save();\n\n        drawContext.clearRect(0, 0, 300, 300);\n\n        // Draw a circle.\n        drawContext.arc(center, center, r, 0, 2 * Math.PI);\n\n        // Add a shadow around the circle\n        drawContext.shadowColor = interfaceConfig.SHADOW_COLOR;\n        drawContext.shadowBlur = getShadowLevel(audioLevel);\n        drawContext.shadowOffsetX = 0;\n        drawContext.shadowOffsetY = 0;\n\n        // Fill the shape.\n        drawContext.fill();\n\n        drawContext.save();\n\n        drawContext.restore();\n\n\n        drawContext.arc(center, center, r, 0, 2 * Math.PI);\n\n        drawContext.clip();\n        drawContext.clearRect(0, 0, 277, 200);\n\n        // Restore the previous context state.\n        drawContext.restore();\n    };\n\n    /**\n     * Resizes the given audio level canvas to match the given thumbnail size.\n     */\n    function resizeAudioLevelCanvas(audioLevelCanvas,\n                                    thumbnailWidth,\n                                    thumbnailHeight) {\n        audioLevelCanvas.width = thumbnailWidth + interfaceConfig.CANVAS_EXTRA;\n        audioLevelCanvas.height = thumbnailHeight + interfaceConfig.CANVAS_EXTRA;\n    }\n\n    /**\n     * Draws the audio level canvas into the cached canvas object.\n     *\n     * @param resourceJid the resource jid indicating the video element for\n     * which we draw the audio level\n     * @param audioLevel the newAudio level to render\n     */\n    function drawAudioLevelCanvas(resourceJid, audioLevel) {\n        if (!audioLevelCanvasCache[resourceJid]) {\n\n            var videoSpanId = getVideoSpanId(resourceJid);\n\n            var audioLevelCanvasOrig = $('#' + videoSpanId + '>canvas').get(0);\n\n            /*\n             * FIXME Testing has shown that audioLevelCanvasOrig may not exist.\n             * In such a case, the method CanvasUtil.cloneCanvas may throw an\n             * error. Since audio levels are frequently updated, the errors have\n             * been observed to pile into the console, strain the CPU.\n             */\n            if (audioLevelCanvasOrig)\n            {\n                audioLevelCanvasCache[resourceJid]\n                    = CanvasUtil.cloneCanvas(audioLevelCanvasOrig);\n            }\n        }\n\n        var canvas = audioLevelCanvasCache[resourceJid];\n\n        if (!canvas)\n            return;\n\n        var drawContext = canvas.getContext('2d');\n\n        drawContext.clearRect(0, 0, canvas.width, canvas.height);\n\n        var shadowLevel = getShadowLevel(audioLevel);\n\n        if (shadowLevel > 0)\n            // drawContext, x, y, w, h, r, shadowColor, shadowLevel\n            CanvasUtil.drawRoundRectGlow(   drawContext,\n                interfaceConfig.CANVAS_EXTRA/2, interfaceConfig.CANVAS_EXTRA/2,\n                canvas.width - interfaceConfig.CANVAS_EXTRA,\n                canvas.height - interfaceConfig.CANVAS_EXTRA,\n                interfaceConfig.CANVAS_RADIUS,\n                interfaceConfig.SHADOW_COLOR,\n                shadowLevel);\n    }\n\n    /**\n     * Returns the shadow/glow level for the given audio level.\n     *\n     * @param audioLevel the audio level from which we determine the shadow\n     * level\n     */\n    function getShadowLevel (audioLevel) {\n        var shadowLevel = 0;\n\n        if (audioLevel <= 0.3) {\n            shadowLevel = Math.round(interfaceConfig.CANVAS_EXTRA/2*(audioLevel/0.3));\n        }\n        else if (audioLevel <= 0.6) {\n            shadowLevel = Math.round(interfaceConfig.CANVAS_EXTRA/2*((audioLevel - 0.3) / 0.3));\n        }\n        else {\n            shadowLevel = Math.round(interfaceConfig.CANVAS_EXTRA/2*((audioLevel - 0.6) / 0.4));\n        }\n        return shadowLevel;\n    }\n\n    /**\n     * Returns the video span id corresponding to the given resourceJid or local\n     * user.\n     */\n    function getVideoSpanId(resourceJid) {\n        var videoSpanId = null;\n        if (resourceJid === AudioLevels.LOCAL_LEVEL\n                || (xmpp.myResource() && resourceJid\n                    === xmpp.myResource()))\n            videoSpanId = 'localVideoContainer';\n        else\n            videoSpanId = 'participant_' + resourceJid;\n\n        return videoSpanId;\n    }\n\n    /**\n     * Indicates that the remote video has been resized.\n     */\n    $(document).bind('remotevideo.resized', function (event, width, height) {\n        var resized = false;\n        $('#remoteVideos>span>canvas').each(function() {\n            var canvas = $(this).get(0);\n            if (canvas.width !== width + interfaceConfig.CANVAS_EXTRA) {\n                canvas.width = width + interfaceConfig.CANVAS_EXTRA;\n                resized = true;\n            }\n\n            if (canvas.heigh !== height + interfaceConfig.CANVAS_EXTRA) {\n                canvas.height = height + interfaceConfig.CANVAS_EXTRA;\n                resized = true;\n            }\n        });\n\n        if (resized)\n            Object.keys(audioLevelCanvasCache).forEach(function (resourceJid) {\n                audioLevelCanvasCache[resourceJid].width\n                    = width + interfaceConfig.CANVAS_EXTRA;\n                audioLevelCanvasCache[resourceJid].height\n                    = height + interfaceConfig.CANVAS_EXTRA;\n            });\n    });\n\n    return my;\n\n})(AudioLevels || {});\n\nmodule.exports = AudioLevels;","/**\n * Utility class for drawing canvas shapes.\n */\nvar CanvasUtil = (function(my) {\n\n    /**\n     * Draws a round rectangle with a glow. The glowWidth indicates the depth\n     * of the glow.\n     *\n     * @param drawContext the context of the canvas to draw to\n     * @param x the x coordinate of the round rectangle\n     * @param y the y coordinate of the round rectangle\n     * @param w the width of the round rectangle\n     * @param h the height of the round rectangle\n     * @param glowColor the color of the glow\n     * @param glowWidth the width of the glow\n     */\n    my.drawRoundRectGlow\n        = function(drawContext, x, y, w, h, r, glowColor, glowWidth) {\n\n        // Save the previous state of the context.\n        drawContext.save();\n\n        if (w < 2 * r) r = w / 2;\n        if (h < 2 * r) r = h / 2;\n\n        // Draw a round rectangle.\n        drawContext.beginPath();\n        drawContext.moveTo(x+r, y);\n        drawContext.arcTo(x+w, y,   x+w, y+h, r);\n        drawContext.arcTo(x+w, y+h, x,   y+h, r);\n        drawContext.arcTo(x,   y+h, x,   y,   r);\n        drawContext.arcTo(x,   y,   x+w, y,   r);\n        drawContext.closePath();\n\n        // Add a shadow around the rectangle\n        drawContext.shadowColor = glowColor;\n        drawContext.shadowBlur = glowWidth;\n        drawContext.shadowOffsetX = 0;\n        drawContext.shadowOffsetY = 0;\n\n        // Fill the shape.\n        drawContext.fill();\n\n        drawContext.save();\n\n        drawContext.restore();\n\n//      1) Uncomment this line to use Composite Operation, which is doing the\n//      same as the clip function below and is also antialiasing the round\n//      border, but is said to be less fast performance wise.\n\n//      drawContext.globalCompositeOperation='destination-out';\n\n        drawContext.beginPath();\n        drawContext.moveTo(x+r, y);\n        drawContext.arcTo(x+w, y,   x+w, y+h, r);\n        drawContext.arcTo(x+w, y+h, x,   y+h, r);\n        drawContext.arcTo(x,   y+h, x,   y,   r);\n        drawContext.arcTo(x,   y,   x+w, y,   r);\n        drawContext.closePath();\n\n//      2) Uncomment this line to use Composite Operation, which is doing the\n//      same as the clip function below and is also antialiasing the round\n//      border, but is said to be less fast performance wise.\n\n//      drawContext.fill();\n\n        // Comment these two lines if choosing to do the same with composite\n        // operation above 1 and 2.\n        drawContext.clip();\n        drawContext.clearRect(0, 0, 277, 200);\n\n        // Restore the previous context state.\n        drawContext.restore();\n    };\n\n    /**\n     * Clones the given canvas.\n     *\n     * @return the new cloned canvas.\n     */\n    my.cloneCanvas = function (oldCanvas) {\n        /*\n         * FIXME Testing has shown that oldCanvas may not exist. In such a case,\n         * the method CanvasUtil.cloneCanvas may throw an error. Since audio\n         * levels are frequently updated, the errors have been observed to pile\n         * into the console, strain the CPU.\n         */\n        if (!oldCanvas)\n            return oldCanvas;\n\n        //create a new canvas\n        var newCanvas = document.createElement('canvas');\n        var context = newCanvas.getContext('2d');\n\n        //set dimensions\n        newCanvas.width = oldCanvas.width;\n        newCanvas.height = oldCanvas.height;\n\n        //apply the old canvas to the new one\n        context.drawImage(oldCanvas, 0, 0);\n\n        //return the new canvas\n        return newCanvas;\n    };\n\n    return my;\n})(CanvasUtil || {});\n\nmodule.exports = CanvasUtil;","/* Initial \"authentication required\" dialog */\nvar authDialog = null;\n/* Loop retry ID that wits for other user to create the room */\nvar authRetryId = null;\nvar authenticationWindow = null;\n\nvar Authentication = {\n    openAuthenticationDialog: function (roomName, intervalCallback, callback) {\n        // This is the loop that will wait for the room to be created by\n        // someone else. 'auth_required.moderator' will bring us back here.\n        authRetryId = window.setTimeout(intervalCallback , 5000);\n        // Show prompt only if it's not open\n        if (authDialog !== null) {\n            return;\n        }\n        // extract room name from 'room@muc.server.net'\n        var room = roomName.substr(0, roomName.indexOf('@'));\n\n        authDialog = messageHandler.openDialog(\n            'Stop',\n                'Authentication is required to create room:<br/><b>' + room +\n                '</b></br> You can either authenticate to create the room or ' +\n                'just wait for someone else to do so.',\n            true,\n            {\n                Authenticate: 'authNow'\n            },\n            function (onSubmitEvent, submitValue) {\n\n                // Do not close the dialog yet\n                onSubmitEvent.preventDefault();\n\n                // Open login popup\n                if (submitValue === 'authNow') {\n                    callback();\n                }\n            }\n        );\n    },\n    closeAuthenticationWindow:function () {\n        if (authenticationWindow) {\n            authenticationWindow.close();\n            authenticationWindow = null;\n        }\n    },\n    focusAuthenticationWindow: function () {\n        // If auth window exists just bring it to the front\n        if (authenticationWindow) {\n            authenticationWindow.focus();\n            return;\n        }\n    },\n    closeAuthenticationDialog: function () {\n        // Close authentication dialog if opened\n        if (authDialog) {\n            UI.messageHandler.closeDialog();\n            authDialog = null;\n        }\n    },\n    createAuthenticationWindow: function (callback, url) {\n        authenticationWindow = messageHandler.openCenteredPopup(\n            url, 910, 660,\n            // On closed\n            function () {\n                // Close authentication dialog if opened\n                if (authDialog) {\n                    messageHandler.closeDialog();\n                    authDialog = null;\n                }\n                callback();\n                authenticationWindow = null;\n            });\n        return authenticationWindow;\n    },\n    stopInterval: function () {\n        // Clear retry interval, so that we don't call 'doJoinAfterFocus' twice\n        if (authRetryId) {\n            window.clearTimeout(authRetryId);\n            authRetryId = null;\n        }\n    }\n};\n\nmodule.exports = Authentication;","var Settings = require(\"../side_pannels/settings/Settings\");\n\nvar users = {};\nvar activeSpeakerJid;\n\nfunction setVisibility(selector, show) {\n    if (selector && selector.length > 0) {\n        selector.css(\"visibility\", show ? \"visible\" : \"hidden\");\n    }\n}\n\nfunction isUserMuted(jid) {\n    // XXX(gp) we may want to rename this method to something like\n    // isUserStreaming, for example.\n    if (jid && jid != xmpp.myJid()) {\n        var resource = Strophe.getResourceFromJid(jid);\n        if (!require(\"../videolayout/VideoLayout\").isInLastN(resource)) {\n            return true;\n        }\n    }\n\n    if (!RTC.remoteStreams[jid] || !RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE]) {\n        return null;\n    }\n    return RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE].muted;\n}\n\nfunction getGravatarUrl(id, size) {\n    if(id === xmpp.myJid() || !id) {\n        id = Settings.getSettings().uid;\n    }\n    return 'https://www.gravatar.com/avatar/' +\n        MD5.hexdigest(id.trim().toLowerCase()) +\n        \"?d=wavatar&size=\" + (size || \"30\");\n}\n\nvar Avatar = {\n\n    /**\n     * Sets the user's avatar in the settings menu(if local user), contact list\n     * and thumbnail\n     * @param jid jid of the user\n     * @param id email or userID to be used as a hash\n     */\n    setUserAvatar: function (jid, id) {\n        if (id) {\n            if (users[jid] === id) {\n                return;\n            }\n            users[jid] = id;\n        }\n        var thumbUrl = getGravatarUrl(users[jid] || jid, 100);\n        var contactListUrl = getGravatarUrl(users[jid] || jid);\n        var resourceJid = Strophe.getResourceFromJid(jid);\n        var thumbnail = $('#participant_' + resourceJid);\n        var avatar = $('#avatar_' + resourceJid);\n\n        // set the avatar in the settings menu if it is local user and get the\n        // local video container\n        if (jid === xmpp.myJid()) {\n            $('#avatar').get(0).src = thumbUrl;\n            thumbnail = $('#localVideoContainer');\n        }\n\n        // set the avatar in the contact list\n        var contact = $('#' + resourceJid + '>img');\n        if (contact && contact.length > 0) {\n            contact.get(0).src = contactListUrl;\n        }\n\n        // set the avatar in the thumbnail\n        if (avatar && avatar.length > 0) {\n            avatar[0].src = thumbUrl;\n        } else {\n            if (thumbnail && thumbnail.length > 0) {\n                avatar = document.createElement('img');\n                avatar.id = 'avatar_' + resourceJid;\n                avatar.className = 'userAvatar';\n                avatar.src = thumbUrl;\n                thumbnail.append(avatar);\n            }\n        }\n\n        //if the user is the current active speaker - update the active speaker\n        // avatar\n        if (jid === activeSpeakerJid) {\n            this.updateActiveSpeakerAvatarSrc(jid);\n        }\n    },\n\n    /**\n     * Hides or shows the user's avatar\n     * @param jid jid of the user\n     * @param show whether we should show the avatar or not\n     * video because there is no dominant speaker and no focused speaker\n     */\n    showUserAvatar: function (jid, show) {\n        if (users[jid]) {\n            var resourceJid = Strophe.getResourceFromJid(jid);\n            var video = $('#participant_' + resourceJid + '>video');\n            var avatar = $('#avatar_' + resourceJid);\n\n            if (jid === xmpp.myJid()) {\n                video = $('#localVideoWrapper>video');\n            }\n            if (show === undefined || show === null) {\n                show = isUserMuted(jid);\n            }\n\n            //if the user is the currently focused, the dominant speaker or if\n            //there is no focused and no dominant speaker and the large video is\n            //currently shown\n            if (activeSpeakerJid === jid && require(\"../videolayout/VideoLayout\").isLargeVideoOnTop()) {\n                setVisibility($(\"#largeVideo\"), !show);\n                setVisibility($('#activeSpeaker'), show);\n                setVisibility(avatar, false);\n                setVisibility(video, false);\n            } else {\n                if (video && video.length > 0) {\n                    setVisibility(video, !show);\n                    setVisibility(avatar, show);\n                }\n            }\n        }\n    },\n\n    /**\n     * Updates the src of the active speaker avatar\n     * @param jid of the current active speaker\n     */\n    updateActiveSpeakerAvatarSrc: function (jid) {\n        if (!jid) {\n            jid = xmpp.findJidFromResource(\n                require(\"../videolayout/VideoLayout\").getLargeVideoState().userResourceJid);\n        }\n        var avatar = $(\"#activeSpeakerAvatar\")[0];\n        var url = getGravatarUrl(users[jid],\n            interfaceConfig.ACTIVE_SPEAKER_AVATAR_SIZE);\n        if (jid === activeSpeakerJid && avatar.src === url) {\n            return;\n        }\n        activeSpeakerJid = jid;\n        var isMuted = isUserMuted(jid);\n        if (jid && isMuted !== null) {\n            avatar.src = url;\n            setVisibility($(\"#largeVideo\"), !isMuted);\n            Avatar.showUserAvatar(jid, isMuted);\n        }\n    }\n\n};\n\n\nmodule.exports = Avatar;","/* global $, config, dockToolbar,\n   setLargeVideoVisible, Util */\n\nvar VideoLayout = require(\"../videolayout/VideoLayout\");\nvar Prezi = require(\"../prezi/Prezi\");\nvar UIUtil = require(\"../util/UIUtil\");\n\nvar etherpadName = null;\nvar etherpadIFrame = null;\nvar domain = null;\nvar options = \"?showControls=true&showChat=false&showLineNumbers=true&useMonospaceFont=false\";\n\n\n/**\n * Resizes the etherpad.\n */\nfunction resize() {\n    if ($('#etherpad>iframe').length) {\n        var remoteVideos = $('#remoteVideos');\n        var availableHeight\n            = window.innerHeight - remoteVideos.outerHeight();\n        var availableWidth = UIUtil.getAvailableVideoWidth();\n\n        $('#etherpad>iframe').width(availableWidth);\n        $('#etherpad>iframe').height(availableHeight);\n    }\n}\n\n/**\n * Shares the Etherpad name with other participants.\n */\nfunction shareEtherpad() {\n    xmpp.addToPresence(\"etherpad\", etherpadName);\n}\n\n/**\n * Creates the Etherpad button and adds it to the toolbar.\n */\nfunction enableEtherpadButton() {\n    if (!$('#etherpadButton').is(\":visible\"))\n        $('#etherpadButton').css({display: 'inline-block'});\n}\n\n/**\n * Creates the IFrame for the etherpad.\n */\nfunction createIFrame() {\n    etherpadIFrame = document.createElement('iframe');\n    etherpadIFrame.src = domain + etherpadName + options;\n    etherpadIFrame.frameBorder = 0;\n    etherpadIFrame.scrolling = \"no\";\n    etherpadIFrame.width = $('#largeVideoContainer').width() || 640;\n    etherpadIFrame.height = $('#largeVideoContainer').height() || 480;\n    etherpadIFrame.setAttribute('style', 'visibility: hidden;');\n\n    document.getElementById('etherpad').appendChild(etherpadIFrame);\n\n    etherpadIFrame.onload = function() {\n\n        document.domain = document.domain;\n        bubbleIframeMouseMove(etherpadIFrame);\n        setTimeout(function() {\n            // the iframes inside of the etherpad are\n            // not yet loaded when the etherpad iframe is loaded\n            var outer = etherpadIFrame.\n                contentDocument.getElementsByName(\"ace_outer\")[0];\n            bubbleIframeMouseMove(outer);\n            var inner = outer.\n                contentDocument.getElementsByName(\"ace_inner\")[0];\n            bubbleIframeMouseMove(inner);\n        }, 2000);\n    };\n}\n\nfunction bubbleIframeMouseMove(iframe){\n    var existingOnMouseMove = iframe.contentWindow.onmousemove;\n    iframe.contentWindow.onmousemove = function(e){\n        if(existingOnMouseMove) existingOnMouseMove(e);\n        var evt = document.createEvent(\"MouseEvents\");\n        var boundingClientRect = iframe.getBoundingClientRect();\n        evt.initMouseEvent(\n            \"mousemove\",\n            true, // bubbles\n            false, // not cancelable\n            window,\n            e.detail,\n            e.screenX,\n            e.screenY,\n                e.clientX + boundingClientRect.left,\n                e.clientY + boundingClientRect.top,\n            e.ctrlKey,\n            e.altKey,\n            e.shiftKey,\n            e.metaKey,\n            e.button,\n            null // no related element\n        );\n        iframe.dispatchEvent(evt);\n    };\n}\n\n\n/**\n * On video selected event.\n */\n$(document).bind('video.selected', function (event, isPresentation) {\n    if (config.etherpad_base && etherpadIFrame && etherpadIFrame.style.visibility !== 'hidden')\n        Etherpad.toggleEtherpad(isPresentation);\n});\n\n\nvar Etherpad = {\n    /**\n     * Initializes the etherpad.\n     */\n    init: function (name) {\n\n        if (config.etherpad_base && !etherpadName) {\n\n            domain = config.etherpad_base;\n\n            if (!name) {\n                // In case we're the focus we generate the name.\n                etherpadName = Math.random().toString(36).substring(7) +\n                                '_' + (new Date().getTime()).toString();\n                shareEtherpad();\n            }\n            else\n                etherpadName = name;\n\n            enableEtherpadButton();\n\n            /**\n             * Resizes the etherpad, when the window is resized.\n             */\n            $(window).resize(function () {\n                resize();\n            });\n        }\n    },\n\n    /**\n     * Opens/hides the Etherpad.\n     */\n    toggleEtherpad: function (isPresentation) {\n        if (!etherpadIFrame)\n            createIFrame();\n\n        var largeVideo = null;\n        if (Prezi.isPresentationVisible())\n            largeVideo = $('#presentation>iframe');\n        else\n            largeVideo = $('#largeVideo');\n\n        if ($('#etherpad>iframe').css('visibility') === 'hidden') {\n            $('#activeSpeaker').css('visibility', 'hidden');\n            largeVideo.fadeOut(300, function () {\n                if (Prezi.isPresentationVisible()) {\n                    largeVideo.css({opacity: '0'});\n                } else {\n                    VideoLayout.setLargeVideoVisible(false);\n                }\n            });\n\n            $('#etherpad>iframe').fadeIn(300, function () {\n                document.body.style.background = '#eeeeee';\n                $('#etherpad>iframe').css({visibility: 'visible'});\n                $('#etherpad').css({zIndex: 2});\n            });\n        }\n        else if ($('#etherpad>iframe')) {\n            $('#etherpad>iframe').fadeOut(300, function () {\n                $('#etherpad>iframe').css({visibility: 'hidden'});\n                $('#etherpad').css({zIndex: 0});\n                document.body.style.background = 'black';\n            });\n\n            if (!isPresentation) {\n                $('#largeVideo').fadeIn(300, function () {\n                    VideoLayout.setLargeVideoVisible(true);\n                });\n            }\n        }\n        resize();\n    },\n\n    isVisible: function() {\n        var etherpadIframe = $('#etherpad>iframe');\n        return etherpadIframe && etherpadIframe.is(':visible');\n    }\n\n};\n\nmodule.exports = Etherpad;\n","var ToolbarToggler = require(\"../toolbars/ToolbarToggler\");\nvar UIUtil = require(\"../util/UIUtil\");\nvar VideoLayout = require(\"../videolayout/VideoLayout\");\nvar messageHandler = require(\"../util/MessageHandler\");\n\nvar preziPlayer = null;\n\nvar Prezi = {\n\n\n    /**\n     * Reloads the current presentation.\n     */\n    reloadPresentation: function() {\n        var iframe = document.getElementById(preziPlayer.options.preziId);\n        iframe.src = iframe.src;\n    },\n\n    /**\n     * Returns <tt>true</tt> if the presentation is visible, <tt>false</tt> -\n     * otherwise.\n     */\n    isPresentationVisible: function () {\n        return ($('#presentation>iframe') != null\n                && $('#presentation>iframe').css('opacity') == 1);\n    },\n\n    /**\n     * Opens the Prezi dialog, from which the user could choose a presentation\n     * to load.\n     */\n    openPreziDialog: function() {\n        var myprezi = xmpp.getPrezi();\n        if (myprezi) {\n            messageHandler.openTwoButtonDialog(\"Remove Prezi\",\n                \"Are you sure you would like to remove your Prezi?\",\n                false,\n                \"Remove\",\n                function(e,v,m,f) {\n                    if(v) {\n                        xmpp.removePreziFromPresence();\n                    }\n                }\n            );\n        }\n        else if (preziPlayer != null) {\n            messageHandler.openTwoButtonDialog(\"Share a Prezi\",\n                \"Another participant is already sharing a Prezi.\" +\n                    \"This conference allows only one Prezi at a time.\",\n                false,\n                \"Ok\",\n                function(e,v,m,f) {\n                    $.prompt.close();\n                }\n            );\n        }\n        else {\n            var openPreziState = {\n                state0: {\n                    html:   '<h2>Share a Prezi</h2>' +\n                            '<input id=\"preziUrl\" type=\"text\" ' +\n                            'placeholder=\"e.g. ' +\n                            'http://prezi.com/wz7vhjycl7e6/my-prezi\" autofocus>',\n                    persistent: false,\n                    buttons: { \"Share\": true , \"Cancel\": false},\n                    defaultButton: 1,\n                    submit: function(e,v,m,f){\n                        e.preventDefault();\n                        if(v)\n                        {\n                            var preziUrl = document.getElementById('preziUrl');\n\n                            if (preziUrl.value)\n                            {\n                                var urlValue\n                                    = encodeURI(Util.escapeHtml(preziUrl.value));\n\n                                if (urlValue.indexOf('http://prezi.com/') != 0\n                                    && urlValue.indexOf('https://prezi.com/') != 0)\n                                {\n                                    $.prompt.goToState('state1');\n                                    return false;\n                                }\n                                else {\n                                    var presIdTmp = urlValue.substring(\n                                            urlValue.indexOf(\"prezi.com/\") + 10);\n                                    if (!isAlphanumeric(presIdTmp)\n                                            || presIdTmp.indexOf('/') < 2) {\n                                        $.prompt.goToState('state1');\n                                        return false;\n                                    }\n                                    else {\n                                        xmpp.addToPresence(\"prezi\", urlValue);\n                                        $.prompt.close();\n                                    }\n                                }\n                            }\n                        }\n                        else\n                            $.prompt.close();\n                    }\n                },\n                state1: {\n                    html:   '<h2>Share a Prezi</h2>' +\n                            'Please provide a correct prezi link.',\n                    persistent: false,\n                    buttons: { \"Back\": true, \"Cancel\": false },\n                    defaultButton: 1,\n                    submit:function(e,v,m,f) {\n                        e.preventDefault();\n                        if(v==0)\n                            $.prompt.close();\n                        else\n                            $.prompt.goToState('state0');\n                    }\n                }\n            };\n            var focusPreziUrl =  function(e) {\n                    document.getElementById('preziUrl').focus();\n                };\n            messageHandler.openDialogWithStates(openPreziState, focusPreziUrl, focusPreziUrl);\n        }\n    }\n\n};\n\n/**\n * A new presentation has been added.\n *\n * @param event the event indicating the add of a presentation\n * @param jid the jid from which the presentation was added\n * @param presUrl url of the presentation\n * @param currentSlide the current slide to which we should move\n */\nfunction presentationAdded(event, jid, presUrl, currentSlide) {\n    console.log(\"presentation added\", presUrl);\n\n    var presId = getPresentationId(presUrl);\n\n    var elementId = 'participant_'\n        + Strophe.getResourceFromJid(jid)\n        + '_' + presId;\n\n    // We explicitly don't specify the peer jid here, because we don't want\n    // this video to be dealt with as a peer related one (for example we\n    // don't want to show a mute/kick menu for this one, etc.).\n    VideoLayout.addRemoteVideoContainer(null, elementId);\n    VideoLayout.resizeThumbnails();\n\n    var controlsEnabled = false;\n    if (jid === xmpp.myJid())\n        controlsEnabled = true;\n\n    setPresentationVisible(true);\n    $('#largeVideoContainer').hover(\n        function (event) {\n            if (Prezi.isPresentationVisible()) {\n                var reloadButtonRight = window.innerWidth\n                    - $('#presentation>iframe').offset().left\n                    - $('#presentation>iframe').width();\n\n                $('#reloadPresentation').css({  right: reloadButtonRight,\n                    display:'inline-block'});\n            }\n        },\n        function (event) {\n            if (!Prezi.isPresentationVisible())\n                $('#reloadPresentation').css({display:'none'});\n            else {\n                var e = event.toElement || event.relatedTarget;\n\n                if (e && e.id != 'reloadPresentation' && e.id != 'header')\n                    $('#reloadPresentation').css({display:'none'});\n            }\n        });\n\n    preziPlayer = new PreziPlayer(\n        'presentation',\n        {preziId: presId,\n            width: getPresentationWidth(),\n            height: getPresentationHeihgt(),\n            controls: controlsEnabled,\n            debug: true\n        });\n\n    $('#presentation>iframe').attr('id', preziPlayer.options.preziId);\n\n    preziPlayer.on(PreziPlayer.EVENT_STATUS, function(event) {\n        console.log(\"prezi status\", event.value);\n        if (event.value == PreziPlayer.STATUS_CONTENT_READY) {\n            if (jid != xmpp.myJid())\n                preziPlayer.flyToStep(currentSlide);\n        }\n    });\n\n    preziPlayer.on(PreziPlayer.EVENT_CURRENT_STEP, function(event) {\n        console.log(\"event value\", event.value);\n        xmpp.addToPresence(\"preziSlide\", event.value);\n    });\n\n    $(\"#\" + elementId).css( 'background-image',\n        'url(../images/avatarprezi.png)');\n    $(\"#\" + elementId).click(\n        function () {\n            setPresentationVisible(true);\n        }\n    );\n};\n\n/**\n * A presentation has been removed.\n *\n * @param event the event indicating the remove of a presentation\n * @param jid the jid for which the presentation was removed\n * @param the url of the presentation\n */\nfunction presentationRemoved(event, jid, presUrl) {\n    console.log('presentation removed', presUrl);\n    var presId = getPresentationId(presUrl);\n    setPresentationVisible(false);\n    $('#participant_'\n        + Strophe.getResourceFromJid(jid)\n        + '_' + presId).remove();\n    $('#presentation>iframe').remove();\n    if (preziPlayer != null) {\n        preziPlayer.destroy();\n        preziPlayer = null;\n    }\n};\n\n/**\n * Indicates if the given string is an alphanumeric string.\n * Note that some special characters are also allowed (-, _ , /, &, ?, =, ;) for the\n * purpose of checking URIs.\n */\nfunction isAlphanumeric(unsafeText) {\n    var regex = /^[a-z0-9-_\\/&\\?=;]+$/i;\n    return regex.test(unsafeText);\n}\n\n/**\n * Returns the presentation id from the given url.\n */\nfunction getPresentationId (presUrl) {\n    var presIdTmp = presUrl.substring(presUrl.indexOf(\"prezi.com/\") + 10);\n    return presIdTmp.substring(0, presIdTmp.indexOf('/'));\n}\n\n/**\n * Returns the presentation width.\n */\nfunction getPresentationWidth() {\n    var availableWidth = UIUtil.getAvailableVideoWidth();\n    var availableHeight = getPresentationHeihgt();\n\n    var aspectRatio = 16.0 / 9.0;\n    if (availableHeight < availableWidth / aspectRatio) {\n        availableWidth = Math.floor(availableHeight * aspectRatio);\n    }\n    return availableWidth;\n}\n\n/**\n * Returns the presentation height.\n */\nfunction getPresentationHeihgt() {\n    var remoteVideos = $('#remoteVideos');\n    return window.innerHeight - remoteVideos.outerHeight();\n}\n\n/**\n * Resizes the presentation iframe.\n */\nfunction resize() {\n    if ($('#presentation>iframe')) {\n        $('#presentation>iframe').width(getPresentationWidth());\n        $('#presentation>iframe').height(getPresentationHeihgt());\n    }\n}\n\n/**\n * Shows/hides a presentation.\n */\nfunction setPresentationVisible(visible) {\n    var prezi = $('#presentation>iframe');\n    if (visible) {\n        // Trigger the video.selected event to indicate a change in the\n        // large video.\n        $(document).trigger(\"video.selected\", [true]);\n\n        $('#largeVideo').fadeOut(300);\n        prezi.fadeIn(300, function() {\n            prezi.css({opacity:'1'});\n            ToolbarToggler.dockToolbar(true);\n            VideoLayout.setLargeVideoVisible(false);\n        });\n        $('#activeSpeaker').css('visibility', 'hidden');\n    }\n    else {\n        if (prezi.css('opacity') == '1') {\n            prezi.fadeOut(300, function () {\n                prezi.css({opacity:'0'});\n                $('#reloadPresentation').css({display:'none'});\n                $('#largeVideo').fadeIn(300, function() {\n                    VideoLayout.setLargeVideoVisible(true);\n                    ToolbarToggler.dockToolbar(false);\n                });\n            });\n        }\n    }\n}\n\n/**\n * Presentation has been removed.\n */\n$(document).bind('presentationremoved.muc', presentationRemoved);\n\n/**\n * Presentation has been added.\n */\n$(document).bind('presentationadded.muc', presentationAdded);\n\n/*\n * Indicates presentation slide change.\n */\n$(document).bind('gotoslide.muc', function (event, jid, presUrl, current) {\n    if (preziPlayer && preziPlayer.getCurrentStep() != current) {\n        preziPlayer.flyToStep(current);\n\n        var animationStepsArray = preziPlayer.getAnimationCountOnSteps();\n        for (var i = 0; i < parseInt(animationStepsArray[current]); i++) {\n            preziPlayer.flyToStep(current, i);\n        }\n    }\n});\n\n/**\n * On video selected event.\n */\n$(document).bind('video.selected', function (event, isPresentation) {\n    if (!isPresentation && $('#presentation>iframe')) {\n        setPresentationVisible(false);\n    }\n});\n\n$(window).resize(function () {\n    resize();\n});\n\nmodule.exports = Prezi;\n","var Chat = require(\"./chat/Chat\");\nvar ContactList = require(\"./contactlist/ContactList\");\nvar Settings = require(\"./settings/Settings\");\nvar SettingsMenu = require(\"./settings/SettingsMenu\");\nvar VideoLayout = require(\"../videolayout/VideoLayout\");\nvar ToolbarToggler = require(\"../toolbars/ToolbarToggler\");\nvar UIUtil = require(\"../util/UIUtil\");\n\n/**\n * Toggler for the chat, contact list, settings menu, etc..\n */\nvar PanelToggler = (function(my) {\n\n    var currentlyOpen = null;\n    var buttons = {\n        '#chatspace': '#chatBottomButton',\n        '#contactlist': '#contactListButton',\n        '#settingsmenu': '#settingsButton'\n    };\n\n    /**\n     * Resizes the video area\n     * @param isClosing whether the side panel is going to be closed or is going to open / remain opened\n     * @param completeFunction a function to be called when the video space is resized\n     */\n    var resizeVideoArea = function(isClosing, completeFunction) {\n        var videospace = $('#videospace');\n\n        var panelSize = isClosing ? [0, 0] : PanelToggler.getPanelSize();\n        var videospaceWidth = window.innerWidth - panelSize[0];\n        var videospaceHeight = window.innerHeight;\n        var videoSize\n            = VideoLayout.getVideoSize(null, null, videospaceWidth, videospaceHeight);\n        var videoWidth = videoSize[0];\n        var videoHeight = videoSize[1];\n        var videoPosition = VideoLayout.getVideoPosition(videoWidth,\n            videoHeight,\n            videospaceWidth,\n            videospaceHeight);\n        var horizontalIndent = videoPosition[0];\n        var verticalIndent = videoPosition[1];\n\n        var thumbnailSize = VideoLayout.calculateThumbnailSize(videospaceWidth);\n        var thumbnailsWidth = thumbnailSize[0];\n        var thumbnailsHeight = thumbnailSize[1];\n        //for chat\n\n        videospace.animate({\n                right: panelSize[0],\n                width: videospaceWidth,\n                height: videospaceHeight\n            },\n            {\n                queue: false,\n                duration: 500,\n                complete: completeFunction\n            });\n\n        $('#remoteVideos').animate({\n                height: thumbnailsHeight\n            },\n            {\n                queue: false,\n                duration: 500\n            });\n\n        $('#remoteVideos>span').animate({\n                height: thumbnailsHeight,\n                width: thumbnailsWidth\n            },\n            {\n                queue: false,\n                duration: 500,\n                complete: function () {\n                    $(document).trigger(\n                        \"remotevideo.resized\",\n                        [thumbnailsWidth,\n                            thumbnailsHeight]);\n                }\n            });\n\n        $('#largeVideoContainer').animate({\n                width: videospaceWidth,\n                height: videospaceHeight\n            },\n            {\n                queue: false,\n                duration: 500\n            });\n\n        $('#largeVideo').animate({\n                width: videoWidth,\n                height: videoHeight,\n                top: verticalIndent,\n                bottom: verticalIndent,\n                left: horizontalIndent,\n                right: horizontalIndent\n            },\n            {\n                queue: false,\n                duration: 500\n            });\n    };\n\n    /**\n     * Toggles the windows in the side panel\n     * @param object the window that should be shown\n     * @param selector the selector for the element containing the panel\n     * @param onOpenComplete function to be called when the panel is opened\n     * @param onOpen function to be called if the window is going to be opened\n     * @param onClose function to be called if the window is going to be closed\n     */\n    var toggle = function(object, selector, onOpenComplete, onOpen, onClose) {\n        UIUtil.buttonClick(buttons[selector], \"active\");\n\n        if (object.isVisible()) {\n            $(\"#toast-container\").animate({\n                    right: '5px'\n                },\n                {\n                    queue: false,\n                    duration: 500\n                });\n            $(selector).hide(\"slide\", {\n                direction: \"right\",\n                queue: false,\n                duration: 500\n            });\n            if(typeof onClose === \"function\") {\n                onClose();\n            }\n\n            currentlyOpen = null;\n        }\n        else {\n            // Undock the toolbar when the chat is shown and if we're in a\n            // video mode.\n            if (VideoLayout.isLargeVideoVisible()) {\n                ToolbarToggler.dockToolbar(false);\n            }\n\n            if(currentlyOpen) {\n                var current = $(currentlyOpen);\n                UIUtil.buttonClick(buttons[currentlyOpen], \"active\");\n                current.css('z-index', 4);\n                setTimeout(function () {\n                    current.css('display', 'none');\n                    current.css('z-index', 5);\n                }, 500);\n            }\n\n            $(\"#toast-container\").animate({\n                    right: (PanelToggler.getPanelSize()[0] + 5) + 'px'\n                },\n                {\n                    queue: false,\n                    duration: 500\n                });\n            $(selector).show(\"slide\", {\n                direction: \"right\",\n                queue: false,\n                duration: 500,\n                complete: onOpenComplete\n            });\n            if(typeof onOpen === \"function\") {\n                onOpen();\n            }\n\n            currentlyOpen = selector;\n        }\n    };\n\n    /**\n     * Opens / closes the chat area.\n     */\n    my.toggleChat = function() {\n        var chatCompleteFunction = Chat.isVisible() ?\n            function() {} : function () {\n            Chat.scrollChatToBottom();\n            $('#chatspace').trigger('shown');\n        };\n\n        resizeVideoArea(Chat.isVisible(), chatCompleteFunction);\n\n        toggle(Chat,\n            '#chatspace',\n            function () {\n                // Request the focus in the nickname field or the chat input field.\n                if ($('#nickname').css('visibility') === 'visible') {\n                    $('#nickinput').focus();\n                } else {\n                    $('#usermsg').focus();\n                }\n            },\n            null,\n            Chat.resizeChat,\n            null);\n    };\n\n    /**\n     * Opens / closes the contact list area.\n     */\n    my.toggleContactList = function () {\n        var completeFunction = ContactList.isVisible() ?\n            function() {} : function () { $('#contactlist').trigger('shown');};\n        resizeVideoArea(ContactList.isVisible(), completeFunction);\n\n        toggle(ContactList,\n            '#contactlist',\n            null,\n            function() {\n                ContactList.setVisualNotification(false);\n            },\n            null);\n    };\n\n    /**\n     * Opens / closes the settings menu\n     */\n    my.toggleSettingsMenu = function() {\n        resizeVideoArea(SettingsMenu.isVisible(), function (){});\n        toggle(SettingsMenu,\n            '#settingsmenu',\n            null,\n            function() {\n                var settings = Settings.getSettings();\n                $('#setDisplayName').get(0).value = settings.displayName;\n                $('#setEmail').get(0).value = settings.email;\n            },\n            null);\n    };\n\n    /**\n     * Returns the size of the side panel.\n     */\n    my.getPanelSize = function () {\n        var availableHeight = window.innerHeight;\n        var availableWidth = window.innerWidth;\n\n        var panelWidth = 200;\n        if (availableWidth * 0.2 < 200) {\n            panelWidth = availableWidth * 0.2;\n        }\n\n        return [panelWidth, availableHeight];\n    };\n\n    my.isVisible = function() {\n        return (Chat.isVisible() || ContactList.isVisible() || SettingsMenu.isVisible());\n    };\n\n    return my;\n\n}(PanelToggler || {}));\n\nmodule.exports = PanelToggler;","/* global $, Util, nickname:true, showToolbar */\nvar Replacement = require(\"./Replacement\");\nvar CommandsProcessor = require(\"./Commands\");\nvar ToolbarToggler = require(\"../../toolbars/ToolbarToggler\");\nvar smileys = require(\"./smileys.json\").smileys;\n\nvar notificationInterval = false;\nvar unreadMessages = 0;\n\n\n/**\n * Shows/hides a visual notification, indicating that a message has arrived.\n */\nfunction setVisualNotification(show) {\n    var unreadMsgElement = document.getElementById('unreadMessages');\n    var unreadMsgBottomElement\n        = document.getElementById('bottomUnreadMessages');\n\n    var glower = $('#chatButton');\n    var bottomGlower = $('#chatBottomButton');\n\n    if (unreadMessages) {\n        unreadMsgElement.innerHTML = unreadMessages.toString();\n        unreadMsgBottomElement.innerHTML = unreadMessages.toString();\n\n        ToolbarToggler.dockToolbar(true);\n\n        var chatButtonElement\n            = document.getElementById('chatButton').parentNode;\n        var leftIndent = (Util.getTextWidth(chatButtonElement) -\n            Util.getTextWidth(unreadMsgElement)) / 2;\n        var topIndent = (Util.getTextHeight(chatButtonElement) -\n            Util.getTextHeight(unreadMsgElement)) / 2 - 3;\n\n        unreadMsgElement.setAttribute(\n            'style',\n                'top:' + topIndent +\n                '; left:' + leftIndent + ';');\n\n        var chatBottomButtonElement\n            = document.getElementById('chatBottomButton').parentNode;\n        var bottomLeftIndent = (Util.getTextWidth(chatBottomButtonElement) -\n            Util.getTextWidth(unreadMsgBottomElement)) / 2;\n        var bottomTopIndent = (Util.getTextHeight(chatBottomButtonElement) -\n            Util.getTextHeight(unreadMsgBottomElement)) / 2 - 2;\n\n        unreadMsgBottomElement.setAttribute(\n            'style',\n                'top:' + bottomTopIndent +\n                '; left:' + bottomLeftIndent + ';');\n\n\n        if (!glower.hasClass('icon-chat-simple')) {\n            glower.removeClass('icon-chat');\n            glower.addClass('icon-chat-simple');\n        }\n    }\n    else {\n        unreadMsgElement.innerHTML = '';\n        unreadMsgBottomElement.innerHTML = '';\n        glower.removeClass('icon-chat-simple');\n        glower.addClass('icon-chat');\n    }\n\n    if (show && !notificationInterval) {\n        notificationInterval = window.setInterval(function () {\n            glower.toggleClass('active');\n            bottomGlower.toggleClass('active glowing');\n        }, 800);\n    }\n    else if (!show && notificationInterval) {\n        window.clearInterval(notificationInterval);\n        notificationInterval = false;\n        glower.removeClass('active');\n        bottomGlower.removeClass('glowing');\n        bottomGlower.addClass('active');\n    }\n}\n\n\n/**\n * Returns the current time in the format it is shown to the user\n * @returns {string}\n */\nfunction getCurrentTime() {\n    var now     = new Date();\n    var hour    = now.getHours();\n    var minute  = now.getMinutes();\n    var second  = now.getSeconds();\n    if(hour.toString().length === 1) {\n        hour = '0'+hour;\n    }\n    if(minute.toString().length === 1) {\n        minute = '0'+minute;\n    }\n    if(second.toString().length === 1) {\n        second = '0'+second;\n    }\n    return hour+':'+minute+':'+second;\n}\n\nfunction toggleSmileys()\n{\n    var smileys = $('#smileysContainer');\n    if(!smileys.is(':visible')) {\n        smileys.show(\"slide\", { direction: \"down\", duration: 300});\n    } else {\n        smileys.hide(\"slide\", { direction: \"down\", duration: 300});\n    }\n    $('#usermsg').focus();\n}\n\nfunction addClickFunction(smiley, number) {\n    smiley.onclick = function addSmileyToMessage() {\n        var usermsg = $('#usermsg');\n        var message = usermsg.val();\n        message += smileys['smiley' + number];\n        usermsg.val(message);\n        usermsg.get(0).setSelectionRange(message.length, message.length);\n        toggleSmileys();\n        usermsg.focus();\n    };\n}\n\n/**\n * Adds the smileys container to the chat\n */\nfunction addSmileys() {\n    var smileysContainer = document.createElement('div');\n    smileysContainer.id = 'smileysContainer';\n    for(var i = 1; i <= 21; i++) {\n        var smileyContainer = document.createElement('div');\n        smileyContainer.id = 'smiley' + i;\n        smileyContainer.className = 'smileyContainer';\n        var smiley = document.createElement('img');\n        smiley.src = 'images/smileys/smiley' + i + '.svg';\n        smiley.className =  'smiley';\n        addClickFunction(smiley, i);\n        smileyContainer.appendChild(smiley);\n        smileysContainer.appendChild(smileyContainer);\n    }\n\n    $(\"#chatspace\").append(smileysContainer);\n}\n\n/**\n * Resizes the chat conversation.\n */\nfunction resizeChatConversation() {\n    var msgareaHeight = $('#usermsg').outerHeight();\n    var chatspace = $('#chatspace');\n    var width = chatspace.width();\n    var chat = $('#chatconversation');\n    var smileys = $('#smileysarea');\n\n    smileys.height(msgareaHeight);\n    $(\"#smileys\").css('bottom', (msgareaHeight - 26) / 2);\n    $('#smileysContainer').css('bottom', msgareaHeight);\n    chat.width(width - 10);\n    chat.height(window.innerHeight - 15 - msgareaHeight);\n}\n\n/**\n * Chat related user interface.\n */\nvar Chat = (function (my) {\n    /**\n     * Initializes chat related interface.\n     */\n    my.init = function () {\n        var storedDisplayName = window.localStorage.displayname;\n        if (storedDisplayName) {\n            nickname = storedDisplayName;\n\n            Chat.setChatConversationMode(true);\n        }\n\n        $('#nickinput').keydown(function (event) {\n            if (event.keyCode === 13) {\n                event.preventDefault();\n                var val = Util.escapeHtml(this.value);\n                this.value = '';\n                if (!nickname) {\n                    nickname = val;\n                    window.localStorage.displayname = nickname;\n\n                    xmpp.addToPresence(\"displayName\", nickname);\n\n                    Chat.setChatConversationMode(true);\n\n                    return;\n                }\n            }\n        });\n\n        $('#usermsg').keydown(function (event) {\n            if (event.keyCode === 13) {\n                event.preventDefault();\n                var value = this.value;\n                $('#usermsg').val('').trigger('autosize.resize');\n                this.focus();\n                var command = new CommandsProcessor(value);\n                if(command.isCommand())\n                {\n                    command.processCommand();\n                }\n                else\n                {\n                    var message = Util.escapeHtml(value);\n                    xmpp.sendChatMessage(message, nickname);\n                }\n            }\n        });\n\n        var onTextAreaResize = function () {\n            resizeChatConversation();\n            Chat.scrollChatToBottom();\n        };\n        $('#usermsg').autosize({callback: onTextAreaResize});\n\n        $(\"#chatspace\").bind(\"shown\",\n            function () {\n                unreadMessages = 0;\n                setVisualNotification(false);\n            });\n\n        addSmileys();\n    };\n\n    /**\n     * Appends the given message to the chat conversation.\n     */\n    my.updateChatConversation = function (from, displayName, message) {\n        var divClassName = '';\n\n        if (xmpp.myJid() === from) {\n            divClassName = \"localuser\";\n        }\n        else {\n            divClassName = \"remoteuser\";\n\n            if (!Chat.isVisible()) {\n                unreadMessages++;\n                Util.playSoundNotification('chatNotification');\n                setVisualNotification(true);\n            }\n        }\n\n        // replace links and smileys\n        // Strophe already escapes special symbols on sending,\n        // so we escape here only tags to avoid double &amp;\n        var escMessage = message.replace(/</g, '&lt;').\n            replace(/>/g, '&gt;').replace(/\\n/g, '<br/>');\n        var escDisplayName = Util.escapeHtml(displayName);\n        message = Replacement.processReplacements(escMessage);\n\n        var messageContainer =\n            '<div class=\"chatmessage\">'+\n                '<img src=\"../images/chatArrow.svg\" class=\"chatArrow\">' +\n                '<div class=\"username ' + divClassName +'\">' + escDisplayName +\n                '</div>' + '<div class=\"timestamp\">' + getCurrentTime() +\n                '</div>' + '<div class=\"usermessage\">' + message + '</div>' +\n            '</div>';\n\n        $('#chatconversation').append(messageContainer);\n        $('#chatconversation').animate(\n                { scrollTop: $('#chatconversation')[0].scrollHeight}, 1000);\n    };\n\n    /**\n     * Appends error message to the conversation\n     * @param errorMessage the received error message.\n     * @param originalText the original message.\n     */\n    my.chatAddError = function(errorMessage, originalText)\n    {\n        errorMessage = Util.escapeHtml(errorMessage);\n        originalText = Util.escapeHtml(originalText);\n\n        $('#chatconversation').append(\n            '<div class=\"errorMessage\"><b>Error: </b>' + 'Your message' +\n            (originalText? (' \\\"'+ originalText + '\\\"') : \"\") +\n            ' was not sent.' +\n            (errorMessage? (' Reason: ' + errorMessage) : '') +  '</div>');\n        $('#chatconversation').animate(\n            { scrollTop: $('#chatconversation')[0].scrollHeight}, 1000);\n    };\n\n    /**\n     * Sets the subject to the UI\n     * @param subject the subject\n     */\n    my.chatSetSubject = function(subject)\n    {\n        if(subject)\n            subject = subject.trim();\n        $('#subject').html(Replacement.linkify(Util.escapeHtml(subject)));\n        if(subject === \"\")\n        {\n            $(\"#subject\").css({display: \"none\"});\n        }\n        else\n        {\n            $(\"#subject\").css({display: \"block\"});\n        }\n    };\n\n\n\n    /**\n     * Sets the chat conversation mode.\n     */\n    my.setChatConversationMode = function (isConversationMode) {\n        if (isConversationMode) {\n            $('#nickname').css({visibility: 'hidden'});\n            $('#chatconversation').css({visibility: 'visible'});\n            $('#usermsg').css({visibility: 'visible'});\n            $('#smileysarea').css({visibility: 'visible'});\n            $('#usermsg').focus();\n        }\n    };\n\n    /**\n     * Resizes the chat area.\n     */\n    my.resizeChat = function () {\n        var chatSize = require(\"../SidePanelToggler\").getPanelSize();\n\n        $('#chatspace').width(chatSize[0]);\n        $('#chatspace').height(chatSize[1]);\n\n        resizeChatConversation();\n    };\n\n    /**\n     * Indicates if the chat is currently visible.\n     */\n    my.isVisible = function () {\n        return $('#chatspace').is(\":visible\");\n    };\n    /**\n     * Shows and hides the window with the smileys\n     */\n    my.toggleSmileys = toggleSmileys;\n\n    /**\n     * Scrolls chat to the bottom.\n     */\n    my.scrollChatToBottom = function() {\n        setTimeout(function () {\n            $('#chatconversation').scrollTop(\n                $('#chatconversation')[0].scrollHeight);\n        }, 5);\n    };\n\n\n    return my;\n}(Chat || {}));\nmodule.exports = Chat;","/**\n * List with supported commands. The keys are the names of the commands and\n * the value is the function that processes the message.\n * @type {{String: function}}\n */\nvar commands = {\n    \"topic\" : processTopic\n};\n\n/**\n * Extracts the command from the message.\n * @param message the received message\n * @returns {string} the command\n */\nfunction getCommand(message)\n{\n    if(message)\n    {\n        for(var command in commands)\n        {\n            if(message.indexOf(\"/\" + command) == 0)\n                return command;\n        }\n    }\n    return \"\";\n};\n\n/**\n * Processes the data for topic command.\n * @param commandArguments the arguments of the topic command.\n */\nfunction processTopic(commandArguments)\n{\n    var topic = Util.escapeHtml(commandArguments);\n    xmpp.setSubject(topic);\n}\n\n/**\n * Constructs new CommandProccessor instance from a message that\n * handles commands received via chat messages.\n * @param message the message\n * @constructor\n */\nfunction CommandsProcessor(message)\n{\n\n\n    var command = getCommand(message);\n\n    /**\n     * Returns the name of the command.\n     * @returns {String} the command\n     */\n    this.getCommand = function()\n    {\n        return command;\n    };\n\n\n    var messageArgument = message.substr(command.length + 2);\n\n    /**\n     * Returns the arguments of the command.\n     * @returns {string}\n     */\n    this.getArgument = function()\n    {\n        return messageArgument;\n    };\n}\n\n/**\n * Checks whether this instance is valid command or not.\n * @returns {boolean}\n */\nCommandsProcessor.prototype.isCommand = function()\n{\n    if(this.getCommand())\n        return true;\n    return false;\n};\n\n/**\n * Processes the command.\n */\nCommandsProcessor.prototype.processCommand = function()\n{\n    if(!this.isCommand())\n        return;\n\n    commands[this.getCommand()](this.getArgument());\n\n};\n\nmodule.exports = CommandsProcessor;","var Smileys = require(\"./smileys.json\");\n/**\n * Processes links and smileys in \"body\"\n */\nfunction processReplacements(body)\n{\n    //make links clickable\n    body = linkify(body);\n\n    //add smileys\n    body = smilify(body);\n\n    return body;\n}\n\n/**\n * Finds and replaces all links in the links in \"body\"\n * with their <a href=\"\"></a>\n */\nfunction linkify(inputText)\n{\n    var replacedText, replacePattern1, replacePattern2, replacePattern3;\n\n    //URLs starting with http://, https://, or ftp://\n    replacePattern1 = /(\\b(https?|ftp):\\/\\/[-A-Z0-9+&@#\\/%?=~_|!:,.;]*[-A-Z0-9+&@#\\/%=~_|])/gim;\n    replacedText = inputText.replace(replacePattern1, '<a href=\"$1\" target=\"_blank\">$1</a>');\n\n    //URLs starting with \"www.\" (without // before it, or it'd re-link the ones done above).\n    replacePattern2 = /(^|[^\\/])(www\\.[\\S]+(\\b|$))/gim;\n    replacedText = replacedText.replace(replacePattern2, '$1<a href=\"http://$2\" target=\"_blank\">$2</a>');\n\n    //Change email addresses to mailto:: links.\n    replacePattern3 = /(([a-zA-Z0-9\\-\\_\\.])+@[a-zA-Z\\_]+?(\\.[a-zA-Z]{2,6})+)/gim;\n    replacedText = replacedText.replace(replacePattern3, '<a href=\"mailto:$1\">$1</a>');\n\n    return replacedText;\n}\n\n/**\n * Replaces common smiley strings with images\n */\nfunction smilify(body)\n{\n    if(!body) {\n        return body;\n    }\n\n    var regexs = Smileys[\"regexs\"];\n    for(var smiley in regexs) {\n        if(regexs.hasOwnProperty(smiley)) {\n            body = body.replace(regexs[smiley],\n                    '<img class=\"smiley\" src=\"images/smileys/' + smiley + '.svg\">');\n        }\n    }\n\n    return body;\n}\n\nmodule.exports = {\n    processReplacements: processReplacements,\n    linkify: linkify\n};\n","module.exports={\n    \"smileys\": {\n        \"smiley1\": \":)\",\n        \"smiley2\": \":(\",\n        \"smiley3\": \":D\",\n        \"smiley4\": \"(y)\",\n        \"smiley5\": \" :P\",\n        \"smiley6\": \"(wave)\",\n        \"smiley7\": \"(blush)\",\n        \"smiley8\": \"(chuckle)\",\n        \"smiley9\": \"(shocked)\",\n        \"smiley10\": \":*\",\n        \"smiley11\": \"(n)\",\n        \"smiley12\": \"(search)\",\n        \"smiley13\": \" <3\",\n        \"smiley14\": \"(oops)\",\n        \"smiley15\": \"(angry)\",\n        \"smiley16\": \"(angel)\",\n        \"smiley17\": \"(sick)\",\n        \"smiley18\": \";(\",\n        \"smiley19\": \"(bomb)\",\n        \"smiley20\": \"(clap)\",\n        \"smiley21\": \" ;)\"\n    },\n    \"regexs\": {\n        \"smiley2\": /(:-\\(\\(|:-\\(|:\\(\\(|:\\(|\\(sad\\))/gi,\n        \"smiley3\": /(:-\\)\\)|:\\)\\)|\\(lol\\)|:-D|:D)/gi,\n        \"smiley1\": /(:-\\)|:\\))/gi,\n        \"smiley4\": /(\\(y\\)|\\(Y\\)|\\(ok\\))/gi,\n        \"smiley5\": /(:-P|:P|:-p|:p)/gi,\n        \"smiley6\": /(\\(wave\\))/gi,\n        \"smiley7\": /(\\(blush\\))/gi,\n        \"smiley8\": /(\\(chuckle\\))/gi,\n        \"smiley9\": /(:-0|\\(shocked\\))/gi,\n        \"smiley10\": /(:-\\*|:\\*|\\(kiss\\))/gi,\n        \"smiley11\": /(\\(n\\))/gi,\n        \"smiley12\": /(\\(search\\))/g,\n        \"smiley13\": /(<3|&lt;3|&amp;lt;3|\\(L\\)|\\(l\\)|\\(H\\)|\\(h\\))/gi,\n        \"smiley14\": /(\\(oops\\))/gi,\n        \"smiley15\": /(\\(angry\\))/gi,\n        \"smiley16\": /(\\(angel\\))/gi,\n        \"smiley17\": /(\\(sick\\))/gi,\n        \"smiley18\": /(;-\\(\\(|;\\(\\(|;-\\(|;\\(|:\"\\(|:\"-\\(|:~-\\(|:~\\(|\\(upset\\))/gi,\n        \"smiley19\": /(\\(bomb\\))/gi,\n        \"smiley20\": /(\\(clap\\))/gi,\n        \"smiley21\": /(;-\\)|;\\)|;-\\)\\)|;\\)\\)|;-D|;D|\\(wink\\))/gi\n    }\n}\n","\nvar numberOfContacts = 0;\nvar notificationInterval;\n\n/**\n * Updates the number of participants in the contact list button and sets\n * the glow\n * @param delta indicates whether a new user has joined (1) or someone has\n * left(-1)\n */\nfunction updateNumberOfParticipants(delta) {\n    //when the user is alone we don't show the number of participants\n    if(numberOfContacts === 0) {\n        $(\"#numberOfParticipants\").text('');\n        numberOfContacts += delta;\n    } else if(numberOfContacts !== 0 && !ContactList.isVisible()) {\n        ContactList.setVisualNotification(true);\n        numberOfContacts += delta;\n        $(\"#numberOfParticipants\").text(numberOfContacts);\n    }\n}\n\n/**\n * Creates the avatar element.\n *\n * @return the newly created avatar element\n */\nfunction createAvatar(id) {\n    var avatar = document.createElement('img');\n    avatar.className = \"icon-avatar avatar\";\n    avatar.src = \"https://www.gravatar.com/avatar/\" + id + \"?d=wavatar&size=30\";\n\n    return avatar;\n}\n\n/**\n * Creates the display name paragraph.\n *\n * @param displayName the display name to set\n */\nfunction createDisplayNameParagraph(displayName) {\n    var p = document.createElement('p');\n    p.innerText = displayName;\n\n    return p;\n}\n\n\nfunction stopGlowing(glower) {\n    window.clearInterval(notificationInterval);\n    notificationInterval = false;\n    glower.removeClass('glowing');\n    if (!ContactList.isVisible()) {\n        glower.removeClass('active');\n    }\n}\n\n\n/**\n * Contact list.\n */\nvar ContactList = {\n    /**\n     * Indicates if the chat is currently visible.\n     *\n     * @return <tt>true</tt> if the chat is currently visible, <tt>false</tt> -\n     * otherwise\n     */\n    isVisible: function () {\n        return $('#contactlist').is(\":visible\");\n    },\n\n    /**\n     * Adds a contact for the given peerJid if such doesn't yet exist.\n     *\n     * @param peerJid the peerJid corresponding to the contact\n     * @param id the user's email or userId used to get the user's avatar\n     */\n    ensureAddContact: function (peerJid, id) {\n        var resourceJid = Strophe.getResourceFromJid(peerJid);\n\n        var contact = $('#contactlist>ul>li[id=\"' + resourceJid + '\"]');\n\n        if (!contact || contact.length <= 0)\n            ContactList.addContact(peerJid, id);\n    },\n\n    /**\n     * Adds a contact for the given peer jid.\n     *\n     * @param peerJid the jid of the contact to add\n     * @param id the email or userId of the user\n     */\n    addContact: function (peerJid, id) {\n        var resourceJid = Strophe.getResourceFromJid(peerJid);\n\n        var contactlist = $('#contactlist>ul');\n\n        var newContact = document.createElement('li');\n        newContact.id = resourceJid;\n        newContact.className = \"clickable\";\n        newContact.onclick = function (event) {\n            if (event.currentTarget.className === \"clickable\") {\n                $(ContactList).trigger('contactclicked', [peerJid]);\n            }\n        };\n\n        newContact.appendChild(createAvatar(id));\n        newContact.appendChild(createDisplayNameParagraph(\"Participant\"));\n\n        var clElement = contactlist.get(0);\n\n        if (resourceJid === xmpp.myResource()\n            && $('#contactlist>ul .title')[0].nextSibling.nextSibling) {\n            clElement.insertBefore(newContact,\n                $('#contactlist>ul .title')[0].nextSibling.nextSibling);\n        }\n        else {\n            clElement.appendChild(newContact);\n        }\n        updateNumberOfParticipants(1);\n    },\n\n    /**\n     * Removes a contact for the given peer jid.\n     *\n     * @param peerJid the peerJid corresponding to the contact to remove\n     */\n    removeContact: function (peerJid) {\n        var resourceJid = Strophe.getResourceFromJid(peerJid);\n\n        var contact = $('#contactlist>ul>li[id=\"' + resourceJid + '\"]');\n\n        if (contact && contact.length > 0) {\n            var contactlist = $('#contactlist>ul');\n\n            contactlist.get(0).removeChild(contact.get(0));\n\n            updateNumberOfParticipants(-1);\n        }\n    },\n\n    setVisualNotification: function (show, stopGlowingIn) {\n        var glower = $('#contactListButton');\n\n        if (show && !notificationInterval) {\n            notificationInterval = window.setInterval(function () {\n                glower.toggleClass('active glowing');\n            }, 800);\n        }\n        else if (!show && notificationInterval) {\n            stopGlowing(glower);\n        }\n        if (stopGlowingIn) {\n            setTimeout(function () {\n                stopGlowing(glower);\n            }, stopGlowingIn);\n        }\n    },\n\n    setClickable: function (resourceJid, isClickable) {\n        var contact = $('#contactlist>ul>li[id=\"' + resourceJid + '\"]');\n        if (isClickable) {\n            contact.addClass('clickable');\n        } else {\n            contact.removeClass('clickable');\n        }\n    },\n\n    onDisplayNameChange: function (peerJid, displayName) {\n        if (peerJid === 'localVideoContainer')\n            peerJid = xmpp.myJid();\n\n        var resourceJid = Strophe.getResourceFromJid(peerJid);\n\n        var contactName = $('#contactlist #' + resourceJid + '>p');\n\n        if (contactName && displayName && displayName.length > 0)\n            contactName.html(displayName);\n    }\n};\n\nmodule.exports = ContactList;","var email = '';\nvar displayName = '';\nvar userId;\n\n\nfunction supportsLocalStorage() {\n    try {\n        return 'localStorage' in window && window.localStorage !== null;\n    } catch (e) {\n        console.log(\"localstorage is not supported\");\n        return false;\n    }\n}\n\n\nfunction generateUniqueId() {\n    function _p8() {\n        return (Math.random().toString(16)+\"000000000\").substr(2,8);\n    }\n    return _p8() + _p8() + _p8() + _p8();\n}\n\nif(supportsLocalStorage()) {\n    if(!window.localStorage.jitsiMeetId) {\n        window.localStorage.jitsiMeetId = generateUniqueId();\n        console.log(\"generated id\", window.localStorage.jitsiMeetId);\n    }\n    userId = window.localStorage.jitsiMeetId || '';\n    email = window.localStorage.email || '';\n    displayName = window.localStorage.displayname || '';\n} else {\n    console.log(\"local storage is not supported\");\n    userId = generateUniqueId();\n}\n\nvar Settings =\n{\n    setDisplayName: function (newDisplayName) {\n        displayName = newDisplayName;\n        window.localStorage.displayname = displayName;\n        return displayName;\n    },\n    setEmail: function(newEmail)\n    {\n        email = newEmail;\n        window.localStorage.email = newEmail;\n        return email;\n    },\n    getSettings: function () {\n        return {\n            email: email,\n            displayName: displayName,\n            uid: userId\n        };\n    }\n};\n\nmodule.exports = Settings;\n","var Avatar = require(\"../../avatar/Avatar\");\nvar Settings = require(\"./Settings\");\n\n\nvar SettingsMenu = {\n\n    update: function() {\n        var newDisplayName = Util.escapeHtml($('#setDisplayName').get(0).value);\n        var newEmail = Util.escapeHtml($('#setEmail').get(0).value);\n\n        if(newDisplayName) {\n            var displayName = Settings.setDisplayName(newDisplayName);\n            xmpp.addToPresence(\"displayName\", displayName, true);\n        }\n\n\n        xmpp.addToPresence(\"email\", newEmail);\n        var email = Settings.setEmail(newEmail);\n\n\n        Avatar.setUserAvatar(xmpp.myJid(), email);\n    },\n\n    isVisible: function() {\n        return $('#settingsmenu').is(':visible');\n    },\n\n    setDisplayName: function(newDisplayName) {\n        var displayName = Settings.setDisplayName(newDisplayName);\n        $('#setDisplayName').get(0).value = displayName;\n    },\n\n    onDisplayNameChange: function(peerJid, newDisplayName) {\n        if(peerJid === 'localVideoContainer' ||\n            peerJid === xmpp.myJid()) {\n            this.setDisplayName(newDisplayName);\n        }\n    }\n};\n\n\nmodule.exports = SettingsMenu;","var PanelToggler = require(\"../side_pannels/SidePanelToggler\");\n\nvar buttonHandlers = {\n    \"bottom_toolbar_contact_list\": function () {\n        BottomToolbar.toggleContactList();\n    },\n    \"bottom_toolbar_film_strip\": function () {\n        BottomToolbar.toggleFilmStrip();\n    },\n    \"bottom_toolbar_chat\": function () {\n        BottomToolbar.toggleChat();\n    }\n};\n\nvar BottomToolbar = (function (my) {\n    my.init = function () {\n        for(var k in buttonHandlers)\n            $(\"#\" + k).click(buttonHandlers[k]);\n    };\n\n    my.toggleChat = function() {\n        PanelToggler.toggleChat();\n    };\n\n    my.toggleContactList = function() {\n        PanelToggler.toggleContactList();\n    };\n\n    my.toggleFilmStrip = function() {\n        var filmstrip = $(\"#remoteVideos\");\n        filmstrip.toggleClass(\"hidden\");\n    };\n\n    $(document).bind(\"remotevideo.resized\", function (event, width, height) {\n        var bottom = (height - $('#bottomToolbar').outerHeight())/2 + 18;\n\n        $('#bottomToolbar').css({bottom: bottom + 'px'});\n    });\n\n    return my;\n}(BottomToolbar || {}));\n\nmodule.exports = BottomToolbar;\n","/* global $, interfaceConfig, Moderator, DesktopStreaming.showDesktopSharingButton */\n\nvar toolbarTimeoutObject,\n    toolbarTimeout = interfaceConfig.INITIAL_TOOLBAR_TIMEOUT;\n\nfunction showDesktopSharingButton() {\n    if (desktopsharing.isDesktopSharingEnabled()) {\n        $('#desktopsharing').css({display: \"inline\"});\n    } else {\n        $('#desktopsharing').css({display: \"none\"});\n    }\n}\n\n/**\n * Hides the toolbar.\n */\nfunction hideToolbar() {\n    var header = $(\"#header\"),\n        bottomToolbar = $(\"#bottomToolbar\");\n    var isToolbarHover = false;\n    header.find('*').each(function () {\n        var id = $(this).attr('id');\n        if ($(\"#\" + id + \":hover\").length > 0) {\n            isToolbarHover = true;\n        }\n    });\n    if ($(\"#bottomToolbar:hover\").length > 0) {\n        isToolbarHover = true;\n    }\n\n    clearTimeout(toolbarTimeoutObject);\n    toolbarTimeoutObject = null;\n\n    if (!isToolbarHover) {\n        header.hide(\"slide\", { direction: \"up\", duration: 300});\n        $('#subject').animate({top: \"-=40\"}, 300);\n        if ($(\"#remoteVideos\").hasClass(\"hidden\")) {\n            bottomToolbar.hide(\n                \"slide\", {direction: \"right\", duration: 300});\n        }\n    }\n    else {\n        toolbarTimeoutObject = setTimeout(hideToolbar, toolbarTimeout);\n    }\n}\n\nvar ToolbarToggler = {\n    /**\n     * Shows the main toolbar.\n     */\n    showToolbar: function () {\n        var header = $(\"#header\"),\n            bottomToolbar = $(\"#bottomToolbar\");\n        if (!header.is(':visible') || !bottomToolbar.is(\":visible\")) {\n            header.show(\"slide\", { direction: \"up\", duration: 300});\n            $('#subject').animate({top: \"+=40\"}, 300);\n            if (!bottomToolbar.is(\":visible\")) {\n                bottomToolbar.show(\n                    \"slide\", {direction: \"right\", duration: 300});\n            }\n\n            if (toolbarTimeoutObject) {\n                clearTimeout(toolbarTimeoutObject);\n                toolbarTimeoutObject = null;\n            }\n            toolbarTimeoutObject = setTimeout(hideToolbar, toolbarTimeout);\n            toolbarTimeout = interfaceConfig.TOOLBAR_TIMEOUT;\n        }\n\n        if (xmpp.isModerator())\n        {\n//            TODO: Enable settings functionality.\n//                  Need to uncomment the settings button in index.html.\n//            $('#settingsButton').css({visibility:\"visible\"});\n        }\n\n        // Show/hide desktop sharing button\n        showDesktopSharingButton();\n    },\n\n\n    /**\n     * Docks/undocks the toolbar.\n     *\n     * @param isDock indicates what operation to perform\n     */\n    dockToolbar: function (isDock) {\n        if (isDock) {\n            // First make sure the toolbar is shown.\n            if (!$('#header').is(':visible')) {\n                this.showToolbar();\n            }\n\n            // Then clear the time out, to dock the toolbar.\n            if (toolbarTimeoutObject) {\n                clearTimeout(toolbarTimeoutObject);\n                toolbarTimeoutObject = null;\n            }\n        }\n        else {\n            if (!$('#header').is(':visible')) {\n                this.showToolbar();\n            }\n            else {\n                toolbarTimeoutObject = setTimeout(hideToolbar, toolbarTimeout);\n            }\n        }\n    },\n\n    showDesktopSharingButton: showDesktopSharingButton\n\n};\n\nmodule.exports = ToolbarToggler;","/* global $, buttonClick, config, lockRoom,\n   setSharedKey, Util */\nvar messageHandler = require(\"../util/MessageHandler\");\nvar BottomToolbar = require(\"./BottomToolbar\");\nvar Prezi = require(\"../prezi/Prezi\");\nvar Etherpad = require(\"../etherpad/Etherpad\");\nvar PanelToggler = require(\"../side_pannels/SidePanelToggler\");\nvar Authentication = require(\"../authentication/Authentication\");\nvar UIUtil = require(\"../util/UIUtil\");\n\nvar roomUrl = null;\nvar sharedKey = '';\nvar UI = null;\n\nvar buttonHandlers =\n{\n    \"toolbar_button_mute\": function () {\n        return UI.toggleAudio();\n    },\n    \"toolbar_button_camera\": function () {\n        return UI.toggleVideo();\n    },\n    \"toolbar_button_authentication\": function () {\n        return Toolbar.authenticateClicked();\n    },\n    \"toolbar_button_record\": function () {\n        return toggleRecording();\n    },\n    \"toolbar_button_security\": function () {\n        return Toolbar.openLockDialog();\n    },\n    \"toolbar_button_link\": function () {\n        return Toolbar.openLinkDialog();\n    },\n    \"toolbar_button_chat\": function () {\n        return BottomToolbar.toggleChat();\n    },\n    \"toolbar_button_prezi\": function () {\n        return Prezi.openPreziDialog();\n    },\n    \"toolbar_button_etherpad\": function () {\n        return Etherpad.toggleEtherpad(0);\n    },\n    \"toolbar_button_desktopsharing\": function () {\n        return desktopsharing.toggleScreenSharing();\n    },\n    \"toolbar_button_fullScreen\": function()\n    {\n        UIUtil.buttonClick(\"#fullScreen\", \"icon-full-screen icon-exit-full-screen\");\n        return Toolbar.toggleFullScreen();\n    },\n    \"toolbar_button_sip\": function () {\n        return callSipButtonClicked();\n    },\n    \"toolbar_button_settings\": function () {\n        PanelToggler.toggleSettingsMenu();\n    },\n    \"toolbar_button_hangup\": function () {\n        return hangup();\n    }\n};\n\nfunction hangup() {\n    xmpp.disposeConference();\n    if(config.enableWelcomePage)\n    {\n        setTimeout(function()\n        {\n            window.localStorage.welcomePageDisabled = false;\n            window.location.pathname = \"/\";\n        }, 10000);\n\n    }\n\n    UI.messageHandler.openDialog(\n        \"Session Terminated\",\n        \"You hung up the call\",\n        true,\n        { \"Join again\": true },\n        function(event, value, message, formVals)\n        {\n            window.location.reload();\n            return false;\n        }\n    );\n}\n\n/**\n * Starts or stops the recording for the conference.\n */\n\nfunction toggleRecording() {\n    xmpp.toggleRecording(function (callback) {\n        UI.messageHandler.openTwoButtonDialog(null,\n                '<h2>Enter recording token</h2>' +\n                '<input id=\"recordingToken\" type=\"text\" ' +\n                'placeholder=\"token\" autofocus>',\n            false,\n            \"Save\",\n            function (e, v, m, f) {\n                if (v) {\n                    var token = document.getElementById('recordingToken');\n\n                    if (token.value) {\n                        callback(Util.escapeHtml(token.value));\n                    }\n                }\n            },\n            function (event) {\n                document.getElementById('recordingToken').focus();\n            },\n            function () {\n            }\n        );\n    }, Toolbar.setRecordingButtonState, Toolbar.setRecordingButtonState);\n}\n\n/**\n * Locks / unlocks the room.\n */\nfunction lockRoom(lock) {\n    var currentSharedKey = '';\n    if (lock)\n        currentSharedKey = sharedKey;\n\n    xmpp.lockRoom(currentSharedKey, function (res) {\n        // password is required\n        if (sharedKey)\n        {\n            console.log('set room password');\n            Toolbar.lockLockButton();\n        }\n        else\n        {\n            console.log('removed room password');\n            Toolbar.unlockLockButton();\n        }\n    }, function (err) {\n        console.warn('setting password failed', err);\n        messageHandler.showError('Lock failed',\n            'Failed to lock conference.',\n            err);\n        Toolbar.setSharedKey('');\n    }, function () {\n        console.warn('room passwords not supported');\n        messageHandler.showError('Warning',\n            'Room passwords are currently not supported.');\n        Toolbar.setSharedKey('');\n    });\n};\n\n/**\n * Invite participants to conference.\n */\nfunction inviteParticipants() {\n    if (roomUrl === null)\n        return;\n\n    var sharedKeyText = \"\";\n    if (sharedKey && sharedKey.length > 0) {\n        sharedKeyText =\n            \"This conference is password protected. Please use the \" +\n            \"following pin when joining:%0D%0A%0D%0A\" +\n            sharedKey + \"%0D%0A%0D%0A\";\n    }\n\n    var conferenceName = roomUrl.substring(roomUrl.lastIndexOf('/') + 1);\n    var subject = \"Invitation to a \" + interfaceConfig.APP_NAME + \" (\" + conferenceName + \")\";\n    var body = \"Hey there, I%27d like to invite you to a \" + interfaceConfig.APP_NAME +\n        \" conference I%27ve just set up.%0D%0A%0D%0A\" +\n        \"Please click on the following link in order\" +\n        \" to join the conference.%0D%0A%0D%0A\" +\n        roomUrl +\n        \"%0D%0A%0D%0A\" +\n        sharedKeyText +\n        \"Note that \" + interfaceConfig.APP_NAME + \" is currently\" +\n        \" only supported by Chromium,\" +\n        \" Google Chrome and Opera, so you need\" +\n        \" to be using one of these browsers.%0D%0A%0D%0A\" +\n        \"Talk to you in a sec!\";\n\n    if (window.localStorage.displayname) {\n        body += \"%0D%0A%0D%0A\" + window.localStorage.displayname;\n    }\n\n    if (interfaceConfig.INVITATION_POWERED_BY) {\n        body += \"%0D%0A%0D%0A--%0D%0Apowered by jitsi.org\";\n    }\n\n    window.open(\"mailto:?subject=\" + subject + \"&body=\" + body, '_blank');\n}\n\nfunction callSipButtonClicked()\n{\n    var defaultNumber\n        = config.defaultSipNumber ? config.defaultSipNumber : '';\n\n    messageHandler.openTwoButtonDialog(null,\n        '<h2>Enter SIP number</h2>' +\n        '<input id=\"sipNumber\" type=\"text\"' +\n        ' value=\"' + defaultNumber + '\" autofocus>',\n        false,\n        \"Dial\",\n        function (e, v, m, f) {\n            if (v) {\n                var numberInput = document.getElementById('sipNumber');\n                if (numberInput.value) {\n                    xmpp.dial(numberInput.value, 'fromnumber',\n                        UI.getRoomName(), sharedKey);\n                }\n            }\n        },\n        function (event) {\n            document.getElementById('sipNumber').focus();\n        }\n    );\n}\n\nvar Toolbar = (function (my) {\n\n    my.init = function (ui) {\n        for(var k in buttonHandlers)\n            $(\"#\" + k).click(buttonHandlers[k]);\n        UI = ui;\n    }\n\n    /**\n     * Sets shared key\n     * @param sKey the shared key\n     */\n    my.setSharedKey = function (sKey) {\n        sharedKey = sKey;\n    };\n\n    my.authenticateClicked = function () {\n        Authentication.focusAuthenticationWindow();\n        // Get authentication URL\n        xmpp.getAuthUrl(UI.getRoomName(), function (url) {\n            // Open popup with authentication URL\n            var authenticationWindow = Authentication.createAuthenticationWindow(function () {\n                // On popup closed - retry room allocation\n                xmpp.allocateConferenceFocus(UI.getRoomName(), UI.checkForNicknameAndJoin);\n            }, url);\n            if (!authenticationWindow) {\n                Toolbar.showAuthenticateButton(true);\n                messageHandler.openMessageDialog(\n                    null, \"Your browser is blocking popup windows from this site.\" +\n                        \" Please enable popups in your browser security settings\" +\n                        \" and try again.\");\n            }\n        });\n    };\n\n    /**\n     * Updates the room invite url.\n     */\n    my.updateRoomUrl = function (newRoomUrl) {\n        roomUrl = newRoomUrl;\n\n        // If the invite dialog has been already opened we update the information.\n        var inviteLink = document.getElementById('inviteLinkRef');\n        if (inviteLink) {\n            inviteLink.value = roomUrl;\n            inviteLink.select();\n            document.getElementById('jqi_state0_buttonInvite').disabled = false;\n        }\n    };\n\n    /**\n     * Disables and enables some of the buttons.\n     */\n    my.setupButtonsFromConfig = function () {\n        if (config.disablePrezi)\n        {\n            $(\"#prezi_button\").css({display: \"none\"});\n        }\n    };\n\n    /**\n     * Opens the lock room dialog.\n     */\n    my.openLockDialog = function () {\n        // Only the focus is able to set a shared key.\n        if (!xmpp.isModerator()) {\n            if (sharedKey) {\n                messageHandler.openMessageDialog(null,\n                        \"This conversation is currently protected by\" +\n                        \" a password. Only the owner of the conference\" +\n                        \" could set a password.\",\n                    false,\n                    \"Password\");\n            } else {\n                messageHandler.openMessageDialog(null,\n                    \"This conversation isn't currently protected by\" +\n                        \" a password. Only the owner of the conference\" +\n                        \" could set a password.\",\n                    false,\n                    \"Password\");\n            }\n        } else {\n            if (sharedKey) {\n                messageHandler.openTwoButtonDialog(null,\n                    \"Are you sure you would like to remove your password?\",\n                    false,\n                    \"Remove\",\n                    function (e, v) {\n                        if (v) {\n                            Toolbar.setSharedKey('');\n                            lockRoom(false);\n                        }\n                    });\n            } else {\n                messageHandler.openTwoButtonDialog(null,\n                    '<h2>Set a password to lock your room</h2>' +\n                        '<input id=\"lockKey\" type=\"text\"' +\n                        'placeholder=\"your password\" autofocus>',\n                    false,\n                    \"Save\",\n                    function (e, v) {\n                        if (v) {\n                            var lockKey = document.getElementById('lockKey');\n\n                            if (lockKey.value) {\n                                Toolbar.setSharedKey(Util.escapeHtml(lockKey.value));\n                                lockRoom(true);\n                            }\n                        }\n                    },\n                    function () {\n                        document.getElementById('lockKey').focus();\n                    }\n                );\n            }\n        }\n    };\n\n    /**\n     * Opens the invite link dialog.\n     */\n    my.openLinkDialog = function () {\n        var inviteLink;\n        if (roomUrl === null) {\n            inviteLink = \"Your conference is currently being created...\";\n        } else {\n            inviteLink = encodeURI(roomUrl);\n        }\n        messageHandler.openTwoButtonDialog(\n            \"Share this link with everyone you want to invite\",\n            '<input id=\"inviteLinkRef\" type=\"text\" value=\"' +\n                inviteLink + '\" onclick=\"this.select();\" readonly>',\n            false,\n            \"Invite\",\n            function (e, v) {\n                if (v) {\n                    if (roomUrl) {\n                        inviteParticipants();\n                    }\n                }\n            },\n            function () {\n                if (roomUrl) {\n                    document.getElementById('inviteLinkRef').select();\n                } else {\n                    document.getElementById('jqi_state0_buttonInvite')\n                        .disabled = true;\n                }\n            }\n        );\n    };\n\n    /**\n     * Opens the settings dialog.\n     */\n    my.openSettingsDialog = function () {\n        messageHandler.openTwoButtonDialog(\n            '<h2>Configure your conference</h2>' +\n                '<input type=\"checkbox\" id=\"initMuted\">' +\n                'Participants join muted<br/>' +\n                '<input type=\"checkbox\" id=\"requireNicknames\">' +\n                'Require nicknames<br/><br/>' +\n                'Set a password to lock your room:' +\n                '<input id=\"lockKey\" type=\"text\" placeholder=\"your password\"' +\n                'autofocus>',\n            null,\n            false,\n            \"Save\",\n            function () {\n                document.getElementById('lockKey').focus();\n            },\n            function (e, v) {\n                if (v) {\n                    if ($('#initMuted').is(\":checked\")) {\n                        // it is checked\n                    }\n\n                    if ($('#requireNicknames').is(\":checked\")) {\n                        // it is checked\n                    }\n                    /*\n                    var lockKey = document.getElementById('lockKey');\n\n                    if (lockKey.value) {\n                        setSharedKey(lockKey.value);\n                        lockRoom(true);\n                    }\n                    */\n                }\n            }\n        );\n    };\n\n    /**\n     * Toggles the application in and out of full screen mode\n     * (a.k.a. presentation mode in Chrome).\n     */\n    my.toggleFullScreen = function () {\n        var fsElement = document.documentElement;\n\n        if (!document.mozFullScreen && !document.webkitIsFullScreen) {\n            //Enter Full Screen\n            if (fsElement.mozRequestFullScreen) {\n                fsElement.mozRequestFullScreen();\n            }\n            else {\n                fsElement.webkitRequestFullScreen(Element.ALLOW_KEYBOARD_INPUT);\n            }\n        } else {\n            //Exit Full Screen\n            if (document.mozCancelFullScreen) {\n                document.mozCancelFullScreen();\n            } else {\n                document.webkitCancelFullScreen();\n            }\n        }\n    };\n    /**\n     * Unlocks the lock button state.\n     */\n    my.unlockLockButton = function () {\n        if ($(\"#lockIcon\").hasClass(\"icon-security-locked\"))\n            UIUtil.buttonClick(\"#lockIcon\", \"icon-security icon-security-locked\");\n    };\n    /**\n     * Updates the lock button state to locked.\n     */\n    my.lockLockButton = function () {\n        if ($(\"#lockIcon\").hasClass(\"icon-security\"))\n            UIUtil.buttonClick(\"#lockIcon\", \"icon-security icon-security-locked\");\n    };\n\n    /**\n     * Shows or hides authentication button\n     * @param show <tt>true</tt> to show or <tt>false</tt> to hide\n     */\n    my.showAuthenticateButton = function (show) {\n        if (show) {\n            $('#authentication').css({display: \"inline\"});\n        }\n        else {\n            $('#authentication').css({display: \"none\"});\n        }\n    };\n\n    // Shows or hides the 'recording' button.\n    my.showRecordingButton = function (show) {\n        if (!config.enableRecording) {\n            return;\n        }\n\n        if (show) {\n            $('#recording').css({display: \"inline\"});\n        }\n        else {\n            $('#recording').css({display: \"none\"});\n        }\n    };\n\n    // Sets the state of the recording button\n    my.setRecordingButtonState = function (isRecording) {\n        if (isRecording) {\n            $('#recordButton').removeClass(\"icon-recEnable\");\n            $('#recordButton').addClass(\"icon-recEnable active\");\n        } else {\n            $('#recordButton').removeClass(\"icon-recEnable active\");\n            $('#recordButton').addClass(\"icon-recEnable\");\n        }\n    };\n\n    // Shows or hides SIP calls button\n    my.showSipCallButton = function (show) {\n        if (xmpp.isSipGatewayEnabled() && show) {\n            $('#sipCallButton').css({display: \"inline\"});\n        } else {\n            $('#sipCallButton').css({display: \"none\"});\n        }\n    };\n\n    /**\n     * Sets the state of the button. The button has blue glow if desktop\n     * streaming is active.\n     * @param active the state of the desktop streaming.\n     */\n    my.changeDesktopSharingButtonState = function (active) {\n        var button = $(\"#desktopsharing > a\");\n        if (active)\n        {\n            button.addClass(\"glow\");\n        }\n        else\n        {\n            button.removeClass(\"glow\");\n        }\n    };\n\n    return my;\n}(Toolbar || {}));\n\nmodule.exports = Toolbar;","var JitsiPopover = (function () {\n    /**\n     * Constructs new JitsiPopover and attaches it to the element\n     * @param element jquery selector\n     * @param options the options for the popover.\n     * @constructor\n     */\n    function JitsiPopover(element, options)\n    {\n        this.options = {\n            skin: \"white\",\n            content: \"\"\n        };\n        if(options)\n        {\n            if(options.skin)\n                this.options.skin = options.skin;\n\n            if(options.content)\n                this.options.content = options.content;\n        }\n\n        this.elementIsHovered = false;\n        this.popoverIsHovered = false;\n        this.popoverShown = false;\n\n        element.data(\"jitsi_popover\", this);\n        this.element = element;\n        this.template = ' <div class=\"jitsipopover ' + this.options.skin +\n            '\"><div class=\"arrow\"></div><div class=\"jitsipopover-content\"></div>' +\n            '<div class=\"jitsiPopupmenuPadding\"></div></div>';\n        var self = this;\n        this.element.on(\"mouseenter\", function () {\n            self.elementIsHovered = true;\n            self.show();\n        }).on(\"mouseleave\", function () {\n            self.elementIsHovered = false;\n            setTimeout(function () {\n                self.hide();\n            }, 10);\n        });\n    }\n\n    /**\n     * Shows the popover\n     */\n    JitsiPopover.prototype.show = function () {\n        this.createPopover();\n        this.popoverShown = true;\n\n    };\n\n    /**\n     * Hides the popover\n     */\n    JitsiPopover.prototype.hide = function () {\n        if(!this.elementIsHovered && !this.popoverIsHovered && this.popoverShown)\n        {\n            this.forceHide();\n        }\n    };\n\n    /**\n     * Hides the popover\n     */\n    JitsiPopover.prototype.forceHide = function () {\n        $(\".jitsipopover\").remove();\n        this.popoverShown = false;\n    };\n\n    /**\n     * Creates the popover html\n     */\n    JitsiPopover.prototype.createPopover = function () {\n        $(\"body\").append(this.template);\n        $(\".jitsipopover > .jitsipopover-content\").html(this.options.content);\n        var self = this;\n        $(\".jitsipopover\").on(\"mouseenter\", function () {\n            self.popoverIsHovered = true;\n        }).on(\"mouseleave\", function () {\n            self.popoverIsHovered = false;\n            self.hide();\n        });\n\n        this.refreshPosition();\n    };\n\n    /**\n     * Refreshes the position of the popover\n     */\n    JitsiPopover.prototype.refreshPosition = function () {\n        $(\".jitsipopover\").position({\n            my: \"bottom\",\n            at: \"top\",\n            collision: \"fit\",\n            of: this.element,\n            using: function (position, elements) {\n                var calcLeft = elements.target.left - elements.element.left + elements.target.width/2;\n                $(\".jitsipopover\").css({top: position.top, left: position.left, display: \"table\"});\n                $(\".jitsipopover > .arrow\").css({left: calcLeft});\n                $(\".jitsipopover > .jitsiPopupmenuPadding\").css({left: calcLeft - 50});\n            }\n        });\n    };\n\n    /**\n     * Updates the content of popover.\n     * @param content new content\n     */\n    JitsiPopover.prototype.updateContent = function (content) {\n        this.options.content = content;\n        if(!this.popoverShown)\n            return;\n        $(\".jitsipopover\").remove();\n        this.createPopover();\n    };\n\n    return JitsiPopover;\n\n\n})();\n\nmodule.exports = JitsiPopover;","/* global $, jQuery */\nvar messageHandler = (function(my) {\n\n    /**\n     * Shows a message to the user.\n     *\n     * @param titleString the title of the message\n     * @param messageString the text of the message\n     */\n    my.openMessageDialog = function(titleString, messageString) {\n        $.prompt(messageString,\n            {\n                title: titleString,\n                persistent: false\n            }\n        );\n    };\n\n    /**\n     * Shows a message to the user with two buttons: first is given as a parameter and the second is Cancel.\n     *\n     * @param titleString the title of the message\n     * @param msgString the text of the message\n     * @param persistent boolean value which determines whether the message is persistent or not\n     * @param leftButton the fist button's text\n     * @param submitFunction function to be called on submit\n     * @param loadedFunction function to be called after the prompt is fully loaded\n     * @param closeFunction function to be called after the prompt is closed\n     */\n    my.openTwoButtonDialog = function(titleString, msgString, persistent, leftButton,\n                                      submitFunction, loadedFunction, closeFunction) {\n        var buttons = {};\n        buttons[leftButton] = true;\n        buttons.Cancel = false;\n        $.prompt(msgString, {\n            title: titleString,\n            persistent: false,\n            buttons: buttons,\n            defaultButton: 1,\n            loaded: loadedFunction,\n            submit: submitFunction,\n            close: closeFunction\n        });\n    };\n\n    /**\n     * Shows a message to the user with two buttons: first is given as a parameter and the second is Cancel.\n     *\n     * @param titleString the title of the message\n     * @param msgString the text of the message\n     * @param persistent boolean value which determines whether the message is persistent or not\n     * @param buttons object with the buttons. The keys must be the name of the button and value is the value\n     * that will be passed to submitFunction\n     * @param submitFunction function to be called on submit\n     * @param loadedFunction function to be called after the prompt is fully loaded\n     */\n    my.openDialog = function (titleString,    msgString, persistent, buttons,\n                              submitFunction, loadedFunction) {\n        var args = {\n            title: titleString,\n            persistent: persistent,\n            buttons: buttons,\n            defaultButton: 1,\n            loaded: loadedFunction,\n            submit: submitFunction\n        };\n        if (persistent) {\n            args.closeText = '';\n        }\n        return $.prompt(msgString, args);\n    };\n\n    /**\n     * Closes currently opened dialog.\n     */\n    my.closeDialog = function () {\n        $.prompt.close();\n    };\n\n    /**\n     * Shows a dialog with different states to the user.\n     *\n     * @param statesObject object containing all the states of the dialog\n     * @param loadedFunction function to be called after the prompt is fully loaded\n     * @param stateChangedFunction function to be called when the state of the dialog is changed\n     */\n    my.openDialogWithStates = function(statesObject, loadedFunction, stateChangedFunction) {\n\n\n        var myPrompt = $.prompt(statesObject);\n\n        myPrompt.on('impromptu:loaded', loadedFunction);\n        myPrompt.on('impromptu:statechanged', stateChangedFunction);\n    };\n\n    /**\n     * Opens new popup window for given <tt>url</tt> centered over current\n     * window.\n     *\n     * @param url the URL to be displayed in the popup window\n     * @param w the width of the popup window\n     * @param h the height of the popup window\n     * @param onPopupClosed optional callback function called when popup window\n     *        has been closed.\n     *\n     * @returns popup window object if opened successfully or undefined\n     *          in case we failed to open it(popup blocked)\n     */\n    my.openCenteredPopup = function (url, w, h, onPopupClosed) {\n        var l = window.screenX + (window.innerWidth / 2) - (w / 2);\n        var t = window.screenY + (window.innerHeight / 2) - (h / 2);\n        var popup = window.open(\n            url, '_blank',\n            'top=' + t + ', left=' + l + ', width=' + w + ', height=' + h + '');\n        if (popup && onPopupClosed) {\n            var pollTimer = window.setInterval(function () {\n                if (popup.closed !== false) {\n                    window.clearInterval(pollTimer);\n                    onPopupClosed();\n                }\n            }, 200);\n        }\n        return popup;\n    };\n\n    /**\n     * Shows a dialog prompting the user to send an error report.\n     *\n     * @param titleString the title of the message\n     * @param msgString the text of the message\n     * @param error the error that is being reported\n     */\n    my.openReportDialog = function(titleString, msgString, error) {\n        my.openMessageDialog(titleString, msgString);\n        console.log(error);\n        //FIXME send the error to the server\n    };\n\n    /**\n     *  Shows an error dialog to the user.\n     * @param title the title of the message\n     * @param message the text of the messafe\n     */\n    my.showError = function(title, message) {\n        if(!(title || message)) {\n            title = title || \"Oops!\";\n            message = message || \"There was some kind of error\";\n        }\n        messageHandler.openMessageDialog(title, message);\n    };\n\n    my.notify = function(displayName, cls, message) {\n        toastr.info(\n            '<span class=\"nickname\">' +\n                displayName +\n            '</span><br>' +\n            '<span class=' + cls + '>' +\n                message +\n            '</span>');\n    };\n\n    return my;\n}(messageHandler || {}));\n\nmodule.exports = messageHandler;\n\n\n","/**\n * Created by hristo on 12/22/14.\n */\nmodule.exports = {\n    /**\n     * Returns the available video width.\n     */\n    getAvailableVideoWidth: function () {\n        var PanelToggler = require(\"../side_pannels/SidePanelToggler\");\n        var rightPanelWidth\n            = PanelToggler.isVisible() ? PanelToggler.getPanelSize()[0] : 0;\n\n        return window.innerWidth - rightPanelWidth;\n    },\n    /**\n     * Changes the style class of the element given by id.\n     */\n    buttonClick: function(id, classname) {\n        $(id).toggleClass(classname); // add the class to the clicked element\n    }\n\n\n};","var JitsiPopover = require(\"../util/JitsiPopover\");\n\n/**\n * Constructs new connection indicator.\n * @param videoContainer the video container associated with the indicator.\n * @constructor\n */\nfunction ConnectionIndicator(videoContainer, jid, VideoLayout)\n{\n    this.videoContainer = videoContainer;\n    this.bandwidth = null;\n    this.packetLoss = null;\n    this.bitrate = null;\n    this.showMoreValue = false;\n    this.resolution = null;\n    this.transport = [];\n    this.popover = null;\n    this.jid = jid;\n    this.create();\n    this.videoLayout = VideoLayout;\n}\n\n/**\n * Values for the connection quality\n * @type {{98: string,\n *         81: string,\n *         64: string,\n *         47: string,\n *         30: string,\n *         0: string}}\n */\nConnectionIndicator.connectionQualityValues = {\n    98: \"18px\", //full\n    81: \"15px\",//4 bars\n    64: \"11px\",//3 bars\n    47: \"7px\",//2 bars\n    30: \"3px\",//1 bar\n    0: \"0px\"//empty\n};\n\nConnectionIndicator.getIP = function(value)\n{\n    return value.substring(0, value.lastIndexOf(\":\"));\n};\n\nConnectionIndicator.getPort = function(value)\n{\n    return value.substring(value.lastIndexOf(\":\") + 1, value.length);\n};\n\nConnectionIndicator.getStringFromArray = function (array) {\n    var res = \"\";\n    for(var i = 0; i < array.length; i++)\n    {\n        res += (i === 0? \"\" : \", \") + array[i];\n    }\n    return res;\n};\n\n/**\n * Generates the html content.\n * @returns {string} the html content.\n */\nConnectionIndicator.prototype.generateText = function () {\n    var downloadBitrate, uploadBitrate, packetLoss, resolution, i;\n\n    if(this.bitrate === null)\n    {\n        downloadBitrate = \"N/A\";\n        uploadBitrate = \"N/A\";\n    }\n    else\n    {\n        downloadBitrate =\n            this.bitrate.download? this.bitrate.download + \" Kbps\" : \"N/A\";\n        uploadBitrate =\n            this.bitrate.upload? this.bitrate.upload + \" Kbps\" : \"N/A\";\n    }\n\n    if(this.packetLoss === null)\n    {\n        packetLoss = \"N/A\";\n    }\n    else\n    {\n\n        packetLoss = \"<span class='jitsipopover_green'>&darr;</span>\" +\n            (this.packetLoss.download !== null? this.packetLoss.download : \"N/A\") +\n            \"% <span class='jitsipopover_orange'>&uarr;</span>\" +\n            (this.packetLoss.upload !== null? this.packetLoss.upload : \"N/A\") + \"%\";\n    }\n\n    var resolutionValue = null;\n    if(this.resolution && this.jid != null)\n    {\n        var keys = Object.keys(this.resolution);\n        if(keys.length == 1)\n        {\n            for(var ssrc in this.resolution)\n            {\n                resolutionValue = this.resolution[ssrc];\n            }\n        }\n        else if(keys.length > 1)\n        {\n            var displayedSsrc = simulcast.getReceivingSSRC(this.jid);\n            resolutionValue = this.resolution[displayedSsrc];\n        }\n    }\n\n    if(this.jid === null)\n    {\n        resolution = \"\";\n        if(this.resolution === null || !Object.keys(this.resolution) ||\n            Object.keys(this.resolution).length === 0)\n        {\n            resolution = \"N/A\";\n        }\n        else\n            for(i in this.resolution)\n            {\n                resolutionValue = this.resolution[i];\n                if(resolutionValue)\n                {\n                    if(resolutionValue.height &&\n                        resolutionValue.width)\n                    {\n                        resolution += (resolution === \"\"? \"\" : \", \") +\n                            resolutionValue.width + \"x\" +\n                            resolutionValue.height;\n                    }\n                }\n            }\n    }\n    else if(!resolutionValue ||\n        !resolutionValue.height ||\n        !resolutionValue.width)\n    {\n        resolution = \"N/A\";\n    }\n    else\n    {\n        resolution = resolutionValue.width + \"x\" + resolutionValue.height;\n    }\n\n    var result = \"<table style='width:100%'>\" +\n        \"<tr>\" +\n        \"<td><span class='jitsipopover_blue'>Bitrate:</span></td>\" +\n        \"<td><span class='jitsipopover_green'>&darr;</span>\" +\n        downloadBitrate + \" <span class='jitsipopover_orange'>&uarr;</span>\" +\n        uploadBitrate + \"</td>\" +\n        \"</tr><tr>\" +\n        \"<td><span class='jitsipopover_blue'>Packet loss: </span></td>\" +\n        \"<td>\" + packetLoss  + \"</td>\" +\n        \"</tr><tr>\" +\n        \"<td><span class='jitsipopover_blue'>Resolution:</span></td>\" +\n        \"<td>\" + resolution + \"</td></tr></table>\";\n\n    if(this.videoContainer.id == \"localVideoContainer\")\n        result += \"<div class=\\\"jitsipopover_showmore\\\" \" +\n            \"onclick = \\\"UI.connectionIndicatorShowMore('\" +\n            this.videoContainer.id + \"')\\\">\" +\n            (this.showMoreValue? \"Show less\" : \"Show More\") + \"</div><br />\";\n\n    if(this.showMoreValue)\n    {\n        var downloadBandwidth, uploadBandwidth, transport;\n        if(this.bandwidth === null)\n        {\n            downloadBandwidth = \"N/A\";\n            uploadBandwidth = \"N/A\";\n        }\n        else\n        {\n            downloadBandwidth = this.bandwidth.download?\n                this.bandwidth.download + \" Kbps\" :\n                \"N/A\";\n            uploadBandwidth = this.bandwidth.upload?\n                this.bandwidth.upload + \" Kbps\" :\n                \"N/A\";\n        }\n\n        if(!this.transport || this.transport.length === 0)\n        {\n            transport = \"<tr>\" +\n                \"<td><span class='jitsipopover_blue'>Address:</span></td>\" +\n                \"<td> N/A</td></tr>\";\n        }\n        else\n        {\n            var data = {remoteIP: [], localIP:[], remotePort:[], localPort:[]};\n            for(i = 0; i < this.transport.length; i++)\n            {\n                var ip =  ConnectionIndicator.getIP(this.transport[i].ip);\n                var port = ConnectionIndicator.getPort(this.transport[i].ip);\n                var localIP =\n                    ConnectionIndicator.getIP(this.transport[i].localip);\n                var localPort =\n                    ConnectionIndicator.getPort(this.transport[i].localip);\n                if(data.remoteIP.indexOf(ip) == -1)\n                {\n                    data.remoteIP.push(ip);\n                }\n\n                if(data.remotePort.indexOf(port) == -1)\n                {\n                    data.remotePort.push(port);\n                }\n\n                if(data.localIP.indexOf(localIP) == -1)\n                {\n                    data.localIP.push(localIP);\n                }\n\n                if(data.localPort.indexOf(localPort) == -1)\n                {\n                    data.localPort.push(localPort);\n                }\n\n            }\n            var localTransport =\n                \"<tr><td><span class='jitsipopover_blue'>Local address\" +\n                (data.localIP.length > 1? \"es\" : \"\") + \": </span></td><td> \" +\n                ConnectionIndicator.getStringFromArray(data.localIP) +\n                \"</td></tr>\";\n            transport =\n                \"<tr><td><span class='jitsipopover_blue'>Remote address\"+\n                (data.remoteIP.length > 1? \"es\" : \"\") + \":</span></td><td> \" +\n                ConnectionIndicator.getStringFromArray(data.remoteIP) +\n                \"</td></tr>\";\n            if(this.transport.length > 1)\n            {\n                transport += \"<tr>\" +\n                    \"<td>\" +\n                    \"<span class='jitsipopover_blue'>Remote ports:</span>\" +\n                    \"</td><td>\";\n                localTransport += \"<tr>\" +\n                    \"<td>\" +\n                    \"<span class='jitsipopover_blue'>Local ports:</span>\" +\n                    \"</td><td>\";\n            }\n            else\n            {\n                transport +=\n                    \"<tr>\" +\n                    \"<td>\" +\n                    \"<span class='jitsipopover_blue'>Remote port:</span>\" +\n                    \"</td><td>\";\n                localTransport +=\n                    \"<tr>\" +\n                    \"<td>\" +\n                    \"<span class='jitsipopover_blue'>Local port:</span>\" +\n                    \"</td><td>\";\n            }\n\n            transport +=\n                ConnectionIndicator.getStringFromArray(data.remotePort);\n            localTransport +=\n                ConnectionIndicator.getStringFromArray(data.localPort);\n            transport += \"</td></tr>\";\n            transport += localTransport + \"</td></tr>\";\n            transport +=\"<tr>\" +\n                \"<td><span class='jitsipopover_blue'>Transport:</span></td>\" +\n                \"<td>\" + this.transport[0].type + \"</td></tr>\";\n\n        }\n\n        result += \"<table  style='width:100%'>\" +\n            \"<tr>\" +\n            \"<td>\" +\n            \"<span class='jitsipopover_blue'>Estimated bandwidth:</span>\" +\n            \"</td><td>\" +\n            \"<span class='jitsipopover_green'>&darr;</span>\" +\n            downloadBandwidth +\n            \" <span class='jitsipopover_orange'>&uarr;</span>\" +\n            uploadBandwidth + \"</td></tr>\";\n\n        result += transport + \"</table>\";\n\n    }\n\n    return result;\n};\n\n/**\n * Shows or hide the additional information.\n */\nConnectionIndicator.prototype.showMore = function () {\n    this.showMoreValue = !this.showMoreValue;\n    this.updatePopoverData();\n};\n\n\nfunction createIcon(classes)\n{\n    var icon = document.createElement(\"span\");\n    for(var i in classes)\n    {\n        icon.classList.add(classes[i]);\n    }\n    icon.appendChild(\n        document.createElement(\"i\")).classList.add(\"icon-connection\");\n    return icon;\n}\n\n/**\n * Creates the indicator\n */\nConnectionIndicator.prototype.create = function () {\n    this.connectionIndicatorContainer = document.createElement(\"div\");\n    this.connectionIndicatorContainer.className = \"connectionindicator\";\n    this.connectionIndicatorContainer.style.display = \"none\";\n    this.videoContainer.appendChild(this.connectionIndicatorContainer);\n    this.popover = new JitsiPopover(\n        $(\"#\" + this.videoContainer.id + \" > .connectionindicator\"),\n        {content: \"<div class=\\\"connection_info\\\">Come back here for \" +\n            \"connection information once the conference starts</div>\",\n            skin: \"black\"});\n\n    this.emptyIcon = this.connectionIndicatorContainer.appendChild(\n        createIcon([\"connection\", \"connection_empty\"]));\n    this.fullIcon = this.connectionIndicatorContainer.appendChild(\n        createIcon([\"connection\", \"connection_full\"]));\n\n};\n\n/**\n * Removes the indicator\n */\nConnectionIndicator.prototype.remove = function()\n{\n    this.connectionIndicatorContainer.remove();\n    this.popover.forceHide();\n\n};\n\n/**\n * Updates the data of the indicator\n * @param percent the percent of connection quality\n * @param object the statistics data.\n */\nConnectionIndicator.prototype.updateConnectionQuality =\nfunction (percent, object) {\n\n    if(percent === null)\n    {\n        this.connectionIndicatorContainer.style.display = \"none\";\n        this.popover.forceHide();\n        return;\n    }\n    else\n    {\n        if(this.connectionIndicatorContainer.style.display == \"none\") {\n            this.connectionIndicatorContainer.style.display = \"block\";\n            this.videoLayout.updateMutePosition(this.videoContainer.id);\n        }\n    }\n    this.bandwidth = object.bandwidth;\n    this.bitrate = object.bitrate;\n    this.packetLoss = object.packetLoss;\n    this.transport = object.transport;\n    if(object.resolution)\n    {\n        this.resolution = object.resolution;\n    }\n    for(var quality in ConnectionIndicator.connectionQualityValues)\n    {\n        if(percent >= quality)\n        {\n            this.fullIcon.style.width =\n                ConnectionIndicator.connectionQualityValues[quality];\n        }\n    }\n    this.updatePopoverData();\n};\n\n/**\n * Updates the resolution\n * @param resolution the new resolution\n */\nConnectionIndicator.prototype.updateResolution = function (resolution) {\n    this.resolution = resolution;\n    this.updatePopoverData();\n};\n\n/**\n * Updates the content of the popover\n */\nConnectionIndicator.prototype.updatePopoverData = function () {\n    this.popover.updateContent(\n            \"<div class=\\\"connection_info\\\">\" + this.generateText() + \"</div>\");\n};\n\n/**\n * Hides the popover\n */\nConnectionIndicator.prototype.hide = function () {\n    this.popover.forceHide();\n};\n\n/**\n * Hides the indicator\n */\nConnectionIndicator.prototype.hideIndicator = function () {\n    this.connectionIndicatorContainer.style.display = \"none\";\n    if(this.popover)\n        this.popover.forceHide();\n};\n\nmodule.exports = ConnectionIndicator;","var AudioLevels = require(\"../audio_levels/AudioLevels\");\nvar Avatar = require(\"../avatar/Avatar\");\nvar Chat = require(\"../side_pannels/chat/Chat\");\nvar ContactList = require(\"../side_pannels/contactlist/ContactList\");\nvar UIUtil = require(\"../util/UIUtil\");\nvar ConnectionIndicator = require(\"./ConnectionIndicator\");\n\nvar currentDominantSpeaker = null;\nvar lastNCount = config.channelLastN;\nvar localLastNCount = config.channelLastN;\nvar localLastNSet = [];\nvar lastNEndpointsCache = [];\nvar lastNPickupJid = null;\nvar largeVideoState = {\n    updateInProgress: false,\n    newSrc: ''\n};\n\n/**\n * Indicates if we have muted our audio before the conference has started.\n * @type {boolean}\n */\nvar preMuted = false;\n\nvar mutedAudios = {};\n\nvar flipXLocalVideo = true;\nvar currentVideoWidth = null;\nvar currentVideoHeight = null;\n\nvar localVideoSrc = null;\n\nvar defaultLocalDisplayName = \"Me\";\n\nfunction videoactive( videoelem) {\n    if (videoelem.attr('id').indexOf('mixedmslabel') === -1) {\n        // ignore mixedmslabela0 and v0\n\n        videoelem.show();\n        VideoLayout.resizeThumbnails();\n\n        var videoParent = videoelem.parent();\n        var parentResourceJid = null;\n        if (videoParent)\n            parentResourceJid\n                = VideoLayout.getPeerContainerResourceJid(videoParent[0]);\n\n        // Update the large video to the last added video only if there's no\n        // current dominant, focused speaker or prezi playing or update it to\n        // the current dominant speaker.\n        if ((!focusedVideoInfo &&\n            !VideoLayout.getDominantSpeakerResourceJid() &&\n            !require(\"../prezi/Prezi\").isPresentationVisible()) ||\n            (parentResourceJid &&\n                VideoLayout.getDominantSpeakerResourceJid() === parentResourceJid)) {\n            VideoLayout.updateLargeVideo(\n                RTC.getVideoSrc(videoelem[0]),\n                1,\n                parentResourceJid);\n        }\n\n        VideoLayout.showModeratorIndicator();\n    }\n}\n\nfunction waitForRemoteVideo(selector, ssrc, stream, jid) {\n    // XXX(gp) so, every call to this function is *always* preceded by a call\n    // to the RTC.attachMediaStream() function but that call is *not* followed\n    // by an update to the videoSrcToSsrc map!\n    //\n    // The above way of doing things results in video SRCs that don't correspond\n    // to any SSRC for a short period of time (to be more precise, for as long\n    // the waitForRemoteVideo takes to complete). This causes problems (see\n    // bellow).\n    //\n    // I'm wondering why we need to do that; i.e. why call RTC.attachMediaStream()\n    // a second time in here and only then update the videoSrcToSsrc map? Why\n    // not simply update the videoSrcToSsrc map when the RTC.attachMediaStream()\n    // is called the first time? I actually do that in the lastN changed event\n    // handler because the \"orphan\" video SRC is causing troubles there. The\n    // purpose of this method would then be to fire the \"videoactive.jingle\".\n    //\n    // Food for though I guess :-)\n\n    if (selector.removed || !selector.parent().is(\":visible\")) {\n        console.warn(\"Media removed before had started\", selector);\n        return;\n    }\n\n    if (stream.id === 'mixedmslabel') return;\n\n    if (selector[0].currentTime > 0) {\n        var videoStream = simulcast.getReceivingVideoStream(stream);\n        RTC.attachMediaStream(selector, videoStream); // FIXME: why do i have to do this for FF?\n\n        // FIXME: add a class that will associate peer Jid, video.src, it's ssrc and video type\n        //        in order to get rid of too many maps\n        if (ssrc && jid) {\n            jid2Ssrc[Strophe.getResourceFromJid(jid)] = ssrc;\n        } else {\n            console.warn(\"No ssrc given for jid\", jid);\n        }\n\n        videoactive(selector);\n    } else {\n        setTimeout(function () {\n            waitForRemoteVideo(selector, ssrc, stream, jid);\n        }, 250);\n    }\n}\n\n/**\n * Returns an array of the video horizontal and vertical indents,\n * so that if fits its parent.\n *\n * @return an array with 2 elements, the horizontal indent and the vertical\n * indent\n */\nfunction getCameraVideoPosition(videoWidth,\n                                videoHeight,\n                                videoSpaceWidth,\n                                videoSpaceHeight) {\n    // Parent height isn't completely calculated when we position the video in\n    // full screen mode and this is why we use the screen height in this case.\n    // Need to think it further at some point and implement it properly.\n    var isFullScreen = document.fullScreen ||\n        document.mozFullScreen ||\n        document.webkitIsFullScreen;\n    if (isFullScreen)\n        videoSpaceHeight = window.innerHeight;\n\n    var horizontalIndent = (videoSpaceWidth - videoWidth) / 2;\n    var verticalIndent = (videoSpaceHeight - videoHeight) / 2;\n\n    return [horizontalIndent, verticalIndent];\n}\n\n/**\n * Returns an array of the video horizontal and vertical indents.\n * Centers horizontally and top aligns vertically.\n *\n * @return an array with 2 elements, the horizontal indent and the vertical\n * indent\n */\nfunction getDesktopVideoPosition(videoWidth,\n                                 videoHeight,\n                                 videoSpaceWidth,\n                                 videoSpaceHeight) {\n\n    var horizontalIndent = (videoSpaceWidth - videoWidth) / 2;\n\n    var verticalIndent = 0;// Top aligned\n\n    return [horizontalIndent, verticalIndent];\n}\n\n\n/**\n * Returns an array of the video dimensions, so that it covers the screen.\n * It leaves no empty areas, but some parts of the video might not be visible.\n *\n * @return an array with 2 elements, the video width and the video height\n */\nfunction getCameraVideoSize(videoWidth,\n                            videoHeight,\n                            videoSpaceWidth,\n                            videoSpaceHeight) {\n    if (!videoWidth)\n        videoWidth = currentVideoWidth;\n    if (!videoHeight)\n        videoHeight = currentVideoHeight;\n\n    var aspectRatio = videoWidth / videoHeight;\n\n    var availableWidth = Math.max(videoWidth, videoSpaceWidth);\n    var availableHeight = Math.max(videoHeight, videoSpaceHeight);\n\n    if (availableWidth / aspectRatio < videoSpaceHeight) {\n        availableHeight = videoSpaceHeight;\n        availableWidth = availableHeight * aspectRatio;\n    }\n\n    if (availableHeight * aspectRatio < videoSpaceWidth) {\n        availableWidth = videoSpaceWidth;\n        availableHeight = availableWidth / aspectRatio;\n    }\n\n    return [availableWidth, availableHeight];\n}\n\n/**\n * Sets the display name for the given video span id.\n */\nfunction setDisplayName(videoSpanId, displayName) {\n    var nameSpan = $('#' + videoSpanId + '>span.displayname');\n    var defaultLocalDisplayName = interfaceConfig.DEFAULT_LOCAL_DISPLAY_NAME;\n\n    // If we already have a display name for this video.\n    if (nameSpan.length > 0) {\n        var nameSpanElement = nameSpan.get(0);\n\n        if (nameSpanElement.id === 'localDisplayName' &&\n            $('#localDisplayName').text() !== displayName) {\n            if (displayName && displayName.length > 0)\n                $('#localDisplayName').html(displayName + ' (me)');\n            else\n                $('#localDisplayName').text(defaultLocalDisplayName);\n        } else {\n            if (displayName && displayName.length > 0)\n                $('#' + videoSpanId + '_name').html(displayName);\n            else\n                $('#' + videoSpanId + '_name').text(interfaceConfig.DEFAULT_REMOTE_DISPLAY_NAME);\n        }\n    } else {\n        var editButton = null;\n\n        nameSpan = document.createElement('span');\n        nameSpan.className = 'displayname';\n        $('#' + videoSpanId)[0].appendChild(nameSpan);\n\n        if (videoSpanId === 'localVideoContainer') {\n            editButton = createEditDisplayNameButton();\n            nameSpan.innerText = defaultLocalDisplayName;\n        }\n        else {\n            nameSpan.innerText = interfaceConfig.DEFAULT_REMOTE_DISPLAY_NAME;\n        }\n\n        if (displayName && displayName.length > 0) {\n            nameSpan.innerText = displayName;\n        }\n\n        if (!editButton) {\n            nameSpan.id = videoSpanId + '_name';\n        } else {\n            nameSpan.id = 'localDisplayName';\n            $('#' + videoSpanId)[0].appendChild(editButton);\n\n            var editableText = document.createElement('input');\n            editableText.className = 'displayname';\n            editableText.type = 'text';\n            editableText.id = 'editDisplayName';\n\n            if (displayName && displayName.length) {\n                editableText.value\n                    = displayName.substring(0, displayName.indexOf(' (me)'));\n            }\n\n            editableText.setAttribute('style', 'display:none;');\n            editableText.setAttribute('placeholder', 'ex. Jane Pink');\n            $('#' + videoSpanId)[0].appendChild(editableText);\n\n            $('#localVideoContainer .displayname')\n                .bind(\"click\", function (e) {\n\n                    e.preventDefault();\n                    e.stopPropagation();\n                    $('#localDisplayName').hide();\n                    $('#editDisplayName').show();\n                    $('#editDisplayName').focus();\n                    $('#editDisplayName').select();\n\n                    $('#editDisplayName').one(\"focusout\", function (e) {\n                        VideoLayout.inputDisplayNameHandler(this.value);\n                    });\n\n                    $('#editDisplayName').on('keydown', function (e) {\n                        if (e.keyCode === 13) {\n                            e.preventDefault();\n                            VideoLayout.inputDisplayNameHandler(this.value);\n                        }\n                    });\n                });\n        }\n    }\n}\n\n/**\n * Gets the selector of video thumbnail container for the user identified by\n * given <tt>userJid</tt>\n * @param resourceJid user's Jid for whom we want to get the video container.\n */\nfunction getParticipantContainer(resourceJid)\n{\n    if (!resourceJid)\n        return null;\n\n    if (resourceJid === xmpp.myResource())\n        return $(\"#localVideoContainer\");\n    else\n        return $(\"#participant_\" + resourceJid);\n}\n\n/**\n * Sets the size and position of the given video element.\n *\n * @param video the video element to position\n * @param width the desired video width\n * @param height the desired video height\n * @param horizontalIndent the left and right indent\n * @param verticalIndent the top and bottom indent\n */\nfunction positionVideo(video,\n                       width,\n                       height,\n                       horizontalIndent,\n                       verticalIndent) {\n    video.width(width);\n    video.height(height);\n    video.css({  top: verticalIndent + 'px',\n        bottom: verticalIndent + 'px',\n        left: horizontalIndent + 'px',\n        right: horizontalIndent + 'px'});\n}\n\n/**\n * Adds the remote video menu element for the given <tt>jid</tt> in the\n * given <tt>parentElement</tt>.\n *\n * @param jid the jid indicating the video for which we're adding a menu.\n * @param parentElement the parent element where this menu will be added\n */\nfunction addRemoteVideoMenu(jid, parentElement) {\n    var spanElement = document.createElement('span');\n    spanElement.className = 'remotevideomenu';\n\n    parentElement.appendChild(spanElement);\n\n    var menuElement = document.createElement('i');\n    menuElement.className = 'fa fa-angle-down';\n    menuElement.title = 'Remote user controls';\n    spanElement.appendChild(menuElement);\n\n//        <ul class=\"popupmenu\">\n//        <li><a href=\"#\">Mute</a></li>\n//        <li><a href=\"#\">Eject</a></li>\n//        </ul>\n\n    var popupmenuElement = document.createElement('ul');\n    popupmenuElement.className = 'popupmenu';\n    popupmenuElement.id\n        = 'remote_popupmenu_' + Strophe.getResourceFromJid(jid);\n    spanElement.appendChild(popupmenuElement);\n\n    var muteMenuItem = document.createElement('li');\n    var muteLinkItem = document.createElement('a');\n\n    var mutedIndicator = \"<i class='icon-mic-disabled'></i>\";\n\n    if (!mutedAudios[jid]) {\n        muteLinkItem.innerHTML = mutedIndicator + 'Mute';\n        muteLinkItem.className = 'mutelink';\n    }\n    else {\n        muteLinkItem.innerHTML = mutedIndicator + ' Muted';\n        muteLinkItem.className = 'mutelink disabled';\n    }\n\n    muteLinkItem.onclick = function(){\n        if ($(this).attr('disabled') != undefined) {\n            event.preventDefault();\n        }\n        var isMute = mutedAudios[jid] == true;\n        xmpp.setMute(jid, !isMute);\n\n        popupmenuElement.setAttribute('style', 'display:none;');\n\n        if (isMute) {\n            this.innerHTML = mutedIndicator + ' Muted';\n            this.className = 'mutelink disabled';\n        }\n        else {\n            this.innerHTML = mutedIndicator + ' Mute';\n            this.className = 'mutelink';\n        }\n    };\n\n    muteMenuItem.appendChild(muteLinkItem);\n    popupmenuElement.appendChild(muteMenuItem);\n\n    var ejectIndicator = \"<i class='fa fa-eject'></i>\";\n\n    var ejectMenuItem = document.createElement('li');\n    var ejectLinkItem = document.createElement('a');\n    ejectLinkItem.innerHTML = ejectIndicator + ' Kick out';\n    ejectLinkItem.onclick = function(){\n        xmpp.eject(jid);\n        popupmenuElement.setAttribute('style', 'display:none;');\n    };\n\n    ejectMenuItem.appendChild(ejectLinkItem);\n    popupmenuElement.appendChild(ejectMenuItem);\n\n    var paddingSpan = document.createElement('span');\n    paddingSpan.className = 'popupmenuPadding';\n    popupmenuElement.appendChild(paddingSpan);\n}\n\n/**\n * Removes remote video menu element from video element identified by\n * given <tt>videoElementId</tt>.\n *\n * @param videoElementId the id of local or remote video element.\n */\nfunction removeRemoteVideoMenu(videoElementId) {\n    var menuSpan = $('#' + videoElementId + '>span.remotevideomenu');\n    if (menuSpan.length) {\n        menuSpan.remove();\n    }\n}\n\n/**\n * Updates the data for the indicator\n * @param id the id of the indicator\n * @param percent the percent for connection quality\n * @param object the data\n */\nfunction updateStatsIndicator(id, percent, object) {\n    if(VideoLayout.connectionIndicators[id])\n        VideoLayout.connectionIndicators[id].updateConnectionQuality(percent, object);\n}\n\n\n/**\n * Returns an array of the video dimensions, so that it keeps it's aspect\n * ratio and fits available area with it's larger dimension. This method\n * ensures that whole video will be visible and can leave empty areas.\n *\n * @return an array with 2 elements, the video width and the video height\n */\nfunction getDesktopVideoSize(videoWidth,\n                             videoHeight,\n                             videoSpaceWidth,\n                             videoSpaceHeight) {\n    if (!videoWidth)\n        videoWidth = currentVideoWidth;\n    if (!videoHeight)\n        videoHeight = currentVideoHeight;\n\n    var aspectRatio = videoWidth / videoHeight;\n\n    var availableWidth = Math.max(videoWidth, videoSpaceWidth);\n    var availableHeight = Math.max(videoHeight, videoSpaceHeight);\n\n    videoSpaceHeight -= $('#remoteVideos').outerHeight();\n\n    if (availableWidth / aspectRatio >= videoSpaceHeight)\n    {\n        availableHeight = videoSpaceHeight;\n        availableWidth = availableHeight * aspectRatio;\n    }\n\n    if (availableHeight * aspectRatio >= videoSpaceWidth)\n    {\n        availableWidth = videoSpaceWidth;\n        availableHeight = availableWidth / aspectRatio;\n    }\n\n    return [availableWidth, availableHeight];\n}\n\n/**\n * Creates the edit display name button.\n *\n * @returns the edit button\n */\nfunction createEditDisplayNameButton() {\n    var editButton = document.createElement('a');\n    editButton.className = 'displayname';\n    Util.setTooltip(editButton,\n        'Click to edit your<br/>display name',\n        \"top\");\n    editButton.innerHTML = '<i class=\"fa fa-pencil\"></i>';\n\n    return editButton;\n}\n\n/**\n * Creates the element indicating the moderator(owner) of the conference.\n *\n * @param parentElement the parent element where the owner indicator will\n * be added\n */\nfunction createModeratorIndicatorElement(parentElement) {\n    var moderatorIndicator = document.createElement('i');\n    moderatorIndicator.className = 'fa fa-star';\n    parentElement.appendChild(moderatorIndicator);\n\n    Util.setTooltip(parentElement,\n        \"The owner of<br/>this conference\",\n        \"top\");\n}\n\n\n/**\n * Checks if video identified by given src is desktop stream.\n * @param videoSrc eg.\n * blob:https%3A//pawel.jitsi.net/9a46e0bd-131e-4d18-9c14-a9264e8db395\n * @returns {boolean}\n */\nfunction isVideoSrcDesktop(jid) {\n    // FIXME: fix this mapping mess...\n    // figure out if large video is desktop stream or just a camera\n\n    if(!jid)\n        return false;\n    var isDesktop = false;\n    if (xmpp.myJid() &&\n        xmpp.myResource() === jid) {\n        // local video\n        isDesktop = desktopsharing.isUsingScreenStream();\n    } else {\n        // Do we have associations...\n        var videoSsrc = jid2Ssrc[jid];\n        if (videoSsrc) {\n            var videoType = ssrc2videoType[videoSsrc];\n            if (videoType) {\n                // Finally there...\n                isDesktop = videoType === 'screen';\n            } else {\n                console.error(\"No video type for ssrc: \" + videoSsrc);\n            }\n        } else {\n            console.error(\"No ssrc for jid: \" + jid);\n        }\n    }\n    return isDesktop;\n}\n\n\n\nvar VideoLayout = (function (my) {\n    my.connectionIndicators = {};\n\n    // By default we use camera\n    my.getVideoSize = getCameraVideoSize;\n    my.getVideoPosition = getCameraVideoPosition;\n\n    my.init = function () {\n        // Listen for large video size updates\n        document.getElementById('largeVideo')\n            .addEventListener('loadedmetadata', function (e) {\n                currentVideoWidth = this.videoWidth;\n                currentVideoHeight = this.videoHeight;\n                VideoLayout.positionLarge(currentVideoWidth, currentVideoHeight);\n            });\n    };\n\n    my.isInLastN = function(resource) {\n        return lastNCount < 0 // lastN is disabled, return true\n            || (lastNCount > 0 && lastNEndpointsCache.length == 0) // lastNEndpoints cache not built yet, return true\n            || (lastNEndpointsCache && lastNEndpointsCache.indexOf(resource) !== -1);\n    };\n\n    my.changeLocalStream = function (stream) {\n        VideoLayout.changeLocalVideo(stream);\n    };\n\n    my.changeLocalAudio = function(stream) {\n        RTC.attachMediaStream($('#localAudio'), stream.getOriginalStream());\n        document.getElementById('localAudio').autoplay = true;\n        document.getElementById('localAudio').volume = 0;\n        if (preMuted) {\n            if(!UI.setAudioMuted(true))\n            {\n                preMuted = mute;\n            }\n            preMuted = false;\n        }\n    };\n\n    my.changeLocalVideo = function(stream) {\n        var flipX = true;\n        if(stream.type == \"desktop\")\n            flipX = false;\n        var localVideo = document.createElement('video');\n        localVideo.id = 'localVideo_' +\n            RTC.getStreamID(stream.getOriginalStream());\n        localVideo.autoplay = true;\n        localVideo.volume = 0; // is it required if audio is separated ?\n        localVideo.oncontextmenu = function () { return false; };\n\n        var localVideoContainer = document.getElementById('localVideoWrapper');\n        localVideoContainer.appendChild(localVideo);\n\n        // Set default display name.\n        setDisplayName('localVideoContainer');\n\n        if(!VideoLayout.connectionIndicators[\"localVideoContainer\"]) {\n            VideoLayout.connectionIndicators[\"localVideoContainer\"]\n                = new ConnectionIndicator($(\"#localVideoContainer\")[0], null, VideoLayout);\n        }\n\n        AudioLevels.updateAudioLevelCanvas(null, VideoLayout);\n\n        var localVideoSelector = $('#' + localVideo.id);\n        // Add click handler to both video and video wrapper elements in case\n        // there's no video.\n        localVideoSelector.click(function (event) {\n            event.stopPropagation();\n            VideoLayout.handleVideoThumbClicked(\n                RTC.getVideoSrc(localVideo),\n                false,\n                xmpp.myResource());\n        });\n        $('#localVideoContainer').click(function (event) {\n            event.stopPropagation();\n            VideoLayout.handleVideoThumbClicked(\n                RTC.getVideoSrc(localVideo),\n                false,\n                xmpp.myResource());\n        });\n\n        // Add hover handler\n        $('#localVideoContainer').hover(\n            function() {\n                VideoLayout.showDisplayName('localVideoContainer', true);\n            },\n            function() {\n                if (!VideoLayout.isLargeVideoVisible()\n                        || RTC.getVideoSrc(localVideo) !== RTC.getVideoSrc($('#largeVideo')[0]))\n                    VideoLayout.showDisplayName('localVideoContainer', false);\n            }\n        );\n        // Add stream ended handler\n        stream.getOriginalStream().onended = function () {\n            localVideoContainer.removeChild(localVideo);\n            VideoLayout.updateRemovedVideo(RTC.getVideoSrc(localVideo));\n        };\n        // Flip video x axis if needed\n        flipXLocalVideo = flipX;\n        if (flipX) {\n            localVideoSelector.addClass(\"flipVideoX\");\n        }\n        // Attach WebRTC stream\n        var videoStream = simulcast.getLocalVideoStream();\n        RTC.attachMediaStream(localVideoSelector, videoStream);\n\n        localVideoSrc = RTC.getVideoSrc(localVideo);\n\n        var myResourceJid = xmpp.myResource();\n\n        VideoLayout.updateLargeVideo(localVideoSrc, 0,\n            myResourceJid);\n\n    };\n\n    /**\n     * Checks if removed video is currently displayed and tries to display\n     * another one instead.\n     * @param removedVideoSrc src stream identifier of the video.\n     */\n    my.updateRemovedVideo = function(removedVideoSrc) {\n        if (removedVideoSrc === RTC.getVideoSrc($('#largeVideo')[0])) {\n            // this is currently displayed as large\n            // pick the last visible video in the row\n            // if nobody else is left, this picks the local video\n            var pick\n                = $('#remoteVideos>span[id!=\"mixedstream\"]:visible:last>video')\n                    .get(0);\n\n            if (!pick) {\n                console.info(\"Last visible video no longer exists\");\n                pick = $('#remoteVideos>span[id!=\"mixedstream\"]>video').get(0);\n\n                if (!pick || !RTC.getVideoSrc(pick)) {\n                    // Try local video\n                    console.info(\"Fallback to local video...\");\n                    pick = $('#remoteVideos>span>span>video').get(0);\n                }\n            }\n\n            // mute if localvideo\n            if (pick) {\n                var container = pick.parentNode;\n                var jid = null;\n                if(container)\n                {\n                    if(container.id == \"localVideoWrapper\")\n                    {\n                        jid = xmpp.myResource();\n                    }\n                    else\n                    {\n                        jid = VideoLayout.getPeerContainerResourceJid(container);\n                    }\n                }\n\n                VideoLayout.updateLargeVideo(RTC.getVideoSrc(pick), pick.volume, jid);\n            } else {\n                console.warn(\"Failed to elect large video\");\n            }\n        }\n    };\n    \n    my.onRemoteStreamAdded = function (stream) {\n        var container;\n        var remotes = document.getElementById('remoteVideos');\n\n        if (stream.peerjid) {\n            VideoLayout.ensurePeerContainerExists(stream.peerjid);\n\n            container  = document.getElementById(\n                    'participant_' + Strophe.getResourceFromJid(stream.peerjid));\n        } else {\n            var id = stream.getOriginalStream().id;\n            if (id !== 'mixedmslabel'\n                // FIXME: default stream is added always with new focus\n                // (to be investigated)\n                && id !== 'default') {\n                console.error('can not associate stream',\n                    id,\n                    'with a participant');\n                // We don't want to add it here since it will cause troubles\n                return;\n            }\n            // FIXME: for the mixed ms we dont need a video -- currently\n            container = document.createElement('span');\n            container.id = 'mixedstream';\n            container.className = 'videocontainer';\n            remotes.appendChild(container);\n            Util.playSoundNotification('userJoined');\n        }\n\n        if (container) {\n            VideoLayout.addRemoteStreamElement( container,\n                stream.sid,\n                stream.getOriginalStream(),\n                stream.peerjid,\n                stream.ssrc);\n        }\n    }\n\n    my.getLargeVideoState = function () {\n        return largeVideoState;\n    };\n\n    /**\n     * Updates the large video with the given new video source.\n     */\n    my.updateLargeVideo = function(newSrc, vol, resourceJid) {\n        console.log('hover in', newSrc);\n\n        if (RTC.getVideoSrc($('#largeVideo')[0]) !== newSrc) {\n\n            $('#activeSpeaker').css('visibility', 'hidden');\n            // Due to the simulcast the localVideoSrc may have changed when the\n            // fadeOut event triggers. In that case the getJidFromVideoSrc and\n            // isVideoSrcDesktop methods will not function correctly.\n            //\n            // Also, again due to the simulcast, the updateLargeVideo method can\n            // be called multiple times almost simultaneously. Therefore, we\n            // store the state here and update only once.\n\n            largeVideoState.newSrc = newSrc;\n            largeVideoState.isVisible = $('#largeVideo').is(':visible');\n            largeVideoState.isDesktop = isVideoSrcDesktop(resourceJid);\n            if(jid2Ssrc[largeVideoState.userResourceJid] ||\n                (xmpp.myResource() &&\n                    largeVideoState.userResourceJid ===\n                    xmpp.myResource())) {\n                largeVideoState.oldResourceJid = largeVideoState.userResourceJid;\n            } else {\n                largeVideoState.oldResourceJid = null;\n            }\n            largeVideoState.userResourceJid = resourceJid;\n\n            // Screen stream is already rotated\n            largeVideoState.flipX = (newSrc === localVideoSrc) && flipXLocalVideo;\n\n            var userChanged = false;\n            if (largeVideoState.oldResourceJid !== largeVideoState.userResourceJid) {\n                userChanged = true;\n                // we want the notification to trigger even if userJid is undefined,\n                // or null.\n                $(document).trigger(\"selectedendpointchanged\", [largeVideoState.userResourceJid]);\n            }\n\n            if (!largeVideoState.updateInProgress) {\n                largeVideoState.updateInProgress = true;\n\n                var doUpdate = function () {\n\n                    Avatar.updateActiveSpeakerAvatarSrc(\n                        xmpp.findJidFromResource(\n                            largeVideoState.userResourceJid));\n\n                    if (!userChanged && largeVideoState.preload &&\n                        largeVideoState.preload !== null &&\n                        RTC.getVideoSrc($(largeVideoState.preload)[0]) === newSrc)\n                    {\n\n                        console.info('Switching to preloaded video');\n                        var attributes = $('#largeVideo').prop(\"attributes\");\n\n                        // loop through largeVideo attributes and apply them on\n                        // preload.\n                        $.each(attributes, function () {\n                            if (this.name !== 'id' && this.name !== 'src') {\n                                largeVideoState.preload.attr(this.name, this.value);\n                            }\n                        });\n\n                        largeVideoState.preload.appendTo($('#largeVideoContainer'));\n                        $('#largeVideo').attr('id', 'previousLargeVideo');\n                        largeVideoState.preload.attr('id', 'largeVideo');\n                        $('#previousLargeVideo').remove();\n\n                        largeVideoState.preload.on('loadedmetadata', function (e) {\n                            currentVideoWidth = this.videoWidth;\n                            currentVideoHeight = this.videoHeight;\n                            VideoLayout.positionLarge(currentVideoWidth, currentVideoHeight);\n                        });\n                        largeVideoState.preload = null;\n                        largeVideoState.preload_ssrc = 0;\n                    } else {\n                        RTC.setVideoSrc($('#largeVideo')[0], largeVideoState.newSrc);\n                    }\n\n                    var videoTransform = document.getElementById('largeVideo')\n                        .style.webkitTransform;\n\n                    if (largeVideoState.flipX && videoTransform !== 'scaleX(-1)') {\n                        document.getElementById('largeVideo').style.webkitTransform\n                            = \"scaleX(-1)\";\n                    }\n                    else if (!largeVideoState.flipX && videoTransform === 'scaleX(-1)') {\n                        document.getElementById('largeVideo').style.webkitTransform\n                            = \"none\";\n                    }\n\n                    // Change the way we'll be measuring and positioning large video\n\n                    VideoLayout.getVideoSize = largeVideoState.isDesktop\n                        ? getDesktopVideoSize\n                        : getCameraVideoSize;\n                    VideoLayout.getVideoPosition = largeVideoState.isDesktop\n                        ? getDesktopVideoPosition\n                        : getCameraVideoPosition;\n\n\n                    // Only if the large video is currently visible.\n                    // Disable previous dominant speaker video.\n                    if (largeVideoState.oldResourceJid) {\n                        VideoLayout.enableDominantSpeaker(\n                            largeVideoState.oldResourceJid,\n                            false);\n                    }\n\n                    // Enable new dominant speaker in the remote videos section.\n                    if (largeVideoState.userResourceJid) {\n                        VideoLayout.enableDominantSpeaker(\n                            largeVideoState.userResourceJid,\n                            true);\n                    }\n\n                    if (userChanged && largeVideoState.isVisible) {\n                        // using \"this\" should be ok because we're called\n                        // from within the fadeOut event.\n                        $(this).fadeIn(300);\n                    }\n\n                    if(userChanged) {\n                        Avatar.showUserAvatar(\n                            xmpp.findJidFromResource(\n                                largeVideoState.oldResourceJid));\n                    }\n\n                    largeVideoState.updateInProgress = false;\n                };\n\n                if (userChanged) {\n                    $('#largeVideo').fadeOut(300, doUpdate);\n                } else {\n                    doUpdate();\n                }\n            }\n        } else {\n            Avatar.showUserAvatar(\n                xmpp.findJidFromResource(\n                    largeVideoState.userResourceJid));\n        }\n\n    };\n\n    my.handleVideoThumbClicked = function(videoSrc,\n                                          noPinnedEndpointChangedEvent, \n                                          resourceJid) {\n        // Restore style for previously focused video\n        var oldContainer = null;\n        if(focusedVideoInfo) {\n            var focusResourceJid = focusedVideoInfo.resourceJid;\n            oldContainer = getParticipantContainer(focusResourceJid);\n        }\n\n        if (oldContainer) {\n            oldContainer.removeClass(\"videoContainerFocused\");\n        }\n\n        // Unlock current focused.\n        if (focusedVideoInfo && focusedVideoInfo.src === videoSrc)\n        {\n            focusedVideoInfo = null;\n            var dominantSpeakerVideo = null;\n            // Enable the currently set dominant speaker.\n            if (currentDominantSpeaker) {\n                dominantSpeakerVideo\n                    = $('#participant_' + currentDominantSpeaker + '>video')\n                        .get(0);\n\n                if (dominantSpeakerVideo) {\n                    VideoLayout.updateLargeVideo(\n                        RTC.getVideoSrc(dominantSpeakerVideo),\n                        1,\n                        currentDominantSpeaker);\n                }\n            }\n\n            if (!noPinnedEndpointChangedEvent) {\n                $(document).trigger(\"pinnedendpointchanged\");\n            }\n            return;\n        }\n\n        // Lock new video\n        focusedVideoInfo = {\n            src: videoSrc,\n            resourceJid: resourceJid\n        };\n\n        // Update focused/pinned interface.\n        if (resourceJid)\n        {\n            var container = getParticipantContainer(resourceJid);\n            container.addClass(\"videoContainerFocused\");\n\n            if (!noPinnedEndpointChangedEvent) {\n                $(document).trigger(\"pinnedendpointchanged\", [resourceJid]);\n            }\n        }\n\n        if ($('#largeVideo').attr('src') === videoSrc &&\n            VideoLayout.isLargeVideoOnTop()) {\n            return;\n        }\n\n        // Triggers a \"video.selected\" event. The \"false\" parameter indicates\n        // this isn't a prezi.\n        $(document).trigger(\"video.selected\", [false]);\n\n        VideoLayout.updateLargeVideo(videoSrc, 1, resourceJid);\n\n        $('audio').each(function (idx, el) {\n            if (el.id.indexOf('mixedmslabel') !== -1) {\n                el.volume = 0;\n                el.volume = 1;\n            }\n        });\n    };\n\n    /**\n     * Positions the large video.\n     *\n     * @param videoWidth the stream video width\n     * @param videoHeight the stream video height\n     */\n    my.positionLarge = function (videoWidth, videoHeight) {\n        var videoSpaceWidth = $('#videospace').width();\n        var videoSpaceHeight = window.innerHeight;\n\n        var videoSize = VideoLayout.getVideoSize(videoWidth,\n                                     videoHeight,\n                                     videoSpaceWidth,\n                                     videoSpaceHeight);\n\n        var largeVideoWidth = videoSize[0];\n        var largeVideoHeight = videoSize[1];\n\n        var videoPosition = VideoLayout.getVideoPosition(largeVideoWidth,\n                                             largeVideoHeight,\n                                             videoSpaceWidth,\n                                             videoSpaceHeight);\n\n        var horizontalIndent = videoPosition[0];\n        var verticalIndent = videoPosition[1];\n\n        positionVideo($('#largeVideo'),\n                      largeVideoWidth,\n                      largeVideoHeight,\n                      horizontalIndent, verticalIndent);\n    };\n\n    /**\n     * Shows/hides the large video.\n     */\n    my.setLargeVideoVisible = function(isVisible) {\n        var resourceJid = largeVideoState.userResourceJid;\n\n        if (isVisible) {\n            $('#largeVideo').css({visibility: 'visible'});\n            $('.watermark').css({visibility: 'visible'});\n            VideoLayout.enableDominantSpeaker(resourceJid, true);\n        }\n        else {\n            $('#largeVideo').css({visibility: 'hidden'});\n            $('#activeSpeaker').css('visibility', 'hidden');\n            $('.watermark').css({visibility: 'hidden'});\n            VideoLayout.enableDominantSpeaker(resourceJid, false);\n            if(focusedVideoInfo) {\n                var focusResourceJid = focusedVideoInfo.resourceJid;\n                var oldContainer = getParticipantContainer(focusResourceJid);\n\n                if (oldContainer && oldContainer.length > 0) {\n                    oldContainer.removeClass(\"videoContainerFocused\");\n                }\n                focusedVideoInfo = null;\n                if(focusResourceJid) {\n                    Avatar.showUserAvatar(\n                        xmpp.findJidFromResource(focusResourceJid));\n                }\n            }\n        }\n    };\n\n    /**\n     * Indicates if the large video is currently visible.\n     *\n     * @return <tt>true</tt> if visible, <tt>false</tt> - otherwise\n     */\n    my.isLargeVideoVisible = function() {\n        return $('#largeVideo').is(':visible');\n    };\n\n    my.isLargeVideoOnTop = function () {\n        var Etherpad = require(\"../etherpad/Etherpad\");\n        var Prezi = require(\"../prezi/Prezi\");\n        return !Prezi.isPresentationVisible() && !Etherpad.isVisible();\n    };\n\n    /**\n     * Checks if container for participant identified by given peerJid exists\n     * in the document and creates it eventually.\n     * \n     * @param peerJid peer Jid to check.\n     * @param userId user email or id for setting the avatar\n     * \n     * @return Returns <tt>true</tt> if the peer container exists,\n     * <tt>false</tt> - otherwise\n     */\n    my.ensurePeerContainerExists = function(peerJid, userId) {\n        ContactList.ensureAddContact(peerJid, userId);\n\n        var resourceJid = Strophe.getResourceFromJid(peerJid);\n\n        var videoSpanId = 'participant_' + resourceJid;\n\n        if (!$('#' + videoSpanId).length) {\n            var container =\n                VideoLayout.addRemoteVideoContainer(peerJid, videoSpanId, userId);\n            Avatar.setUserAvatar(peerJid, userId);\n            // Set default display name.\n            setDisplayName(videoSpanId);\n\n            VideoLayout.connectionIndicators[videoSpanId] =\n                new ConnectionIndicator(container, peerJid, VideoLayout);\n\n            var nickfield = document.createElement('span');\n            nickfield.className = \"nick\";\n            nickfield.appendChild(document.createTextNode(resourceJid));\n            container.appendChild(nickfield);\n\n            // In case this is not currently in the last n we don't show it.\n            if (localLastNCount\n                && localLastNCount > 0\n                && $('#remoteVideos>span').length >= localLastNCount + 2) {\n                showPeerContainer(resourceJid, 'hide');\n            }\n            else\n                VideoLayout.resizeThumbnails();\n        }\n    };\n\n    my.addRemoteVideoContainer = function(peerJid, spanId) {\n        var container = document.createElement('span');\n        container.id = spanId;\n        container.className = 'videocontainer';\n        var remotes = document.getElementById('remoteVideos');\n\n        // If the peerJid is null then this video span couldn't be directly\n        // associated with a participant (this could happen in the case of prezi).\n        if (xmpp.isModerator() && peerJid !== null)\n            addRemoteVideoMenu(peerJid, container);\n\n        remotes.appendChild(container);\n        AudioLevels.updateAudioLevelCanvas(peerJid, VideoLayout);\n\n        return container;\n    };\n\n    /**\n     * Creates an audio or video stream element.\n     */\n    my.createStreamElement = function (sid, stream) {\n        var isVideo = stream.getVideoTracks().length > 0;\n\n        var element = isVideo\n                        ? document.createElement('video')\n                        : document.createElement('audio');\n        var id = (isVideo ? 'remoteVideo_' : 'remoteAudio_')\n                    + sid + '_' + RTC.getStreamID(stream);\n\n        element.id = id;\n        element.autoplay = true;\n        element.oncontextmenu = function () { return false; };\n\n        return element;\n    };\n\n    my.addRemoteStreamElement\n        = function (container, sid, stream, peerJid, thessrc) {\n        var newElementId = null;\n\n        var isVideo = stream.getVideoTracks().length > 0;\n\n        if (container) {\n            var streamElement = VideoLayout.createStreamElement(sid, stream);\n            newElementId = streamElement.id;\n\n            container.appendChild(streamElement);\n\n            var sel = $('#' + newElementId);\n            sel.hide();\n\n            // If the container is currently visible we attach the stream.\n            if (!isVideo\n                || (container.offsetParent !== null && isVideo)) {\n                var videoStream = simulcast.getReceivingVideoStream(stream);\n                RTC.attachMediaStream(sel, videoStream);\n\n                if (isVideo)\n                    waitForRemoteVideo(sel, thessrc, stream, peerJid);\n            }\n\n            stream.onended = function () {\n                console.log('stream ended', this);\n\n                VideoLayout.removeRemoteStreamElement(\n                    stream, isVideo, container);\n\n                // NOTE(gp) it seems that under certain circumstances, the\n                // onended event is not fired and thus the contact list is not\n                // updated.\n                //\n                // The onended event of a stream should be fired when the SSRCs\n                // corresponding to that stream are removed from the SDP; but\n                // this doesn't seem to always be the case, resulting in ghost\n                // contacts.\n                //\n                // In an attempt to fix the ghost contacts problem, I'm moving\n                // the removeContact() method call in app.js, inside the\n                // 'muc.left' event handler.\n\n                //if (peerJid)\n                //    ContactList.removeContact(peerJid);\n            };\n\n            // Add click handler.\n            container.onclick = function (event) {\n                /*\n                 * FIXME It turns out that videoThumb may not exist (if there is\n                 * no actual video).\n                 */\n                var videoThumb = $('#' + container.id + '>video').get(0);\n                if (videoThumb) {\n                    VideoLayout.handleVideoThumbClicked(\n                        RTC.getVideoSrc(videoThumb),\n                        false,\n                        Strophe.getResourceFromJid(peerJid));\n                }\n\n                event.stopPropagation();\n                event.preventDefault();\n                return false;\n            };\n\n            // Add hover handler\n            $(container).hover(\n                function() {\n                    VideoLayout.showDisplayName(container.id, true);\n                },\n                function() {\n                    var videoSrc = null;\n                    if ($('#' + container.id + '>video')\n                            && $('#' + container.id + '>video').length > 0) {\n                        videoSrc = RTC.getVideoSrc($('#' + container.id + '>video').get(0));\n                    }\n\n                    // If the video has been \"pinned\" by the user we want to\n                    // keep the display name on place.\n                    if (!VideoLayout.isLargeVideoVisible()\n                            || videoSrc !== RTC.getVideoSrc($('#largeVideo')[0]))\n                        VideoLayout.showDisplayName(container.id, false);\n                }\n            );\n        }\n\n        return newElementId;\n    };\n\n    /**\n     * Removes the remote stream element corresponding to the given stream and\n     * parent container.\n     * \n     * @param stream the stream\n     * @param isVideo <tt>true</tt> if given <tt>stream</tt> is a video one.\n     * @param container\n     */\n    my.removeRemoteStreamElement = function (stream, isVideo, container) {\n        if (!container)\n            return;\n\n        var select = null;\n        var removedVideoSrc = null;\n        if (isVideo) {\n            select = $('#' + container.id + '>video');\n            removedVideoSrc = RTC.getVideoSrc(select.get(0));\n        }\n        else\n            select = $('#' + container.id + '>audio');\n\n\n        // Mark video as removed to cancel waiting loop(if video is removed\n        // before has started)\n        select.removed = true;\n        select.remove();\n\n        var audioCount = $('#' + container.id + '>audio').length;\n        var videoCount = $('#' + container.id + '>video').length;\n\n        if (!audioCount && !videoCount) {\n            console.log(\"Remove whole user\", container.id);\n            if(VideoLayout.connectionIndicators[container.id])\n                VideoLayout.connectionIndicators[container.id].remove();\n            // Remove whole container\n            container.remove();\n\n            Util.playSoundNotification('userLeft');\n            VideoLayout.resizeThumbnails();\n        }\n\n        if (removedVideoSrc)\n            VideoLayout.updateRemovedVideo(removedVideoSrc);\n    };\n\n    /**\n     * Show/hide peer container for the given resourceJid.\n     */\n    function showPeerContainer(resourceJid, state) {\n        var peerContainer = $('#participant_' + resourceJid);\n\n        if (!peerContainer)\n            return;\n\n        var isHide = state === 'hide';\n        var resizeThumbnails = false;\n\n        if (!isHide) {\n            if (!peerContainer.is(':visible')) {\n                resizeThumbnails = true;\n                peerContainer.show();\n            }\n\n            if (state == 'show')\n            {\n                // peerContainer.css('-webkit-filter', '');\n                var jid = xmpp.findJidFromResource(resourceJid);\n                Avatar.showUserAvatar(jid, false);\n            }\n            else // if (state == 'avatar')\n            {\n                // peerContainer.css('-webkit-filter', 'grayscale(100%)');\n                var jid = xmpp.findJidFromResource(resourceJid);\n                Avatar.showUserAvatar(jid, true);\n            }\n        }\n        else if (peerContainer.is(':visible') && isHide)\n        {\n            resizeThumbnails = true;\n            peerContainer.hide();\n            if(VideoLayout.connectionIndicators['participant_' + resourceJid])\n                VideoLayout.connectionIndicators['participant_' + resourceJid].hide();\n        }\n\n        if (resizeThumbnails) {\n            VideoLayout.resizeThumbnails();\n        }\n\n        // We want to be able to pin a participant from the contact list, even\n        // if he's not in the lastN set!\n        // ContactList.setClickable(resourceJid, !isHide);\n\n    };\n\n    my.inputDisplayNameHandler = function (name) {\n        if (name && nickname !== name) {\n            nickname = name;\n            window.localStorage.displayname = nickname;\n            xmpp.addToPresence(\"displayName\", nickname);\n\n            Chat.setChatConversationMode(true);\n        }\n\n        if (!$('#localDisplayName').is(\":visible\")) {\n            if (nickname)\n                $('#localDisplayName').text(nickname + \" (me)\");\n            else\n                $('#localDisplayName')\n                    .text(interfaceConfig.DEFAULT_LOCAL_DISPLAY_NAME);\n            $('#localDisplayName').show();\n        }\n\n        $('#editDisplayName').hide();\n    };\n\n    /**\n     * Shows/hides the display name on the remote video.\n     * @param videoSpanId the identifier of the video span element\n     * @param isShow indicates if the display name should be shown or hidden\n     */\n    my.showDisplayName = function(videoSpanId, isShow) {\n        var nameSpan = $('#' + videoSpanId + '>span.displayname').get(0);\n        if (isShow) {\n            if (nameSpan && nameSpan.innerHTML && nameSpan.innerHTML.length) \n                nameSpan.setAttribute(\"style\", \"display:inline-block;\");\n        }\n        else {\n            if (nameSpan)\n                nameSpan.setAttribute(\"style\", \"display:none;\");\n        }\n    };\n\n    /**\n     * Shows the presence status message for the given video.\n     */\n    my.setPresenceStatus = function (videoSpanId, statusMsg) {\n\n        if (!$('#' + videoSpanId).length) {\n            // No container\n            return;\n        }\n\n        var statusSpan = $('#' + videoSpanId + '>span.status');\n        if (!statusSpan.length) {\n            //Add status span\n            statusSpan = document.createElement('span');\n            statusSpan.className = 'status';\n            statusSpan.id = videoSpanId + '_status';\n            $('#' + videoSpanId)[0].appendChild(statusSpan);\n\n            statusSpan = $('#' + videoSpanId + '>span.status');\n        }\n\n        // Display status\n        if (statusMsg && statusMsg.length) {\n            $('#' + videoSpanId + '_status').text(statusMsg);\n            statusSpan.get(0).setAttribute(\"style\", \"display:inline-block;\");\n        }\n        else {\n            // Hide\n            statusSpan.get(0).setAttribute(\"style\", \"display:none;\");\n        }\n    };\n\n    /**\n     * Shows a visual indicator for the moderator of the conference.\n     */\n    my.showModeratorIndicator = function () {\n\n        var isModerator = xmpp.isModerator();\n        if (isModerator) {\n            var indicatorSpan = $('#localVideoContainer .focusindicator');\n\n            if (indicatorSpan.children().length === 0)\n            {\n                createModeratorIndicatorElement(indicatorSpan[0]);\n            }\n        }\n\n        var members = xmpp.getMembers();\n\n        Object.keys(members).forEach(function (jid) {\n\n            if (Strophe.getResourceFromJid(jid) === 'focus') {\n                // Skip server side focus\n                return;\n            }\n\n            var resourceJid = Strophe.getResourceFromJid(jid);\n            var videoSpanId = 'participant_' + resourceJid;\n            var videoContainer = document.getElementById(videoSpanId);\n\n            if (!videoContainer) {\n                console.error(\"No video container for \" + jid);\n                return;\n            }\n\n            var member = members[jid];\n\n            if (member.role === 'moderator') {\n                // Remove menu if peer is moderator\n                var menuSpan = $('#' + videoSpanId + '>span.remotevideomenu');\n                if (menuSpan.length) {\n                    removeRemoteVideoMenu(videoSpanId);\n                }\n                // Show moderator indicator\n                var indicatorSpan\n                    = $('#' + videoSpanId + ' .focusindicator');\n\n                if (!indicatorSpan || indicatorSpan.length === 0) {\n                    indicatorSpan = document.createElement('span');\n                    indicatorSpan.className = 'focusindicator';\n\n                    videoContainer.appendChild(indicatorSpan);\n\n                    createModeratorIndicatorElement(indicatorSpan);\n                }\n            } else if (isModerator) {\n                // We are moderator, but user is not - add menu\n                if ($('#remote_popupmenu_' + resourceJid).length <= 0) {\n                    addRemoteVideoMenu(\n                        jid,\n                        document.getElementById('participant_' + resourceJid));\n                }\n            }\n        });\n    };\n\n    /**\n     * Shows video muted indicator over small videos.\n     */\n    my.showVideoIndicator = function(videoSpanId, isMuted) {\n        var videoMutedSpan = $('#' + videoSpanId + '>span.videoMuted');\n\n        if (isMuted === 'false') {\n            if (videoMutedSpan.length > 0) {\n                videoMutedSpan.remove();\n            }\n        }\n        else {\n            if(videoMutedSpan.length == 0) {\n                videoMutedSpan = document.createElement('span');\n                videoMutedSpan.className = 'videoMuted';\n\n                $('#' + videoSpanId)[0].appendChild(videoMutedSpan);\n\n                var mutedIndicator = document.createElement('i');\n                mutedIndicator.className = 'icon-camera-disabled';\n                Util.setTooltip(mutedIndicator,\n                    \"Participant has<br/>stopped the camera.\",\n                    \"top\");\n                videoMutedSpan.appendChild(mutedIndicator);\n            }\n\n            VideoLayout.updateMutePosition(videoSpanId);\n\n        }\n    };\n\n    my.updateMutePosition = function (videoSpanId) {\n        var audioMutedSpan = $('#' + videoSpanId + '>span.audioMuted');\n        var connectionIndicator = $('#' + videoSpanId + '>div.connectionindicator');\n        var videoMutedSpan = $('#' + videoSpanId + '>span.videoMuted');\n        if(connectionIndicator.length > 0\n            && connectionIndicator[0].style.display != \"none\") {\n            audioMutedSpan.css({right: \"23px\"});\n            videoMutedSpan.css({right: ((audioMutedSpan.length > 0? 23 : 0) + 30) + \"px\"});\n        }\n        else\n        {\n            audioMutedSpan.css({right: \"0px\"});\n            videoMutedSpan.css({right: (audioMutedSpan.length > 0? 30 : 0) + \"px\"});\n        }\n    }\n    /**\n     * Shows audio muted indicator over small videos.\n     * @param {string} isMuted\n     */\n    my.showAudioIndicator = function(videoSpanId, isMuted) {\n        var audioMutedSpan = $('#' + videoSpanId + '>span.audioMuted');\n\n        if (isMuted === 'false') {\n            if (audioMutedSpan.length > 0) {\n                audioMutedSpan.popover('hide');\n                audioMutedSpan.remove();\n            }\n        }\n        else {\n            if(audioMutedSpan.length == 0 ) {\n                audioMutedSpan = document.createElement('span');\n                audioMutedSpan.className = 'audioMuted';\n                Util.setTooltip(audioMutedSpan,\n                    \"Participant is muted\",\n                    \"top\");\n\n                $('#' + videoSpanId)[0].appendChild(audioMutedSpan);\n                var mutedIndicator = document.createElement('i');\n                mutedIndicator.className = 'icon-mic-disabled';\n                audioMutedSpan.appendChild(mutedIndicator);\n\n            }\n            VideoLayout.updateMutePosition(videoSpanId);\n        }\n    };\n\n    /*\n     * Shows or hides the audio muted indicator over the local thumbnail video.\n     * @param {boolean} isMuted\n     */\n    my.showLocalAudioIndicator = function(isMuted) {\n        VideoLayout.showAudioIndicator('localVideoContainer', isMuted.toString());\n    };\n\n    /**\n     * Resizes the large video container.\n     */\n    my.resizeLargeVideoContainer = function () {\n        Chat.resizeChat();\n        var availableHeight = window.innerHeight;\n        var availableWidth = UIUtil.getAvailableVideoWidth();\n\n        if (availableWidth < 0 || availableHeight < 0) return;\n\n        $('#videospace').width(availableWidth);\n        $('#videospace').height(availableHeight);\n        $('#largeVideoContainer').width(availableWidth);\n        $('#largeVideoContainer').height(availableHeight);\n\n        var avatarSize = interfaceConfig.ACTIVE_SPEAKER_AVATAR_SIZE;\n        var top = availableHeight / 2 - avatarSize / 4 * 3;\n        $('#activeSpeaker').css('top', top);\n\n        VideoLayout.resizeThumbnails();\n    };\n\n    /**\n     * Resizes thumbnails.\n     */\n    my.resizeThumbnails = function() {\n        var videoSpaceWidth = $('#remoteVideos').width();\n\n        var thumbnailSize = VideoLayout.calculateThumbnailSize(videoSpaceWidth);\n        var width = thumbnailSize[0];\n        var height = thumbnailSize[1];\n\n        // size videos so that while keeping AR and max height, we have a\n        // nice fit\n        $('#remoteVideos').height(height);\n        $('#remoteVideos>span').width(width);\n        $('#remoteVideos>span').height(height);\n\n        $('.userAvatar').css('left', (width - height) / 2);\n\n        $(document).trigger(\"remotevideo.resized\", [width, height]);\n    };\n\n    /**\n     * Enables the dominant speaker UI.\n     *\n     * @param resourceJid the jid indicating the video element to\n     * activate/deactivate\n     * @param isEnable indicates if the dominant speaker should be enabled or\n     * disabled\n     */\n    my.enableDominantSpeaker = function(resourceJid, isEnable) {\n\n        var videoSpanId = null;\n        var videoContainerId = null;\n        if (resourceJid\n                === xmpp.myResource()) {\n            videoSpanId = 'localVideoWrapper';\n            videoContainerId = 'localVideoContainer';\n        }\n        else {\n            videoSpanId = 'participant_' + resourceJid;\n            videoContainerId = videoSpanId;\n        }\n\n        var displayName = resourceJid;\n        var nameSpan = $('#' + videoContainerId + '>span.displayname');\n        if (nameSpan.length > 0)\n            displayName = nameSpan.html();\n\n        console.log(\"UI enable dominant speaker\",\n            displayName,\n            resourceJid,\n            isEnable);\n\n        videoSpan = document.getElementById(videoContainerId);\n\n        if (!videoSpan) {\n            return;\n        }\n\n        var video = $('#' + videoSpanId + '>video');\n\n        if (video && video.length > 0) {\n            if (isEnable) {\n                var isLargeVideoVisible = VideoLayout.isLargeVideoOnTop();\n                VideoLayout.showDisplayName(videoContainerId, isLargeVideoVisible);\n\n                if (!videoSpan.classList.contains(\"dominantspeaker\"))\n                    videoSpan.classList.add(\"dominantspeaker\");\n            }\n            else {\n                VideoLayout.showDisplayName(videoContainerId, false);\n\n                if (videoSpan.classList.contains(\"dominantspeaker\"))\n                    videoSpan.classList.remove(\"dominantspeaker\");\n            }\n\n            Avatar.showUserAvatar(\n                xmpp.findJidFromResource(resourceJid));\n        }\n    };\n\n    /**\n     * Calculates the thumbnail size.\n     *\n     * @param videoSpaceWidth the width of the video space\n     */\n    my.calculateThumbnailSize = function (videoSpaceWidth) {\n        // Calculate the available height, which is the inner window height minus\n       // 39px for the header minus 2px for the delimiter lines on the top and\n       // bottom of the large video, minus the 36px space inside the remoteVideos\n       // container used for highlighting shadow.\n       var availableHeight = 100;\n\n        var numvids = $('#remoteVideos>span:visible').length;\n        if (localLastNCount && localLastNCount > 0) {\n            numvids = Math.min(localLastNCount + 1, numvids);\n        }\n\n       // Remove the 3px borders arround videos and border around the remote\n       // videos area and the 4 pixels between the local video and the others\n       //TODO: Find out where the 4 pixels come from and remove them\n       var availableWinWidth = videoSpaceWidth - 2 * 3 * numvids - 70 - 4;\n\n       var availableWidth = availableWinWidth / numvids;\n       var aspectRatio = 16.0 / 9.0;\n       var maxHeight = Math.min(160, availableHeight);\n       availableHeight = Math.min(maxHeight, availableWidth / aspectRatio);\n       if (availableHeight < availableWidth / aspectRatio) {\n           availableWidth = Math.floor(availableHeight * aspectRatio);\n       }\n\n       return [availableWidth, availableHeight];\n   };\n\n    /**\n     * Updates the remote video menu.\n     *\n     * @param jid the jid indicating the video for which we're adding a menu.\n     * @param isMuted indicates the current mute state\n     */\n    my.updateRemoteVideoMenu = function(jid, isMuted) {\n        var muteMenuItem\n            = $('#remote_popupmenu_'\n                    + Strophe.getResourceFromJid(jid)\n                    + '>li>a.mutelink');\n\n        var mutedIndicator = \"<i class='icon-mic-disabled'></i>\";\n\n        if (muteMenuItem.length) {\n            var muteLink = muteMenuItem.get(0);\n\n            if (isMuted === 'true') {\n                muteLink.innerHTML = mutedIndicator + ' Muted';\n                muteLink.className = 'mutelink disabled';\n            }\n            else {\n                muteLink.innerHTML = mutedIndicator + ' Mute';\n                muteLink.className = 'mutelink';\n            }\n        }\n    };\n\n    /**\n     * Returns the current dominant speaker resource jid.\n     */\n    my.getDominantSpeakerResourceJid = function () {\n        return currentDominantSpeaker;\n    };\n\n    /**\n     * Returns the corresponding resource jid to the given peer container\n     * DOM element.\n     *\n     * @return the corresponding resource jid to the given peer container\n     * DOM element\n     */\n    my.getPeerContainerResourceJid = function (containerElement) {\n        var i = containerElement.id.indexOf('participant_');\n\n        if (i >= 0)\n            return containerElement.id.substring(i + 12); \n    };\n\n    /**\n     * On contact list item clicked.\n     */\n    $(ContactList).bind('contactclicked', function(event, jid) {\n        if (!jid) {\n            return;\n        }\n\n        var resource = Strophe.getResourceFromJid(jid);\n        var videoContainer = $(\"#participant_\" + resource);\n        if (videoContainer.length > 0) {\n            var videoThumb = $('video', videoContainer).get(0);\n            // It is not always the case that a videoThumb exists (if there is\n            // no actual video).\n            if (videoThumb) {\n                if (videoThumb.src && videoThumb.src != '') {\n\n                    // We have a video src, great! Let's update the large video\n                    // now.\n\n                    VideoLayout.handleVideoThumbClicked(\n                        videoThumb.src,\n                        false,\n                        Strophe.getResourceFromJid(jid));\n                } else {\n\n                    // If we don't have a video src for jid, there's absolutely\n                    // no point in calling handleVideoThumbClicked; Quite\n                    // simply, it won't work because it needs an src to attach\n                    // to the large video.\n                    //\n                    // Instead, we trigger the pinned endpoint changed event to\n                    // let the bridge adjust its lastN set for myjid and store\n                    // the pinned user in the lastNPickupJid variable to be\n                    // picked up later by the lastN changed event handler.\n\n                    lastNPickupJid = jid;\n                    $(document).trigger(\"pinnedendpointchanged\", [jid]);\n                }\n            } else if (jid == xmpp.myJid()) {\n                $(\"#localVideoContainer\").click();\n            }\n        }\n    });\n\n    /**\n     * On audio muted event.\n     */\n    $(document).bind('audiomuted.muc', function (event, jid, isMuted) {\n        /*\n         // FIXME: but focus can not mute in this case ? - check\n        if (jid === xmpp.myJid()) {\n\n            // The local mute indicator is controlled locally\n            return;\n        }*/\n        var videoSpanId = null;\n        if (jid === xmpp.myJid()) {\n            videoSpanId = 'localVideoContainer';\n        } else {\n            VideoLayout.ensurePeerContainerExists(jid);\n            videoSpanId = 'participant_' + Strophe.getResourceFromJid(jid);\n        }\n\n        mutedAudios[jid] = isMuted;\n\n        if (xmpp.isModerator()) {\n            VideoLayout.updateRemoteVideoMenu(jid, isMuted);\n        }\n\n        if (videoSpanId)\n            VideoLayout.showAudioIndicator(videoSpanId, isMuted);\n    });\n\n    /**\n     * On video muted event.\n     */\n    $(document).bind('videomuted.muc', function (event, jid, value) {\n        var isMuted = (value === \"true\");\n        if(!RTC.muteRemoteVideoStream(jid, isMuted))\n            return;\n\n        Avatar.showUserAvatar(jid, isMuted);\n        var videoSpanId = null;\n        if (jid === xmpp.myJid()) {\n            videoSpanId = 'localVideoContainer';\n        } else {\n            VideoLayout.ensurePeerContainerExists(jid);\n            videoSpanId = 'participant_' + Strophe.getResourceFromJid(jid);\n        }\n\n        if (videoSpanId)\n            VideoLayout.showVideoIndicator(videoSpanId, value);\n    });\n\n    /**\n     * Display name changed.\n     */\n    my.onDisplayNameChanged =\n                    function (jid, displayName, status) {\n        var name = null;\n        if (jid === 'localVideoContainer'\n            || jid === xmpp.myJid()) {\n            name = nickname;\n            setDisplayName('localVideoContainer',\n                           displayName);\n        } else {\n            VideoLayout.ensurePeerContainerExists(jid);\n            name = $('#participant_' + Strophe.getResourceFromJid(jid) + \"_name\").text();\n            setDisplayName(\n                'participant_' + Strophe.getResourceFromJid(jid),\n                displayName,\n                status);\n        }\n\n        if(jid === 'localVideoContainer')\n            jid = xmpp.myJid();\n        if(!name || name != displayName)\n            API.triggerEvent(\"displayNameChange\",{jid: jid, displayname: displayName});\n    };\n\n    /**\n     * On dominant speaker changed event.\n     */\n    $(document).bind('dominantspeakerchanged', function (event, resourceJid) {\n        // We ignore local user events.\n        if (resourceJid\n                === xmpp.myResource())\n            return;\n\n        // Update the current dominant speaker.\n        if (resourceJid !== currentDominantSpeaker) {\n            var oldSpeakerVideoSpanId = \"participant_\" + currentDominantSpeaker,\n                newSpeakerVideoSpanId = \"participant_\" + resourceJid;\n            if($(\"#\" + oldSpeakerVideoSpanId + \">span.displayname\").text() ===\n                interfaceConfig.DEFAULT_DOMINANT_SPEAKER_DISPLAY_NAME) {\n                setDisplayName(oldSpeakerVideoSpanId, null);\n            }\n            if($(\"#\" + newSpeakerVideoSpanId + \">span.displayname\").text() ===\n                interfaceConfig.DEFAULT_REMOTE_DISPLAY_NAME) {\n                setDisplayName(newSpeakerVideoSpanId,\n                    interfaceConfig.DEFAULT_DOMINANT_SPEAKER_DISPLAY_NAME);\n            }\n            currentDominantSpeaker = resourceJid;\n        } else {\n            return;\n        }\n\n        // Obtain container for new dominant speaker.\n        var container  = document.getElementById(\n                'participant_' + resourceJid);\n\n        // Local video will not have container found, but that's ok\n        // since we don't want to switch to local video.\n        if (container && !focusedVideoInfo)\n        {\n            var video = container.getElementsByTagName(\"video\");\n\n            // Update the large video if the video source is already available,\n            // otherwise wait for the \"videoactive.jingle\" event.\n            if (video.length && video[0].currentTime > 0)\n                VideoLayout.updateLargeVideo(RTC.getVideoSrc(video[0]), resourceJid);\n        }\n    });\n\n    /**\n     * On last N change event.\n     *\n     * @param event the event that notified us\n     * @param lastNEndpoints the list of last N endpoints\n     * @param endpointsEnteringLastN the list currently entering last N\n     * endpoints\n     */\n    $(document).bind('lastnchanged', function ( event,\n                                                lastNEndpoints,\n                                                endpointsEnteringLastN,\n                                                stream) {\n        if (lastNCount !== lastNEndpoints.length)\n            lastNCount = lastNEndpoints.length;\n\n        lastNEndpointsCache = lastNEndpoints;\n\n        // Say A, B, C, D, E, and F are in a conference and LastN = 3.\n        //\n        // If LastN drops to, say, 2, because of adaptivity, then E should see\n        // thumbnails for A, B and C. A and B are in E's server side LastN set,\n        // so E sees them. C is only in E's local LastN set.\n        //\n        // If F starts talking and LastN = 3, then E should see thumbnails for\n        // F, A, B. B gets \"ejected\" from E's server side LastN set, but it\n        // enters E's local LastN ejecting C.\n\n        // Increase the local LastN set size, if necessary.\n        if (lastNCount > localLastNCount) {\n            localLastNCount = lastNCount;\n        }\n\n        // Update the local LastN set preserving the order in which the\n        // endpoints appeared in the LastN/local LastN set.\n\n        var nextLocalLastNSet = lastNEndpoints.slice(0);\n        for (var i = 0; i < localLastNSet.length; i++) {\n            if (nextLocalLastNSet.length >= localLastNCount) {\n                break;\n            }\n\n            var resourceJid = localLastNSet[i];\n            if (nextLocalLastNSet.indexOf(resourceJid) === -1) {\n                nextLocalLastNSet.push(resourceJid);\n            }\n        }\n\n        localLastNSet = nextLocalLastNSet;\n\n        var updateLargeVideo = false;\n\n        // Handle LastN/local LastN changes.\n        $('#remoteVideos>span').each(function( index, element ) {\n            var resourceJid = VideoLayout.getPeerContainerResourceJid(element);\n\n            var isReceived = true;\n            if (resourceJid\n                && lastNEndpoints.indexOf(resourceJid) < 0\n                && localLastNSet.indexOf(resourceJid) < 0) {\n                console.log(\"Remove from last N\", resourceJid);\n                showPeerContainer(resourceJid, 'hide');\n                isReceived = false;\n            } else if (resourceJid\n                && $('#participant_' + resourceJid).is(':visible')\n                && lastNEndpoints.indexOf(resourceJid) < 0\n                && localLastNSet.indexOf(resourceJid) >= 0) {\n                showPeerContainer(resourceJid, 'avatar');\n                isReceived = false;\n            }\n\n            if (!isReceived) {\n                // resourceJid has dropped out of the server side lastN set, so\n                // it is no longer being received. If resourceJid was being\n                // displayed in the large video we have to switch to another\n                // user.\n                var largeVideoResource = largeVideoState.userResourceJid;\n                if (!updateLargeVideo && resourceJid === largeVideoResource) {\n                    updateLargeVideo = true;\n                }\n            }\n        });\n\n        if (!endpointsEnteringLastN || endpointsEnteringLastN.length < 0)\n            endpointsEnteringLastN = lastNEndpoints;\n\n        if (endpointsEnteringLastN && endpointsEnteringLastN.length > 0) {\n            endpointsEnteringLastN.forEach(function (resourceJid) {\n\n                var isVisible = $('#participant_' + resourceJid).is(':visible');\n                showPeerContainer(resourceJid, 'show');\n                if (!isVisible) {\n                    console.log(\"Add to last N\", resourceJid);\n\n                    var jid = xmpp.findJidFromResource(resourceJid);\n                    var mediaStream = RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE];\n                    var sel = $('#participant_' + resourceJid + '>video');\n\n                    var videoStream = simulcast.getReceivingVideoStream(\n                        mediaStream.stream);\n                    RTC.attachMediaStream(sel, videoStream);\n                    if (lastNPickupJid == mediaStream.peerjid) {\n                        // Clean up the lastN pickup jid.\n                        lastNPickupJid = null;\n\n                        // Don't fire the events again, they've already\n                        // been fired in the contact list click handler.\n                        VideoLayout.handleVideoThumbClicked(\n                            $(sel).attr('src'),\n                            false,\n                            Strophe.getResourceFromJid(mediaStream.peerjid));\n\n                        updateLargeVideo = false;\n                    }\n                    waitForRemoteVideo(sel, mediaStream.ssrc, mediaStream.stream, resourceJid);\n                }\n            })\n        }\n\n        // The endpoint that was being shown in the large video has dropped out\n        // of the lastN set and there was no lastN pickup jid. We need to update\n        // the large video now.\n\n        if (updateLargeVideo) {\n\n            var resource, container, src;\n            var myResource\n                = xmpp.myResource();\n\n            // Find out which endpoint to show in the large video.\n            for (var i = 0; i < lastNEndpoints.length; i++) {\n                resource = lastNEndpoints[i];\n                if (!resource || resource === myResource)\n                    continue;\n\n                container = $(\"#participant_\" + resource);\n                if (container.length == 0)\n                    continue;\n\n                src = $('video', container).attr('src');\n                if (!src)\n                    continue;\n\n                // videoSrcToSsrc needs to be update for this call to succeed.\n                VideoLayout.updateLargeVideo(src);\n                break;\n\n            }\n        }\n    });\n\n    $(document).bind('simulcastlayerschanging', function (event, endpointSimulcastLayers) {\n        endpointSimulcastLayers.forEach(function (esl) {\n\n            var resource = esl.endpoint;\n\n            // if lastN is enabled *and* the endpoint is *not* in the lastN set,\n            // then ignore the event (= do not preload anything).\n            //\n            // The bridge could probably stop sending this message if it's for\n            // an endpoint that's not in lastN.\n\n            if (lastNCount != -1\n                && (lastNCount < 1 || lastNEndpointsCache.indexOf(resource) === -1)) {\n                return;\n            }\n\n            var primarySSRC = esl.simulcastLayer.primarySSRC;\n\n            // Get session and stream from primary ssrc.\n            var res = simulcast.getReceivingVideoStreamBySSRC(primarySSRC);\n            var sid = res.sid;\n            var electedStream = res.stream;\n\n            if (sid && electedStream) {\n                var msid = simulcast.getRemoteVideoStreamIdBySSRC(primarySSRC);\n\n                console.info([esl, primarySSRC, msid, sid, electedStream]);\n\n                var msidParts = msid.split(' ');\n\n                var preload = (Strophe.getResourceFromJid(ssrc2jid[primarySSRC]) == largeVideoState.userResourceJid);\n\n                if (preload) {\n                    if (largeVideoState.preload)\n                    {\n                        $(largeVideoState.preload).remove();\n                    }\n                    console.info('Preloading remote video');\n                    largeVideoState.preload = $('<video autoplay></video>');\n                    // ssrcs are unique in an rtp session\n                    largeVideoState.preload_ssrc = primarySSRC;\n\n                    RTC.attachMediaStream(largeVideoState.preload, electedStream)\n                }\n\n            } else {\n                console.error('Could not find a stream or a session.', sid, electedStream);\n            }\n        });\n    });\n\n    /**\n     * On simulcast layers changed event.\n     */\n    $(document).bind('simulcastlayerschanged', function (event, endpointSimulcastLayers) {\n        endpointSimulcastLayers.forEach(function (esl) {\n\n            var resource = esl.endpoint;\n\n            // if lastN is enabled *and* the endpoint is *not* in the lastN set,\n            // then ignore the event (= do not change large video/thumbnail\n            // SRCs).\n            //\n            // Note that even if we ignore the \"changed\" event in this event\n            // handler, the bridge must continue sending these events because\n            // the simulcast code in simulcast.js uses it to know what's going\n            // to be streamed by the bridge when/if the endpoint gets back into\n            // the lastN set.\n\n            if (lastNCount != -1\n                && (lastNCount < 1 || lastNEndpointsCache.indexOf(resource) === -1)) {\n                return;\n            }\n\n            var primarySSRC = esl.simulcastLayer.primarySSRC;\n\n            // Get session and stream from primary ssrc.\n            var res = simulcast.getReceivingVideoStreamBySSRC(primarySSRC);\n            var sid = res.sid;\n            var electedStream = res.stream;\n\n            if (sid && electedStream) {\n                var msid = simulcast.getRemoteVideoStreamIdBySSRC(primarySSRC);\n\n                console.info('Switching simulcast substream.');\n                console.info([esl, primarySSRC, msid, sid, electedStream]);\n\n                var msidParts = msid.split(' ');\n                var selRemoteVideo = $(['#', 'remoteVideo_', sid, '_', msidParts[0]].join(''));\n\n                var updateLargeVideo = (Strophe.getResourceFromJid(ssrc2jid[primarySSRC])\n                    == largeVideoState.userResourceJid);\n                var updateFocusedVideoSrc = (focusedVideoInfo && focusedVideoInfo.src && focusedVideoInfo.src != '' &&\n                    (RTC.getVideoSrc(selRemoteVideo[0]) == focusedVideoInfo.src));\n\n                var electedStreamUrl;\n                if (largeVideoState.preload_ssrc == primarySSRC)\n                {\n                    RTC.setVideoSrc(selRemoteVideo[0], RTC.getVideoSrc(largeVideoState.preload[0]));\n                }\n                else\n                {\n                    if (largeVideoState.preload\n                        && largeVideoState.preload != null) {\n                        $(largeVideoState.preload).remove();\n                    }\n\n                    largeVideoState.preload_ssrc = 0;\n\n                    RTC.attachMediaStream(selRemoteVideo, electedStream);\n                }\n\n                var jid = ssrc2jid[primarySSRC];\n                jid2Ssrc[jid] = primarySSRC;\n\n                if (updateLargeVideo) {\n                    VideoLayout.updateLargeVideo(RTC.getVideoSrc(selRemoteVideo[0]), null,\n                        Strophe.getResourceFromJid(jid));\n                }\n\n                if (updateFocusedVideoSrc) {\n                    focusedVideoInfo.src = RTC.getVideoSrc(selRemoteVideo[0]);\n                }\n\n                var videoId;\n                if(resource == xmpp.myResource())\n                {\n                    videoId = \"localVideoContainer\";\n                }\n                else\n                {\n                    videoId = \"participant_\" + resource;\n                }\n                var connectionIndicator = VideoLayout.connectionIndicators[videoId];\n                if(connectionIndicator)\n                    connectionIndicator.updatePopoverData();\n\n            } else {\n                console.error('Could not find a stream or a sid.', sid, electedStream);\n            }\n        });\n    });\n\n    /**\n     * Updates local stats\n     * @param percent\n     * @param object\n     */\n    my.updateLocalConnectionStats = function (percent, object) {\n        var resolution = null;\n        if(object.resolution !== null)\n        {\n            resolution = object.resolution;\n            object.resolution = resolution[xmpp.myJid()];\n            delete resolution[xmpp.myJid()];\n        }\n        updateStatsIndicator(\"localVideoContainer\", percent, object);\n        for(var jid in resolution)\n        {\n            if(resolution[jid] === null)\n                continue;\n            var id = 'participant_' + Strophe.getResourceFromJid(jid);\n            if(VideoLayout.connectionIndicators[id])\n            {\n                VideoLayout.connectionIndicators[id].updateResolution(resolution[jid]);\n            }\n        }\n\n    };\n\n    /**\n     * Updates remote stats.\n     * @param jid the jid associated with the stats\n     * @param percent the connection quality percent\n     * @param object the stats data\n     */\n    my.updateConnectionStats = function (jid, percent, object) {\n        var resourceJid = Strophe.getResourceFromJid(jid);\n\n        var videoSpanId = 'participant_' + resourceJid;\n        updateStatsIndicator(videoSpanId, percent, object);\n    };\n\n    /**\n     * Removes the connection\n     * @param jid\n     */\n    my.removeConnectionIndicator = function (jid) {\n        if(VideoLayout.connectionIndicators['participant_' + Strophe.getResourceFromJid(jid)])\n            VideoLayout.connectionIndicators['participant_' + Strophe.getResourceFromJid(jid)].remove();\n    };\n\n    /**\n     * Hides the connection indicator\n     * @param jid\n     */\n    my.hideConnectionIndicator = function (jid) {\n        if(VideoLayout.connectionIndicators['participant_' + Strophe.getResourceFromJid(jid)])\n            VideoLayout.connectionIndicators['participant_' + Strophe.getResourceFromJid(jid)].hide();\n    };\n\n    /**\n     * Hides all the indicators\n     */\n    my.onStatsStop = function () {\n        for(var indicator in VideoLayout.connectionIndicators)\n        {\n            VideoLayout.connectionIndicators[indicator].hideIndicator();\n        }\n    };\n\n    return my;\n}(VideoLayout || {}));\n\nmodule.exports = VideoLayout;","//var nouns = [\n//];\nvar pluralNouns = [\n    \"Aliens\", \"Animals\", \"Antelopes\", \"Ants\", \"Apes\", \"Apples\", \"Baboons\", \"Bacteria\", \"Badgers\", \"Bananas\", \"Bats\",\n    \"Bears\", \"Birds\", \"Bonobos\", \"Brides\", \"Bugs\", \"Bulls\", \"Butterflies\", \"Cheetahs\",\n    \"Cherries\", \"Chicken\", \"Children\", \"Chimps\", \"Clowns\", \"Cows\", \"Creatures\", \"Dinosaurs\", \"Dogs\", \"Dolphins\",\n    \"Donkeys\", \"Dragons\", \"Ducks\", \"Dwarfs\", \"Eagles\", \"Elephants\", \"Elves\", \"FAIL\", \"Fathers\",\n    \"Fish\", \"Flowers\", \"Frogs\", \"Fruit\", \"Fungi\", \"Galaxies\", \"Geese\", \"Goats\",\n    \"Gorillas\", \"Hedgehogs\", \"Hippos\", \"Horses\", \"Hunters\", \"Insects\", \"Kids\", \"Knights\",\n    \"Lemons\", \"Lemurs\", \"Leopards\", \"LifeForms\", \"Lions\", \"Lizards\", \"Mice\", \"Monkeys\", \"Monsters\",\n    \"Mushrooms\", \"Octopodes\", \"Oranges\", \"Orangutans\", \"Organisms\", \"Pants\", \"Parrots\", \"Penguins\",\n    \"People\", \"Pigeons\", \"Pigs\", \"Pineapples\", \"Plants\", \"Potatoes\", \"Priests\", \"Rats\", \"Reptiles\", \"Reptilians\",\n    \"Rhinos\", \"Seagulls\", \"Sheep\", \"Siblings\", \"Snakes\", \"Spaghetti\", \"Spiders\", \"Squid\", \"Squirrels\",\n    \"Stars\", \"Students\", \"Teachers\", \"Tigers\", \"Tomatoes\", \"Trees\", \"Vampires\", \"Vegetables\", \"Viruses\", \"Vulcans\",\n    \"Warewolves\", \"Weasels\", \"Whales\", \"Witches\", \"Wizards\", \"Wolves\", \"Workers\", \"Worms\", \"Zebras\"\n];\n//var places = [\n//\"Pub\", \"University\", \"Airport\", \"Library\", \"Mall\", \"Theater\", \"Stadium\", \"Office\", \"Show\", \"Gallows\", \"Beach\",\n// \"Cemetery\", \"Hospital\", \"Reception\", \"Restaurant\", \"Bar\", \"Church\", \"House\", \"School\", \"Square\", \"Village\",\n// \"Cinema\", \"Movies\", \"Party\", \"Restroom\", \"End\", \"Jail\", \"PostOffice\", \"Station\", \"Circus\", \"Gates\", \"Entrance\",\n// \"Bridge\"\n//];\nvar verbs = [\n    \"Abandon\", \"Adapt\", \"Advertise\", \"Answer\", \"Anticipate\", \"Appreciate\",\n    \"Approach\", \"Argue\", \"Ask\", \"Bite\", \"Blossom\", \"Blush\", \"Breathe\", \"Breed\", \"Bribe\", \"Burn\", \"Calculate\",\n    \"Clean\", \"Code\", \"Communicate\", \"Compute\", \"Confess\", \"Confiscate\", \"Conjugate\", \"Conjure\", \"Consume\",\n    \"Contemplate\", \"Crawl\", \"Dance\", \"Delegate\", \"Devour\", \"Develop\", \"Differ\", \"Discuss\",\n    \"Dissolve\", \"Drink\", \"Eat\", \"Elaborate\", \"Emancipate\", \"Estimate\", \"Expire\", \"Extinguish\",\n    \"Extract\", \"FAIL\", \"Facilitate\", \"Fall\", \"Feed\", \"Finish\", \"Floss\", \"Fly\", \"Follow\", \"Fragment\", \"Freeze\",\n    \"Gather\", \"Glow\", \"Grow\", \"Hex\", \"Hide\", \"Hug\", \"Hurry\", \"Improve\", \"Intersect\", \"Investigate\", \"Jinx\",\n    \"Joke\", \"Jubilate\", \"Kiss\", \"Laugh\", \"Manage\", \"Meet\", \"Merge\", \"Move\", \"Object\", \"Observe\", \"Offer\",\n    \"Paint\", \"Participate\", \"Party\", \"Perform\", \"Plan\", \"Pursue\", \"Pierce\", \"Play\", \"Postpone\", \"Pray\", \"Proclaim\",\n    \"Question\", \"Read\", \"Reckon\", \"Rejoice\", \"Represent\", \"Resize\", \"Rhyme\", \"Scream\", \"Search\", \"Select\", \"Share\", \"Shoot\",\n    \"Shout\", \"Signal\", \"Sing\", \"Skate\", \"Sleep\", \"Smile\", \"Smoke\", \"Solve\", \"Spell\", \"Steer\", \"Stink\",\n    \"Substitute\", \"Swim\", \"Taste\", \"Teach\", \"Terminate\", \"Think\", \"Type\", \"Unite\", \"Vanish\", \"Worship\"\n];\nvar adverbs = [\n    \"Absently\", \"Accurately\", \"Accusingly\", \"Adorably\", \"AllTheTime\", \"Alone\", \"Always\", \"Amazingly\", \"Angrily\",\n    \"Anxiously\", \"Anywhere\", \"Appallingly\", \"Apparently\", \"Articulately\", \"Astonishingly\", \"Badly\", \"Barely\",\n    \"Beautifully\", \"Blindly\", \"Bravely\", \"Brightly\", \"Briskly\", \"Brutally\", \"Calmly\", \"Carefully\", \"Casually\",\n    \"Cautiously\", \"Cleverly\", \"Constantly\", \"Correctly\", \"Crazily\", \"Curiously\", \"Cynically\", \"Daily\",\n    \"Dangerously\", \"Deliberately\", \"Delicately\", \"Desperately\", \"Discreetly\", \"Eagerly\", \"Easily\", \"Euphoricly\",\n    \"Evenly\", \"Everywhere\", \"Exactly\", \"Expectantly\", \"Extensively\", \"FAIL\", \"Ferociously\", \"Fiercely\", \"Finely\",\n    \"Flatly\", \"Frequently\", \"Frighteningly\", \"Gently\", \"Gloriously\", \"Grimly\", \"Guiltily\", \"Happily\",\n    \"Hard\", \"Hastily\", \"Heroically\", \"High\", \"Highly\", \"Hourly\", \"Humbly\", \"Hysterically\", \"Immensely\",\n    \"Impartially\", \"Impolitely\", \"Indifferently\", \"Intensely\", \"Jealously\", \"Jovially\", \"Kindly\", \"Lazily\",\n    \"Lightly\", \"Loudly\", \"Lovingly\", \"Loyally\", \"Magnificently\", \"Malevolently\", \"Merrily\", \"Mightily\", \"Miserably\",\n    \"Mysteriously\", \"NOT\", \"Nervously\", \"Nicely\", \"Nowhere\", \"Objectively\", \"Obnoxiously\", \"Obsessively\",\n    \"Obviously\", \"Often\", \"Painfully\", \"Patiently\", \"Playfully\", \"Politely\", \"Poorly\", \"Precisely\", \"Promptly\",\n    \"Quickly\", \"Quietly\", \"Randomly\", \"Rapidly\", \"Rarely\", \"Recklessly\", \"Regularly\", \"Remorsefully\", \"Responsibly\",\n    \"Rudely\", \"Ruthlessly\", \"Sadly\", \"Scornfully\", \"Seamlessly\", \"Seldom\", \"Selfishly\", \"Seriously\", \"Shakily\",\n    \"Sharply\", \"Sideways\", \"Silently\", \"Sleepily\", \"Slightly\", \"Slowly\", \"Slyly\", \"Smoothly\", \"Softly\", \"Solemnly\", \"Steadily\", \"Sternly\", \"Strangely\", \"Strongly\", \"Stunningly\", \"Surely\", \"Tenderly\", \"Thoughtfully\",\n    \"Tightly\", \"Uneasily\", \"Vanishingly\", \"Violently\", \"Warmly\", \"Weakly\", \"Wearily\", \"Weekly\", \"Weirdly\", \"Well\",\n    \"Well\", \"Wickedly\", \"Wildly\", \"Wisely\", \"Wonderfully\", \"Yearly\"\n];\nvar adjectives = [\n    \"Abominable\", \"Accurate\", \"Adorable\", \"All\", \"Alleged\", \"Ancient\", \"Angry\", \"Angry\", \"Anxious\", \"Appalling\",\n    \"Apparent\", \"Astonishing\", \"Attractive\", \"Awesome\", \"Baby\", \"Bad\", \"Beautiful\", \"Benign\", \"Big\", \"Bitter\",\n    \"Blind\", \"Blue\", \"Bold\", \"Brave\", \"Bright\", \"Brisk\", \"Calm\", \"Camouflaged\", \"Casual\", \"Cautious\",\n    \"Choppy\", \"Chosen\", \"Clever\", \"Cold\", \"Cool\", \"Crawly\", \"Crazy\", \"Creepy\", \"Cruel\", \"Curious\", \"Cynical\",\n    \"Dangerous\", \"Dark\", \"Delicate\", \"Desperate\", \"Difficult\", \"Discreet\", \"Disguised\", \"Dizzy\",\n    \"Dumb\", \"Eager\", \"Easy\", \"Edgy\", \"Electric\", \"Elegant\", \"Emancipated\", \"Enormous\", \"Euphoric\", \"Evil\",\n    \"FAIL\", \"Fast\", \"Ferocious\", \"Fierce\", \"Fine\", \"Flawed\", \"Flying\", \"Foolish\", \"Foxy\",\n    \"Freezing\", \"Funny\", \"Furious\", \"Gentle\", \"Glorious\", \"Golden\", \"Good\", \"Green\", \"Green\", \"Guilty\",\n    \"Hairy\", \"Happy\", \"Hard\", \"Hasty\", \"Hazy\", \"Heroic\", \"Hostile\", \"Hot\", \"Humble\", \"Humongous\",\n    \"Humorous\", \"Hysterical\", \"Idealistic\", \"Ignorant\", \"Immense\", \"Impartial\", \"Impolite\", \"Indifferent\",\n    \"Infuriated\", \"Insightful\", \"Intense\", \"Interesting\", \"Intimidated\", \"Intriguing\", \"Jealous\", \"Jolly\", \"Jovial\",\n    \"Jumpy\", \"Kind\", \"Laughing\", \"Lazy\", \"Liquid\", \"Lonely\", \"Longing\", \"Loud\", \"Loving\", \"Loyal\", \"Macabre\", \"Mad\",\n    \"Magical\", \"Magnificent\", \"Malevolent\", \"Medieval\", \"Memorable\", \"Mere\", \"Merry\", \"Mighty\",\n    \"Mischievous\", \"Miserable\", \"Modified\", \"Moody\", \"Most\", \"Mysterious\", \"Mystical\", \"Needy\",\n    \"Nervous\", \"Nice\", \"Objective\", \"Obnoxious\", \"Obsessive\", \"Obvious\", \"Opinionated\", \"Orange\",\n    \"Painful\", \"Passionate\", \"Perfect\", \"Pink\", \"Playful\", \"Poisonous\", \"Polite\", \"Poor\", \"Popular\", \"Powerful\",\n    \"Precise\", \"Preserved\", \"Pretty\", \"Purple\", \"Quick\", \"Quiet\", \"Random\", \"Rapid\", \"Rare\", \"Real\",\n    \"Reassuring\", \"Reckless\", \"Red\", \"Regular\", \"Remorseful\", \"Responsible\", \"Rich\", \"Rude\", \"Ruthless\",\n    \"Sad\", \"Scared\", \"Scary\", \"Scornful\", \"Screaming\", \"Selfish\", \"Serious\", \"Shady\", \"Shaky\", \"Sharp\",\n    \"Shiny\", \"Shy\", \"Simple\", \"Sleepy\", \"Slow\", \"Sly\", \"Small\", \"Smart\", \"Smelly\", \"Smiling\", \"Smooth\",\n    \"Smug\", \"Sober\", \"Soft\", \"Solemn\", \"Square\", \"Square\", \"Steady\", \"Strange\", \"Strong\",\n    \"Stunning\", \"Subjective\", \"Successful\", \"Surly\", \"Sweet\", \"Tactful\", \"Tense\",\n    \"Thoughtful\", \"Tight\", \"Tiny\", \"Tolerant\", \"Uneasy\", \"Unique\", \"Unseen\", \"Warm\", \"Weak\",\n    \"Weird\", \"WellCooked\", \"Wild\", \"Wise\", \"Witty\", \"Wonderful\", \"Worried\", \"Yellow\", \"Young\",\n    \"Zealous\"\n    ];\n//var pronouns = [\n//];\n//var conjunctions = [\n//\"And\", \"Or\", \"For\", \"Above\", \"Before\", \"Against\", \"Between\"\n//];\n\n/*\n * Maps a string (category name) to the array of words from that category.\n */\nvar CATEGORIES =\n{\n    //\"_NOUN_\": nouns,\n    \"_PLURALNOUN_\": pluralNouns,\n    //\"_PLACE_\": places,\n    \"_VERB_\": verbs,\n    \"_ADVERB_\": adverbs,\n    \"_ADJECTIVE_\": adjectives\n    //\"_PRONOUN_\": pronouns,\n    //\"_CONJUNCTION_\": conjunctions,\n};\n\nvar PATTERNS = [\n    \"_ADJECTIVE__PLURALNOUN__VERB__ADVERB_\"\n\n    // BeautifulFungiOrSpaghetti\n    //\"_ADJECTIVE__PLURALNOUN__CONJUNCTION__PLURALNOUN_\",\n\n    // AmazinglyScaryToy\n    //\"_ADVERB__ADJECTIVE__NOUN_\",\n\n    // NeitherTrashNorRifle\n    //\"Neither_NOUN_Nor_NOUN_\",\n    //\"Either_NOUN_Or_NOUN_\",\n\n    // EitherCopulateOrInvestigate\n    //\"Either_VERB_Or_VERB_\",\n    //\"Neither_VERB_Nor_VERB_\",\n\n    //\"The_ADJECTIVE__ADJECTIVE__NOUN_\",\n    //\"The_ADVERB__ADJECTIVE__NOUN_\",\n    //\"The_ADVERB__ADJECTIVE__NOUN_s\",\n    //\"The_ADVERB__ADJECTIVE__PLURALNOUN__VERB_\",\n\n    // WolvesComputeBadly\n    //\"_PLURALNOUN__VERB__ADVERB_\",\n\n    // UniteFacilitateAndMerge\n    //\"_VERB__VERB_And_VERB_\",\n\n    //NastyWitchesAtThePub\n    //\"_ADJECTIVE__PLURALNOUN_AtThe_PLACE_\",\n];\n\n\n/*\n * Returns a random element from the array 'arr'\n */\nfunction randomElement(arr)\n{\n    return arr[Math.floor(Math.random() * arr.length)];\n}\n\n/*\n * Returns true if the string 's' contains one of the\n * template strings.\n */\nfunction hasTemplate(s)\n{\n    for (var template in CATEGORIES){\n        if (s.indexOf(template) >= 0){\n            return true;\n        }\n    }\n}\n\n/**\n * Generates new room name.\n */\nvar RoomNameGenerator = {\n    generateRoomWithoutSeparator: function()\n    {\n        // Note that if more than one pattern is available, the choice of 'name' won't be random (names from patterns\n        // with fewer options will have higher probability of being chosen that names from patterns with more options).\n        var name = randomElement(PATTERNS);\n        var word;\n        while (hasTemplate(name)){\n            for (var template in CATEGORIES){\n                word = randomElement(CATEGORIES[template]);\n                name = name.replace(template, word);\n            }\n        }\n\n        return name;\n    }\n}\n\nmodule.exports = RoomNameGenerator;\n","var animateTimeout, updateTimeout;\n\nvar RoomNameGenerator = require(\"./RoomnameGenerator\");\n\nfunction enter_room()\n{\n    var val = $(\"#enter_room_field\").val();\n    if(!val) {\n        val = $(\"#enter_room_field\").attr(\"room_name\");\n    }\n    if (val) {\n        window.location.pathname = \"/\" + val;\n    }\n}\n\nfunction animate(word) {\n    var currentVal = $(\"#enter_room_field\").attr(\"placeholder\");\n    $(\"#enter_room_field\").attr(\"placeholder\", currentVal + word.substr(0, 1));\n    animateTimeout = setTimeout(function() {\n        animate(word.substring(1, word.length))\n    }, 70);\n}\n\nfunction update_roomname()\n{\n    var word = RoomNameGenerator.generateRoomWithoutSeparator();\n    $(\"#enter_room_field\").attr(\"room_name\", word);\n    $(\"#enter_room_field\").attr(\"placeholder\", \"\");\n    clearTimeout(animateTimeout);\n    animate(word);\n    updateTimeout = setTimeout(update_roomname, 10000);\n}\n\n\nfunction setupWelcomePage()\n{\n    $(\"#videoconference_page\").hide();\n    $(\"#domain_name\").text(\n            window.location.protocol + \"//\" + window.location.host + \"/\");\n    $(\"span[name='appName']\").text(interfaceConfig.APP_NAME);\n\n    if (interfaceConfig.SHOW_JITSI_WATERMARK) {\n        var leftWatermarkDiv\n            = $(\"#welcome_page_header div[class='watermark leftwatermark']\");\n        if(leftWatermarkDiv && leftWatermarkDiv.length > 0)\n        {\n            leftWatermarkDiv.css({display: 'block'});\n            leftWatermarkDiv.parent().get(0).href\n                = interfaceConfig.JITSI_WATERMARK_LINK;\n        }\n\n    }\n\n    if (interfaceConfig.SHOW_BRAND_WATERMARK) {\n        var rightWatermarkDiv\n            = $(\"#welcome_page_header div[class='watermark rightwatermark']\");\n        if(rightWatermarkDiv && rightWatermarkDiv.length > 0) {\n            rightWatermarkDiv.css({display: 'block'});\n            rightWatermarkDiv.parent().get(0).href\n                = interfaceConfig.BRAND_WATERMARK_LINK;\n            rightWatermarkDiv.get(0).style.backgroundImage\n                = \"url(images/rightwatermark.png)\";\n        }\n    }\n\n    if (interfaceConfig.SHOW_POWERED_BY) {\n        $(\"#welcome_page_header>a[class='poweredby']\")\n            .css({display: 'block'});\n    }\n\n    $(\"#enter_room_button\").click(function()\n    {\n        enter_room();\n    });\n\n    $(\"#enter_room_field\").keydown(function (event) {\n        if (event.keyCode === 13 /* enter */) {\n            enter_room();\n        }\n    });\n\n    if (!(interfaceConfig.GENERATE_ROOMNAMES_ON_WELCOME_PAGE === false)){\n        var updateTimeout;\n        var animateTimeout;\n        $(\"#reload_roomname\").click(function () {\n            clearTimeout(updateTimeout);\n            clearTimeout(animateTimeout);\n            update_roomname();\n        });\n        $(\"#reload_roomname\").show();\n\n\n        update_roomname();\n    }\n\n    $(\"#disable_welcome\").click(function () {\n        window.localStorage.welcomePageDisabled\n            = $(\"#disable_welcome\").is(\":checked\");\n    });\n\n}\n\nmodule.exports = setupWelcomePage;"]}
diff --git a/libs/modules/connectionquality.bundle.js b/libs/modules/connectionquality.bundle.js
index 50b92a66d..ee43400f6 100644
--- a/libs/modules/connectionquality.bundle.js
+++ b/libs/modules/connectionquality.bundle.js
@@ -30,8 +30,7 @@ function startSendingStats() {
* Sends statistics to other participants
*/
function sendStats() {
- connection.emuc.addConnectionInfoToPresence(convertToMUCStats(stats));
- connection.emuc.sendPresence();
+ xmpp.addToPresence("connectionQuality", convertToMUCStats(stats));
}
/**
@@ -119,4 +118,5 @@ var ConnectionQuality = {
module.exports = ConnectionQuality;
},{}]},{},[1])(1)
-});
\ No newline at end of file
+});
+//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi91c3IvbG9jYWwvbGliL25vZGVfbW9kdWxlcy9icm93c2VyaWZ5L25vZGVfbW9kdWxlcy9icm93c2VyLXBhY2svX3ByZWx1ZGUuanMiLCIvVXNlcnMvaHJpc3RvL0RvY3VtZW50cy93b3Jrc3BhY2Uvaml0c2ktbWVldC9tb2R1bGVzL2Nvbm5lY3Rpb25xdWFsaXR5L2Nvbm5lY3Rpb25xdWFsaXR5LmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBO0FDQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJmaWxlIjoiZ2VuZXJhdGVkLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXNDb250ZW50IjpbIihmdW5jdGlvbiBlKHQsbixyKXtmdW5jdGlvbiBzKG8sdSl7aWYoIW5bb10pe2lmKCF0W29dKXt2YXIgYT10eXBlb2YgcmVxdWlyZT09XCJmdW5jdGlvblwiJiZyZXF1aXJlO2lmKCF1JiZhKXJldHVybiBhKG8sITApO2lmKGkpcmV0dXJuIGkobywhMCk7dmFyIGY9bmV3IEVycm9yKFwiQ2Fubm90IGZpbmQgbW9kdWxlICdcIitvK1wiJ1wiKTt0aHJvdyBmLmNvZGU9XCJNT0RVTEVfTk9UX0ZPVU5EXCIsZn12YXIgbD1uW29dPXtleHBvcnRzOnt9fTt0W29dWzBdLmNhbGwobC5leHBvcnRzLGZ1bmN0aW9uKGUpe3ZhciBuPXRbb11bMV1bZV07cmV0dXJuIHMobj9uOmUpfSxsLGwuZXhwb3J0cyxlLHQsbixyKX1yZXR1cm4gbltvXS5leHBvcnRzfXZhciBpPXR5cGVvZiByZXF1aXJlPT1cImZ1bmN0aW9uXCImJnJlcXVpcmU7Zm9yKHZhciBvPTA7bzxyLmxlbmd0aDtvKyspcyhyW29dKTtyZXR1cm4gc30pIiwiLyoqXG4gKiBsb2NhbCBzdGF0c1xuICogQHR5cGUge3t9fVxuICovXG52YXIgc3RhdHMgPSB7fTtcblxuLyoqXG4gKiByZW1vdGUgc3RhdHNcbiAqIEB0eXBlIHt7fX1cbiAqL1xudmFyIHJlbW90ZVN0YXRzID0ge307XG5cbi8qKlxuICogSW50ZXJ2YWwgZm9yIHNlbmRpbmcgc3RhdGlzdGljcyB0byBvdGhlciBwYXJ0aWNpcGFudHNcbiAqIEB0eXBlIHtudWxsfVxuICovXG52YXIgc2VuZEludGVydmFsSWQgPSBudWxsO1xuXG5cbi8qKlxuICogU3RhcnQgc3RhdGlzdGljcyBzZW5kaW5nLlxuICovXG5mdW5jdGlvbiBzdGFydFNlbmRpbmdTdGF0cygpIHtcbiAgICBzZW5kU3RhdHMoKTtcbiAgICBzZW5kSW50ZXJ2YWxJZCA9IHNldEludGVydmFsKHNlbmRTdGF0cywgMTAwMDApO1xufVxuXG4vKipcbiAqIFNlbmRzIHN0YXRpc3RpY3MgdG8gb3RoZXIgcGFydGljaXBhbnRzXG4gKi9cbmZ1bmN0aW9uIHNlbmRTdGF0cygpIHtcbiAgICB4bXBwLmFkZFRvUHJlc2VuY2UoXCJjb25uZWN0aW9uUXVhbGl0eVwiLCBjb252ZXJ0VG9NVUNTdGF0cyhzdGF0cykpO1xufVxuXG4vKipcbiAqIENvbnZlcnRzIHN0YXRpc3RpY3MgdG8gZm9ybWF0IGZvciBzZW5kaW5nIHRocm91Z2ggWE1QUFxuICogQHBhcmFtIHN0YXRzIHRoZSBzdGF0aXN0aWNzXG4gKiBAcmV0dXJucyB7e2JpdHJhdGVfZG9ud2xvYWQ6ICosIGJpdHJhdGVfdXBscG9hZDogKiwgcGFja2V0TG9zc190b3RhbDogKiwgcGFja2V0TG9zc19kb3dubG9hZDogKiwgcGFja2V0TG9zc191cGxvYWQ6ICp9fVxuICovXG5mdW5jdGlvbiBjb252ZXJ0VG9NVUNTdGF0cyhzdGF0cykge1xuICAgIHJldHVybiB7XG4gICAgICAgIFwiYml0cmF0ZV9kb3dubG9hZFwiOiBzdGF0cy5iaXRyYXRlLmRvd25sb2FkLFxuICAgICAgICBcImJpdHJhdGVfdXBsb2FkXCI6IHN0YXRzLmJpdHJhdGUudXBsb2FkLFxuICAgICAgICBcInBhY2tldExvc3NfdG90YWxcIjogc3RhdHMucGFja2V0TG9zcy50b3RhbCxcbiAgICAgICAgXCJwYWNrZXRMb3NzX2Rvd25sb2FkXCI6IHN0YXRzLnBhY2tldExvc3MuZG93bmxvYWQsXG4gICAgICAgIFwicGFja2V0TG9zc191cGxvYWRcIjogc3RhdHMucGFja2V0TG9zcy51cGxvYWRcbiAgICB9O1xufVxuXG4vKipcbiAqIENvbnZlcnRzIHN0YXRpdGlzdGljcyB0byBmb3JtYXQgdXNlZCBieSBWaWRlb0xheW91dFxuICogQHBhcmFtIHN0YXRzXG4gKiBAcmV0dXJucyB7e2JpdHJhdGU6IHtkb3dubG9hZDogKiwgdXBsb2FkOiAqfSwgcGFja2V0TG9zczoge3RvdGFsOiAqLCBkb3dubG9hZDogKiwgdXBsb2FkOiAqfX19XG4gKi9cbmZ1bmN0aW9uIHBhcnNlTVVDU3RhdHMoc3RhdHMpIHtcbiAgICByZXR1cm4ge1xuICAgICAgICBiaXRyYXRlOiB7XG4gICAgICAgICAgICBkb3dubG9hZDogc3RhdHMuYml0cmF0ZV9kb3dubG9hZCxcbiAgICAgICAgICAgIHVwbG9hZDogc3RhdHMuYml0cmF0ZV91cGxvYWRcbiAgICAgICAgfSxcbiAgICAgICAgcGFja2V0TG9zczoge1xuICAgICAgICAgICAgdG90YWw6IHN0YXRzLnBhY2tldExvc3NfdG90YWwsXG4gICAgICAgICAgICBkb3dubG9hZDogc3RhdHMucGFja2V0TG9zc19kb3dubG9hZCxcbiAgICAgICAgICAgIHVwbG9hZDogc3RhdHMucGFja2V0TG9zc191cGxvYWRcbiAgICAgICAgfVxuICAgIH07XG59XG5cblxudmFyIENvbm5lY3Rpb25RdWFsaXR5ID0ge1xuICAgIC8qKlxuICAgICAqIFVwZGF0ZXMgdGhlIGxvY2FsIHN0YXRpc3RpY3NcbiAgICAgKiBAcGFyYW0gZGF0YSBuZXcgc3RhdGlzdGljc1xuICAgICAqL1xuICAgIHVwZGF0ZUxvY2FsU3RhdHM6IGZ1bmN0aW9uIChkYXRhKSB7XG4gICAgICAgIHN0YXRzID0gZGF0YTtcbiAgICAgICAgVUkudXBkYXRlTG9jYWxDb25uZWN0aW9uU3RhdHMoMTAwIC0gc3RhdHMucGFja2V0TG9zcy50b3RhbCwgc3RhdHMpO1xuICAgICAgICBpZiAoc2VuZEludGVydmFsSWQgPT0gbnVsbCkge1xuICAgICAgICAgICAgc3RhcnRTZW5kaW5nU3RhdHMoKTtcbiAgICAgICAgfVxuICAgIH0sXG5cbiAgICAvKipcbiAgICAgKiBVcGRhdGVzIHJlbW90ZSBzdGF0aXN0aWNzXG4gICAgICogQHBhcmFtIGppZCB0aGUgamlkIGFzc29jaWF0ZWQgd2l0aCB0aGUgc3RhdGlzdGljc1xuICAgICAqIEBwYXJhbSBkYXRhIHRoZSBzdGF0aXN0aWNzXG4gICAgICovXG4gICAgdXBkYXRlUmVtb3RlU3RhdHM6IGZ1bmN0aW9uIChqaWQsIGRhdGEpIHtcbiAgICAgICAgaWYgKGRhdGEgPT0gbnVsbCB8fCBkYXRhLnBhY2tldExvc3NfdG90YWwgPT0gbnVsbCkge1xuICAgICAgICAgICAgVUkudXBkYXRlQ29ubmVjdGlvblN0YXRzKGppZCwgbnVsbCwgbnVsbCk7XG4gICAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cbiAgICAgICAgcmVtb3RlU3RhdHNbamlkXSA9IHBhcnNlTVVDU3RhdHMoZGF0YSk7XG5cbiAgICAgICAgVUkudXBkYXRlQ29ubmVjdGlvblN0YXRzKGppZCwgMTAwIC0gZGF0YS5wYWNrZXRMb3NzX3RvdGFsLCByZW1vdGVTdGF0c1tqaWRdKTtcblxuICAgIH0sXG5cbiAgICAvKipcbiAgICAgKiBTdG9wcyBzdGF0aXN0aWNzIHNlbmRpbmcuXG4gICAgICovXG4gICAgc3RvcFNlbmRpbmdTdGF0czogZnVuY3Rpb24gKCkge1xuICAgICAgICBjbGVhckludGVydmFsKHNlbmRJbnRlcnZhbElkKTtcbiAgICAgICAgc2VuZEludGVydmFsSWQgPSBudWxsO1xuICAgICAgICAvL25vdGlmeSBVSSBhYm91dCBzdG9wcGluZyBzdGF0aXN0aWNzIGdhdGhlcmluZ1xuICAgICAgICBVSS5vblN0YXRzU3RvcCgpO1xuICAgIH0sXG5cbiAgICAvKipcbiAgICAgKiBSZXR1cm5zIHRoZSBsb2NhbCBzdGF0aXN0aWNzLlxuICAgICAqL1xuICAgIGdldFN0YXRzOiBmdW5jdGlvbiAoKSB7XG4gICAgICAgIHJldHVybiBzdGF0cztcbiAgICB9XG5cbn07XG5cbm1vZHVsZS5leHBvcnRzID0gQ29ubmVjdGlvblF1YWxpdHk7Il19
diff --git a/libs/modules/desktopsharing.bundle.js b/libs/modules/desktopsharing.bundle.js
index 3ce0d817c..dc8e493b5 100644
--- a/libs/modules/desktopsharing.bundle.js
+++ b/libs/modules/desktopsharing.bundle.js
@@ -1,5 +1,5 @@
!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var f;"undefined"!=typeof window?f=window:"undefined"!=typeof global?f=global:"undefined"!=typeof self&&(f=self),f.desktopsharing=e()}}(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o 0) {
+ var ice = SDPUtil.iceparams(this.localSDP.media[mid], this.localSDP.session);
+ ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
+ cand.c('content', {creator: this.initiator == this.me ? 'initiator' : 'responder',
+ name: (cands[0].sdpMid? cands[0].sdpMid : mline.media)
+ }).c('transport', ice);
+ for (var i = 0; i < cands.length; i++) {
+ cand.c('candidate', SDPUtil.candidateToJingle(cands[i].candidate)).up();
+ }
+ // add fingerprint
+ if (SDPUtil.find_line(this.localSDP.media[mid], 'a=fingerprint:', this.localSDP.session)) {
+ var tmp = SDPUtil.parse_fingerprint(SDPUtil.find_line(this.localSDP.media[mid], 'a=fingerprint:', this.localSDP.session));
+ tmp.required = true;
+ cand.c(
+ 'fingerprint',
+ {xmlns: 'urn:xmpp:jingle:apps:dtls:0'})
+ .t(tmp.fingerprint);
+ delete tmp.fingerprint;
+ cand.attrs(tmp);
+ cand.up();
+ }
+ cand.up(); // transport
+ cand.up(); // content
+ }
+ }
+ // might merge last-candidate notification into this, but it is called alot later. See webrtc issue #2340
+ //console.log('was this the last candidate', this.lasticecandidate);
+ this.connection.sendIQ(cand,
+ function () {
+ var ack = {};
+ ack.source = 'transportinfo';
+ $(document).trigger('ack.jingle', [this.sid, ack]);
+ },
+ function (stanza) {
+ var error = ($(stanza).find('error').length) ? {
+ code: $(stanza).find('error').attr('code'),
+ reason: $(stanza).find('error :first')[0].tagName,
+ }:{};
+ error.source = 'transportinfo';
+ JingleSession.onJingleError(this.sid, error);
+ },
+ 10000);
+};
+
+
+JingleSession.prototype.sendOffer = function () {
+ //console.log('sendOffer...');
+ var self = this;
+ this.peerconnection.createOffer(function (sdp) {
+ self.createdOffer(sdp);
+ },
+ function (e) {
+ console.error('createOffer failed', e);
+ },
+ this.media_constraints
+ );
+};
+
+JingleSession.prototype.createdOffer = function (sdp) {
+ //console.log('createdOffer', sdp);
+ var self = this;
+ this.localSDP = new SDP(sdp.sdp);
+ //this.localSDP.mangle();
+ var sendJingle = function () {
+ var init = $iq({to: this.peerjid,
+ type: 'set'})
+ .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+ action: 'session-initiate',
+ initiator: this.initiator,
+ sid: this.sid});
+ self.localSDP.toJingle(init, this.initiator == this.me ? 'initiator' : 'responder', this.localStreamsSSRC);
+ self.connection.sendIQ(init,
+ function () {
+ var ack = {};
+ ack.source = 'offer';
+ $(document).trigger('ack.jingle', [self.sid, ack]);
+ },
+ function (stanza) {
+ self.state = 'error';
+ self.peerconnection.close();
+ var error = ($(stanza).find('error').length) ? {
+ code: $(stanza).find('error').attr('code'),
+ reason: $(stanza).find('error :first')[0].tagName,
+ }:{};
+ error.source = 'offer';
+ JingleSession.onJingleError(self.sid, error);
+ },
+ 10000);
+ }
+ sdp.sdp = this.localSDP.raw;
+ this.peerconnection.setLocalDescription(sdp,
+ function () {
+ if(self.usetrickle)
+ {
+ sendJingle();
+ }
+ self.setLocalDescription();
+ //console.log('setLocalDescription success');
+ },
+ function (e) {
+ console.error('setLocalDescription failed', e);
+ }
+ );
+ var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');
+ for (var i = 0; i < cands.length; i++) {
+ var cand = SDPUtil.parse_icecandidate(cands[i]);
+ if (cand.type == 'srflx') {
+ this.hadstuncandidate = true;
+ } else if (cand.type == 'relay') {
+ this.hadturncandidate = true;
+ }
+ }
+};
+
+JingleSession.prototype.setRemoteDescription = function (elem, desctype) {
+ //console.log('setting remote description... ', desctype);
+ this.remoteSDP = new SDP('');
+ this.remoteSDP.fromJingle(elem);
+ if (this.peerconnection.remoteDescription !== null) {
+ console.log('setRemoteDescription when remote description is not null, should be pranswer', this.peerconnection.remoteDescription);
+ if (this.peerconnection.remoteDescription.type == 'pranswer') {
+ var pranswer = new SDP(this.peerconnection.remoteDescription.sdp);
+ for (var i = 0; i < pranswer.media.length; i++) {
+ // make sure we have ice ufrag and pwd
+ if (!SDPUtil.find_line(this.remoteSDP.media[i], 'a=ice-ufrag:', this.remoteSDP.session)) {
+ if (SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session)) {
+ this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session) + '\r\n';
+ } else {
+ console.warn('no ice ufrag?');
+ }
+ if (SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session)) {
+ this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session) + '\r\n';
+ } else {
+ console.warn('no ice pwd?');
+ }
+ }
+ // copy over candidates
+ var lines = SDPUtil.find_lines(pranswer.media[i], 'a=candidate:');
+ for (var j = 0; j < lines.length; j++) {
+ this.remoteSDP.media[i] += lines[j] + '\r\n';
+ }
+ }
+ this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');
+ }
+ }
+ var remotedesc = new RTCSessionDescription({type: desctype, sdp: this.remoteSDP.raw});
+
+ this.peerconnection.setRemoteDescription(remotedesc,
+ function () {
+ //console.log('setRemoteDescription success');
+ },
+ function (e) {
+ console.error('setRemoteDescription error', e);
+ JingleSession.onJingleFatalError(self, e);
+ }
+ );
+};
+
+JingleSession.prototype.addIceCandidate = function (elem) {
+ var self = this;
+ if (this.peerconnection.signalingState == 'closed') {
+ return;
+ }
+ if (!this.peerconnection.remoteDescription && this.peerconnection.signalingState == 'have-local-offer') {
+ console.log('trickle ice candidate arriving before session accept...');
+ // create a PRANSWER for setRemoteDescription
+ if (!this.remoteSDP) {
+ var cobbled = 'v=0\r\n' +
+ 'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\r\n' +// FIXME
+ 's=-\r\n' +
+ 't=0 0\r\n';
+ // first, take some things from the local description
+ for (var i = 0; i < this.localSDP.media.length; i++) {
+ cobbled += SDPUtil.find_line(this.localSDP.media[i], 'm=') + '\r\n';
+ cobbled += SDPUtil.find_lines(this.localSDP.media[i], 'a=rtpmap:').join('\r\n') + '\r\n';
+ if (SDPUtil.find_line(this.localSDP.media[i], 'a=mid:')) {
+ cobbled += SDPUtil.find_line(this.localSDP.media[i], 'a=mid:') + '\r\n';
+ }
+ cobbled += 'a=inactive\r\n';
+ }
+ this.remoteSDP = new SDP(cobbled);
+ }
+ // then add things like ice and dtls from remote candidate
+ elem.each(function () {
+ for (var i = 0; i < self.remoteSDP.media.length; i++) {
+ if (SDPUtil.find_line(self.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
+ self.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
+ if (!SDPUtil.find_line(self.remoteSDP.media[i], 'a=ice-ufrag:')) {
+ var tmp = $(this).find('transport');
+ self.remoteSDP.media[i] += 'a=ice-ufrag:' + tmp.attr('ufrag') + '\r\n';
+ self.remoteSDP.media[i] += 'a=ice-pwd:' + tmp.attr('pwd') + '\r\n';
+ tmp = $(this).find('transport>fingerprint');
+ if (tmp.length) {
+ self.remoteSDP.media[i] += 'a=fingerprint:' + tmp.attr('hash') + ' ' + tmp.text() + '\r\n';
+ } else {
+ console.log('no dtls fingerprint (webrtc issue #1718?)');
+ self.remoteSDP.media[i] += 'a=crypto:1 AES_CM_128_HMAC_SHA1_80 inline:BAADBAADBAADBAADBAADBAADBAADBAADBAADBAAD\r\n';
+ }
+ break;
+ }
+ }
+ }
+ });
+ this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');
+
+ // we need a complete SDP with ice-ufrag/ice-pwd in all parts
+ // this makes the assumption that the PRANSWER is constructed such that the ice-ufrag is in all mediaparts
+ // but it could be in the session part as well. since the code above constructs this sdp this can't happen however
+ var iscomplete = this.remoteSDP.media.filter(function (mediapart) {
+ return SDPUtil.find_line(mediapart, 'a=ice-ufrag:');
+ }).length == this.remoteSDP.media.length;
+
+ if (iscomplete) {
+ console.log('setting pranswer');
+ try {
+ this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'pranswer', sdp: this.remoteSDP.raw }),
+ function() {
+ },
+ function(e) {
+ console.log('setRemoteDescription pranswer failed', e.toString());
+ });
+ } catch (e) {
+ console.error('setting pranswer failed', e);
+ }
+ } else {
+ //console.log('not yet setting pranswer');
+ }
+ }
+ // operate on each content element
+ elem.each(function () {
+ // would love to deactivate this, but firefox still requires it
+ var idx = -1;
+ var i;
+ for (i = 0; i < self.remoteSDP.media.length; i++) {
+ if (SDPUtil.find_line(self.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
+ self.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
+ idx = i;
+ break;
+ }
+ }
+ if (idx == -1) { // fall back to localdescription
+ for (i = 0; i < self.localSDP.media.length; i++) {
+ if (SDPUtil.find_line(self.localSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
+ self.localSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
+ idx = i;
+ break;
+ }
+ }
+ }
+ var name = $(this).attr('name');
+ // TODO: check ice-pwd and ice-ufrag?
+ $(this).find('transport>candidate').each(function () {
+ var line, candidate;
+ line = SDPUtil.candidateFromJingle(this);
+ candidate = new RTCIceCandidate({sdpMLineIndex: idx,
+ sdpMid: name,
+ candidate: line});
+ try {
+ self.peerconnection.addIceCandidate(candidate);
+ } catch (e) {
+ console.error('addIceCandidate failed', e.toString(), line);
+ }
+ });
+ });
+};
+
+JingleSession.prototype.sendAnswer = function (provisional) {
+ //console.log('createAnswer', provisional);
+ var self = this;
+ this.peerconnection.createAnswer(
+ function (sdp) {
+ self.createdAnswer(sdp, provisional);
+ },
+ function (e) {
+ console.error('createAnswer failed', e);
+ },
+ this.media_constraints
+ );
+};
+
+JingleSession.prototype.createdAnswer = function (sdp, provisional) {
+ //console.log('createAnswer callback');
+ var self = this;
+ this.localSDP = new SDP(sdp.sdp);
+ //this.localSDP.mangle();
+ this.usepranswer = provisional === true;
+ if (this.usetrickle) {
+ if (this.usepranswer) {
+ sdp.type = 'pranswer';
+ for (var i = 0; i < this.localSDP.media.length; i++) {
+ this.localSDP.media[i] = this.localSDP.media[i].replace('a=sendrecv\r\n', 'a=inactive\r\n');
+ }
+ this.localSDP.raw = this.localSDP.session + '\r\n' + this.localSDP.media.join('');
+ }
+ }
+ var self = this;
+ var sendJingle = function (ssrcs) {
+
+ var accept = $iq({to: self.peerjid,
+ type: 'set'})
+ .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+ action: 'session-accept',
+ initiator: self.initiator,
+ responder: self.responder,
+ sid: self.sid });
+ var publicLocalDesc = simulcast.reverseTransformLocalDescription(sdp);
+ var publicLocalSDP = new SDP(publicLocalDesc.sdp);
+ publicLocalSDP.toJingle(accept, self.initiator == self.me ? 'initiator' : 'responder', ssrcs);
+ self.connection.sendIQ(accept,
+ function () {
+ var ack = {};
+ ack.source = 'answer';
+ $(document).trigger('ack.jingle', [self.sid, ack]);
+ },
+ function (stanza) {
+ var error = ($(stanza).find('error').length) ? {
+ code: $(stanza).find('error').attr('code'),
+ reason: $(stanza).find('error :first')[0].tagName,
+ }:{};
+ error.source = 'answer';
+ JingleSession.onJingleError(self.sid, error);
+ },
+ 10000);
+ }
+ sdp.sdp = this.localSDP.raw;
+ this.peerconnection.setLocalDescription(sdp,
+ function () {
+
+ //console.log('setLocalDescription success');
+ if (self.usetrickle && !self.usepranswer) {
+ sendJingle();
+ }
+ self.setLocalDescription();
+ },
+ function (e) {
+ console.error('setLocalDescription failed', e);
+ }
+ );
+ var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');
+ for (var j = 0; j < cands.length; j++) {
+ var cand = SDPUtil.parse_icecandidate(cands[j]);
+ if (cand.type == 'srflx') {
+ this.hadstuncandidate = true;
+ } else if (cand.type == 'relay') {
+ this.hadturncandidate = true;
+ }
+ }
+};
+
+JingleSession.prototype.sendTerminate = function (reason, text) {
+ var self = this,
+ term = $iq({to: this.peerjid,
+ type: 'set'})
+ .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+ action: 'session-terminate',
+ initiator: this.initiator,
+ sid: this.sid})
+ .c('reason')
+ .c(reason || 'success');
+
+ if (text) {
+ term.up().c('text').t(text);
+ }
+
+ this.connection.sendIQ(term,
+ function () {
+ self.peerconnection.close();
+ self.peerconnection = null;
+ self.terminate();
+ var ack = {};
+ ack.source = 'terminate';
+ $(document).trigger('ack.jingle', [self.sid, ack]);
+ },
+ function (stanza) {
+ var error = ($(stanza).find('error').length) ? {
+ code: $(stanza).find('error').attr('code'),
+ reason: $(stanza).find('error :first')[0].tagName,
+ }:{};
+ $(document).trigger('ack.jingle', [self.sid, error]);
+ },
+ 10000);
+ if (this.statsinterval !== null) {
+ window.clearInterval(this.statsinterval);
+ this.statsinterval = null;
+ }
+};
+
+JingleSession.prototype.addSource = function (elem, fromJid) {
+
+ var self = this;
+ // FIXME: dirty waiting
+ if (!this.peerconnection.localDescription)
+ {
+ console.warn("addSource - localDescription not ready yet")
+ setTimeout(function()
+ {
+ self.addSource(elem, fromJid);
+ },
+ 200
+ );
+ return;
+ }
+
+ console.log('addssrc', new Date().getTime());
+ console.log('ice', this.peerconnection.iceConnectionState);
+ var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
+ var mySdp = new SDP(this.peerconnection.localDescription.sdp);
+
+ $(elem).each(function (idx, content) {
+ var name = $(content).attr('name');
+ var lines = '';
+ tmp = $(content).find('ssrc-group[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]').each(function() {
+ var semantics = this.getAttribute('semantics');
+ var ssrcs = $(this).find('>source').map(function () {
+ return this.getAttribute('ssrc');
+ }).get();
+
+ if (ssrcs.length != 0) {
+ lines += 'a=ssrc-group:' + semantics + ' ' + ssrcs.join(' ') + '\r\n';
+ }
+ });
+ tmp = $(content).find('source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]'); // can handle both >source and >description>source
+ tmp.each(function () {
+ var ssrc = $(this).attr('ssrc');
+ if(mySdp.containsSSRC(ssrc)){
+ /**
+ * This happens when multiple participants change their streams at the same time and
+ * ColibriFocus.modifySources have to wait for stable state. In the meantime multiple
+ * addssrc are scheduled for update IQ. See
+ */
+ console.warn("Got add stream request for my own ssrc: "+ssrc);
+ return;
+ }
+ $(this).find('>parameter').each(function () {
+ lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');
+ if ($(this).attr('value') && $(this).attr('value').length)
+ lines += ':' + $(this).attr('value');
+ lines += '\r\n';
+ });
+ });
+ sdp.media.forEach(function(media, idx) {
+ if (!SDPUtil.find_line(media, 'a=mid:' + name))
+ return;
+ sdp.media[idx] += lines;
+ if (!self.addssrc[idx]) self.addssrc[idx] = '';
+ self.addssrc[idx] += lines;
+ });
+ sdp.raw = sdp.session + sdp.media.join('');
+ });
+ this.modifySources();
+};
+
+JingleSession.prototype.removeSource = function (elem, fromJid) {
+
+ var self = this;
+ // FIXME: dirty waiting
+ if (!this.peerconnection.localDescription)
+ {
+ console.warn("removeSource - localDescription not ready yet")
+ setTimeout(function()
+ {
+ self.removeSource(elem, fromJid);
+ },
+ 200
+ );
+ return;
+ }
+
+ console.log('removessrc', new Date().getTime());
+ console.log('ice', this.peerconnection.iceConnectionState);
+ var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
+ var mySdp = new SDP(this.peerconnection.localDescription.sdp);
+
+ $(elem).each(function (idx, content) {
+ var name = $(content).attr('name');
+ var lines = '';
+ tmp = $(content).find('ssrc-group[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]').each(function() {
+ var semantics = this.getAttribute('semantics');
+ var ssrcs = $(this).find('>source').map(function () {
+ return this.getAttribute('ssrc');
+ }).get();
+
+ if (ssrcs.length != 0) {
+ lines += 'a=ssrc-group:' + semantics + ' ' + ssrcs.join(' ') + '\r\n';
+ }
+ });
+ tmp = $(content).find('source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]'); // can handle both >source and >description>source
+ tmp.each(function () {
+ var ssrc = $(this).attr('ssrc');
+ // This should never happen, but can be useful for bug detection
+ if(mySdp.containsSSRC(ssrc)){
+ console.error("Got remove stream request for my own ssrc: "+ssrc);
+ return;
+ }
+ $(this).find('>parameter').each(function () {
+ lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');
+ if ($(this).attr('value') && $(this).attr('value').length)
+ lines += ':' + $(this).attr('value');
+ lines += '\r\n';
+ });
+ });
+ sdp.media.forEach(function(media, idx) {
+ if (!SDPUtil.find_line(media, 'a=mid:' + name))
+ return;
+ sdp.media[idx] += lines;
+ if (!self.removessrc[idx]) self.removessrc[idx] = '';
+ self.removessrc[idx] += lines;
+ });
+ sdp.raw = sdp.session + sdp.media.join('');
+ });
+ this.modifySources();
+};
+
+JingleSession.prototype.modifySources = function (successCallback) {
+ var self = this;
+ if (this.peerconnection.signalingState == 'closed') return;
+ if (!(this.addssrc.length || this.removessrc.length || this.pendingop !== null || this.switchstreams)){
+ // There is nothing to do since scheduled job might have been executed by another succeeding call
+ this.setLocalDescription();
+ if(successCallback){
+ successCallback();
+ }
+ return;
+ }
+
+ // FIXME: this is a big hack
+ // https://code.google.com/p/webrtc/issues/detail?id=2688
+ // ^ has been fixed.
+ if (!(this.peerconnection.signalingState == 'stable' && this.peerconnection.iceConnectionState == 'connected')) {
+ console.warn('modifySources not yet', this.peerconnection.signalingState, this.peerconnection.iceConnectionState);
+ this.wait = true;
+ window.setTimeout(function() { self.modifySources(successCallback); }, 250);
+ return;
+ }
+ if (this.wait) {
+ window.setTimeout(function() { self.modifySources(successCallback); }, 2500);
+ this.wait = false;
+ return;
+ }
+
+ // Reset switch streams flag
+ this.switchstreams = false;
+
+ var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
+
+ // add sources
+ this.addssrc.forEach(function(lines, idx) {
+ sdp.media[idx] += lines;
+ });
+ this.addssrc = [];
+
+ // remove sources
+ this.removessrc.forEach(function(lines, idx) {
+ lines = lines.split('\r\n');
+ lines.pop(); // remove empty last element;
+ lines.forEach(function(line) {
+ sdp.media[idx] = sdp.media[idx].replace(line + '\r\n', '');
+ });
+ });
+ this.removessrc = [];
+
+ // FIXME:
+ // this was a hack for the situation when only one peer exists
+ // in the conference.
+ // check if still required and remove
+ if (sdp.media[0])
+ sdp.media[0] = sdp.media[0].replace('a=recvonly', 'a=sendrecv');
+ if (sdp.media[1])
+ sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');
+
+ sdp.raw = sdp.session + sdp.media.join('');
+ this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'offer', sdp: sdp.raw}),
+ function() {
+
+ if(self.signalingState == 'closed') {
+ console.error("createAnswer attempt on closed state");
+ return;
+ }
+
+ self.peerconnection.createAnswer(
+ function(modifiedAnswer) {
+ // change video direction, see https://github.com/jitsi/jitmeet/issues/41
+ if (self.pendingop !== null) {
+ var sdp = new SDP(modifiedAnswer.sdp);
+ if (sdp.media.length > 1) {
+ switch(self.pendingop) {
+ case 'mute':
+ sdp.media[1] = sdp.media[1].replace('a=sendrecv', 'a=recvonly');
+ break;
+ case 'unmute':
+ sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');
+ break;
+ }
+ sdp.raw = sdp.session + sdp.media.join('');
+ modifiedAnswer.sdp = sdp.raw;
+ }
+ 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.peerconnection.setLocalDescription(modifiedAnswer,
+ function() {
+ //console.log('modified setLocalDescription ok');
+ self.setLocalDescription();
+ if(successCallback){
+ successCallback();
+ }
+ },
+ function(error) {
+ console.error('modified setLocalDescription failed', error);
+ }
+ );
+ },
+ function(error) {
+ console.error('modified answer failed', error);
+ }
+ );
+ },
+ function(error) {
+ console.error('modify failed', error);
+ }
+ );
+};
+
+/**
+ * Switches video streams.
+ * @param new_stream new stream that will be used as video of this session.
+ * @param oldStream old video stream of this session.
+ * @param success_callback callback executed after successful stream switch.
+ */
+JingleSession.prototype.switchStreams = function (new_stream, oldStream, success_callback) {
+
+ var self = this;
+
+ // Remember SDP to figure out added/removed SSRCs
+ var oldSdp = null;
+ if(self.peerconnection) {
+ if(self.peerconnection.localDescription) {
+ oldSdp = new SDP(self.peerconnection.localDescription.sdp);
+ }
+ self.peerconnection.removeStream(oldStream, true);
+ self.peerconnection.addStream(new_stream);
+ }
+
+ RTC.switchVideoStreams(new_stream, oldStream);
+
+ // Conference is not active
+ if(!oldSdp || !self.peerconnection) {
+ success_callback();
+ return;
+ }
+
+ self.switchstreams = true;
+ self.modifySources(function() {
+ console.log('modify sources done');
+
+ success_callback();
+
+ var newSdp = new SDP(self.peerconnection.localDescription.sdp);
+ console.log("SDPs", oldSdp, newSdp);
+ self.notifyMySSRCUpdate(oldSdp, newSdp);
+ });
+};
+
+/**
+ * Figures out added/removed ssrcs and send update IQs.
+ * @param old_sdp SDP object for old description.
+ * @param new_sdp SDP object for new description.
+ */
+JingleSession.prototype.notifyMySSRCUpdate = function (old_sdp, new_sdp) {
+
+ if (!(this.peerconnection.signalingState == 'stable' &&
+ this.peerconnection.iceConnectionState == 'connected')){
+ console.log("Too early to send updates");
+ return;
+ }
+
+ // send source-remove IQ.
+ sdpDiffer = new SDPDiffer(new_sdp, old_sdp);
+ var remove = $iq({to: this.peerjid, type: 'set'})
+ .c('jingle', {
+ xmlns: 'urn:xmpp:jingle:1',
+ action: 'source-remove',
+ initiator: this.initiator,
+ sid: this.sid
+ }
+ );
+ var removed = sdpDiffer.toJingle(remove);
+ if (removed) {
+ this.connection.sendIQ(remove,
+ function (res) {
+ console.info('got remove result', res);
+ },
+ function (err) {
+ console.error('got remove error', err);
+ }
+ );
+ } else {
+ console.log('removal not necessary');
+ }
+
+ // send source-add IQ.
+ var sdpDiffer = new SDPDiffer(old_sdp, new_sdp);
+ var add = $iq({to: this.peerjid, type: 'set'})
+ .c('jingle', {
+ xmlns: 'urn:xmpp:jingle:1',
+ action: 'source-add',
+ initiator: this.initiator,
+ sid: this.sid
+ }
+ );
+ var added = sdpDiffer.toJingle(add);
+ if (added) {
+ this.connection.sendIQ(add,
+ function (res) {
+ console.info('got add result', res);
+ },
+ function (err) {
+ console.error('got add error', err);
+ }
+ );
+ } else {
+ console.log('addition not necessary');
+ }
+};
+
+/**
+ * Determines whether the (local) video is mute i.e. all video tracks are
+ * disabled.
+ *
+ * @return true if the (local) video is mute i.e. all video tracks are
+ * disabled; otherwise, false
+ */
+JingleSession.prototype.isVideoMute = function () {
+ var tracks = RTC.localVideo.getVideoTracks();
+ var mute = true;
+
+ for (var i = 0; i < tracks.length; ++i) {
+ if (tracks[i].enabled) {
+ mute = false;
+ break;
+ }
+ }
+ return mute;
+};
+
+/**
+ * Mutes/unmutes the (local) video i.e. enables/disables all video tracks.
+ *
+ * @param mute true to mute the (local) video i.e. to disable all video
+ * tracks; otherwise, false
+ * @param callback a function to be invoked with mute after all video
+ * tracks have been enabled/disabled. The function may, optionally, return
+ * another function which is to be invoked after the whole mute/unmute operation
+ * has completed successfully.
+ * @param options an object which specifies optional arguments such as the
+ * boolean key byUser with default value true which
+ * specifies whether the method was initiated in response to a user command (in
+ * contrast to an automatic decision made by the application logic)
+ */
+JingleSession.prototype.setVideoMute = function (mute, callback, options) {
+ var byUser;
+
+ if (options) {
+ byUser = options.byUser;
+ if (typeof byUser === 'undefined') {
+ byUser = true;
+ }
+ } else {
+ byUser = true;
+ }
+ // The user's command to mute the (local) video takes precedence over any
+ // automatic decision made by the application logic.
+ if (byUser) {
+ this.videoMuteByUser = mute;
+ } else if (this.videoMuteByUser) {
+ return;
+ }
+
+ var self = this;
+ var localCallback = function (mute) {
+ self.connection.emuc.addVideoInfoToPresence(mute);
+ self.connection.emuc.sendPresence();
+ return callback(mute)
+ };
+
+ if (mute == RTC.localVideo.isMuted())
+ {
+ // Even if no change occurs, the specified callback is to be executed.
+ // The specified callback may, optionally, return a successCallback
+ // which is to be executed as well.
+ var successCallback = localCallback(mute);
+
+ if (successCallback) {
+ successCallback();
+ }
+ } else {
+ RTC.localVideo.setMute(!mute);
+
+ this.hardMuteVideo(mute);
+
+ this.modifySources(localCallback(mute));
+ }
+};
+
+// SDP-based mute by going recvonly/sendrecv
+// FIXME: should probably black out the screen as well
+JingleSession.prototype.toggleVideoMute = function (callback) {
+ this.service.setVideoMute(RTC.localVideo.isMuted(), callback);
+};
+
+JingleSession.prototype.hardMuteVideo = function (muted) {
+ this.pendingop = muted ? 'mute' : 'unmute';
+};
+
+JingleSession.prototype.sendMute = function (muted, content) {
+ var info = $iq({to: this.peerjid,
+ type: 'set'})
+ .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+ action: 'session-info',
+ initiator: this.initiator,
+ sid: this.sid });
+ info.c(muted ? 'mute' : 'unmute', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});
+ info.attrs({'creator': this.me == this.initiator ? 'creator' : 'responder'});
+ if (content) {
+ info.attrs({'name': content});
+ }
+ this.connection.send(info);
+};
+
+JingleSession.prototype.sendRinging = function () {
+ var info = $iq({to: this.peerjid,
+ type: 'set'})
+ .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+ action: 'session-info',
+ initiator: this.initiator,
+ sid: this.sid });
+ info.c('ringing', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});
+ this.connection.send(info);
+};
+
+JingleSession.prototype.getStats = function (interval) {
+ var self = this;
+ var recv = {audio: 0, video: 0};
+ var lost = {audio: 0, video: 0};
+ var lastrecv = {audio: 0, video: 0};
+ var lastlost = {audio: 0, video: 0};
+ var loss = {audio: 0, video: 0};
+ var delta = {audio: 0, video: 0};
+ this.statsinterval = window.setInterval(function () {
+ if (self && self.peerconnection && self.peerconnection.getStats) {
+ self.peerconnection.getStats(function (stats) {
+ var results = stats.result();
+ // TODO: there are so much statistics you can get from this..
+ for (var i = 0; i < results.length; ++i) {
+ if (results[i].type == 'ssrc') {
+ var packetsrecv = results[i].stat('packetsReceived');
+ var packetslost = results[i].stat('packetsLost');
+ if (packetsrecv && packetslost) {
+ packetsrecv = parseInt(packetsrecv, 10);
+ packetslost = parseInt(packetslost, 10);
+
+ if (results[i].stat('googFrameRateReceived')) {
+ lastlost.video = lost.video;
+ lastrecv.video = recv.video;
+ recv.video = packetsrecv;
+ lost.video = packetslost;
+ } else {
+ lastlost.audio = lost.audio;
+ lastrecv.audio = recv.audio;
+ recv.audio = packetsrecv;
+ lost.audio = packetslost;
+ }
+ }
+ }
+ }
+ delta.audio = recv.audio - lastrecv.audio;
+ delta.video = recv.video - lastrecv.video;
+ loss.audio = (delta.audio > 0) ? Math.ceil(100 * (lost.audio - lastlost.audio) / delta.audio) : 0;
+ loss.video = (delta.video > 0) ? Math.ceil(100 * (lost.video - lastlost.video) / delta.video) : 0;
+ $(document).trigger('packetloss.jingle', [self.sid, loss]);
+ });
+ }
+ }, interval || 3000);
+ return this.statsinterval;
+};
+
+JingleSession.onJingleError = function (session, error)
+{
+ console.error("Jingle error", error);
+}
+
+JingleSession.onJingleFatalError = function (session, error)
+{
+ this.service.sessionTerminated = true;
+ connection.emuc.doLeave();
+ UI.messageHandler.showError( "Sorry",
+ "Internal application error[setRemoteDescription]");
+}
+
+JingleSession.prototype.setLocalDescription = function () {
+ // put our ssrcs into presence so other clients can identify our stream
+ var newssrcs = [];
+ var media = simulcast.parseMedia(this.peerconnection.localDescription);
+ media.forEach(function (media) {
+
+ if(Object.keys(media.sources).length > 0) {
+ // TODO(gp) maybe exclude FID streams?
+ Object.keys(media.sources).forEach(function (ssrc) {
+ newssrcs.push({
+ 'ssrc': ssrc,
+ 'type': media.type,
+ 'direction': media.direction
+ });
+ });
+ }
+ else if(this.localStreamsSSRC && this.localStreamsSSRC[media.type])
+ {
+ newssrcs.push({
+ 'ssrc': this.localStreamsSSRC[media.type],
+ 'type': media.type,
+ 'direction': media.direction
+ });
+ }
+
+ });
+
+ console.log('new ssrcs', newssrcs);
+
+ // Have to clear presence map to get rid of removed streams
+ this.connection.emuc.clearPresenceMedia();
+
+ if (newssrcs.length > 0) {
+ for (var i = 1; i <= newssrcs.length; i ++) {
+ // Change video type to screen
+ if (newssrcs[i-1].type === 'video' && desktopsharing.isUsingScreenStream()) {
+ newssrcs[i-1].type = 'screen';
+ }
+ this.connection.emuc.addMediaToPresence(i,
+ newssrcs[i-1].type, newssrcs[i-1].ssrc, newssrcs[i-1].direction);
+ }
+
+ this.connection.emuc.sendPresence();
+ }
+}
+
+// an attempt to work around https://github.com/jitsi/jitmeet/issues/32
+function sendKeyframe(pc) {
+ console.log('sendkeyframe', pc.iceConnectionState);
+ if (pc.iceConnectionState !== 'connected') return; // safe...
+ pc.setRemoteDescription(
+ pc.remoteDescription,
+ function () {
+ pc.createAnswer(
+ function (modifiedAnswer) {
+ pc.setLocalDescription(
+ modifiedAnswer,
+ function () {
+ // noop
+ },
+ function (error) {
+ console.log('triggerKeyframe setLocalDescription failed', error);
+ UI.messageHandler.showError();
+ }
+ );
+ },
+ function (error) {
+ console.log('triggerKeyframe createAnswer failed', error);
+ UI.messageHandler.showError();
+ }
+ );
+ },
+ function (error) {
+ console.log('triggerKeyframe setRemoteDescription failed', error);
+ UI.messageHandler.showError();
+ }
+ );
+}
+
+
+JingleSession.prototype.remoteStreamAdded = function (data) {
+ var self = this;
+ var thessrc;
+
+ // look up an associated JID for a stream id
+ if (data.stream.id && data.stream.id.indexOf('mixedmslabel') === -1) {
+ // look only at a=ssrc: and _not_ at a=ssrc-group: lines
+
+ var ssrclines
+ = SDPUtil.find_lines(this.peerconnection.remoteDescription.sdp, 'a=ssrc:');
+ ssrclines = ssrclines.filter(function (line) {
+ // NOTE(gp) previously we filtered on the mslabel, but that property
+ // is not always present.
+ // return line.indexOf('mslabel:' + data.stream.label) !== -1;
+
+ return ((line.indexOf('msid:' + data.stream.id) !== -1));
+ });
+ if (ssrclines.length) {
+ thessrc = ssrclines[0].substring(7).split(' ')[0];
+
+ // We signal our streams (through Jingle to the focus) before we set
+ // our presence (through which peers associate remote streams to
+ // jids). So, it might arrive that a remote stream is added but
+ // ssrc2jid is not yet updated and thus data.peerjid cannot be
+ // successfully set. Here we wait for up to a second for the
+ // presence to arrive.
+
+ if (!ssrc2jid[thessrc]) {
+ // TODO(gp) limit wait duration to 1 sec.
+ setTimeout(function(d) {
+ return function() {
+ self.remoteStreamAdded(d);
+ }
+ }(data), 250);
+ return;
+ }
+
+ // ok to overwrite the one from focus? might save work in colibri.js
+ console.log('associated jid', ssrc2jid[thessrc], data.peerjid);
+ if (ssrc2jid[thessrc]) {
+ data.peerjid = ssrc2jid[thessrc];
+ }
+ }
+ }
+
+ //TODO: this code should be removed when firefox implement multistream support
+ if(RTC.getBrowserType() == RTCBrowserType.RTC_BROWSER_FIREFOX)
+ {
+ if((notReceivedSSRCs.length == 0) ||
+ !ssrc2jid[notReceivedSSRCs[notReceivedSSRCs.length - 1]])
+ {
+ // TODO(gp) limit wait duration to 1 sec.
+ setTimeout(function(d) {
+ return function() {
+ self.remoteStreamAdded(d);
+ }
+ }(data), 250);
+ return;
+ }
+
+ thessrc = notReceivedSSRCs.pop();
+ if (ssrc2jid[thessrc]) {
+ data.peerjid = ssrc2jid[thessrc];
+ }
+ }
+
+ RTC.createRemoteStream(data, this.sid, thessrc);
+
+ var isVideo = data.stream.getVideoTracks().length > 0;
+ // an attempt to work around https://github.com/jitsi/jitmeet/issues/32
+ if (isVideo &&
+ data.peerjid && this.peerjid === data.peerjid &&
+ data.stream.getVideoTracks().length === 0 &&
+ RTC.localVideo.getTracks().length > 0) {
+ window.setTimeout(function () {
+ sendKeyframe(self.peerconnection);
+ }, 3000);
+ }
+}
+
+module.exports = JingleSession;
+},{"./SDP":2,"./SDPDiffer":3,"./SDPUtil":4,"./TraceablePeerConnection":5}],2:[function(require,module,exports){
+/* jshint -W117 */
+var SDPUtil = require("./SDPUtil");
+
+// SDP STUFF
+function SDP(sdp) {
+ this.media = sdp.split('\r\nm=');
+ for (var i = 1; i < this.media.length; i++) {
+ this.media[i] = 'm=' + this.media[i];
+ if (i != this.media.length - 1) {
+ this.media[i] += '\r\n';
+ }
+ }
+ this.session = this.media.shift() + '\r\n';
+ this.raw = this.session + this.media.join('');
+}
+/**
+ * Returns map of MediaChannel mapped per channel idx.
+ */
+SDP.prototype.getMediaSsrcMap = function() {
+ var self = this;
+ var media_ssrcs = {};
+ var tmp;
+ for (var mediaindex = 0; mediaindex < self.media.length; mediaindex++) {
+ tmp = SDPUtil.find_lines(self.media[mediaindex], 'a=ssrc:');
+ var mid = SDPUtil.parse_mid(SDPUtil.find_line(self.media[mediaindex], 'a=mid:'));
+ var media = {
+ mediaindex: mediaindex,
+ mid: mid,
+ ssrcs: {},
+ ssrcGroups: []
+ };
+ media_ssrcs[mediaindex] = media;
+ tmp.forEach(function (line) {
+ var linessrc = line.substring(7).split(' ')[0];
+ // allocate new ChannelSsrc
+ if(!media.ssrcs[linessrc]) {
+ media.ssrcs[linessrc] = {
+ ssrc: linessrc,
+ lines: []
+ };
+ }
+ media.ssrcs[linessrc].lines.push(line);
+ });
+ tmp = SDPUtil.find_lines(self.media[mediaindex], 'a=ssrc-group:');
+ tmp.forEach(function(line){
+ var semantics = line.substr(0, idx).substr(13);
+ var ssrcs = line.substr(14 + semantics.length).split(' ');
+ if (ssrcs.length != 0) {
+ media.ssrcGroups.push({
+ semantics: semantics,
+ ssrcs: ssrcs
+ });
+ }
+ });
+ }
+ return media_ssrcs;
+};
+/**
+ * Returns true if this SDP contains given SSRC.
+ * @param ssrc the ssrc to check.
+ * @returns {boolean} true if this SDP contains given SSRC.
+ */
+SDP.prototype.containsSSRC = function(ssrc) {
+ var medias = this.getMediaSsrcMap();
+ var contains = false;
+ Object.keys(medias).forEach(function(mediaindex){
+ var media = medias[mediaindex];
+ //console.log("Check", channel, ssrc);
+ if(Object.keys(media.ssrcs).indexOf(ssrc) != -1){
+ contains = true;
+ }
+ });
+ return contains;
+};
+
+
+// remove iSAC and CN from SDP
+SDP.prototype.mangle = function () {
+ var i, j, mline, lines, rtpmap, newdesc;
+ for (i = 0; i < this.media.length; i++) {
+ lines = this.media[i].split('\r\n');
+ lines.pop(); // remove empty last element
+ mline = SDPUtil.parse_mline(lines.shift());
+ if (mline.media != 'audio')
+ continue;
+ newdesc = '';
+ mline.fmt.length = 0;
+ for (j = 0; j < lines.length; j++) {
+ if (lines[j].substr(0, 9) == 'a=rtpmap:') {
+ rtpmap = SDPUtil.parse_rtpmap(lines[j]);
+ if (rtpmap.name == 'CN' || rtpmap.name == 'ISAC')
+ continue;
+ mline.fmt.push(rtpmap.id);
+ newdesc += lines[j] + '\r\n';
+ } else {
+ newdesc += lines[j] + '\r\n';
+ }
+ }
+ this.media[i] = SDPUtil.build_mline(mline) + '\r\n';
+ this.media[i] += newdesc;
+ }
+ this.raw = this.session + this.media.join('');
+};
+
+// remove lines matching prefix from session section
+SDP.prototype.removeSessionLines = function(prefix) {
+ var self = this;
+ var lines = SDPUtil.find_lines(this.session, prefix);
+ lines.forEach(function(line) {
+ self.session = self.session.replace(line + '\r\n', '');
+ });
+ this.raw = this.session + this.media.join('');
+ return lines;
+}
+// remove lines matching prefix from a media section specified by mediaindex
+// TODO: non-numeric mediaindex could match mid
+SDP.prototype.removeMediaLines = function(mediaindex, prefix) {
+ var self = this;
+ var lines = SDPUtil.find_lines(this.media[mediaindex], prefix);
+ lines.forEach(function(line) {
+ self.media[mediaindex] = self.media[mediaindex].replace(line + '\r\n', '');
+ });
+ this.raw = this.session + this.media.join('');
+ return lines;
+}
+
+// add content's to a jingle element
+SDP.prototype.toJingle = function (elem, thecreator, ssrcs) {
+// console.log("SSRC" + ssrcs["audio"] + " - " + ssrcs["video"]);
+ var i, j, k, mline, ssrc, rtpmap, tmp, line, lines;
+ var self = this;
+ // new bundle plan
+ if (SDPUtil.find_line(this.session, 'a=group:')) {
+ lines = SDPUtil.find_lines(this.session, 'a=group:');
+ for (i = 0; i < lines.length; i++) {
+ tmp = lines[i].split(' ');
+ var semantics = tmp.shift().substr(8);
+ elem.c('group', {xmlns: 'urn:xmpp:jingle:apps:grouping:0', semantics:semantics});
+ for (j = 0; j < tmp.length; j++) {
+ elem.c('content', {name: tmp[j]}).up();
+ }
+ elem.up();
+ }
+ }
+ for (i = 0; i < this.media.length; i++) {
+ mline = SDPUtil.parse_mline(this.media[i].split('\r\n')[0]);
+ if (!(mline.media === 'audio' ||
+ mline.media === 'video' ||
+ mline.media === 'application'))
+ {
+ continue;
+ }
+ if (SDPUtil.find_line(this.media[i], 'a=ssrc:')) {
+ ssrc = SDPUtil.find_line(this.media[i], 'a=ssrc:').substring(7).split(' ')[0]; // take the first
+ } else {
+ if(ssrcs && ssrcs[mline.media])
+ {
+ ssrc = ssrcs[mline.media];
+ }
+ else
+ ssrc = false;
+ }
+
+ elem.c('content', {creator: thecreator, name: mline.media});
+ if (SDPUtil.find_line(this.media[i], 'a=mid:')) {
+ // prefer identifier from a=mid if present
+ var mid = SDPUtil.parse_mid(SDPUtil.find_line(this.media[i], 'a=mid:'));
+ elem.attrs({ name: mid });
+ }
+
+ if (SDPUtil.find_line(this.media[i], 'a=rtpmap:').length)
+ {
+ elem.c('description',
+ {xmlns: 'urn:xmpp:jingle:apps:rtp:1',
+ media: mline.media });
+ if (ssrc) {
+ elem.attrs({ssrc: ssrc});
+ }
+ for (j = 0; j < mline.fmt.length; j++) {
+ rtpmap = SDPUtil.find_line(this.media[i], 'a=rtpmap:' + mline.fmt[j]);
+ elem.c('payload-type', SDPUtil.parse_rtpmap(rtpmap));
+ // put any 'a=fmtp:' + mline.fmt[j] lines into
+ if (SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j])) {
+ tmp = SDPUtil.parse_fmtp(SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j]));
+ for (k = 0; k < tmp.length; k++) {
+ elem.c('parameter', tmp[k]).up();
+ }
+ }
+ this.RtcpFbToJingle(i, elem, mline.fmt[j]); // XEP-0293 -- map a=rtcp-fb
+
+ elem.up();
+ }
+ if (SDPUtil.find_line(this.media[i], 'a=crypto:', this.session)) {
+ elem.c('encryption', {required: 1});
+ var crypto = SDPUtil.find_lines(this.media[i], 'a=crypto:', this.session);
+ crypto.forEach(function(line) {
+ elem.c('crypto', SDPUtil.parse_crypto(line)).up();
+ });
+ elem.up(); // end of encryption
+ }
+
+ if (ssrc) {
+ // new style mapping
+ elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
+ // FIXME: group by ssrc and support multiple different ssrcs
+ var ssrclines = SDPUtil.find_lines(this.media[i], 'a=ssrc:');
+ if(ssrclines.length > 0) {
+ ssrclines.forEach(function (line) {
+ idx = line.indexOf(' ');
+ var linessrc = line.substr(0, idx).substr(7);
+ if (linessrc != ssrc) {
+ elem.up();
+ ssrc = linessrc;
+ elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
+ }
+ var kv = line.substr(idx + 1);
+ elem.c('parameter');
+ if (kv.indexOf(':') == -1) {
+ elem.attrs({ name: kv });
+ } else {
+ elem.attrs({ name: kv.split(':', 2)[0] });
+ elem.attrs({ value: kv.split(':', 2)[1] });
+ }
+ elem.up();
+ });
+ elem.up();
+ }
+ else
+ {
+ elem.up();
+ elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
+ elem.c('parameter');
+ elem.attrs({name: "cname", value:Math.random().toString(36).substring(7)});
+ elem.up();
+ var msid = null;
+ if(mline.media == "audio")
+ {
+ msid = RTC.localAudio.getId();
+ }
+ else
+ {
+ msid = RTC.localVideo.getId();
+ }
+ if(msid != null)
+ {
+ msid = msid.replace(/[\{,\}]/g,"");
+ elem.c('parameter');
+ elem.attrs({name: "msid", value:msid});
+ elem.up();
+ elem.c('parameter');
+ elem.attrs({name: "mslabel", value:msid});
+ elem.up();
+ elem.c('parameter');
+ elem.attrs({name: "label", value:msid});
+ elem.up();
+ elem.up();
+ }
+
+
+ }
+
+ // XEP-0339 handle ssrc-group attributes
+ var ssrc_group_lines = SDPUtil.find_lines(this.media[i], 'a=ssrc-group:');
+ ssrc_group_lines.forEach(function(line) {
+ idx = line.indexOf(' ');
+ var semantics = line.substr(0, idx).substr(13);
+ var ssrcs = line.substr(14 + semantics.length).split(' ');
+ if (ssrcs.length != 0) {
+ elem.c('ssrc-group', { semantics: semantics, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
+ ssrcs.forEach(function(ssrc) {
+ elem.c('source', { ssrc: ssrc })
+ .up();
+ });
+ elem.up();
+ }
+ });
+ }
+
+ if (SDPUtil.find_line(this.media[i], 'a=rtcp-mux')) {
+ elem.c('rtcp-mux').up();
+ }
+
+ // XEP-0293 -- map a=rtcp-fb:*
+ this.RtcpFbToJingle(i, elem, '*');
+
+ // XEP-0294
+ if (SDPUtil.find_line(this.media[i], 'a=extmap:')) {
+ lines = SDPUtil.find_lines(this.media[i], 'a=extmap:');
+ for (j = 0; j < lines.length; j++) {
+ tmp = SDPUtil.parse_extmap(lines[j]);
+ elem.c('rtp-hdrext', { xmlns: 'urn:xmpp:jingle:apps:rtp:rtp-hdrext:0',
+ uri: tmp.uri,
+ id: tmp.value });
+ if (tmp.hasOwnProperty('direction')) {
+ switch (tmp.direction) {
+ case 'sendonly':
+ elem.attrs({senders: 'responder'});
+ break;
+ case 'recvonly':
+ elem.attrs({senders: 'initiator'});
+ break;
+ case 'sendrecv':
+ elem.attrs({senders: 'both'});
+ break;
+ case 'inactive':
+ elem.attrs({senders: 'none'});
+ break;
+ }
+ }
+ // TODO: handle params
+ elem.up();
+ }
+ }
+ elem.up(); // end of description
+ }
+
+ // map ice-ufrag/pwd, dtls fingerprint, candidates
+ this.TransportToJingle(i, elem);
+
+ if (SDPUtil.find_line(this.media[i], 'a=sendrecv', this.session)) {
+ elem.attrs({senders: 'both'});
+ } else if (SDPUtil.find_line(this.media[i], 'a=sendonly', this.session)) {
+ elem.attrs({senders: 'initiator'});
+ } else if (SDPUtil.find_line(this.media[i], 'a=recvonly', this.session)) {
+ elem.attrs({senders: 'responder'});
+ } else if (SDPUtil.find_line(this.media[i], 'a=inactive', this.session)) {
+ elem.attrs({senders: 'none'});
+ }
+ if (mline.port == '0') {
+ // estos hack to reject an m-line
+ elem.attrs({senders: 'rejected'});
+ }
+ elem.up(); // end of content
+ }
+ elem.up();
+ return elem;
+};
+
+SDP.prototype.TransportToJingle = function (mediaindex, elem) {
+ var i = mediaindex;
+ var tmp;
+ var self = this;
+ elem.c('transport');
+
+ // XEP-0343 DTLS/SCTP
+ if (SDPUtil.find_line(this.media[mediaindex], 'a=sctpmap:').length)
+ {
+ var sctpmap = SDPUtil.find_line(
+ this.media[i], 'a=sctpmap:', self.session);
+ if (sctpmap)
+ {
+ var sctpAttrs = SDPUtil.parse_sctpmap(sctpmap);
+ elem.c('sctpmap',
+ {
+ xmlns: 'urn:xmpp:jingle:transports:dtls-sctp:1',
+ number: sctpAttrs[0], /* SCTP port */
+ protocol: sctpAttrs[1], /* protocol */
+ });
+ // Optional stream count attribute
+ if (sctpAttrs.length > 2)
+ elem.attrs({ streams: sctpAttrs[2]});
+ elem.up();
+ }
+ }
+ // XEP-0320
+ var fingerprints = SDPUtil.find_lines(this.media[mediaindex], 'a=fingerprint:', this.session);
+ fingerprints.forEach(function(line) {
+ tmp = SDPUtil.parse_fingerprint(line);
+ tmp.xmlns = 'urn:xmpp:jingle:apps:dtls:0';
+ elem.c('fingerprint').t(tmp.fingerprint);
+ delete tmp.fingerprint;
+ line = SDPUtil.find_line(self.media[mediaindex], 'a=setup:', self.session);
+ if (line) {
+ tmp.setup = line.substr(8);
+ }
+ elem.attrs(tmp);
+ elem.up(); // end of fingerprint
+ });
+ tmp = SDPUtil.iceparams(this.media[mediaindex], this.session);
+ if (tmp) {
+ tmp.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
+ elem.attrs(tmp);
+ // XEP-0176
+ if (SDPUtil.find_line(this.media[mediaindex], 'a=candidate:', this.session)) { // add any a=candidate lines
+ var lines = SDPUtil.find_lines(this.media[mediaindex], 'a=candidate:', this.session);
+ lines.forEach(function (line) {
+ elem.c('candidate', SDPUtil.candidateToJingle(line)).up();
+ });
+ }
+ }
+ elem.up(); // end of transport
+}
+
+SDP.prototype.RtcpFbToJingle = function (mediaindex, elem, payloadtype) { // XEP-0293
+ var lines = SDPUtil.find_lines(this.media[mediaindex], 'a=rtcp-fb:' + payloadtype);
+ lines.forEach(function (line) {
+ var tmp = SDPUtil.parse_rtcpfb(line);
+ if (tmp.type == 'trr-int') {
+ elem.c('rtcp-fb-trr-int', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', value: tmp.params[0]});
+ elem.up();
+ } else {
+ elem.c('rtcp-fb', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', type: tmp.type});
+ if (tmp.params.length > 0) {
+ elem.attrs({'subtype': tmp.params[0]});
+ }
+ elem.up();
+ }
+ });
+};
+
+SDP.prototype.RtcpFbFromJingle = function (elem, payloadtype) { // XEP-0293
+ var media = '';
+ var tmp = elem.find('>rtcp-fb-trr-int[xmlns="urn:xmpp:jingle:apps:rtp:rtcp-fb:0"]');
+ if (tmp.length) {
+ media += 'a=rtcp-fb:' + '*' + ' ' + 'trr-int' + ' ';
+ if (tmp.attr('value')) {
+ media += tmp.attr('value');
+ } else {
+ media += '0';
+ }
+ media += '\r\n';
+ }
+ tmp = elem.find('>rtcp-fb[xmlns="urn:xmpp:jingle:apps:rtp:rtcp-fb:0"]');
+ tmp.each(function () {
+ media += 'a=rtcp-fb:' + payloadtype + ' ' + $(this).attr('type');
+ if ($(this).attr('subtype')) {
+ media += ' ' + $(this).attr('subtype');
+ }
+ media += '\r\n';
+ });
+ return media;
+};
+
+// construct an SDP from a jingle stanza
+SDP.prototype.fromJingle = function (jingle) {
+ var self = this;
+ this.raw = 'v=0\r\n' +
+ 'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\r\n' +// FIXME
+ 's=-\r\n' +
+ 't=0 0\r\n';
+ // http://tools.ietf.org/html/draft-ietf-mmusic-sdp-bundle-negotiation-04#section-8
+ if ($(jingle).find('>group[xmlns="urn:xmpp:jingle:apps:grouping:0"]').length) {
+ $(jingle).find('>group[xmlns="urn:xmpp:jingle:apps:grouping:0"]').each(function (idx, group) {
+ var contents = $(group).find('>content').map(function (idx, content) {
+ return content.getAttribute('name');
+ }).get();
+ if (contents.length > 0) {
+ self.raw += 'a=group:' + (group.getAttribute('semantics') || group.getAttribute('type')) + ' ' + contents.join(' ') + '\r\n';
+ }
+ });
+ }
+
+ this.session = this.raw;
+ jingle.find('>content').each(function () {
+ var m = self.jingle2media($(this));
+ self.media.push(m);
+ });
+
+ // reconstruct msid-semantic -- apparently not necessary
+ /*
+ var msid = SDPUtil.parse_ssrc(this.raw);
+ if (msid.hasOwnProperty('mslabel')) {
+ this.session += "a=msid-semantic: WMS " + msid.mslabel + "\r\n";
+ }
+ */
+
+ this.raw = this.session + this.media.join('');
+};
+
+// translate a jingle content element into an an SDP media part
+SDP.prototype.jingle2media = function (content) {
+ var media = '',
+ desc = content.find('description'),
+ ssrc = desc.attr('ssrc'),
+ self = this,
+ tmp;
+ var sctp = content.find(
+ '>transport>sctpmap[xmlns="urn:xmpp:jingle:transports:dtls-sctp:1"]');
+
+ tmp = { media: desc.attr('media') };
+ tmp.port = '1';
+ if (content.attr('senders') == 'rejected') {
+ // estos hack to reject an m-line.
+ tmp.port = '0';
+ }
+ if (content.find('>transport>fingerprint').length || desc.find('encryption').length) {
+ if (sctp.length)
+ tmp.proto = 'DTLS/SCTP';
+ else
+ tmp.proto = 'RTP/SAVPF';
+ } else {
+ tmp.proto = 'RTP/AVPF';
+ }
+ if (!sctp.length)
+ {
+ tmp.fmt = desc.find('payload-type').map(
+ function () { return this.getAttribute('id'); }).get();
+ media += SDPUtil.build_mline(tmp) + '\r\n';
+ }
+ else
+ {
+ media += 'm=application 1 DTLS/SCTP ' + sctp.attr('number') + '\r\n';
+ media += 'a=sctpmap:' + sctp.attr('number') +
+ ' ' + sctp.attr('protocol');
+
+ var streamCount = sctp.attr('streams');
+ if (streamCount)
+ media += ' ' + streamCount + '\r\n';
+ else
+ media += '\r\n';
+ }
+
+ media += 'c=IN IP4 0.0.0.0\r\n';
+ if (!sctp.length)
+ media += 'a=rtcp:1 IN IP4 0.0.0.0\r\n';
+ tmp = content.find('>transport[xmlns="urn:xmpp:jingle:transports:ice-udp:1"]');
+ if (tmp.length) {
+ if (tmp.attr('ufrag')) {
+ media += SDPUtil.build_iceufrag(tmp.attr('ufrag')) + '\r\n';
+ }
+ if (tmp.attr('pwd')) {
+ media += SDPUtil.build_icepwd(tmp.attr('pwd')) + '\r\n';
+ }
+ tmp.find('>fingerprint').each(function () {
+ // FIXME: check namespace at some point
+ media += 'a=fingerprint:' + this.getAttribute('hash');
+ media += ' ' + $(this).text();
+ media += '\r\n';
+ if (this.getAttribute('setup')) {
+ media += 'a=setup:' + this.getAttribute('setup') + '\r\n';
+ }
+ });
+ }
+ switch (content.attr('senders')) {
+ case 'initiator':
+ media += 'a=sendonly\r\n';
+ break;
+ case 'responder':
+ media += 'a=recvonly\r\n';
+ break;
+ case 'none':
+ media += 'a=inactive\r\n';
+ break;
+ case 'both':
+ media += 'a=sendrecv\r\n';
+ break;
+ }
+ media += 'a=mid:' + content.attr('name') + '\r\n';
+
+ //
+ // see http://code.google.com/p/libjingle/issues/detail?id=309 -- no spec though
+ // and http://mail.jabber.org/pipermail/jingle/2011-December/001761.html
+ if (desc.find('rtcp-mux').length) {
+ media += 'a=rtcp-mux\r\n';
+ }
+
+ if (desc.find('encryption').length) {
+ desc.find('encryption>crypto').each(function () {
+ media += 'a=crypto:' + this.getAttribute('tag');
+ media += ' ' + this.getAttribute('crypto-suite');
+ media += ' ' + this.getAttribute('key-params');
+ if (this.getAttribute('session-params')) {
+ media += ' ' + this.getAttribute('session-params');
+ }
+ media += '\r\n';
+ });
+ }
+ desc.find('payload-type').each(function () {
+ media += SDPUtil.build_rtpmap(this) + '\r\n';
+ if ($(this).find('>parameter').length) {
+ media += 'a=fmtp:' + this.getAttribute('id') + ' ';
+ media += $(this).find('parameter').map(function () { return (this.getAttribute('name') ? (this.getAttribute('name') + '=') : '') + this.getAttribute('value'); }).get().join('; ');
+ media += '\r\n';
+ }
+ // xep-0293
+ media += self.RtcpFbFromJingle($(this), this.getAttribute('id'));
+ });
+
+ // xep-0293
+ media += self.RtcpFbFromJingle(desc, '*');
+
+ // xep-0294
+ tmp = desc.find('>rtp-hdrext[xmlns="urn:xmpp:jingle:apps:rtp:rtp-hdrext:0"]');
+ tmp.each(function () {
+ media += 'a=extmap:' + this.getAttribute('id') + ' ' + this.getAttribute('uri') + '\r\n';
+ });
+
+ content.find('>transport[xmlns="urn:xmpp:jingle:transports:ice-udp:1"]>candidate').each(function () {
+ media += SDPUtil.candidateFromJingle(this);
+ });
+
+ // XEP-0339 handle ssrc-group attributes
+ tmp = content.find('description>ssrc-group[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]').each(function() {
+ var semantics = this.getAttribute('semantics');
+ var ssrcs = $(this).find('>source').map(function() {
+ return this.getAttribute('ssrc');
+ }).get();
+
+ if (ssrcs.length != 0) {
+ media += 'a=ssrc-group:' + semantics + ' ' + ssrcs.join(' ') + '\r\n';
+ }
+ });
+
+ tmp = content.find('description>source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]');
+ tmp.each(function () {
+ var ssrc = this.getAttribute('ssrc');
+ $(this).find('>parameter').each(function () {
+ media += 'a=ssrc:' + ssrc + ' ' + this.getAttribute('name');
+ if (this.getAttribute('value') && this.getAttribute('value').length)
+ media += ':' + this.getAttribute('value');
+ media += '\r\n';
+ });
+ });
+
+ return media;
+};
+
+
+module.exports = SDP;
+
+
+},{"./SDPUtil":4}],3:[function(require,module,exports){
+function SDPDiffer(mySDP, otherSDP) {
+ this.mySDP = mySDP;
+ this.otherSDP = otherSDP;
+}
+
+/**
+ * Returns map of MediaChannel that contains only media not contained in otherSdp. Mapped by channel idx.
+ * @param otherSdp the other SDP to check ssrc with.
+ */
+SDPDiffer.prototype.getNewMedia = function() {
+
+ // this could be useful in Array.prototype.
+ function arrayEquals(array) {
+ // if the other array is a falsy value, return
+ if (!array)
+ return false;
+
+ // compare lengths - can save a lot of time
+ if (this.length != array.length)
+ return false;
+
+ for (var i = 0, l=this.length; i < l; i++) {
+ // Check if we have nested arrays
+ if (this[i] instanceof Array && array[i] instanceof Array) {
+ // recurse into the nested arrays
+ if (!this[i].equals(array[i]))
+ return false;
+ }
+ else if (this[i] != array[i]) {
+ // Warning - two different object instances will never be equal: {x:20} != {x:20}
+ return false;
+ }
+ }
+ return true;
+ }
+
+ var myMedias = this.mySDP.getMediaSsrcMap();
+ var othersMedias = this.otherSDP.getMediaSsrcMap();
+ var newMedia = {};
+ Object.keys(othersMedias).forEach(function(othersMediaIdx) {
+ var myMedia = myMedias[othersMediaIdx];
+ var othersMedia = othersMedias[othersMediaIdx];
+ if(!myMedia && othersMedia) {
+ // Add whole channel
+ newMedia[othersMediaIdx] = othersMedia;
+ return;
+ }
+ // Look for new ssrcs accross the channel
+ Object.keys(othersMedia.ssrcs).forEach(function(ssrc) {
+ if(Object.keys(myMedia.ssrcs).indexOf(ssrc) === -1) {
+ // Allocate channel if we've found ssrc that doesn't exist in our channel
+ if(!newMedia[othersMediaIdx]){
+ newMedia[othersMediaIdx] = {
+ mediaindex: othersMedia.mediaindex,
+ mid: othersMedia.mid,
+ ssrcs: {},
+ ssrcGroups: []
+ };
+ }
+ newMedia[othersMediaIdx].ssrcs[ssrc] = othersMedia.ssrcs[ssrc];
+ }
+ });
+
+ // Look for new ssrc groups across the channels
+ othersMedia.ssrcGroups.forEach(function(otherSsrcGroup){
+
+ // try to match the other ssrc-group with an ssrc-group of ours
+ var matched = false;
+ for (var i = 0; i < myMedia.ssrcGroups.length; i++) {
+ var mySsrcGroup = myMedia.ssrcGroups[i];
+ if (otherSsrcGroup.semantics == mySsrcGroup.semantics
+ && arrayEquals.apply(otherSsrcGroup.ssrcs, [mySsrcGroup.ssrcs])) {
+
+ matched = true;
+ break;
+ }
+ }
+
+ if (!matched) {
+ // Allocate channel if we've found an ssrc-group that doesn't
+ // exist in our channel
+
+ if(!newMedia[othersMediaIdx]){
+ newMedia[othersMediaIdx] = {
+ mediaindex: othersMedia.mediaindex,
+ mid: othersMedia.mid,
+ ssrcs: {},
+ ssrcGroups: []
+ };
+ }
+ newMedia[othersMediaIdx].ssrcGroups.push(otherSsrcGroup);
+ }
+ });
+ });
+ return newMedia;
+};
+
+/**
+ * Sends SSRC update IQ.
+ * @param sdpMediaSsrcs SSRCs map obtained from SDP.getNewMedia. Cntains SSRCs to add/remove.
+ * @param sid session identifier that will be put into the IQ.
+ * @param initiator initiator identifier.
+ * @param toJid destination Jid
+ * @param isAdd indicates if this is remove or add operation.
+ */
+SDPDiffer.prototype.toJingle = function(modify) {
+ var sdpMediaSsrcs = this.getNewMedia();
+ var self = this;
+
+ // FIXME: only announce video ssrcs since we mix audio and dont need
+ // the audio ssrcs therefore
+ var modified = false;
+ Object.keys(sdpMediaSsrcs).forEach(function(mediaindex){
+ modified = true;
+ var media = sdpMediaSsrcs[mediaindex];
+ modify.c('content', {name: media.mid});
+
+ modify.c('description', {xmlns:'urn:xmpp:jingle:apps:rtp:1', media: media.mid});
+ // FIXME: not completly sure this operates on blocks and / or handles different ssrcs correctly
+ // generate sources from lines
+ Object.keys(media.ssrcs).forEach(function(ssrcNum) {
+ var mediaSsrc = media.ssrcs[ssrcNum];
+ modify.c('source', { xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
+ modify.attrs({ssrc: mediaSsrc.ssrc});
+ // iterate over ssrc lines
+ mediaSsrc.lines.forEach(function (line) {
+ var idx = line.indexOf(' ');
+ var kv = line.substr(idx + 1);
+ modify.c('parameter');
+ if (kv.indexOf(':') == -1) {
+ modify.attrs({ name: kv });
+ } else {
+ modify.attrs({ name: kv.split(':', 2)[0] });
+ modify.attrs({ value: kv.split(':', 2)[1] });
+ }
+ modify.up(); // end of parameter
+ });
+ modify.up(); // end of source
+ });
+
+ // generate source groups from lines
+ media.ssrcGroups.forEach(function(ssrcGroup) {
+ if (ssrcGroup.ssrcs.length != 0) {
+
+ modify.c('ssrc-group', {
+ semantics: ssrcGroup.semantics,
+ xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0'
+ });
+
+ ssrcGroup.ssrcs.forEach(function (ssrc) {
+ modify.c('source', { ssrc: ssrc })
+ .up(); // end of source
+ });
+ modify.up(); // end of ssrc-group
+ }
+ });
+
+ modify.up(); // end of description
+ modify.up(); // end of content
+ });
+
+ return modified;
+};
+
+module.exports = SDPDiffer;
+},{}],4:[function(require,module,exports){
+SDPUtil = {
+ iceparams: function (mediadesc, sessiondesc) {
+ var data = null;
+ if (SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc) &&
+ SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc)) {
+ data = {
+ ufrag: SDPUtil.parse_iceufrag(SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc)),
+ pwd: SDPUtil.parse_icepwd(SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc))
+ };
+ }
+ return data;
+ },
+ parse_iceufrag: function (line) {
+ return line.substring(12);
+ },
+ build_iceufrag: function (frag) {
+ return 'a=ice-ufrag:' + frag;
+ },
+ parse_icepwd: function (line) {
+ return line.substring(10);
+ },
+ build_icepwd: function (pwd) {
+ return 'a=ice-pwd:' + pwd;
+ },
+ parse_mid: function (line) {
+ return line.substring(6);
+ },
+ parse_mline: function (line) {
+ var parts = line.substring(2).split(' '),
+ data = {};
+ data.media = parts.shift();
+ data.port = parts.shift();
+ data.proto = parts.shift();
+ if (parts[parts.length - 1] === '') { // trailing whitespace
+ parts.pop();
+ }
+ data.fmt = parts;
+ return data;
+ },
+ build_mline: function (mline) {
+ return 'm=' + mline.media + ' ' + mline.port + ' ' + mline.proto + ' ' + mline.fmt.join(' ');
+ },
+ parse_rtpmap: function (line) {
+ var parts = line.substring(9).split(' '),
+ data = {};
+ data.id = parts.shift();
+ parts = parts[0].split('/');
+ data.name = parts.shift();
+ data.clockrate = parts.shift();
+ data.channels = parts.length ? parts.shift() : '1';
+ return data;
+ },
+ /**
+ * Parses SDP line "a=sctpmap:..." and extracts SCTP port from it.
+ * @param line eg. "a=sctpmap:5000 webrtc-datachannel"
+ * @returns [SCTP port number, protocol, streams]
+ */
+ parse_sctpmap: function (line)
+ {
+ var parts = line.substring(10).split(' ');
+ var sctpPort = parts[0];
+ var protocol = parts[1];
+ // Stream count is optional
+ var streamCount = parts.length > 2 ? parts[2] : null;
+ return [sctpPort, protocol, streamCount];// SCTP port
+ },
+ build_rtpmap: function (el) {
+ var line = 'a=rtpmap:' + el.getAttribute('id') + ' ' + el.getAttribute('name') + '/' + el.getAttribute('clockrate');
+ if (el.getAttribute('channels') && el.getAttribute('channels') != '1') {
+ line += '/' + el.getAttribute('channels');
+ }
+ return line;
+ },
+ parse_crypto: function (line) {
+ var parts = line.substring(9).split(' '),
+ data = {};
+ data.tag = parts.shift();
+ data['crypto-suite'] = parts.shift();
+ data['key-params'] = parts.shift();
+ if (parts.length) {
+ data['session-params'] = parts.join(' ');
+ }
+ return data;
+ },
+ parse_fingerprint: function (line) { // RFC 4572
+ var parts = line.substring(14).split(' '),
+ data = {};
+ data.hash = parts.shift();
+ data.fingerprint = parts.shift();
+ // TODO assert that fingerprint satisfies 2UHEX *(":" 2UHEX) ?
+ return data;
+ },
+ parse_fmtp: function (line) {
+ var parts = line.split(' '),
+ i, key, value,
+ data = [];
+ parts.shift();
+ parts = parts.join(' ').split(';');
+ for (i = 0; i < parts.length; i++) {
+ key = parts[i].split('=')[0];
+ while (key.length && key[0] == ' ') {
+ key = key.substring(1);
+ }
+ value = parts[i].split('=')[1];
+ if (key && value) {
+ data.push({name: key, value: value});
+ } else if (key) {
+ // rfc 4733 (DTMF) style stuff
+ data.push({name: '', value: key});
+ }
+ }
+ return data;
+ },
+ parse_icecandidate: function (line) {
+ var candidate = {},
+ elems = line.split(' ');
+ candidate.foundation = elems[0].substring(12);
+ candidate.component = elems[1];
+ candidate.protocol = elems[2].toLowerCase();
+ candidate.priority = elems[3];
+ candidate.ip = elems[4];
+ candidate.port = elems[5];
+ // elems[6] => "typ"
+ candidate.type = elems[7];
+ candidate.generation = 0; // default value, may be overwritten below
+ for (var i = 8; i < elems.length; i += 2) {
+ switch (elems[i]) {
+ case 'raddr':
+ candidate['rel-addr'] = elems[i + 1];
+ break;
+ case 'rport':
+ candidate['rel-port'] = elems[i + 1];
+ break;
+ case 'generation':
+ candidate.generation = elems[i + 1];
+ break;
+ case 'tcptype':
+ candidate.tcptype = elems[i + 1];
+ break;
+ default: // TODO
+ console.log('parse_icecandidate not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
+ }
+ }
+ candidate.network = '1';
+ candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
+ return candidate;
+ },
+ build_icecandidate: function (cand) {
+ var line = ['a=candidate:' + cand.foundation, cand.component, cand.protocol, cand.priority, cand.ip, cand.port, 'typ', cand.type].join(' ');
+ line += ' ';
+ switch (cand.type) {
+ case 'srflx':
+ case 'prflx':
+ case 'relay':
+ if (cand.hasOwnAttribute('rel-addr') && cand.hasOwnAttribute('rel-port')) {
+ line += 'raddr';
+ line += ' ';
+ line += cand['rel-addr'];
+ line += ' ';
+ line += 'rport';
+ line += ' ';
+ line += cand['rel-port'];
+ line += ' ';
+ }
+ break;
+ }
+ if (cand.hasOwnAttribute('tcptype')) {
+ line += 'tcptype';
+ line += ' ';
+ line += cand.tcptype;
+ line += ' ';
+ }
+ line += 'generation';
+ line += ' ';
+ line += cand.hasOwnAttribute('generation') ? cand.generation : '0';
+ return line;
+ },
+ parse_ssrc: function (desc) {
+ // proprietary mapping of a=ssrc lines
+ // TODO: see "Jingle RTP Source Description" by Juberti and P. Thatcher on google docs
+ // and parse according to that
+ var lines = desc.split('\r\n'),
+ data = {};
+ for (var i = 0; i < lines.length; i++) {
+ if (lines[i].substring(0, 7) == 'a=ssrc:') {
+ var idx = lines[i].indexOf(' ');
+ data[lines[i].substr(idx + 1).split(':', 2)[0]] = lines[i].substr(idx + 1).split(':', 2)[1];
+ }
+ }
+ return data;
+ },
+ parse_rtcpfb: function (line) {
+ var parts = line.substr(10).split(' ');
+ var data = {};
+ data.pt = parts.shift();
+ data.type = parts.shift();
+ data.params = parts;
+ return data;
+ },
+ parse_extmap: function (line) {
+ var parts = line.substr(9).split(' ');
+ var data = {};
+ data.value = parts.shift();
+ if (data.value.indexOf('/') != -1) {
+ data.direction = data.value.substr(data.value.indexOf('/') + 1);
+ data.value = data.value.substr(0, data.value.indexOf('/'));
+ } else {
+ data.direction = 'both';
+ }
+ data.uri = parts.shift();
+ data.params = parts;
+ return data;
+ },
+ find_line: function (haystack, needle, sessionpart) {
+ var lines = haystack.split('\r\n');
+ for (var i = 0; i < lines.length; i++) {
+ if (lines[i].substring(0, needle.length) == needle) {
+ return lines[i];
+ }
+ }
+ if (!sessionpart) {
+ return false;
+ }
+ // search session part
+ lines = sessionpart.split('\r\n');
+ for (var j = 0; j < lines.length; j++) {
+ if (lines[j].substring(0, needle.length) == needle) {
+ return lines[j];
+ }
+ }
+ return false;
+ },
+ find_lines: function (haystack, needle, sessionpart) {
+ var lines = haystack.split('\r\n'),
+ needles = [];
+ for (var i = 0; i < lines.length; i++) {
+ if (lines[i].substring(0, needle.length) == needle)
+ needles.push(lines[i]);
+ }
+ if (needles.length || !sessionpart) {
+ return needles;
+ }
+ // search session part
+ lines = sessionpart.split('\r\n');
+ for (var j = 0; j < lines.length; j++) {
+ if (lines[j].substring(0, needle.length) == needle) {
+ needles.push(lines[j]);
+ }
+ }
+ return needles;
+ },
+ candidateToJingle: function (line) {
+ // a=candidate:2979166662 1 udp 2113937151 192.168.2.100 57698 typ host generation 0
+ //
+ if (line.indexOf('candidate:') === 0) {
+ line = 'a=' + line;
+ } else if (line.substring(0, 12) != 'a=candidate:') {
+ console.log('parseCandidate called with a line that is not a candidate line');
+ console.log(line);
+ return null;
+ }
+ if (line.substring(line.length - 2) == '\r\n') // chomp it
+ line = line.substring(0, line.length - 2);
+ var candidate = {},
+ elems = line.split(' '),
+ i;
+ if (elems[6] != 'typ') {
+ console.log('did not find typ in the right place');
+ console.log(line);
+ return null;
+ }
+ candidate.foundation = elems[0].substring(12);
+ candidate.component = elems[1];
+ candidate.protocol = elems[2].toLowerCase();
+ candidate.priority = elems[3];
+ candidate.ip = elems[4];
+ candidate.port = elems[5];
+ // elems[6] => "typ"
+ candidate.type = elems[7];
+
+ candidate.generation = '0'; // default, may be overwritten below
+ for (i = 8; i < elems.length; i += 2) {
+ switch (elems[i]) {
+ case 'raddr':
+ candidate['rel-addr'] = elems[i + 1];
+ break;
+ case 'rport':
+ candidate['rel-port'] = elems[i + 1];
+ break;
+ case 'generation':
+ candidate.generation = elems[i + 1];
+ break;
+ case 'tcptype':
+ candidate.tcptype = elems[i + 1];
+ break;
+ default: // TODO
+ console.log('not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
+ }
+ }
+ candidate.network = '1';
+ candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
+ return candidate;
+ },
+ candidateFromJingle: function (cand) {
+ var line = 'a=candidate:';
+ line += cand.getAttribute('foundation');
+ line += ' ';
+ line += cand.getAttribute('component');
+ line += ' ';
+ line += cand.getAttribute('protocol'); //.toUpperCase(); // chrome M23 doesn't like this
+ line += ' ';
+ line += cand.getAttribute('priority');
+ line += ' ';
+ line += cand.getAttribute('ip');
+ line += ' ';
+ line += cand.getAttribute('port');
+ line += ' ';
+ line += 'typ';
+ line += ' ' + cand.getAttribute('type');
+ line += ' ';
+ switch (cand.getAttribute('type')) {
+ case 'srflx':
+ case 'prflx':
+ case 'relay':
+ if (cand.getAttribute('rel-addr') && cand.getAttribute('rel-port')) {
+ line += 'raddr';
+ line += ' ';
+ line += cand.getAttribute('rel-addr');
+ line += ' ';
+ line += 'rport';
+ line += ' ';
+ line += cand.getAttribute('rel-port');
+ line += ' ';
+ }
+ break;
+ }
+ if (cand.getAttribute('protocol').toLowerCase() == 'tcp') {
+ line += 'tcptype';
+ line += ' ';
+ line += cand.getAttribute('tcptype');
+ line += ' ';
+ }
+ line += 'generation';
+ line += ' ';
+ line += cand.getAttribute('generation') || '0';
+ return line + '\r\n';
+ }
+};
+module.exports = SDPUtil;
+},{}],5:[function(require,module,exports){
+function TraceablePeerConnection(ice_config, constraints) {
+ var self = this;
+ var RTCPeerconnection = navigator.mozGetUserMedia ? mozRTCPeerConnection : webkitRTCPeerConnection;
+ this.peerconnection = new RTCPeerconnection(ice_config, constraints);
+ this.updateLog = [];
+ this.stats = {};
+ this.statsinterval = null;
+ this.maxstats = 0; // limit to 300 values, i.e. 5 minutes; set to 0 to disable
+
+ // override as desired
+ this.trace = function (what, info) {
+ //console.warn('WTRACE', what, info);
+ self.updateLog.push({
+ time: new Date(),
+ type: what,
+ value: info || ""
+ });
+ };
+ this.onicecandidate = null;
+ this.peerconnection.onicecandidate = function (event) {
+ self.trace('onicecandidate', JSON.stringify(event.candidate, null, ' '));
+ if (self.onicecandidate !== null) {
+ self.onicecandidate(event);
+ }
+ };
+ this.onaddstream = null;
+ this.peerconnection.onaddstream = function (event) {
+ self.trace('onaddstream', event.stream.id);
+ if (self.onaddstream !== null) {
+ self.onaddstream(event);
+ }
+ };
+ this.onremovestream = null;
+ this.peerconnection.onremovestream = function (event) {
+ self.trace('onremovestream', event.stream.id);
+ if (self.onremovestream !== null) {
+ self.onremovestream(event);
+ }
+ };
+ this.onsignalingstatechange = null;
+ this.peerconnection.onsignalingstatechange = function (event) {
+ self.trace('onsignalingstatechange', self.signalingState);
+ if (self.onsignalingstatechange !== null) {
+ self.onsignalingstatechange(event);
+ }
+ };
+ this.oniceconnectionstatechange = null;
+ this.peerconnection.oniceconnectionstatechange = function (event) {
+ self.trace('oniceconnectionstatechange', self.iceConnectionState);
+ if (self.oniceconnectionstatechange !== null) {
+ self.oniceconnectionstatechange(event);
+ }
+ };
+ this.onnegotiationneeded = null;
+ this.peerconnection.onnegotiationneeded = function (event) {
+ self.trace('onnegotiationneeded');
+ if (self.onnegotiationneeded !== null) {
+ self.onnegotiationneeded(event);
+ }
+ };
+ self.ondatachannel = null;
+ this.peerconnection.ondatachannel = function (event) {
+ self.trace('ondatachannel', event);
+ if (self.ondatachannel !== null) {
+ self.ondatachannel(event);
+ }
+ };
+ if (!navigator.mozGetUserMedia && this.maxstats) {
+ this.statsinterval = window.setInterval(function() {
+ self.peerconnection.getStats(function(stats) {
+ var results = stats.result();
+ for (var i = 0; i < results.length; ++i) {
+ //console.log(results[i].type, results[i].id, results[i].names())
+ var now = new Date();
+ results[i].names().forEach(function (name) {
+ var id = results[i].id + '-' + name;
+ if (!self.stats[id]) {
+ self.stats[id] = {
+ startTime: now,
+ endTime: now,
+ values: [],
+ times: []
+ };
+ }
+ self.stats[id].values.push(results[i].stat(name));
+ self.stats[id].times.push(now.getTime());
+ if (self.stats[id].values.length > self.maxstats) {
+ self.stats[id].values.shift();
+ self.stats[id].times.shift();
+ }
+ self.stats[id].endTime = now;
+ });
+ }
+ });
+
+ }, 1000);
+ }
+};
+
+dumpSDP = function(description) {
+ return 'type: ' + description.type + '\r\n' + description.sdp;
+}
+
+if (TraceablePeerConnection.prototype.__defineGetter__ !== undefined) {
+ TraceablePeerConnection.prototype.__defineGetter__('signalingState', function() { return this.peerconnection.signalingState; });
+ TraceablePeerConnection.prototype.__defineGetter__('iceConnectionState', function() { return this.peerconnection.iceConnectionState; });
+ TraceablePeerConnection.prototype.__defineGetter__('localDescription', function() {
+ var publicLocalDescription = simulcast.reverseTransformLocalDescription(this.peerconnection.localDescription);
+ return publicLocalDescription;
+ });
+ TraceablePeerConnection.prototype.__defineGetter__('remoteDescription', function() {
+ var publicRemoteDescription = simulcast.reverseTransformRemoteDescription(this.peerconnection.remoteDescription);
+ return publicRemoteDescription;
+ });
+}
+
+TraceablePeerConnection.prototype.addStream = function (stream) {
+ this.trace('addStream', stream.id);
+ simulcast.resetSender();
+ try
+ {
+ this.peerconnection.addStream(stream);
+ }
+ catch (e)
+ {
+ console.error(e);
+ return;
+ }
+};
+
+TraceablePeerConnection.prototype.removeStream = function (stream, stopStreams) {
+ this.trace('removeStream', stream.id);
+ simulcast.resetSender();
+ if(stopStreams) {
+ stream.getAudioTracks().forEach(function (track) {
+ track.stop();
+ });
+ stream.getVideoTracks().forEach(function (track) {
+ track.stop();
+ });
+ }
+ this.peerconnection.removeStream(stream);
+};
+
+TraceablePeerConnection.prototype.createDataChannel = function (label, opts) {
+ this.trace('createDataChannel', label, opts);
+ return this.peerconnection.createDataChannel(label, opts);
+};
+
+TraceablePeerConnection.prototype.setLocalDescription = function (description, successCallback, failureCallback) {
+ var self = this;
+ description = simulcast.transformLocalDescription(description);
+ this.trace('setLocalDescription', dumpSDP(description));
+ this.peerconnection.setLocalDescription(description,
+ function () {
+ self.trace('setLocalDescriptionOnSuccess');
+ successCallback();
+ },
+ function (err) {
+ self.trace('setLocalDescriptionOnFailure', err);
+ failureCallback(err);
+ }
+ );
+ /*
+ if (this.statsinterval === null && this.maxstats > 0) {
+ // start gathering stats
+ }
+ */
+};
+
+TraceablePeerConnection.prototype.setRemoteDescription = function (description, successCallback, failureCallback) {
+ var self = this;
+ description = simulcast.transformRemoteDescription(description);
+ this.trace('setRemoteDescription', dumpSDP(description));
+ this.peerconnection.setRemoteDescription(description,
+ function () {
+ self.trace('setRemoteDescriptionOnSuccess');
+ successCallback();
+ },
+ function (err) {
+ self.trace('setRemoteDescriptionOnFailure', err);
+ failureCallback(err);
+ }
+ );
+ /*
+ if (this.statsinterval === null && this.maxstats > 0) {
+ // start gathering stats
+ }
+ */
+};
+
+TraceablePeerConnection.prototype.close = function () {
+ this.trace('stop');
+ if (this.statsinterval !== null) {
+ window.clearInterval(this.statsinterval);
+ this.statsinterval = null;
+ }
+ this.peerconnection.close();
+};
+
+TraceablePeerConnection.prototype.createOffer = function (successCallback, failureCallback, constraints) {
+ var self = this;
+ this.trace('createOffer', JSON.stringify(constraints, null, ' '));
+ this.peerconnection.createOffer(
+ function (offer) {
+ self.trace('createOfferOnSuccess', dumpSDP(offer));
+ successCallback(offer);
+ },
+ function(err) {
+ self.trace('createOfferOnFailure', err);
+ failureCallback(err);
+ },
+ constraints
+ );
+};
+
+TraceablePeerConnection.prototype.createAnswer = function (successCallback, failureCallback, constraints) {
+ var self = this;
+ this.trace('createAnswer', JSON.stringify(constraints, null, ' '));
+ this.peerconnection.createAnswer(
+ function (answer) {
+ answer = simulcast.transformAnswer(answer);
+ self.trace('createAnswerOnSuccess', dumpSDP(answer));
+ successCallback(answer);
+ },
+ function(err) {
+ self.trace('createAnswerOnFailure', err);
+ failureCallback(err);
+ },
+ constraints
+ );
+};
+
+TraceablePeerConnection.prototype.addIceCandidate = function (candidate, successCallback, failureCallback) {
+ var self = this;
+ this.trace('addIceCandidate', JSON.stringify(candidate, null, ' '));
+ this.peerconnection.addIceCandidate(candidate);
+ /* maybe later
+ this.peerconnection.addIceCandidate(candidate,
+ function () {
+ self.trace('addIceCandidateOnSuccess');
+ successCallback();
+ },
+ function (err) {
+ self.trace('addIceCandidateOnFailure', err);
+ failureCallback(err);
+ }
+ );
+ */
+};
+
+TraceablePeerConnection.prototype.getStats = function(callback, errback) {
+ if (navigator.mozGetUserMedia) {
+ // ignore for now...
+ if(!errback)
+ errback = function () {
+
+ }
+ this.peerconnection.getStats(null,callback,errback);
+ } else {
+ this.peerconnection.getStats(callback);
+ }
+};
+
+module.exports = TraceablePeerConnection;
+
+
+},{}],6:[function(require,module,exports){
+/* global $, $iq, config, connection, UI, messageHandler,
+ roomName, sessionTerminated, Strophe, Util */
+/**
+ * Contains logic responsible for enabling/disabling functionality available
+ * only to moderator users.
+ */
+var connection = null;
+var focusUserJid;
+var getNextTimeout = Util.createExpBackoffTimer(1000);
+var getNextErrorTimeout = Util.createExpBackoffTimer(1000);
+// External authentication stuff
+var externalAuthEnabled = false;
+// Sip gateway can be enabled by configuring Jigasi host in config.js or
+// it will be enabled automatically if focus detects the component through
+// service discovery.
+var sipGatewayEnabled = config.hosts.call_control !== undefined;
+
+var Moderator = {
+ isModerator: function () {
+ return connection && connection.emuc.isModerator();
+ },
+
+ isPeerModerator: function (peerJid) {
+ return connection &&
+ connection.emuc.getMemberRole(peerJid) === 'moderator';
+ },
+
+ isExternalAuthEnabled: function () {
+ return externalAuthEnabled;
+ },
+
+ isSipGatewayEnabled: function () {
+ return sipGatewayEnabled;
+ },
+
+ setConnection: function (con) {
+ connection = con;
+ },
+
+ init: function (xmpp) {
+ this.xmppService = xmpp;
+ this.onLocalRoleChange = function (from, member, pres) {
+ UI.onModeratorStatusChanged(Moderator.isModerator());
+ };
+ },
+
+ onMucLeft: function (jid) {
+ console.info("Someone left is it focus ? " + jid);
+ var resource = Strophe.getResourceFromJid(jid);
+ if (resource === 'focus' && !this.xmppService.sessionTerminated) {
+ console.info(
+ "Focus has left the room - leaving conference");
+ //hangUp();
+ // We'd rather reload to have everything re-initialized
+ // FIXME: show some message before reload
+ location.reload();
+ }
+ },
+
+ setFocusUserJid: function (focusJid) {
+ if (!focusUserJid) {
+ focusUserJid = focusJid;
+ console.info("Focus jid set to: " + focusUserJid);
+ }
+ },
+
+ getFocusUserJid: function () {
+ return focusUserJid;
+ },
+
+ getFocusComponent: function () {
+ // Get focus component address
+ var focusComponent = config.hosts.focus;
+ // If not specified use default: 'focus.domain'
+ if (!focusComponent) {
+ focusComponent = 'focus.' + config.hosts.domain;
+ }
+ return focusComponent;
+ },
+
+ createConferenceIq: function (roomName) {
+ // Generate create conference IQ
+ var elem = $iq({to: Moderator.getFocusComponent(), type: 'set'});
+ elem.c('conference', {
+ xmlns: 'http://jitsi.org/protocol/focus',
+ room: roomName
+ });
+ if (config.hosts.bridge !== undefined) {
+ elem.c(
+ 'property',
+ { name: 'bridge', value: config.hosts.bridge})
+ .up();
+ }
+ // Tell the focus we have Jigasi configured
+ if (config.hosts.call_control !== undefined) {
+ elem.c(
+ 'property',
+ { name: 'call_control', value: config.hosts.call_control})
+ .up();
+ }
+ if (config.channelLastN !== undefined) {
+ elem.c(
+ 'property',
+ { name: 'channelLastN', value: config.channelLastN})
+ .up();
+ }
+ if (config.adaptiveLastN !== undefined) {
+ elem.c(
+ 'property',
+ { name: 'adaptiveLastN', value: config.adaptiveLastN})
+ .up();
+ }
+ if (config.adaptiveSimulcast !== undefined) {
+ elem.c(
+ 'property',
+ { name: 'adaptiveSimulcast', value: config.adaptiveSimulcast})
+ .up();
+ }
+ if (config.openSctp !== undefined) {
+ elem.c(
+ 'property',
+ { name: 'openSctp', value: config.openSctp})
+ .up();
+ }
+ if (config.enableFirefoxSupport !== undefined) {
+ elem.c(
+ 'property',
+ { name: 'enableFirefoxHacks',
+ value: config.enableFirefoxSupport})
+ .up();
+ }
+ elem.up();
+ return elem;
+ },
+
+ parseConfigOptions: function (resultIq) {
+
+ Moderator.setFocusUserJid(
+ $(resultIq).find('conference').attr('focusjid'));
+
+ var extAuthParam
+ = $(resultIq).find('>conference>property[name=\'externalAuth\']');
+ if (extAuthParam.length) {
+ externalAuthEnabled = extAuthParam.attr('value') === 'true';
+ }
+
+ console.info("External authentication enabled: " + externalAuthEnabled);
+
+ // Check if focus has auto-detected Jigasi component(this will be also
+ // included if we have passed our host from the config)
+ if ($(resultIq).find(
+ '>conference>property[name=\'sipGatewayEnabled\']').length) {
+ sipGatewayEnabled = true;
+ }
+
+ console.info("Sip gateway enabled: " + sipGatewayEnabled);
+ },
+
+ // FIXME: we need to show the fact that we're waiting for the focus
+ // to the user(or that focus is not available)
+ allocateConferenceFocus: function (roomName, callback) {
+ // Try to use focus user JID from the config
+ Moderator.setFocusUserJid(config.focusUserJid);
+ // Send create conference IQ
+ var iq = Moderator.createConferenceIq(roomName);
+ connection.sendIQ(
+ iq,
+ function (result) {
+ if ('true' === $(result).find('conference').attr('ready')) {
+ // Reset both timers
+ getNextTimeout(true);
+ getNextErrorTimeout(true);
+ // Setup config options
+ Moderator.parseConfigOptions(result);
+ // Exec callback
+ callback();
+ } else {
+ var waitMs = getNextTimeout();
+ console.info("Waiting for the focus... " + waitMs);
+ // Reset error timeout
+ getNextErrorTimeout(true);
+ window.setTimeout(
+ function () {
+ Moderator.allocateConferenceFocus(
+ roomName, callback);
+ }, waitMs);
+ }
+ },
+ function (error) {
+ // Not authorized to create new room
+ if ($(error).find('>error>not-authorized').length) {
+ console.warn("Unauthorized to start the conference");
+ UI.onAuthenticationRequired(function () {
+ Moderator.allocateConferenceFocus(roomName, callback);
+ });
+ return;
+ }
+ var waitMs = getNextErrorTimeout();
+ console.error("Focus error, retry after " + waitMs, error);
+ // Show message
+ UI.messageHandler.notify(
+ 'Conference focus', 'disconnected',
+ Moderator.getFocusComponent() +
+ ' not available - retry in ' +
+ (waitMs / 1000) + ' sec');
+ // Reset response timeout
+ getNextTimeout(true);
+ window.setTimeout(
+ function () {
+ Moderator.allocateConferenceFocus(roomName, callback);
+ }, waitMs);
+ }
+ );
+ },
+
+ getAuthUrl: function (roomName, urlCallback) {
+ var iq = $iq({to: Moderator.getFocusComponent(), type: 'get'});
+ iq.c('auth-url', {
+ xmlns: 'http://jitsi.org/protocol/focus',
+ room: roomName
+ });
+ connection.sendIQ(
+ iq,
+ function (result) {
+ var url = $(result).find('auth-url').attr('url');
+ if (url) {
+ console.info("Got auth url: " + url);
+ urlCallback(url);
+ } else {
+ console.error(
+ "Failed to get auth url fro mthe focus", result);
+ }
+ },
+ function (error) {
+ console.error("Get auth url error", error);
+ }
+ );
+ }
+};
+
+module.exports = Moderator;
+
+
+
+
+},{}],7:[function(require,module,exports){
+/* global $, $iq, config, connection, focusMucJid, messageHandler, Moderator,
+ Toolbar, Util */
+var Moderator = require("./moderator");
+
+
+var recordingToken = null;
+var recordingEnabled;
+
+/**
+ * Whether to use a jirecon component for recording, or use the videobridge
+ * through COLIBRI.
+ */
+var useJirecon = (typeof config.hosts.jirecon != "undefined");
+
+/**
+ * The ID of the jirecon recording session. Jirecon generates it when we
+ * initially start recording, and it needs to be used in subsequent requests
+ * to jirecon.
+ */
+var jireconRid = null;
+
+function setRecordingToken(token) {
+ recordingToken = token;
+}
+
+function setRecording(state, token, callback) {
+ if (useJirecon){
+ this.setRecordingJirecon(state, token, callback);
+ } else {
+ this.setRecordingColibri(state, token, callback);
+ }
+}
+
+function setRecordingJirecon(state, token, callback) {
+ if (state == recordingEnabled){
+ return;
+ }
+
+ var iq = $iq({to: config.hosts.jirecon, type: 'set'})
+ .c('recording', {xmlns: 'http://jitsi.org/protocol/jirecon',
+ action: state ? 'start' : 'stop',
+ mucjid: connection.emuc.roomjid});
+ if (!state){
+ iq.attrs({rid: jireconRid});
+ }
+
+ console.log('Start recording');
+
+ connection.sendIQ(
+ iq,
+ function (result) {
+ // TODO wait for an IQ with the real status, since this is
+ // provisional?
+ jireconRid = $(result).find('recording').attr('rid');
+ console.log('Recording ' + (state ? 'started' : 'stopped') +
+ '(jirecon)' + result);
+ recordingEnabled = state;
+ if (!state){
+ jireconRid = null;
+ }
+
+ callback(state);
+ },
+ function (error) {
+ console.log('Failed to start recording, error: ', error);
+ callback(recordingEnabled);
+ });
+}
+
+// Sends a COLIBRI message which enables or disables (according to 'state')
+// the recording on the bridge. Waits for the result IQ and calls 'callback'
+// with the new recording state, according to the IQ.
+function setRecordingColibri(state, token, callback) {
+ var elem = $iq({to: focusMucJid, type: 'set'});
+ elem.c('conference', {
+ xmlns: 'http://jitsi.org/protocol/colibri'
+ });
+ elem.c('recording', {state: state, token: token});
+
+ connection.sendIQ(elem,
+ function (result) {
+ console.log('Set recording "', state, '". Result:', result);
+ var recordingElem = $(result).find('>conference>recording');
+ var newState = ('true' === recordingElem.attr('state'));
+
+ recordingEnabled = newState;
+ callback(newState);
+ },
+ function (error) {
+ console.warn(error);
+ callback(recordingEnabled);
+ }
+ );
+}
+
+var Recording = {
+ toggleRecording: function (tokenEmptyCallback,
+ startingCallback, startedCallback) {
+ if (!Moderator.isModerator()) {
+ console.log(
+ 'non-focus, or conference not yet organized:' +
+ ' not enabling recording');
+ return;
+ }
+
+ // Jirecon does not (currently) support a token.
+ if (!recordingToken && !useJirecon) {
+ tokenEmptyCallback(function (value) {
+ setRecordingToken(value);
+ this.toggleRecording();
+ });
+
+ return;
+ }
+
+ var oldState = recordingEnabled;
+ startingCallback(!oldState);
+ setRecording(!oldState,
+ recordingToken,
+ function (state) {
+ console.log("New recording state: ", state);
+ if (state === oldState) {
+ // FIXME: new focus:
+ // this will not work when moderator changes
+ // during active session. Then it will assume that
+ // recording status has changed to true, but it might have
+ // been already true(and we only received actual status from
+ // the focus).
+ //
+ // SO we start with status null, so that it is initialized
+ // here and will fail only after second click, so if invalid
+ // token was used we have to press the button twice before
+ // current status will be fetched and token will be reset.
+ //
+ // Reliable way would be to return authentication error.
+ // Or status update when moderator connects.
+ // Or we have to stop recording session when current
+ // moderator leaves the room.
+
+ // Failed to change, reset the token because it might
+ // have been wrong
+ setRecordingToken(null);
+ }
+ startedCallback(state);
+
+ }
+ );
+ }
+
+}
+
+module.exports = Recording;
+},{"./moderator":6}],8:[function(require,module,exports){
+/* jshint -W117 */
+/* a simple MUC connection plugin
+ * can only handle a single MUC room
+ */
+
+var bridgeIsDown = false;
+
+var Moderator = require("./moderator");
+
+module.exports = function(XMPP, eventEmitter) {
+ Strophe.addConnectionPlugin('emuc', {
+ connection: null,
+ roomjid: null,
+ myroomjid: null,
+ members: {},
+ list_members: [], // so we can elect a new focus
+ presMap: {},
+ preziMap: {},
+ joined: false,
+ isOwner: false,
+ role: null,
+ init: function (conn) {
+ this.connection = conn;
+ },
+ initPresenceMap: function (myroomjid) {
+ this.presMap['to'] = myroomjid;
+ this.presMap['xns'] = 'http://jabber.org/protocol/muc';
+ },
+ doJoin: function (jid, password) {
+ this.myroomjid = jid;
+
+ console.info("Joined MUC as " + this.myroomjid);
+
+ this.initPresenceMap(this.myroomjid);
+
+ if (!this.roomjid) {
+ this.roomjid = Strophe.getBareJidFromJid(jid);
+ // add handlers (just once)
+ this.connection.addHandler(this.onPresence.bind(this), null, 'presence', null, null, this.roomjid, {matchBare: true});
+ this.connection.addHandler(this.onPresenceUnavailable.bind(this), null, 'presence', 'unavailable', null, this.roomjid, {matchBare: true});
+ this.connection.addHandler(this.onPresenceError.bind(this), null, 'presence', 'error', null, this.roomjid, {matchBare: true});
+ this.connection.addHandler(this.onMessage.bind(this), null, 'message', null, null, this.roomjid, {matchBare: true});
+ }
+ if (password !== undefined) {
+ this.presMap['password'] = password;
+ }
+ this.sendPresence();
+ },
+ doLeave: function () {
+ console.log("do leave", this.myroomjid);
+ var pres = $pres({to: this.myroomjid, type: 'unavailable' });
+ this.presMap.length = 0;
+ this.connection.send(pres);
+ },
+ createNonAnonymousRoom: function () {
+ // http://xmpp.org/extensions/xep-0045.html#createroom-reserved
+
+ var getForm = $iq({type: 'get', to: this.roomjid})
+ .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'})
+ .c('x', {xmlns: 'jabber:x:data', type: 'submit'});
+
+ this.connection.sendIQ(getForm, function (form) {
+
+ if (!$(form).find(
+ '>query>x[xmlns="jabber:x:data"]' +
+ '>field[var="muc#roomconfig_whois"]').length) {
+
+ console.error('non-anonymous rooms not supported');
+ return;
+ }
+
+ var formSubmit = $iq({to: this.roomjid, type: 'set'})
+ .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});
+
+ formSubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});
+
+ formSubmit.c('field', {'var': 'FORM_TYPE'})
+ .c('value')
+ .t('http://jabber.org/protocol/muc#roomconfig').up().up();
+
+ formSubmit.c('field', {'var': 'muc#roomconfig_whois'})
+ .c('value').t('anyone').up().up();
+
+ this.connection.sendIQ(formSubmit);
+
+ }, function (error) {
+ console.error("Error getting room configuration form");
+ });
+ },
+ onPresence: function (pres) {
+ var from = pres.getAttribute('from');
+
+ // What is this for? A workaround for something?
+ if (pres.getAttribute('type')) {
+ return true;
+ }
+
+ // Parse etherpad tag.
+ var etherpad = $(pres).find('>etherpad');
+ if (etherpad.length) {
+ if (config.etherpad_base && !Moderator.isModerator()) {
+ UI.initEtherpad(etherpad.text());
+ }
+ }
+
+ // Parse prezi tag.
+ var presentation = $(pres).find('>prezi');
+ if (presentation.length) {
+ var url = presentation.attr('url');
+ var current = presentation.find('>current').text();
+
+ console.log('presentation info received from', from, url);
+
+ if (this.preziMap[from] == null) {
+ this.preziMap[from] = url;
+
+ $(document).trigger('presentationadded.muc', [from, url, current]);
+ }
+ else {
+ $(document).trigger('gotoslide.muc', [from, url, current]);
+ }
+ }
+ else if (this.preziMap[from] != null) {
+ var url = this.preziMap[from];
+ delete this.preziMap[from];
+ $(document).trigger('presentationremoved.muc', [from, url]);
+ }
+
+ // Parse audio info tag.
+ var audioMuted = $(pres).find('>audiomuted');
+ if (audioMuted.length) {
+ $(document).trigger('audiomuted.muc', [from, audioMuted.text()]);
+ }
+
+ // Parse video info tag.
+ var videoMuted = $(pres).find('>videomuted');
+ if (videoMuted.length) {
+ $(document).trigger('videomuted.muc', [from, videoMuted.text()]);
+ }
+
+ var stats = $(pres).find('>stats');
+ if (stats.length) {
+ var statsObj = {};
+ Strophe.forEachChild(stats[0], "stat", function (el) {
+ statsObj[el.getAttribute("name")] = el.getAttribute("value");
+ });
+ connectionquality.updateRemoteStats(from, statsObj);
+ }
+
+ // Parse status.
+ if ($(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="201"]').length) {
+ this.isOwner = true;
+ this.createNonAnonymousRoom();
+ }
+
+ // Parse roles.
+ var member = {};
+ member.show = $(pres).find('>show').text();
+ member.status = $(pres).find('>status').text();
+ var tmp = $(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>item');
+ member.affiliation = tmp.attr('affiliation');
+ member.role = tmp.attr('role');
+
+ // Focus recognition
+ member.jid = tmp.attr('jid');
+ member.isFocus = false;
+ if (member.jid
+ && member.jid.indexOf(Moderator.getFocusUserJid() + "/") == 0) {
+ member.isFocus = true;
+ }
+
+ var nicktag = $(pres).find('>nick[xmlns="http://jabber.org/protocol/nick"]');
+ member.displayName = (nicktag.length > 0 ? nicktag.html() : null);
+
+ if (from == this.myroomjid) {
+ if (member.affiliation == 'owner') this.isOwner = true;
+ if (this.role !== member.role) {
+ this.role = member.role;
+ if (Moderator.onLocalRoleChange)
+ Moderator.onLocalRoleChange(from, member, pres);
+ UI.onLocalRoleChange(from, member, pres);
+ }
+ if (!this.joined) {
+ this.joined = true;
+ eventEmitter.emit(XMPPEvents.MUC_JOINED, from, member);
+ this.list_members.push(from);
+ }
+ } else if (this.members[from] === undefined) {
+ // new participant
+ this.members[from] = member;
+ this.list_members.push(from);
+ console.log('entered', from, member);
+ if (member.isFocus) {
+ focusMucJid = from;
+ console.info("Ignore focus: " + from + ", real JID: " + member.jid);
+ }
+ else {
+ var id = $(pres).find('>userID').text();
+ var email = $(pres).find('>email');
+ if (email.length > 0) {
+ id = email.text();
+ }
+ UI.onMucEntered(from, id, member.displayName);
+ API.triggerEvent("participantJoined", {jid: from});
+ }
+ } else {
+ // Presence update for existing participant
+ // Watch role change:
+ if (this.members[from].role != member.role) {
+ this.members[from].role = member.role;
+ UI.onMucRoleChanged(member.role, member.displayName);
+ }
+ }
+
+ // Always trigger presence to update bindings
+ $(document).trigger('presence.muc', [from, member, pres]);
+ this.parsePresence(from, member, pres);
+
+ // Trigger status message update
+ if (member.status) {
+ UI.onMucPresenceStatus(from, member);
+ }
+
+ return true;
+ },
+ onPresenceUnavailable: function (pres) {
+ var from = pres.getAttribute('from');
+ // Status code 110 indicates that this notification is "self-presence".
+ if (!$(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="110"]').length) {
+ delete this.members[from];
+ this.list_members.splice(this.list_members.indexOf(from), 1);
+ this.onParticipantLeft(from);
+ }
+ // If the status code is 110 this means we're leaving and we would like
+ // to remove everyone else from our view, so we trigger the event.
+ else if (this.list_members.length > 1) {
+ for (var i = 0; i < this.list_members.length; i++) {
+ var member = this.list_members[i];
+ delete this.members[i];
+ this.list_members.splice(i, 1);
+ this.onParticipantLeft(member);
+ }
+ }
+ if ($(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="307"]').length) {
+ $(document).trigger('kicked.muc', [from]);
+ if (this.myroomjid === from) {
+ XMPP.disposeConference(false);
+ eventEmitter.emit(XMPPEvents.KICKED);
+ }
+ }
+ return true;
+ },
+ onPresenceError: function (pres) {
+ var from = pres.getAttribute('from');
+ if ($(pres).find('>error[type="auth"]>not-authorized[xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"]').length) {
+ console.log('on password required', from);
+ var self = this;
+ UI.onPasswordReqiured(function (value) {
+ self.doJoin(from, value);
+ });
+ } else if ($(pres).find(
+ '>error[type="cancel"]>not-allowed[xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"]').length) {
+ var toDomain = Strophe.getDomainFromJid(pres.getAttribute('to'));
+ if (toDomain === config.hosts.anonymousdomain) {
+ // we are connected with anonymous domain and only non anonymous users can create rooms
+ // we must authorize the user
+ XMPP.promptLogin();
+ } else {
+ console.warn('onPresError ', pres);
+ UI.messageHandler.openReportDialog(null,
+ 'Oops! Something went wrong and we couldn`t connect to the conference.',
+ pres);
+ }
+ } else {
+ console.warn('onPresError ', pres);
+ UI.messageHandler.openReportDialog(null,
+ 'Oops! Something went wrong and we couldn`t connect to the conference.',
+ pres);
+ }
+ return true;
+ },
+ sendMessage: function (body, nickname) {
+ var msg = $msg({to: this.roomjid, type: 'groupchat'});
+ msg.c('body', body).up();
+ if (nickname) {
+ msg.c('nick', {xmlns: 'http://jabber.org/protocol/nick'}).t(nickname).up().up();
+ }
+ this.connection.send(msg);
+ API.triggerEvent("outgoingMessage", {"message": body});
+ },
+ setSubject: function (subject) {
+ var msg = $msg({to: this.roomjid, type: 'groupchat'});
+ msg.c('subject', subject);
+ this.connection.send(msg);
+ console.log("topic changed to " + subject);
+ },
+ onMessage: function (msg) {
+ // FIXME: this is a hack. but jingle on muc makes nickchanges hard
+ var from = msg.getAttribute('from');
+ var nick = $(msg).find('>nick[xmlns="http://jabber.org/protocol/nick"]').text() || Strophe.getResourceFromJid(from);
+
+ var txt = $(msg).find('>body').text();
+ var type = msg.getAttribute("type");
+ if (type == "error") {
+ UI.chatAddError($(msg).find('>text').text(), txt);
+ return true;
+ }
+
+ var subject = $(msg).find('>subject');
+ if (subject.length) {
+ var subjectText = subject.text();
+ if (subjectText || subjectText == "") {
+ UI.chatSetSubject(subjectText);
+ console.log("Subject is changed to " + subjectText);
+ }
+ }
+
+
+ if (txt) {
+ console.log('chat', nick, txt);
+ UI.updateChatConversation(from, nick, txt);
+ if (from != this.myroomjid)
+ API.triggerEvent("incomingMessage",
+ {"from": from, "nick": nick, "message": txt});
+ }
+ return true;
+ },
+ lockRoom: function (key, onSuccess, onError, onNotSupported) {
+ //http://xmpp.org/extensions/xep-0045.html#roomconfig
+ var ob = this;
+ this.connection.sendIQ($iq({to: this.roomjid, type: 'get'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'}),
+ function (res) {
+ if ($(res).find('>query>x[xmlns="jabber:x:data"]>field[var="muc#roomconfig_roomsecret"]').length) {
+ var formsubmit = $iq({to: ob.roomjid, type: 'set'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});
+ formsubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});
+ formsubmit.c('field', {'var': 'FORM_TYPE'}).c('value').t('http://jabber.org/protocol/muc#roomconfig').up().up();
+ formsubmit.c('field', {'var': 'muc#roomconfig_roomsecret'}).c('value').t(key).up().up();
+ // Fixes a bug in prosody 0.9.+ https://code.google.com/p/lxmppd/issues/detail?id=373
+ formsubmit.c('field', {'var': 'muc#roomconfig_whois'}).c('value').t('anyone').up().up();
+ // FIXME: is muc#roomconfig_passwordprotectedroom required?
+ this.connection.sendIQ(formsubmit,
+ onSuccess,
+ onError);
+ } else {
+ onNotSupported();
+ }
+ }, onError);
+ },
+ kick: function (jid) {
+ var kickIQ = $iq({to: this.roomjid, type: 'set'})
+ .c('query', {xmlns: 'http://jabber.org/protocol/muc#admin'})
+ .c('item', {nick: Strophe.getResourceFromJid(jid), role: 'none'})
+ .c('reason').t('You have been kicked.').up().up().up();
+
+ this.connection.sendIQ(
+ kickIQ,
+ function (result) {
+ console.log('Kick participant with jid: ', jid, result);
+ },
+ function (error) {
+ console.log('Kick participant error: ', error);
+ });
+ },
+ sendPresence: function () {
+ var pres = $pres({to: this.presMap['to'] });
+ pres.c('x', {xmlns: this.presMap['xns']});
+
+ if (this.presMap['password']) {
+ pres.c('password').t(this.presMap['password']).up();
+ }
+
+ pres.up();
+
+ // Send XEP-0115 'c' stanza that contains our capabilities info
+ if (this.connection.caps) {
+ this.connection.caps.node = config.clientNode;
+ pres.c('c', this.connection.caps.generateCapsAttrs()).up();
+ }
+
+ pres.c('user-agent', {xmlns: 'http://jitsi.org/jitmeet/user-agent'})
+ .t(navigator.userAgent).up();
+
+ if (this.presMap['bridgeIsDown']) {
+ pres.c('bridgeIsDown').up();
+ }
+
+ if (this.presMap['email']) {
+ pres.c('email').t(this.presMap['email']).up();
+ }
+
+ if (this.presMap['userId']) {
+ pres.c('userId').t(this.presMap['userId']).up();
+ }
+
+ if (this.presMap['displayName']) {
+ // XEP-0172
+ pres.c('nick', {xmlns: 'http://jabber.org/protocol/nick'})
+ .t(this.presMap['displayName']).up();
+ }
+
+ if (this.presMap['audions']) {
+ pres.c('audiomuted', {xmlns: this.presMap['audions']})
+ .t(this.presMap['audiomuted']).up();
+ }
+
+ if (this.presMap['videons']) {
+ pres.c('videomuted', {xmlns: this.presMap['videons']})
+ .t(this.presMap['videomuted']).up();
+ }
+
+ if (this.presMap['statsns']) {
+ var stats = pres.c('stats', {xmlns: this.presMap['statsns']});
+ for (var stat in this.presMap["stats"])
+ if (this.presMap["stats"][stat] != null)
+ stats.c("stat", {name: stat, value: this.presMap["stats"][stat]}).up();
+ pres.up();
+ }
+
+ if (this.presMap['prezins']) {
+ pres.c('prezi',
+ {xmlns: this.presMap['prezins'],
+ 'url': this.presMap['preziurl']})
+ .c('current').t(this.presMap['prezicurrent']).up().up();
+ }
+
+ if (this.presMap['etherpadns']) {
+ pres.c('etherpad', {xmlns: this.presMap['etherpadns']})
+ .t(this.presMap['etherpadname']).up();
+ }
+
+ if (this.presMap['medians']) {
+ pres.c('media', {xmlns: this.presMap['medians']});
+ var sourceNumber = 0;
+ Object.keys(this.presMap).forEach(function (key) {
+ if (key.indexOf('source') >= 0) {
+ sourceNumber++;
+ }
+ });
+ if (sourceNumber > 0)
+ for (var i = 1; i <= sourceNumber / 3; i++) {
+ pres.c('source',
+ {type: this.presMap['source' + i + '_type'],
+ ssrc: this.presMap['source' + i + '_ssrc'],
+ direction: this.presMap['source' + i + '_direction']
+ || 'sendrecv' }
+ ).up();
+ }
+ }
+
+ pres.up();
+// console.debug(pres.toString());
+ this.connection.send(pres);
+ },
+ addDisplayNameToPresence: function (displayName) {
+ this.presMap['displayName'] = displayName;
+ },
+ addMediaToPresence: function (sourceNumber, mtype, ssrcs, direction) {
+ if (!this.presMap['medians'])
+ this.presMap['medians'] = 'http://estos.de/ns/mjs';
+
+ this.presMap['source' + sourceNumber + '_type'] = mtype;
+ this.presMap['source' + sourceNumber + '_ssrc'] = ssrcs;
+ this.presMap['source' + sourceNumber + '_direction'] = direction;
+ },
+ clearPresenceMedia: function () {
+ var self = this;
+ Object.keys(this.presMap).forEach(function (key) {
+ if (key.indexOf('source') != -1) {
+ delete self.presMap[key];
+ }
+ });
+ },
+ addPreziToPresence: function (url, currentSlide) {
+ this.presMap['prezins'] = 'http://jitsi.org/jitmeet/prezi';
+ this.presMap['preziurl'] = url;
+ this.presMap['prezicurrent'] = currentSlide;
+ },
+ removePreziFromPresence: function () {
+ delete this.presMap['prezins'];
+ delete this.presMap['preziurl'];
+ delete this.presMap['prezicurrent'];
+ },
+ addCurrentSlideToPresence: function (currentSlide) {
+ this.presMap['prezicurrent'] = currentSlide;
+ },
+ getPrezi: function (roomjid) {
+ return this.preziMap[roomjid];
+ },
+ addEtherpadToPresence: function (etherpadName) {
+ this.presMap['etherpadns'] = 'http://jitsi.org/jitmeet/etherpad';
+ this.presMap['etherpadname'] = etherpadName;
+ },
+ addAudioInfoToPresence: function (isMuted) {
+ this.presMap['audions'] = 'http://jitsi.org/jitmeet/audio';
+ this.presMap['audiomuted'] = isMuted.toString();
+ },
+ addVideoInfoToPresence: function (isMuted) {
+ this.presMap['videons'] = 'http://jitsi.org/jitmeet/video';
+ this.presMap['videomuted'] = isMuted.toString();
+ },
+ addConnectionInfoToPresence: function (stats) {
+ this.presMap['statsns'] = 'http://jitsi.org/jitmeet/stats';
+ this.presMap['stats'] = stats;
+ },
+ findJidFromResource: function (resourceJid) {
+ if (resourceJid &&
+ resourceJid === Strophe.getResourceFromJid(this.myroomjid)) {
+ return this.myroomjid;
+ }
+ var peerJid = null;
+ Object.keys(this.members).some(function (jid) {
+ peerJid = jid;
+ return Strophe.getResourceFromJid(jid) === resourceJid;
+ });
+ return peerJid;
+ },
+ addBridgeIsDownToPresence: function () {
+ this.presMap['bridgeIsDown'] = true;
+ },
+ addEmailToPresence: function (email) {
+ this.presMap['email'] = email;
+ },
+ addUserIdToPresence: function (userId) {
+ this.presMap['userId'] = userId;
+ },
+ isModerator: function () {
+ return this.role === 'moderator';
+ },
+ getMemberRole: function (peerJid) {
+ if (this.members[peerJid]) {
+ return this.members[peerJid].role;
+ }
+ return null;
+ },
+ onParticipantLeft: function (jid) {
+ UI.onMucLeft(jid);
+
+ API.triggerEvent("participantLeft", {jid: jid});
+
+ delete jid2Ssrc[jid];
+
+ this.connection.jingle.terminateByJid(jid);
+
+ if (this.getPrezi(jid)) {
+ $(document).trigger('presentationremoved.muc',
+ [jid, this.getPrezi(jid)]);
+ }
+
+ Moderator.onMucLeft(jid);
+ },
+ parsePresence: function (from, memeber, pres) {
+ if($(pres).find(">bridgeIsDown").length > 0 && !bridgeIsDown) {
+ bridgeIsDown = true;
+ eventEmitter.emit(XMPPEvents.BRIDGE_DOWN);
+ }
+
+ if(memeber.isFocus)
+ return;
+
+ // Remove old ssrcs coming from the jid
+ Object.keys(ssrc2jid).forEach(function (ssrc) {
+ if (ssrc2jid[ssrc] == jid) {
+ delete ssrc2jid[ssrc];
+ delete ssrc2videoType[ssrc];
+ }
+ });
+
+ var changedStreams = [];
+ $(pres).find('>media[xmlns="http://estos.de/ns/mjs"]>source').each(function (idx, ssrc) {
+ //console.log(jid, 'assoc ssrc', ssrc.getAttribute('type'), ssrc.getAttribute('ssrc'));
+ var ssrcV = ssrc.getAttribute('ssrc');
+ ssrc2jid[ssrcV] = from;
+ notReceivedSSRCs.push(ssrcV);
+
+ var type = ssrc.getAttribute('type');
+ ssrc2videoType[ssrcV] = type;
+
+ var direction = ssrc.getAttribute('direction');
+
+ changedStreams.push({type: type, direction: direction});
+
+ });
+
+ eventEmitter.emit(XMPPEvents.CHANGED_STREAMS, from, changedStreams);
+
+ var displayName = !config.displayJids
+ ? memeber.displayName : Strophe.getResourceFromJid(from);
+
+ if (displayName && displayName.length > 0)
+ {
+// $(document).trigger('displaynamechanged',
+// [jid, displayName]);
+ eventEmitter.emit(XMPPEvents.DISPLAY_NAME_CHANGED, from, displayName);
+ }
+
+
+ var id = $(pres).find('>userID').text();
+ var email = $(pres).find('>email');
+ if(email.length > 0) {
+ id = email.text();
+ }
+
+ eventEmitter.emit(XMPPEvents.USER_ID_CHANGED, from, id);
+ }
+ });
+};
+
+
+},{"./moderator":6}],9:[function(require,module,exports){
+/* jshint -W117 */
+
+var JingleSession = require("./JingleSession");
+
+function CallIncomingJingle(sid, connection) {
+ var sess = connection.jingle.sessions[sid];
+
+ // TODO: do we check activecall == null?
+ activecall = sess;
+
+ statistics.onConferenceCreated(sess);
+ RTC.onConferenceCreated(sess);
+
+ // TODO: check affiliation and/or role
+ console.log('emuc data for', sess.peerjid, connection.emuc.members[sess.peerjid]);
+ sess.usedrip = true; // not-so-naive trickle ice
+ sess.sendAnswer();
+ sess.accept();
+
+};
+
+module.exports = function(XMPP)
+{
+ Strophe.addConnectionPlugin('jingle', {
+ connection: null,
+ sessions: {},
+ jid2session: {},
+ ice_config: {iceServers: []},
+ pc_constraints: {},
+ media_constraints: {
+ mandatory: {
+ 'OfferToReceiveAudio': true,
+ 'OfferToReceiveVideo': true
+ }
+ // MozDontOfferDataChannel: true when this is firefox
+ },
+ init: function (conn) {
+ this.connection = conn;
+ if (this.connection.disco) {
+ // http://xmpp.org/extensions/xep-0167.html#support
+ // http://xmpp.org/extensions/xep-0176.html#support
+ this.connection.disco.addFeature('urn:xmpp:jingle:1');
+ this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:1');
+ this.connection.disco.addFeature('urn:xmpp:jingle:transports:ice-udp:1');
+ this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:audio');
+ this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:video');
+
+
+ // this is dealt with by SDP O/A so we don't need to annouce this
+ //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtcp-fb:0'); // XEP-0293
+ //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtp-hdrext:0'); // XEP-0294
+ if (config.useRtcpMux) {
+ this.connection.disco.addFeature('urn:ietf:rfc:5761'); // rtcp-mux
+ }
+ if (config.useBundle) {
+ this.connection.disco.addFeature('urn:ietf:rfc:5888'); // a=group, e.g. bundle
+ }
+ //this.connection.disco.addFeature('urn:ietf:rfc:5576'); // a=ssrc
+ }
+ this.connection.addHandler(this.onJingle.bind(this), 'urn:xmpp:jingle:1', 'iq', 'set', null, null);
+ },
+ 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: fromJid,
+ id: iq.getAttribute('id')
+ });
+ console.log('on jingle ' + action + ' from ' + fromJid, iq);
+ var sess = this.sessions[sid];
+ if ('session-initiate' != action) {
+ if (sess === null) {
+ ack.type = 'error';
+ ack.c('error', {type: 'cancel'})
+ .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()
+ .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
+ this.connection.send(ack);
+ return true;
+ }
+ // compare from to sess.peerjid (bare jid comparison for later compat with message-mode)
+ // local jid is not checked
+ 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()
+ .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
+ this.connection.send(ack);
+ return true;
+ }
+ } else if (sess !== undefined) {
+ // existing session with same session id
+ // this might be out-of-order if the sess.peerjid is the same as from
+ ack.type = 'error';
+ ack.c('error', {type: 'cancel'})
+ .c('service-unavailable', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up();
+ console.warn('duplicate session id', sid);
+ this.connection.send(ack);
+ return true;
+ }
+ // FIXME: check for a defined action
+ this.connection.send(ack);
+ // see http://xmpp.org/extensions/xep-0166.html#concepts-session
+ switch (action) {
+ case 'session-initiate':
+ sess = new JingleSession(
+ $(iq).attr('to'), $(iq).find('jingle').attr('sid'),
+ this.connection, XMPP);
+ // configure session
+
+ sess.media_constraints = this.media_constraints;
+ sess.pc_constraints = this.pc_constraints;
+ sess.ice_config = this.ice_config;
+
+ sess.initiate(fromJid, false);
+ // FIXME: setRemoteDescription should only be done when this call is to be accepted
+ sess.setRemoteDescription($(iq).find('>jingle'), 'offer');
+
+ this.sessions[sess.sid] = sess;
+ this.jid2session[sess.peerjid] = sess;
+
+ // the callback should either
+ // .sendAnswer and .accept
+ // or .sendTerminate -- not necessarily synchronus
+ CallIncomingJingle(sess.sid, this.connection);
+ break;
+ case 'session-accept':
+ sess.setRemoteDescription($(iq).find('>jingle'), 'answer');
+ sess.accept();
+ $(document).trigger('callaccepted.jingle', [sess.sid]);
+ break;
+ case 'session-terminate':
+ // If this is not the focus sending the terminate, we have
+ // nothing more to do here.
+ if (Object.keys(this.sessions).length < 1
+ || !(this.sessions[Object.keys(this.sessions)[0]]
+ instanceof JingleSession))
+ {
+ break;
+ }
+ console.log('terminating...', sess.sid);
+ sess.terminate();
+ this.terminate(sess.sid);
+ if ($(iq).find('>jingle>reason').length) {
+ $(document).trigger('callterminated.jingle', [
+ sess.sid,
+ sess.peerjid,
+ $(iq).find('>jingle>reason>:first')[0].tagName,
+ $(iq).find('>jingle>reason>text').text()
+ ]);
+ } else {
+ $(document).trigger('callterminated.jingle',
+ [sess.sid, sess.peerjid]);
+ }
+ break;
+ case 'transport-info':
+ sess.addIceCandidate($(iq).find('>jingle>content'));
+ break;
+ case 'session-info':
+ var affected;
+ if ($(iq).find('>jingle>ringing[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
+ $(document).trigger('ringing.jingle', [sess.sid]);
+ } else if ($(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
+ affected = $(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
+ $(document).trigger('mute.jingle', [sess.sid, affected]);
+ } else if ($(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
+ affected = $(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
+ $(document).trigger('unmute.jingle', [sess.sid, affected]);
+ }
+ break;
+ case 'addsource': // FIXME: proprietary, un-jingleish
+ case 'source-add': // FIXME: proprietary
+ sess.addSource($(iq).find('>jingle>content'), fromJid);
+ break;
+ case 'removesource': // FIXME: proprietary, un-jingleish
+ case 'source-remove': // FIXME: proprietary
+ sess.removeSource($(iq).find('>jingle>content'), fromJid);
+ break;
+ default:
+ console.warn('jingle action not implemented', action);
+ break;
+ }
+ return true;
+ },
+ initiate: function (peerjid, myjid) { // initiate a new jinglesession to peerjid
+ var sess = new JingleSession(myjid || this.connection.jid,
+ Math.random().toString(36).substr(2, 12), // random string
+ this.connection, XMPP);
+ // configure session
+
+ sess.media_constraints = this.media_constraints;
+ sess.pc_constraints = this.pc_constraints;
+ sess.ice_config = this.ice_config;
+
+ sess.initiate(peerjid, true);
+ this.sessions[sess.sid] = sess;
+ this.jid2session[sess.peerjid] = sess;
+ sess.sendOffer();
+ return sess;
+ },
+ terminate: function (sid, reason, text) { // terminate by sessionid (or all sessions)
+ if (sid === null || sid === undefined) {
+ for (sid in this.sessions) {
+ if (this.sessions[sid].state != 'ended') {
+ this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
+ this.sessions[sid].terminate();
+ }
+ delete this.jid2session[this.sessions[sid].peerjid];
+ delete this.sessions[sid];
+ }
+ } else if (this.sessions.hasOwnProperty(sid)) {
+ if (this.sessions[sid].state != 'ended') {
+ this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
+ this.sessions[sid].terminate();
+ }
+ delete this.jid2session[this.sessions[sid].peerjid];
+ delete this.sessions[sid];
+ }
+ },
+ // Used to terminate a session when an unavailable presence is received.
+ terminateByJid: function (jid) {
+ if (this.jid2session.hasOwnProperty(jid)) {
+ var sess = this.jid2session[jid];
+ if (sess) {
+ sess.terminate();
+ console.log('peer went away silently', jid);
+ delete this.sessions[sess.sid];
+ delete this.jid2session[jid];
+ $(document).trigger('callterminated.jingle',
+ [sess.sid, jid], 'gone');
+ }
+ }
+ },
+ terminateRemoteByJid: function (jid, reason) {
+ if (this.jid2session.hasOwnProperty(jid)) {
+ var sess = this.jid2session[jid];
+ if (sess) {
+ sess.sendTerminate(reason || (!sess.active()) ? 'kick' : null);
+ sess.terminate();
+ console.log('terminate peer with jid', sess.sid, jid);
+ delete this.sessions[sess.sid];
+ delete this.jid2session[jid];
+ $(document).trigger('callterminated.jingle',
+ [sess.sid, jid, 'kicked']);
+ }
+ }
+ },
+ getStunAndTurnCredentials: function () {
+ // get stun and turn configuration from server via xep-0215
+ // uses time-limited credentials as described in
+ // http://tools.ietf.org/html/draft-uberti-behave-turn-rest-00
+ //
+ // see https://code.google.com/p/prosody-modules/source/browse/mod_turncredentials/mod_turncredentials.lua
+ // for a prosody module which implements this
+ //
+ // currently, this doesn't work with updateIce and therefore credentials with a long
+ // validity have to be fetched before creating the peerconnection
+ // TODO: implement refresh via updateIce as described in
+ // https://code.google.com/p/webrtc/issues/detail?id=1650
+ var self = this;
+ this.connection.sendIQ(
+ $iq({type: 'get', to: this.connection.domain})
+ .c('services', {xmlns: 'urn:xmpp:extdisco:1'}).c('service', {host: 'turn.' + this.connection.domain}),
+ function (res) {
+ var iceservers = [];
+ $(res).find('>services>service').each(function (idx, el) {
+ el = $(el);
+ var dict = {};
+ var type = el.attr('type');
+ switch (type) {
+ case 'stun':
+ dict.url = 'stun:' + el.attr('host');
+ if (el.attr('port')) {
+ dict.url += ':' + el.attr('port');
+ }
+ iceservers.push(dict);
+ break;
+ case 'turn':
+ case 'turns':
+ dict.url = type + ':';
+ if (el.attr('username')) { // https://code.google.com/p/webrtc/issues/detail?id=1508
+ if (navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./) && parseInt(navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./)[2], 10) < 28) {
+ dict.url += el.attr('username') + '@';
+ } else {
+ dict.username = el.attr('username'); // only works in M28
+ }
+ }
+ dict.url += el.attr('host');
+ if (el.attr('port') && el.attr('port') != '3478') {
+ dict.url += ':' + el.attr('port');
+ }
+ if (el.attr('transport') && el.attr('transport') != 'udp') {
+ dict.url += '?transport=' + el.attr('transport');
+ }
+ if (el.attr('password')) {
+ dict.credential = el.attr('password');
+ }
+ iceservers.push(dict);
+ break;
+ }
+ });
+ self.ice_config.iceServers = iceservers;
+ },
+ function (err) {
+ console.warn('getting turn credentials failed', err);
+ console.warn('is mod_turncredentials or similar installed?');
+ }
+ );
+ // implement push?
+ },
+
+ /**
+ * Populates the log data
+ */
+ populateData: function () {
+ var data = {};
+ Object.keys(this.sessions).forEach(function (sid) {
+ var session = this.sessions[sid];
+ if (session.peerconnection && session.peerconnection.updateLog) {
+ // FIXME: should probably be a .dump call
+ data["jingle_" + session.sid] = {
+ updateLog: session.peerconnection.updateLog,
+ stats: session.peerconnection.stats,
+ url: window.location.href
+ };
+ }
+ });
+ return data;
+ }
+ });
+};
+
+
+},{"./JingleSession":1}],10:[function(require,module,exports){
+/* global Strophe */
+module.exports = function () {
+
+ Strophe.addConnectionPlugin('logger', {
+ // logs raw stanzas and makes them available for download as JSON
+ connection: null,
+ log: [],
+ init: function (conn) {
+ this.connection = conn;
+ this.connection.rawInput = this.log_incoming.bind(this);
+ this.connection.rawOutput = this.log_outgoing.bind(this);
+ },
+ log_incoming: function (stanza) {
+ this.log.push([new Date().getTime(), 'incoming', stanza]);
+ },
+ log_outgoing: function (stanza) {
+ this.log.push([new Date().getTime(), 'outgoing', stanza]);
+ }
+ });
+};
+},{}],11:[function(require,module,exports){
+/* global $, $iq, config, connection, focusMucJid, forceMuted,
+ setAudioMuted, Strophe */
+/**
+ * Moderate connection plugin.
+ */
+module.exports = function (XMPP) {
+ Strophe.addConnectionPlugin('moderate', {
+ connection: null,
+ init: function (conn) {
+ this.connection = conn;
+
+ this.connection.addHandler(this.onMute.bind(this),
+ 'http://jitsi.org/jitmeet/audio',
+ 'iq',
+ 'set',
+ null,
+ null);
+ },
+ setMute: function (jid, mute) {
+ console.info("set mute", mute);
+ var iqToFocus = $iq({to: focusMucJid, type: 'set'})
+ .c('mute', {
+ xmlns: 'http://jitsi.org/jitmeet/audio',
+ jid: jid
+ })
+ .t(mute.toString())
+ .up();
+
+ this.connection.sendIQ(
+ iqToFocus,
+ function (result) {
+ console.log('set mute', result);
+ },
+ function (error) {
+ console.log('set mute error', error);
+ });
+ },
+ onMute: function (iq) {
+ var from = iq.getAttribute('from');
+ if (from !== focusMucJid) {
+ console.warn("Ignored mute from non focus peer");
+ return false;
+ }
+ var mute = $(iq).find('mute');
+ if (mute.length) {
+ var doMuteAudio = mute.text() === "true";
+ UI.setAudioMuted(doMuteAudio);
+ XMPP.forceMuted = doMuteAudio;
+ }
+ return true;
+ },
+ eject: function (jid) {
+ // We're not the focus, so can't terminate
+ //connection.jingle.terminateRemoteByJid(jid, 'kick');
+ this.connection.emuc.kick(jid);
+ }
+ });
+}
+},{}],12:[function(require,module,exports){
+/* jshint -W117 */
+module.exports = function() {
+ Strophe.addConnectionPlugin('rayo',
+ {
+ RAYO_XMLNS: 'urn:xmpp:rayo:1',
+ connection: null,
+ init: function (conn) {
+ this.connection = conn;
+ if (this.connection.disco) {
+ this.connection.disco.addFeature('urn:xmpp:rayo:client:1');
+ }
+
+ this.connection.addHandler(
+ this.onRayo.bind(this), this.RAYO_XMLNS, 'iq', 'set', null, null);
+ },
+ onRayo: function (iq) {
+ console.info("Rayo IQ", iq);
+ },
+ dial: function (to, from, roomName, roomPass) {
+ var self = this;
+ var req = $iq(
+ {
+ type: 'set',
+ to: focusMucJid
+ }
+ );
+ req.c('dial',
+ {
+ xmlns: this.RAYO_XMLNS,
+ to: to,
+ from: from
+ });
+ req.c('header',
+ {
+ name: 'JvbRoomName',
+ value: roomName
+ }).up();
+
+ if (roomPass !== null && roomPass.length) {
+
+ req.c('header',
+ {
+ name: 'JvbRoomPassword',
+ value: roomPass
+ }).up();
+ }
+
+ this.connection.sendIQ(
+ req,
+ function (result) {
+ console.info('Dial result ', result);
+
+ var resource = $(result).find('ref').attr('uri');
+ this.call_resource = resource.substr('xmpp:'.length);
+ console.info(
+ "Received call resource: " + this.call_resource);
+ },
+ function (error) {
+ console.info('Dial error ', error);
+ }
+ );
+ },
+ hang_up: function () {
+ if (!this.call_resource) {
+ console.warn("No call in progress");
+ return;
+ }
+
+ var self = this;
+ var req = $iq(
+ {
+ type: 'set',
+ to: this.call_resource
+ }
+ );
+ req.c('hangup',
+ {
+ xmlns: this.RAYO_XMLNS
+ });
+
+ this.connection.sendIQ(
+ req,
+ function (result) {
+ console.info('Hangup result ', result);
+ self.call_resource = null;
+ },
+ function (error) {
+ console.info('Hangup error ', error);
+ self.call_resource = null;
+ }
+ );
+ }
+ }
+ );
+};
+
+},{}],13:[function(require,module,exports){
+/**
+ * Strophe logger implementation. Logs from level WARN and above.
+ */
+module.exports = function () {
+
+ Strophe.log = function (level, msg) {
+ switch (level) {
+ case Strophe.LogLevel.WARN:
+ console.warn("Strophe: " + msg);
+ break;
+ case Strophe.LogLevel.ERROR:
+ case Strophe.LogLevel.FATAL:
+ console.error("Strophe: " + msg);
+ break;
+ }
+ };
+
+ Strophe.getStatusString = function (status) {
+ switch (status) {
+ case Strophe.Status.ERROR:
+ return "ERROR";
+ case Strophe.Status.CONNECTING:
+ return "CONNECTING";
+ case Strophe.Status.CONNFAIL:
+ return "CONNFAIL";
+ case Strophe.Status.AUTHENTICATING:
+ return "AUTHENTICATING";
+ case Strophe.Status.AUTHFAIL:
+ return "AUTHFAIL";
+ case Strophe.Status.CONNECTED:
+ return "CONNECTED";
+ case Strophe.Status.DISCONNECTED:
+ return "DISCONNECTED";
+ case Strophe.Status.DISCONNECTING:
+ return "DISCONNECTING";
+ case Strophe.Status.ATTACHED:
+ return "ATTACHED";
+ default:
+ return "unknown";
+ }
+ };
+};
+
+},{}],14:[function(require,module,exports){
+var Moderator = require("./moderator");
+var EventEmitter = require("events");
+var Recording = require("./recording");
+var SDP = require("./SDP");
+
+var eventEmitter = new EventEmitter();
+var connection = null;
+var authenticatedUser = false;
+var activecall = null;
+
+function connect(jid, password, uiCredentials) {
+ var bosh
+ = uiCredentials.bosh || config.bosh || '/http-bind';
+ connection = new Strophe.Connection(bosh);
+ Moderator.setConnection(connection);
+
+ var settings = UI.getSettings();
+ var email = settings.email;
+ var displayName = settings.displayName;
+ if(email) {
+ connection.emuc.addEmailToPresence(email);
+ } else {
+ connection.emuc.addUserIdToPresence(settings.uid);
+ }
+ if(displayName) {
+ connection.emuc.addDisplayNameToPresence(displayName);
+ }
+
+ if (connection.disco) {
+ // for chrome, add multistream cap
+ }
+ connection.jingle.pc_constraints = RTC.getPCConstraints();
+ if (config.useIPv6) {
+ // https://code.google.com/p/webrtc/issues/detail?id=2828
+ if (!connection.jingle.pc_constraints.optional)
+ connection.jingle.pc_constraints.optional = [];
+ connection.jingle.pc_constraints.optional.push({googIPv6: true});
+ }
+
+ if(!password)
+ password = uiCredentials.password;
+
+ var anonymousConnectionFailed = false;
+ connection.connect(jid, password, function (status, msg) {
+ console.log('Strophe status changed to',
+ Strophe.getStatusString(status));
+ if (status === Strophe.Status.CONNECTED) {
+ if (config.useStunTurn) {
+ connection.jingle.getStunAndTurnCredentials();
+ }
+ UI.disableConnect();
+
+ console.info("My Jabber ID: " + connection.jid);
+
+ if(password)
+ authenticatedUser = true;
+ maybeDoJoin();
+ } else if (status === Strophe.Status.CONNFAIL) {
+ if(msg === 'x-strophe-bad-non-anon-jid') {
+ anonymousConnectionFailed = true;
+ }
+ } else if (status === Strophe.Status.DISCONNECTED) {
+ if(anonymousConnectionFailed) {
+ // prompt user for username and password
+ XMPP.promptLogin();
+ }
+ } else if (status === Strophe.Status.AUTHFAIL) {
+ // wrong password or username, prompt user
+ XMPP.promptLogin();
+
+ }
+ });
+}
+
+
+
+function maybeDoJoin() {
+ if (connection && connection.connected &&
+ Strophe.getResourceFromJid(connection.jid)
+ && (RTC.localAudio || RTC.localVideo)) {
+ // .connected is true while connecting?
+ doJoin();
+ }
+}
+
+function doJoin() {
+ var roomName = UI.generateRoomName();
+
+ Moderator.allocateConferenceFocus(
+ roomName, UI.checkForNicknameAndJoin);
+}
+
+function initStrophePlugins()
+{
+ require("./strophe.emuc")(XMPP, eventEmitter);
+ require("./strophe.jingle")();
+ require("./strophe.moderate")(XMPP);
+ require("./strophe.util")();
+ require("./strophe.rayo")();
+ require("./strophe.logger")();
+}
+
+function registerListeners() {
+ RTC.addStreamListener(maybeDoJoin,
+ StreamEventTypes.EVENT_TYPE_LOCAL_CREATED);
+}
+
+function setupEvents() {
+ $(window).bind('beforeunload', function () {
+ if (connection && connection.connected) {
+ // ensure signout
+ $.ajax({
+ type: 'POST',
+ url: config.bosh,
+ async: false,
+ cache: false,
+ contentType: 'application/xml',
+ data: "" +
+ "" +
+ "",
+ success: function (data) {
+ console.log('signed out');
+ console.log(data);
+ },
+ error: function (XMLHttpRequest, textStatus, errorThrown) {
+ console.log('signout error',
+ textStatus + ' (' + errorThrown + ')');
+ }
+ });
+ }
+ XMPP.disposeConference(true);
+ });
+}
+
+var XMPP = {
+ sessionTerminated: false,
+ /**
+ * Remembers if we were muted by the focus.
+ * @type {boolean}
+ */
+ forceMuted: false,
+ start: function (uiCredentials) {
+ setupEvents();
+ initStrophePlugins();
+ registerListeners();
+ Moderator.init();
+ var jid = uiCredentials.jid ||
+ config.hosts.anonymousdomain ||
+ config.hosts.domain ||
+ window.location.hostname;
+ connect(jid, null, uiCredentials);
+ },
+ promptLogin: function () {
+ UI.showLoginPopup(connect);
+ },
+ joinRooom: function(roomName, useNicks, nick)
+ {
+ var roomjid;
+ roomjid = roomName;
+
+ if (useNicks) {
+ if (nick) {
+ roomjid += '/' + nick;
+ } else {
+ roomjid += '/' + Strophe.getNodeFromJid(connection.jid);
+ }
+ } else {
+
+ var tmpJid = Strophe.getNodeFromJid(connection.jid);
+
+ if(!authenticatedUser)
+ tmpJid = tmpJid.substr(0, 8);
+
+ roomjid += '/' + tmpJid;
+ }
+ connection.emuc.doJoin(roomjid);
+ },
+ myJid: function () {
+ if(!connection)
+ return null;
+ return connection.emuc.myroomjid;
+ },
+ myResource: function () {
+ if(!connection || ! connection.emuc.myroomjid)
+ return null;
+ return Strophe.getResourceFromJid(connection.emuc.myroomjid);
+ },
+ disposeConference: function (onUnload) {
+ eventEmitter.emit(XMPPEvents.DISPOSE_CONFERENCE, onUnload);
+ var handler = activecall;
+ if (handler && handler.peerconnection) {
+ // FIXME: probably removing streams is not required and close() should
+ // be enough
+ if (RTC.localAudio) {
+ handler.peerconnection.removeStream(RTC.localAudio.getOriginalStream(), onUnload);
+ }
+ if (RTC.localVideo) {
+ handler.peerconnection.removeStream(RTC.localVideo.getOriginalStream(), onUnload);
+ }
+ handler.peerconnection.close();
+ }
+ activecall = null;
+ if(!onUnload)
+ {
+ this.sessionTerminated = true;
+ connection.emuc.doLeave();
+ }
+ },
+ addListener: function(type, listener)
+ {
+ eventEmitter.on(type, listener);
+ },
+ removeListener: function (type, listener) {
+ eventEmitter.removeListener(type, listener);
+ },
+ allocateConferenceFocus: function(roomName, callback) {
+ Moderator.allocateConferenceFocus(roomName, callback);
+ },
+ isModerator: function () {
+ return Moderator.isModerator();
+ },
+ isSipGatewayEnabled: function () {
+ return Moderator.isSipGatewayEnabled();
+ },
+ isExternalAuthEnabled: function () {
+ return Moderator.isExternalAuthEnabled();
+ },
+ switchStreams: function (stream, oldStream, callback) {
+ if (activecall) {
+ // FIXME: will block switchInProgress on true value in case of exception
+ activecall.switchStreams(stream, oldStream, callback);
+ } else {
+ // We are done immediately
+ console.error("No conference handler");
+ UI.messageHandler.showError('Error',
+ 'Unable to switch video stream.');
+ callback();
+ }
+ },
+ setVideoMute: function (mute, callback, options) {
+ if(activecall && connection && RTC.localVideo)
+ {
+ activecall.setVideoMute(mute, callback, options);
+ }
+ },
+ setAudioMute: function (mute, callback) {
+ if (!(connection && RTC.localAudio)) {
+ return false;
+ }
+
+
+ if (this.forceMuted && !mute) {
+ console.info("Asking focus for unmute");
+ connection.moderate.setMute(connection.emuc.myroomjid, mute);
+ // FIXME: wait for result before resetting muted status
+ this.forceMuted = false;
+ }
+
+ if (mute == RTC.localAudio.isMuted()) {
+ // Nothing to do
+ return true;
+ }
+
+ // It is not clear what is the right way to handle multiple tracks.
+ // So at least make sure that they are all muted or all unmuted and
+ // that we send presence just once.
+ RTC.localAudio.mute();
+ // isMuted is the opposite of audioEnabled
+ connection.emuc.addAudioInfoToPresence(mute);
+ connection.emuc.sendPresence();
+ callback();
+ return true;
+ },
+ // Really mute video, i.e. dont even send black frames
+ muteVideo: function (pc, unmute) {
+ // FIXME: this probably needs another of those lovely state safeguards...
+ // which checks for iceconn == connected and sigstate == stable
+ pc.setRemoteDescription(pc.remoteDescription,
+ function () {
+ pc.createAnswer(
+ function (answer) {
+ var sdp = new SDP(answer.sdp);
+ if (sdp.media.length > 1) {
+ if (unmute)
+ sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');
+ else
+ sdp.media[1] = sdp.media[1].replace('a=sendrecv', 'a=recvonly');
+ sdp.raw = sdp.session + sdp.media.join('');
+ answer.sdp = sdp.raw;
+ }
+ pc.setLocalDescription(answer,
+ function () {
+ console.log('mute SLD ok');
+ },
+ function (error) {
+ console.log('mute SLD error');
+ UI.messageHandler.showError('Error',
+ 'Oops! Something went wrong and we failed to ' +
+ 'mute! (SLD Failure)');
+ }
+ );
+ },
+ function (error) {
+ console.log(error);
+ UI.messageHandler.showError();
+ }
+ );
+ },
+ function (error) {
+ console.log('muteVideo SRD error');
+ UI.messageHandler.showError('Error',
+ 'Oops! Something went wrong and we failed to stop video!' +
+ '(SRD Failure)');
+
+ }
+ );
+ },
+ toggleRecording: function (tokenEmptyCallback,
+ startingCallback, startedCallback) {
+ Recording.toggleRecording(tokenEmptyCallback,
+ startingCallback, startedCallback);
+ },
+ addToPresence: function (name, value, dontSend) {
+ switch (name)
+ {
+ case "displayName":
+ connection.emuc.addDisplayNameToPresence(value);
+ break;
+ case "etherpad":
+ connection.emuc.addEtherpadToPresence(value);
+ break;
+ case "prezi":
+ connection.emuc.addPreziToPresence(value, 0);
+ break;
+ case "preziSlide":
+ connection.emuc.addCurrentSlideToPresence(value);
+ break;
+ case "connectionQuality":
+ connection.emuc.addConnectionInfoToPresence(value);
+ break;
+ case "email":
+ connection.emuc.addEmailToPresence(value);
+ default :
+ console.log("Unknown tag for presence.");
+ return;
+ }
+ if(!dontSend)
+ connection.emuc.sendPresence();
+ },
+ sendLogs: function (content) {
+ // XEP-0337-ish
+ var message = $msg({to: focusMucJid, type: 'normal'});
+ message.c('log', { xmlns: 'urn:xmpp:eventlog',
+ id: 'PeerConnectionStats'});
+ message.c('message').t(content).up();
+ if (deflate) {
+ message.c('tag', {name: "deflated", value: "true"}).up();
+ }
+ message.up();
+
+ connection.send(message);
+ },
+ populateData: function () {
+ var data = {};
+ if (connection.jingle) {
+ data = connection.jingle.populateData();
+ }
+ return data;
+ },
+ getLogger: function () {
+ if(connection.logger)
+ return connection.logger.log;
+ return null;
+ },
+ getPrezi: function () {
+ return connection.emuc.getPrezi(this.myJid());
+ },
+ removePreziFromPresence: function () {
+ connection.emuc.removePreziFromPresence();
+ connection.emuc.sendPresence();
+ },
+ sendChatMessage: function (message, nickname) {
+ connection.emuc.sendMessage(message, nickname);
+ },
+ setSubject: function (topic) {
+ connection.emuc.setSubject(topic);
+ },
+ lockRoom: function (key, onSuccess, onError, onNotSupported) {
+ connection.emuc.lockRoom(key, onSuccess, onError, onNotSupported);
+ },
+ dial: function (to, from, roomName,roomPass) {
+ connection.rayo.dial(to, from, roomName,roomPass);
+ },
+ setMute: function (jid, mute) {
+ connection.moderate.setMute(jid, mute);
+ },
+ eject: function (jid) {
+ connection.moderate.eject(jid);
+ },
+ findJidFromResource: function (resource) {
+ connection.emuc.findJidFromResource(resource);
+ },
+ getMembers: function () {
+ return connection.emuc.members;
+ }
+
+};
+
+module.exports = XMPP;
+},{"./SDP":2,"./moderator":6,"./recording":7,"./strophe.emuc":8,"./strophe.jingle":9,"./strophe.logger":10,"./strophe.moderate":11,"./strophe.rayo":12,"./strophe.util":13,"events":15}],15:[function(require,module,exports){
+// Copyright Joyent, Inc. and other Node contributors.
+//
+// Permission is hereby granted, free of charge, to any person obtaining a
+// copy of this software and associated documentation files (the
+// "Software"), to deal in the Software without restriction, including
+// without limitation the rights to use, copy, modify, merge, publish,
+// distribute, sublicense, and/or sell copies of the Software, and to permit
+// persons to whom the Software is furnished to do so, subject to the
+// following conditions:
+//
+// The above copyright notice and this permission notice shall be included
+// in all copies or substantial portions of the Software.
+//
+// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
+// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
+// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
+// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
+// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
+// USE OR OTHER DEALINGS IN THE SOFTWARE.
+
+function EventEmitter() {
+ this._events = this._events || {};
+ this._maxListeners = this._maxListeners || undefined;
+}
+module.exports = EventEmitter;
+
+// Backwards-compat with node 0.10.x
+EventEmitter.EventEmitter = EventEmitter;
+
+EventEmitter.prototype._events = undefined;
+EventEmitter.prototype._maxListeners = undefined;
+
+// By default EventEmitters will print a warning if more than 10 listeners are
+// added to it. This is a useful default which helps finding memory leaks.
+EventEmitter.defaultMaxListeners = 10;
+
+// Obviously not all Emitters should be limited to 10. This function allows
+// that to be increased. Set to zero for unlimited.
+EventEmitter.prototype.setMaxListeners = function(n) {
+ if (!isNumber(n) || n < 0 || isNaN(n))
+ throw TypeError('n must be a positive number');
+ this._maxListeners = n;
+ return this;
+};
+
+EventEmitter.prototype.emit = function(type) {
+ var er, handler, len, args, i, listeners;
+
+ if (!this._events)
+ this._events = {};
+
+ // If there is no 'error' event listener then throw.
+ if (type === 'error') {
+ if (!this._events.error ||
+ (isObject(this._events.error) && !this._events.error.length)) {
+ er = arguments[1];
+ if (er instanceof Error) {
+ throw er; // Unhandled 'error' event
+ } else {
+ throw TypeError('Uncaught, unspecified "error" event.');
+ }
+ return false;
+ }
+ }
+
+ handler = this._events[type];
+
+ if (isUndefined(handler))
+ return false;
+
+ if (isFunction(handler)) {
+ switch (arguments.length) {
+ // fast cases
+ case 1:
+ handler.call(this);
+ break;
+ case 2:
+ handler.call(this, arguments[1]);
+ break;
+ case 3:
+ handler.call(this, arguments[1], arguments[2]);
+ break;
+ // slower
+ default:
+ len = arguments.length;
+ args = new Array(len - 1);
+ for (i = 1; i < len; i++)
+ args[i - 1] = arguments[i];
+ handler.apply(this, args);
+ }
+ } else if (isObject(handler)) {
+ len = arguments.length;
+ args = new Array(len - 1);
+ for (i = 1; i < len; i++)
+ args[i - 1] = arguments[i];
+
+ listeners = handler.slice();
+ len = listeners.length;
+ for (i = 0; i < len; i++)
+ listeners[i].apply(this, args);
+ }
+
+ return true;
+};
+
+EventEmitter.prototype.addListener = function(type, listener) {
+ var m;
+
+ if (!isFunction(listener))
+ throw TypeError('listener must be a function');
+
+ if (!this._events)
+ this._events = {};
+
+ // To avoid recursion in the case that type === "newListener"! Before
+ // adding it to the listeners, first emit "newListener".
+ if (this._events.newListener)
+ this.emit('newListener', type,
+ isFunction(listener.listener) ?
+ listener.listener : listener);
+
+ if (!this._events[type])
+ // Optimize the case of one listener. Don't need the extra array object.
+ this._events[type] = listener;
+ else if (isObject(this._events[type]))
+ // If we've already got an array, just append.
+ this._events[type].push(listener);
+ else
+ // Adding the second element, need to change to array.
+ this._events[type] = [this._events[type], listener];
+
+ // Check for listener leak
+ if (isObject(this._events[type]) && !this._events[type].warned) {
+ var m;
+ if (!isUndefined(this._maxListeners)) {
+ m = this._maxListeners;
+ } else {
+ m = EventEmitter.defaultMaxListeners;
+ }
+
+ if (m && m > 0 && this._events[type].length > m) {
+ this._events[type].warned = true;
+ console.error('(node) warning: possible EventEmitter memory ' +
+ 'leak detected. %d listeners added. ' +
+ 'Use emitter.setMaxListeners() to increase limit.',
+ this._events[type].length);
+ if (typeof console.trace === 'function') {
+ // not supported in IE 10
+ console.trace();
+ }
+ }
+ }
+
+ return this;
+};
+
+EventEmitter.prototype.on = EventEmitter.prototype.addListener;
+
+EventEmitter.prototype.once = function(type, listener) {
+ if (!isFunction(listener))
+ throw TypeError('listener must be a function');
+
+ var fired = false;
+
+ function g() {
+ this.removeListener(type, g);
+
+ if (!fired) {
+ fired = true;
+ listener.apply(this, arguments);
+ }
+ }
+
+ g.listener = listener;
+ this.on(type, g);
+
+ return this;
+};
+
+// emits a 'removeListener' event iff the listener was removed
+EventEmitter.prototype.removeListener = function(type, listener) {
+ var list, position, length, i;
+
+ if (!isFunction(listener))
+ throw TypeError('listener must be a function');
+
+ if (!this._events || !this._events[type])
+ return this;
+
+ list = this._events[type];
+ length = list.length;
+ position = -1;
+
+ if (list === listener ||
+ (isFunction(list.listener) && list.listener === listener)) {
+ delete this._events[type];
+ if (this._events.removeListener)
+ this.emit('removeListener', type, listener);
+
+ } else if (isObject(list)) {
+ for (i = length; i-- > 0;) {
+ if (list[i] === listener ||
+ (list[i].listener && list[i].listener === listener)) {
+ position = i;
+ break;
+ }
+ }
+
+ if (position < 0)
+ return this;
+
+ if (list.length === 1) {
+ list.length = 0;
+ delete this._events[type];
+ } else {
+ list.splice(position, 1);
+ }
+
+ if (this._events.removeListener)
+ this.emit('removeListener', type, listener);
+ }
+
+ return this;
+};
+
+EventEmitter.prototype.removeAllListeners = function(type) {
+ var key, listeners;
+
+ if (!this._events)
+ return this;
+
+ // not listening for removeListener, no need to emit
+ if (!this._events.removeListener) {
+ if (arguments.length === 0)
+ this._events = {};
+ else if (this._events[type])
+ delete this._events[type];
+ return this;
+ }
+
+ // emit removeListener for all listeners on all events
+ if (arguments.length === 0) {
+ for (key in this._events) {
+ if (key === 'removeListener') continue;
+ this.removeAllListeners(key);
+ }
+ this.removeAllListeners('removeListener');
+ this._events = {};
+ return this;
+ }
+
+ listeners = this._events[type];
+
+ if (isFunction(listeners)) {
+ this.removeListener(type, listeners);
+ } else {
+ // LIFO order
+ while (listeners.length)
+ this.removeListener(type, listeners[listeners.length - 1]);
+ }
+ delete this._events[type];
+
+ return this;
+};
+
+EventEmitter.prototype.listeners = function(type) {
+ var ret;
+ if (!this._events || !this._events[type])
+ ret = [];
+ else if (isFunction(this._events[type]))
+ ret = [this._events[type]];
+ else
+ ret = this._events[type].slice();
+ return ret;
+};
+
+EventEmitter.listenerCount = function(emitter, type) {
+ var ret;
+ if (!emitter._events || !emitter._events[type])
+ ret = 0;
+ else if (isFunction(emitter._events[type]))
+ ret = 1;
+ else
+ ret = emitter._events[type].length;
+ return ret;
+};
+
+function isFunction(arg) {
+ return typeof arg === 'function';
+}
+
+function isNumber(arg) {
+ return typeof arg === 'number';
+}
+
+function isObject(arg) {
+ return typeof arg === 'object' && arg !== null;
+}
+
+function isUndefined(arg) {
+ return arg === void 0;
+}
+
+},{}]},{},[14])(14)
+});
+//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/usr/local/lib/node_modules/browserify/node_modules/browser-pack/_prelude.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/JingleSession.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/SDP.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/SDPDiffer.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/SDPUtil.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/TraceablePeerConnection.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/moderator.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/recording.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/strophe.emuc.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/strophe.jingle.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/strophe.logger.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/strophe.moderate.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/strophe.rayo.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/strophe.util.js","/Users/hristo/Documents/workspace/jitsi-meet/modules/xmpp/xmpp.js","/usr/local/lib/node_modules/browserify/node_modules/events/events.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC52CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5mBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5VA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/lBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9UA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3ZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","/* jshint -W117 */\nvar TraceablePeerConnection = require(\"./TraceablePeerConnection\");\nvar SDPDiffer = require(\"./SDPDiffer\");\nvar SDPUtil = require(\"./SDPUtil\");\nvar SDP = require(\"./SDP\");\n\n// Jingle stuff\nfunction JingleSession(me, sid, connection, service) {\n    this.me = me;\n    this.sid = sid;\n    this.connection = connection;\n    this.initiator = null;\n    this.responder = null;\n    this.isInitiator = null;\n    this.peerjid = null;\n    this.state = null;\n    this.localSDP = null;\n    this.remoteSDP = null;\n    this.relayedStreams = [];\n    this.startTime = null;\n    this.stopTime = null;\n    this.media_constraints = null;\n    this.pc_constraints = null;\n    this.ice_config = {};\n    this.drip_container = [];\n    this.service = service;\n\n    this.usetrickle = true;\n    this.usepranswer = false; // early transport warmup -- mind you, this might fail. depends on webrtc issue 1718\n    this.usedrip = false; // dripping is sending trickle candidates not one-by-one\n\n    this.hadstuncandidate = false;\n    this.hadturncandidate = false;\n    this.lasticecandidate = false;\n\n    this.statsinterval = null;\n\n    this.reason = null;\n\n    this.addssrc = [];\n    this.removessrc = [];\n    this.pendingop = null;\n    this.switchstreams = false;\n\n    this.wait = true;\n    this.localStreamsSSRC = null;\n\n    /**\n     * The indicator which determines whether the (local) video has been muted\n     * in response to a user command in contrast to an automatic decision made\n     * by the application logic.\n     */\n    this.videoMuteByUser = false;\n}\n\nJingleSession.prototype.initiate = function (peerjid, isInitiator) {\n    var self = this;\n    if (this.state !== null) {\n        console.error('attempt to initiate on session ' + this.sid +\n            'in state ' + this.state);\n        return;\n    }\n    this.isInitiator = isInitiator;\n    this.state = 'pending';\n    this.initiator = isInitiator ? this.me : peerjid;\n    this.responder = !isInitiator ? this.me : peerjid;\n    this.peerjid = peerjid;\n    this.hadstuncandidate = false;\n    this.hadturncandidate = false;\n    this.lasticecandidate = false;\n\n    this.peerconnection\n        = new TraceablePeerConnection(\n            this.connection.jingle.ice_config,\n            this.connection.jingle.pc_constraints );\n\n    this.peerconnection.onicecandidate = function (event) {\n        self.sendIceCandidate(event.candidate);\n    };\n    this.peerconnection.onaddstream = function (event) {\n        console.log(\"REMOTE STREAM ADDED: \" + event.stream + \" - \" + event.stream.id);\n        self.remoteStreamAdded(event);\n    };\n    this.peerconnection.onremovestream = function (event) {\n        // Remove the stream from remoteStreams\n        // FIXME: remotestreamremoved.jingle not defined anywhere(unused)\n        $(document).trigger('remotestreamremoved.jingle', [event, self.sid]);\n    };\n    this.peerconnection.onsignalingstatechange = function (event) {\n        if (!(self && self.peerconnection)) return;\n    };\n    this.peerconnection.oniceconnectionstatechange = function (event) {\n        if (!(self && self.peerconnection)) return;\n        switch (self.peerconnection.iceConnectionState) {\n            case 'connected':\n                this.startTime = new Date();\n                break;\n            case 'disconnected':\n                this.stopTime = new Date();\n                break;\n        }\n        onIceConnectionStateChange(self.sid, self);\n    };\n    // add any local and relayed stream\n    RTC.localStreams.forEach(function(stream) {\n        self.peerconnection.addStream(stream.getOriginalStream());\n    });\n    this.relayedStreams.forEach(function(stream) {\n        self.peerconnection.addStream(stream);\n    });\n};\n\nfunction onIceConnectionStateChange(sid, session) {\n    switch (session.peerconnection.iceConnectionState) {\n        case 'checking':\n            session.timeChecking = (new Date()).getTime();\n            session.firstconnect = true;\n            break;\n        case 'completed': // on caller side\n        case 'connected':\n            if (session.firstconnect) {\n                session.firstconnect = false;\n                var metadata = {};\n                metadata.setupTime\n                    = (new Date()).getTime() - session.timeChecking;\n                session.peerconnection.getStats(function (res) {\n                    if(res && res.result) {\n                        res.result().forEach(function (report) {\n                            if (report.type == 'googCandidatePair' &&\n                                report.stat('googActiveConnection') == 'true') {\n                                metadata.localCandidateType\n                                    = report.stat('googLocalCandidateType');\n                                metadata.remoteCandidateType\n                                    = report.stat('googRemoteCandidateType');\n\n                                // log pair as well so we can get nice pie\n                                // charts\n                                metadata.candidatePair\n                                    = report.stat('googLocalCandidateType') +\n                                        ';' +\n                                        report.stat('googRemoteCandidateType');\n\n                                if (report.stat('googRemoteAddress').indexOf('[') === 0)\n                                {\n                                    metadata.ipv6 = true;\n                                }\n                            }\n                        });\n                    }\n                });\n            }\n            break;\n    }\n}\n\nJingleSession.prototype.accept = function () {\n    var self = this;\n    this.state = 'active';\n\n    var pranswer = this.peerconnection.localDescription;\n    if (!pranswer || pranswer.type != 'pranswer') {\n        return;\n    }\n    console.log('going from pranswer to answer');\n    if (this.usetrickle) {\n        // remove candidates already sent from session-accept\n        var lines = SDPUtil.find_lines(pranswer.sdp, 'a=candidate:');\n        for (var i = 0; i < lines.length; i++) {\n            pranswer.sdp = pranswer.sdp.replace(lines[i] + '\\r\\n', '');\n        }\n    }\n    while (SDPUtil.find_line(pranswer.sdp, 'a=inactive')) {\n        // FIXME: change any inactive to sendrecv or whatever they were originally\n        pranswer.sdp = pranswer.sdp.replace('a=inactive', 'a=sendrecv');\n    }\n    pranswer = simulcast.reverseTransformLocalDescription(pranswer);\n    var prsdp = new SDP(pranswer.sdp);\n    var accept = $iq({to: this.peerjid,\n        type: 'set'})\n        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n            action: 'session-accept',\n            initiator: this.initiator,\n            responder: this.responder,\n            sid: this.sid });\n    prsdp.toJingle(accept, this.initiator == this.me ? 'initiator' : 'responder', this.localStreamsSSRC);\n    var sdp = this.peerconnection.localDescription.sdp;\n    while (SDPUtil.find_line(sdp, 'a=inactive')) {\n        // FIXME: change any inactive to sendrecv or whatever they were originally\n        sdp = sdp.replace('a=inactive', 'a=sendrecv');\n    }\n    var self = this;\n    this.peerconnection.setLocalDescription(new RTCSessionDescription({type: 'answer', sdp: sdp}),\n        function () {\n            //console.log('setLocalDescription success');\n            self.setLocalDescription();\n\n            self.connection.sendIQ(accept,\n                function () {\n                    var ack = {};\n                    ack.source = 'answer';\n                    $(document).trigger('ack.jingle', [self.sid, ack]);\n                },\n                function (stanza) {\n                    var error = ($(stanza).find('error').length) ? {\n                        code: $(stanza).find('error').attr('code'),\n                        reason: $(stanza).find('error :first')[0].tagName\n                    }:{};\n                    error.source = 'answer';\n                    JingleSession.onJingleError(self.sid, error);\n                },\n                10000);\n        },\n        function (e) {\n            console.error('setLocalDescription failed', e);\n        }\n    );\n};\n\nJingleSession.prototype.terminate = function (reason) {\n    this.state = 'ended';\n    this.reason = reason;\n    this.peerconnection.close();\n    if (this.statsinterval !== null) {\n        window.clearInterval(this.statsinterval);\n        this.statsinterval = null;\n    }\n};\n\nJingleSession.prototype.active = function () {\n    return this.state == 'active';\n};\n\nJingleSession.prototype.sendIceCandidate = function (candidate) {\n    var self = this;\n    if (candidate && !this.lasticecandidate) {\n        var ice = SDPUtil.iceparams(this.localSDP.media[candidate.sdpMLineIndex], this.localSDP.session);\n        var jcand = SDPUtil.candidateToJingle(candidate.candidate);\n        if (!(ice && jcand)) {\n            console.error('failed to get ice && jcand');\n            return;\n        }\n        ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';\n\n        if (jcand.type === 'srflx') {\n            this.hadstuncandidate = true;\n        } else if (jcand.type === 'relay') {\n            this.hadturncandidate = true;\n        }\n\n        if (this.usetrickle) {\n            if (this.usedrip) {\n                if (this.drip_container.length === 0) {\n                    // start 20ms callout\n                    window.setTimeout(function () {\n                        if (self.drip_container.length === 0) return;\n                        self.sendIceCandidates(self.drip_container);\n                        self.drip_container = [];\n                    }, 20);\n\n                }\n                this.drip_container.push(candidate);\n                return;\n            } else {\n                self.sendIceCandidate([candidate]);\n            }\n        }\n    } else {\n        //console.log('sendIceCandidate: last candidate.');\n        if (!this.usetrickle) {\n            //console.log('should send full offer now...');\n            var init = $iq({to: this.peerjid,\n                type: 'set'})\n                .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n                    action: this.peerconnection.localDescription.type == 'offer' ? 'session-initiate' : 'session-accept',\n                    initiator: this.initiator,\n                    sid: this.sid});\n            this.localSDP = new SDP(this.peerconnection.localDescription.sdp);\n            var self = this;\n            var sendJingle = function (ssrc) {\n                if(!ssrc)\n                    ssrc = {};\n                self.localSDP.toJingle(init, self.initiator == self.me ? 'initiator' : 'responder', ssrc);\n                self.connection.sendIQ(init,\n                    function () {\n                        //console.log('session initiate ack');\n                        var ack = {};\n                        ack.source = 'offer';\n                        $(document).trigger('ack.jingle', [self.sid, ack]);\n                    },\n                    function (stanza) {\n                        self.state = 'error';\n                        self.peerconnection.close();\n                        var error = ($(stanza).find('error').length) ? {\n                            code: $(stanza).find('error').attr('code'),\n                            reason: $(stanza).find('error :first')[0].tagName,\n                        }:{};\n                        error.source = 'offer';\n                        JingleSession.onJingleError(self.sid, error);\n                    },\n                    10000);\n            }\n            sendJingle();\n        }\n        this.lasticecandidate = true;\n        console.log('Have we encountered any srflx candidates? ' + this.hadstuncandidate);\n        console.log('Have we encountered any relay candidates? ' + this.hadturncandidate);\n\n        if (!(this.hadstuncandidate || this.hadturncandidate) && this.peerconnection.signalingState != 'closed') {\n            $(document).trigger('nostuncandidates.jingle', [this.sid]);\n        }\n    }\n};\n\nJingleSession.prototype.sendIceCandidates = function (candidates) {\n    console.log('sendIceCandidates', candidates);\n    var cand = $iq({to: this.peerjid, type: 'set'})\n        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n            action: 'transport-info',\n            initiator: this.initiator,\n            sid: this.sid});\n    for (var mid = 0; mid < this.localSDP.media.length; mid++) {\n        var cands = candidates.filter(function (el) { return el.sdpMLineIndex == mid; });\n        var mline = SDPUtil.parse_mline(this.localSDP.media[mid].split('\\r\\n')[0]);\n        if (cands.length > 0) {\n            var ice = SDPUtil.iceparams(this.localSDP.media[mid], this.localSDP.session);\n            ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';\n            cand.c('content', {creator: this.initiator == this.me ? 'initiator' : 'responder',\n                name: (cands[0].sdpMid? cands[0].sdpMid : mline.media)\n            }).c('transport', ice);\n            for (var i = 0; i < cands.length; i++) {\n                cand.c('candidate', SDPUtil.candidateToJingle(cands[i].candidate)).up();\n            }\n            // add fingerprint\n            if (SDPUtil.find_line(this.localSDP.media[mid], 'a=fingerprint:', this.localSDP.session)) {\n                var tmp = SDPUtil.parse_fingerprint(SDPUtil.find_line(this.localSDP.media[mid], 'a=fingerprint:', this.localSDP.session));\n                tmp.required = true;\n                cand.c(\n                    'fingerprint',\n                    {xmlns: 'urn:xmpp:jingle:apps:dtls:0'})\n                    .t(tmp.fingerprint);\n                delete tmp.fingerprint;\n                cand.attrs(tmp);\n                cand.up();\n            }\n            cand.up(); // transport\n            cand.up(); // content\n        }\n    }\n    // might merge last-candidate notification into this, but it is called alot later. See webrtc issue #2340\n    //console.log('was this the last candidate', this.lasticecandidate);\n    this.connection.sendIQ(cand,\n        function () {\n            var ack = {};\n            ack.source = 'transportinfo';\n            $(document).trigger('ack.jingle', [this.sid, ack]);\n        },\n        function (stanza) {\n            var error = ($(stanza).find('error').length) ? {\n                code: $(stanza).find('error').attr('code'),\n                reason: $(stanza).find('error :first')[0].tagName,\n            }:{};\n            error.source = 'transportinfo';\n            JingleSession.onJingleError(this.sid, error);\n        },\n        10000);\n};\n\n\nJingleSession.prototype.sendOffer = function () {\n    //console.log('sendOffer...');\n    var self = this;\n    this.peerconnection.createOffer(function (sdp) {\n            self.createdOffer(sdp);\n        },\n        function (e) {\n            console.error('createOffer failed', e);\n        },\n        this.media_constraints\n    );\n};\n\nJingleSession.prototype.createdOffer = function (sdp) {\n    //console.log('createdOffer', sdp);\n    var self = this;\n    this.localSDP = new SDP(sdp.sdp);\n    //this.localSDP.mangle();\n    var sendJingle = function () {\n        var init = $iq({to: this.peerjid,\n            type: 'set'})\n            .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n                action: 'session-initiate',\n                initiator: this.initiator,\n                sid: this.sid});\n        self.localSDP.toJingle(init, this.initiator == this.me ? 'initiator' : 'responder', this.localStreamsSSRC);\n        self.connection.sendIQ(init,\n            function () {\n                var ack = {};\n                ack.source = 'offer';\n                $(document).trigger('ack.jingle', [self.sid, ack]);\n            },\n            function (stanza) {\n                self.state = 'error';\n                self.peerconnection.close();\n                var error = ($(stanza).find('error').length) ? {\n                    code: $(stanza).find('error').attr('code'),\n                    reason: $(stanza).find('error :first')[0].tagName,\n                }:{};\n                error.source = 'offer';\n                JingleSession.onJingleError(self.sid, error);\n            },\n            10000);\n    }\n    sdp.sdp = this.localSDP.raw;\n    this.peerconnection.setLocalDescription(sdp,\n        function () {\n            if(self.usetrickle)\n            {\n                sendJingle();\n            }\n            self.setLocalDescription();\n            //console.log('setLocalDescription success');\n        },\n        function (e) {\n            console.error('setLocalDescription failed', e);\n        }\n    );\n    var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');\n    for (var i = 0; i < cands.length; i++) {\n        var cand = SDPUtil.parse_icecandidate(cands[i]);\n        if (cand.type == 'srflx') {\n            this.hadstuncandidate = true;\n        } else if (cand.type == 'relay') {\n            this.hadturncandidate = true;\n        }\n    }\n};\n\nJingleSession.prototype.setRemoteDescription = function (elem, desctype) {\n    //console.log('setting remote description... ', desctype);\n    this.remoteSDP = new SDP('');\n    this.remoteSDP.fromJingle(elem);\n    if (this.peerconnection.remoteDescription !== null) {\n        console.log('setRemoteDescription when remote description is not null, should be pranswer', this.peerconnection.remoteDescription);\n        if (this.peerconnection.remoteDescription.type == 'pranswer') {\n            var pranswer = new SDP(this.peerconnection.remoteDescription.sdp);\n            for (var i = 0; i < pranswer.media.length; i++) {\n                // make sure we have ice ufrag and pwd\n                if (!SDPUtil.find_line(this.remoteSDP.media[i], 'a=ice-ufrag:', this.remoteSDP.session)) {\n                    if (SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session)) {\n                        this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session) + '\\r\\n';\n                    } else {\n                        console.warn('no ice ufrag?');\n                    }\n                    if (SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session)) {\n                        this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session) + '\\r\\n';\n                    } else {\n                        console.warn('no ice pwd?');\n                    }\n                }\n                // copy over candidates\n                var lines = SDPUtil.find_lines(pranswer.media[i], 'a=candidate:');\n                for (var j = 0; j < lines.length; j++) {\n                    this.remoteSDP.media[i] += lines[j] + '\\r\\n';\n                }\n            }\n            this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');\n        }\n    }\n    var remotedesc = new RTCSessionDescription({type: desctype, sdp: this.remoteSDP.raw});\n\n    this.peerconnection.setRemoteDescription(remotedesc,\n        function () {\n            //console.log('setRemoteDescription success');\n        },\n        function (e) {\n            console.error('setRemoteDescription error', e);\n            JingleSession.onJingleFatalError(self, e);\n        }\n    );\n};\n\nJingleSession.prototype.addIceCandidate = function (elem) {\n    var self = this;\n    if (this.peerconnection.signalingState == 'closed') {\n        return;\n    }\n    if (!this.peerconnection.remoteDescription && this.peerconnection.signalingState == 'have-local-offer') {\n        console.log('trickle ice candidate arriving before session accept...');\n        // create a PRANSWER for setRemoteDescription\n        if (!this.remoteSDP) {\n            var cobbled = 'v=0\\r\\n' +\n                'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\\r\\n' +// FIXME\n                's=-\\r\\n' +\n                't=0 0\\r\\n';\n            // first, take some things from the local description\n            for (var i = 0; i < this.localSDP.media.length; i++) {\n                cobbled += SDPUtil.find_line(this.localSDP.media[i], 'm=') + '\\r\\n';\n                cobbled += SDPUtil.find_lines(this.localSDP.media[i], 'a=rtpmap:').join('\\r\\n') + '\\r\\n';\n                if (SDPUtil.find_line(this.localSDP.media[i], 'a=mid:')) {\n                    cobbled += SDPUtil.find_line(this.localSDP.media[i], 'a=mid:') + '\\r\\n';\n                }\n                cobbled += 'a=inactive\\r\\n';\n            }\n            this.remoteSDP = new SDP(cobbled);\n        }\n        // then add things like ice and dtls from remote candidate\n        elem.each(function () {\n            for (var i = 0; i < self.remoteSDP.media.length; i++) {\n                if (SDPUtil.find_line(self.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||\n                    self.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {\n                    if (!SDPUtil.find_line(self.remoteSDP.media[i], 'a=ice-ufrag:')) {\n                        var tmp = $(this).find('transport');\n                        self.remoteSDP.media[i] += 'a=ice-ufrag:' + tmp.attr('ufrag') + '\\r\\n';\n                        self.remoteSDP.media[i] += 'a=ice-pwd:' + tmp.attr('pwd') + '\\r\\n';\n                        tmp = $(this).find('transport>fingerprint');\n                        if (tmp.length) {\n                            self.remoteSDP.media[i] += 'a=fingerprint:' + tmp.attr('hash') + ' ' + tmp.text() + '\\r\\n';\n                        } else {\n                            console.log('no dtls fingerprint (webrtc issue #1718?)');\n                            self.remoteSDP.media[i] += 'a=crypto:1 AES_CM_128_HMAC_SHA1_80 inline:BAADBAADBAADBAADBAADBAADBAADBAADBAADBAAD\\r\\n';\n                        }\n                        break;\n                    }\n                }\n            }\n        });\n        this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');\n\n        // we need a complete SDP with ice-ufrag/ice-pwd in all parts\n        // this makes the assumption that the PRANSWER is constructed such that the ice-ufrag is in all mediaparts\n        // but it could be in the session part as well. since the code above constructs this sdp this can't happen however\n        var iscomplete = this.remoteSDP.media.filter(function (mediapart) {\n            return SDPUtil.find_line(mediapart, 'a=ice-ufrag:');\n        }).length == this.remoteSDP.media.length;\n\n        if (iscomplete) {\n            console.log('setting pranswer');\n            try {\n                this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'pranswer', sdp: this.remoteSDP.raw }),\n                    function() {\n                    },\n                    function(e) {\n                        console.log('setRemoteDescription pranswer failed', e.toString());\n                    });\n            } catch (e) {\n                console.error('setting pranswer failed', e);\n            }\n        } else {\n            //console.log('not yet setting pranswer');\n        }\n    }\n    // operate on each content element\n    elem.each(function () {\n        // would love to deactivate this, but firefox still requires it\n        var idx = -1;\n        var i;\n        for (i = 0; i < self.remoteSDP.media.length; i++) {\n            if (SDPUtil.find_line(self.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||\n                self.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {\n                idx = i;\n                break;\n            }\n        }\n        if (idx == -1) { // fall back to localdescription\n            for (i = 0; i < self.localSDP.media.length; i++) {\n                if (SDPUtil.find_line(self.localSDP.media[i], 'a=mid:' + $(this).attr('name')) ||\n                    self.localSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {\n                    idx = i;\n                    break;\n                }\n            }\n        }\n        var name = $(this).attr('name');\n        // TODO: check ice-pwd and ice-ufrag?\n        $(this).find('transport>candidate').each(function () {\n            var line, candidate;\n            line = SDPUtil.candidateFromJingle(this);\n            candidate = new RTCIceCandidate({sdpMLineIndex: idx,\n                sdpMid: name,\n                candidate: line});\n            try {\n                self.peerconnection.addIceCandidate(candidate);\n            } catch (e) {\n                console.error('addIceCandidate failed', e.toString(), line);\n            }\n        });\n    });\n};\n\nJingleSession.prototype.sendAnswer = function (provisional) {\n    //console.log('createAnswer', provisional);\n    var self = this;\n    this.peerconnection.createAnswer(\n        function (sdp) {\n            self.createdAnswer(sdp, provisional);\n        },\n        function (e) {\n            console.error('createAnswer failed', e);\n        },\n        this.media_constraints\n    );\n};\n\nJingleSession.prototype.createdAnswer = function (sdp, provisional) {\n    //console.log('createAnswer callback');\n    var self = this;\n    this.localSDP = new SDP(sdp.sdp);\n    //this.localSDP.mangle();\n    this.usepranswer = provisional === true;\n    if (this.usetrickle) {\n        if (this.usepranswer) {\n            sdp.type = 'pranswer';\n            for (var i = 0; i < this.localSDP.media.length; i++) {\n                this.localSDP.media[i] = this.localSDP.media[i].replace('a=sendrecv\\r\\n', 'a=inactive\\r\\n');\n            }\n            this.localSDP.raw = this.localSDP.session + '\\r\\n' + this.localSDP.media.join('');\n        }\n    }\n    var self = this;\n    var sendJingle = function (ssrcs) {\n\n                var accept = $iq({to: self.peerjid,\n                    type: 'set'})\n                    .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n                        action: 'session-accept',\n                        initiator: self.initiator,\n                        responder: self.responder,\n                        sid: self.sid });\n                var publicLocalDesc = simulcast.reverseTransformLocalDescription(sdp);\n                var publicLocalSDP = new SDP(publicLocalDesc.sdp);\n                publicLocalSDP.toJingle(accept, self.initiator == self.me ? 'initiator' : 'responder', ssrcs);\n                self.connection.sendIQ(accept,\n                    function () {\n                        var ack = {};\n                        ack.source = 'answer';\n                        $(document).trigger('ack.jingle', [self.sid, ack]);\n                    },\n                    function (stanza) {\n                        var error = ($(stanza).find('error').length) ? {\n                            code: $(stanza).find('error').attr('code'),\n                            reason: $(stanza).find('error :first')[0].tagName,\n                        }:{};\n                        error.source = 'answer';\n                        JingleSession.onJingleError(self.sid, error);\n                    },\n                    10000);\n    }\n    sdp.sdp = this.localSDP.raw;\n    this.peerconnection.setLocalDescription(sdp,\n        function () {\n\n            //console.log('setLocalDescription success');\n            if (self.usetrickle && !self.usepranswer) {\n                sendJingle();\n            }\n            self.setLocalDescription();\n        },\n        function (e) {\n            console.error('setLocalDescription failed', e);\n        }\n    );\n    var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');\n    for (var j = 0; j < cands.length; j++) {\n        var cand = SDPUtil.parse_icecandidate(cands[j]);\n        if (cand.type == 'srflx') {\n            this.hadstuncandidate = true;\n        } else if (cand.type == 'relay') {\n            this.hadturncandidate = true;\n        }\n    }\n};\n\nJingleSession.prototype.sendTerminate = function (reason, text) {\n    var self = this,\n        term = $iq({to: this.peerjid,\n            type: 'set'})\n            .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n                action: 'session-terminate',\n                initiator: this.initiator,\n                sid: this.sid})\n            .c('reason')\n            .c(reason || 'success');\n\n    if (text) {\n        term.up().c('text').t(text);\n    }\n\n    this.connection.sendIQ(term,\n        function () {\n            self.peerconnection.close();\n            self.peerconnection = null;\n            self.terminate();\n            var ack = {};\n            ack.source = 'terminate';\n            $(document).trigger('ack.jingle', [self.sid, ack]);\n        },\n        function (stanza) {\n            var error = ($(stanza).find('error').length) ? {\n                code: $(stanza).find('error').attr('code'),\n                reason: $(stanza).find('error :first')[0].tagName,\n            }:{};\n            $(document).trigger('ack.jingle', [self.sid, error]);\n        },\n        10000);\n    if (this.statsinterval !== null) {\n        window.clearInterval(this.statsinterval);\n        this.statsinterval = null;\n    }\n};\n\nJingleSession.prototype.addSource = function (elem, fromJid) {\n\n    var self = this;\n    // FIXME: dirty waiting\n    if (!this.peerconnection.localDescription)\n    {\n        console.warn(\"addSource - localDescription not ready yet\")\n        setTimeout(function()\n            {\n                self.addSource(elem, fromJid);\n            },\n            200\n        );\n        return;\n    }\n\n    console.log('addssrc', new Date().getTime());\n    console.log('ice', this.peerconnection.iceConnectionState);\n    var sdp = new SDP(this.peerconnection.remoteDescription.sdp);\n    var mySdp = new SDP(this.peerconnection.localDescription.sdp);\n\n    $(elem).each(function (idx, content) {\n        var name = $(content).attr('name');\n        var lines = '';\n        tmp = $(content).find('ssrc-group[xmlns=\"urn:xmpp:jingle:apps:rtp:ssma:0\"]').each(function() {\n            var semantics = this.getAttribute('semantics');\n            var ssrcs = $(this).find('>source').map(function () {\n                return this.getAttribute('ssrc');\n            }).get();\n\n            if (ssrcs.length != 0) {\n                lines += 'a=ssrc-group:' + semantics + ' ' + ssrcs.join(' ') + '\\r\\n';\n            }\n        });\n        tmp = $(content).find('source[xmlns=\"urn:xmpp:jingle:apps:rtp:ssma:0\"]'); // can handle both >source and >description>source\n        tmp.each(function () {\n            var ssrc = $(this).attr('ssrc');\n            if(mySdp.containsSSRC(ssrc)){\n                /**\n                 * This happens when multiple participants change their streams at the same time and\n                 * ColibriFocus.modifySources have to wait for stable state. In the meantime multiple\n                 * addssrc are scheduled for update IQ. See\n                 */\n                console.warn(\"Got add stream request for my own ssrc: \"+ssrc);\n                return;\n            }\n            $(this).find('>parameter').each(function () {\n                lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');\n                if ($(this).attr('value') && $(this).attr('value').length)\n                    lines += ':' + $(this).attr('value');\n                lines += '\\r\\n';\n            });\n        });\n        sdp.media.forEach(function(media, idx) {\n            if (!SDPUtil.find_line(media, 'a=mid:' + name))\n                return;\n            sdp.media[idx] += lines;\n            if (!self.addssrc[idx]) self.addssrc[idx] = '';\n            self.addssrc[idx] += lines;\n        });\n        sdp.raw = sdp.session + sdp.media.join('');\n    });\n    this.modifySources();\n};\n\nJingleSession.prototype.removeSource = function (elem, fromJid) {\n\n    var self = this;\n    // FIXME: dirty waiting\n    if (!this.peerconnection.localDescription)\n    {\n        console.warn(\"removeSource - localDescription not ready yet\")\n        setTimeout(function()\n            {\n                self.removeSource(elem, fromJid);\n            },\n            200\n        );\n        return;\n    }\n\n    console.log('removessrc', new Date().getTime());\n    console.log('ice', this.peerconnection.iceConnectionState);\n    var sdp = new SDP(this.peerconnection.remoteDescription.sdp);\n    var mySdp = new SDP(this.peerconnection.localDescription.sdp);\n\n    $(elem).each(function (idx, content) {\n        var name = $(content).attr('name');\n        var lines = '';\n        tmp = $(content).find('ssrc-group[xmlns=\"urn:xmpp:jingle:apps:rtp:ssma:0\"]').each(function() {\n            var semantics = this.getAttribute('semantics');\n            var ssrcs = $(this).find('>source').map(function () {\n                return this.getAttribute('ssrc');\n            }).get();\n\n            if (ssrcs.length != 0) {\n                lines += 'a=ssrc-group:' + semantics + ' ' + ssrcs.join(' ') + '\\r\\n';\n            }\n        });\n        tmp = $(content).find('source[xmlns=\"urn:xmpp:jingle:apps:rtp:ssma:0\"]'); // can handle both >source and >description>source\n        tmp.each(function () {\n            var ssrc = $(this).attr('ssrc');\n            // This should never happen, but can be useful for bug detection\n            if(mySdp.containsSSRC(ssrc)){\n                console.error(\"Got remove stream request for my own ssrc: \"+ssrc);\n                return;\n            }\n            $(this).find('>parameter').each(function () {\n                lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');\n                if ($(this).attr('value') && $(this).attr('value').length)\n                    lines += ':' + $(this).attr('value');\n                lines += '\\r\\n';\n            });\n        });\n        sdp.media.forEach(function(media, idx) {\n            if (!SDPUtil.find_line(media, 'a=mid:' + name))\n                return;\n            sdp.media[idx] += lines;\n            if (!self.removessrc[idx]) self.removessrc[idx] = '';\n            self.removessrc[idx] += lines;\n        });\n        sdp.raw = sdp.session + sdp.media.join('');\n    });\n    this.modifySources();\n};\n\nJingleSession.prototype.modifySources = function (successCallback) {\n    var self = this;\n    if (this.peerconnection.signalingState == 'closed') return;\n    if (!(this.addssrc.length || this.removessrc.length || this.pendingop !== null || this.switchstreams)){\n        // There is nothing to do since scheduled job might have been executed by another succeeding call\n        this.setLocalDescription();\n        if(successCallback){\n            successCallback();\n        }\n        return;\n    }\n\n    // FIXME: this is a big hack\n    // https://code.google.com/p/webrtc/issues/detail?id=2688\n    // ^ has been fixed.\n    if (!(this.peerconnection.signalingState == 'stable' && this.peerconnection.iceConnectionState == 'connected')) {\n        console.warn('modifySources not yet', this.peerconnection.signalingState, this.peerconnection.iceConnectionState);\n        this.wait = true;\n        window.setTimeout(function() { self.modifySources(successCallback); }, 250);\n        return;\n    }\n    if (this.wait) {\n        window.setTimeout(function() { self.modifySources(successCallback); }, 2500);\n        this.wait = false;\n        return;\n    }\n\n    // Reset switch streams flag\n    this.switchstreams = false;\n\n    var sdp = new SDP(this.peerconnection.remoteDescription.sdp);\n\n    // add sources\n    this.addssrc.forEach(function(lines, idx) {\n        sdp.media[idx] += lines;\n    });\n    this.addssrc = [];\n\n    // remove sources\n    this.removessrc.forEach(function(lines, idx) {\n        lines = lines.split('\\r\\n');\n        lines.pop(); // remove empty last element;\n        lines.forEach(function(line) {\n            sdp.media[idx] = sdp.media[idx].replace(line + '\\r\\n', '');\n        });\n    });\n    this.removessrc = [];\n\n    // FIXME:\n    // this was a hack for the situation when only one peer exists\n    // in the conference.\n    // check if still required and remove\n    if (sdp.media[0])\n        sdp.media[0] = sdp.media[0].replace('a=recvonly', 'a=sendrecv');\n    if (sdp.media[1])\n        sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');\n\n    sdp.raw = sdp.session + sdp.media.join('');\n    this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'offer', sdp: sdp.raw}),\n        function() {\n\n            if(self.signalingState == 'closed') {\n                console.error(\"createAnswer attempt on closed state\");\n                return;\n            }\n\n            self.peerconnection.createAnswer(\n                function(modifiedAnswer) {\n                    // change video direction, see https://github.com/jitsi/jitmeet/issues/41\n                    if (self.pendingop !== null) {\n                        var sdp = new SDP(modifiedAnswer.sdp);\n                        if (sdp.media.length > 1) {\n                            switch(self.pendingop) {\n                                case 'mute':\n                                    sdp.media[1] = sdp.media[1].replace('a=sendrecv', 'a=recvonly');\n                                    break;\n                                case 'unmute':\n                                    sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');\n                                    break;\n                            }\n                            sdp.raw = sdp.session + sdp.media.join('');\n                            modifiedAnswer.sdp = sdp.raw;\n                        }\n                        self.pendingop = null;\n                    }\n\n                    // FIXME: pushing down an answer while ice connection state\n                    // is still checking is bad...\n                    //console.log(self.peerconnection.iceConnectionState);\n\n                    // trying to work around another chrome bug\n                    //modifiedAnswer.sdp = modifiedAnswer.sdp.replace(/a=setup:active/g, 'a=setup:actpass');\n                    self.peerconnection.setLocalDescription(modifiedAnswer,\n                        function() {\n                            //console.log('modified setLocalDescription ok');\n                            self.setLocalDescription();\n                            if(successCallback){\n                                successCallback();\n                            }\n                        },\n                        function(error) {\n                            console.error('modified setLocalDescription failed', error);\n                        }\n                    );\n                },\n                function(error) {\n                    console.error('modified answer failed', error);\n                }\n            );\n        },\n        function(error) {\n            console.error('modify failed', error);\n        }\n    );\n};\n\n/**\n * Switches video streams.\n * @param new_stream new stream that will be used as video of this session.\n * @param oldStream old video stream of this session.\n * @param success_callback callback executed after successful stream switch.\n */\nJingleSession.prototype.switchStreams = function (new_stream, oldStream, success_callback) {\n\n    var self = this;\n\n    // Remember SDP to figure out added/removed SSRCs\n    var oldSdp = null;\n    if(self.peerconnection) {\n        if(self.peerconnection.localDescription) {\n            oldSdp = new SDP(self.peerconnection.localDescription.sdp);\n        }\n        self.peerconnection.removeStream(oldStream, true);\n        self.peerconnection.addStream(new_stream);\n    }\n\n    RTC.switchVideoStreams(new_stream, oldStream);\n\n    // Conference is not active\n    if(!oldSdp || !self.peerconnection) {\n        success_callback();\n        return;\n    }\n\n    self.switchstreams = true;\n    self.modifySources(function() {\n        console.log('modify sources done');\n\n        success_callback();\n\n        var newSdp = new SDP(self.peerconnection.localDescription.sdp);\n        console.log(\"SDPs\", oldSdp, newSdp);\n        self.notifyMySSRCUpdate(oldSdp, newSdp);\n    });\n};\n\n/**\n * Figures out added/removed ssrcs and send update IQs.\n * @param old_sdp SDP object for old description.\n * @param new_sdp SDP object for new description.\n */\nJingleSession.prototype.notifyMySSRCUpdate = function (old_sdp, new_sdp) {\n\n    if (!(this.peerconnection.signalingState == 'stable' &&\n        this.peerconnection.iceConnectionState == 'connected')){\n        console.log(\"Too early to send updates\");\n        return;\n    }\n\n    // send source-remove IQ.\n    sdpDiffer = new SDPDiffer(new_sdp, old_sdp);\n    var remove = $iq({to: this.peerjid, type: 'set'})\n        .c('jingle', {\n            xmlns: 'urn:xmpp:jingle:1',\n            action: 'source-remove',\n            initiator: this.initiator,\n            sid: this.sid\n        }\n    );\n    var removed = sdpDiffer.toJingle(remove);\n    if (removed) {\n        this.connection.sendIQ(remove,\n            function (res) {\n                console.info('got remove result', res);\n            },\n            function (err) {\n                console.error('got remove error', err);\n            }\n        );\n    } else {\n        console.log('removal not necessary');\n    }\n\n    // send source-add IQ.\n    var sdpDiffer = new SDPDiffer(old_sdp, new_sdp);\n    var add = $iq({to: this.peerjid, type: 'set'})\n        .c('jingle', {\n            xmlns: 'urn:xmpp:jingle:1',\n            action: 'source-add',\n            initiator: this.initiator,\n            sid: this.sid\n        }\n    );\n    var added = sdpDiffer.toJingle(add);\n    if (added) {\n        this.connection.sendIQ(add,\n            function (res) {\n                console.info('got add result', res);\n            },\n            function (err) {\n                console.error('got add error', err);\n            }\n        );\n    } else {\n        console.log('addition not necessary');\n    }\n};\n\n/**\n * Determines whether the (local) video is mute i.e. all video tracks are\n * disabled.\n *\n * @return <tt>true</tt> if the (local) video is mute i.e. all video tracks are\n * disabled; otherwise, <tt>false</tt>\n */\nJingleSession.prototype.isVideoMute = function () {\n    var tracks = RTC.localVideo.getVideoTracks();\n    var mute = true;\n\n    for (var i = 0; i < tracks.length; ++i) {\n        if (tracks[i].enabled) {\n            mute = false;\n            break;\n        }\n    }\n    return mute;\n};\n\n/**\n * Mutes/unmutes the (local) video i.e. enables/disables all video tracks.\n *\n * @param mute <tt>true</tt> to mute the (local) video i.e. to disable all video\n * tracks; otherwise, <tt>false</tt>\n * @param callback a function to be invoked with <tt>mute</tt> after all video\n * tracks have been enabled/disabled. The function may, optionally, return\n * another function which is to be invoked after the whole mute/unmute operation\n * has completed successfully.\n * @param options an object which specifies optional arguments such as the\n * <tt>boolean</tt> key <tt>byUser</tt> with default value <tt>true</tt> which\n * specifies whether the method was initiated in response to a user command (in\n * contrast to an automatic decision made by the application logic)\n */\nJingleSession.prototype.setVideoMute = function (mute, callback, options) {\n    var byUser;\n\n    if (options) {\n        byUser = options.byUser;\n        if (typeof byUser === 'undefined') {\n            byUser = true;\n        }\n    } else {\n        byUser = true;\n    }\n    // The user's command to mute the (local) video takes precedence over any\n    // automatic decision made by the application logic.\n    if (byUser) {\n        this.videoMuteByUser = mute;\n    } else if (this.videoMuteByUser) {\n        return;\n    }\n\n    var self = this;\n    var localCallback = function (mute) {\n        self.connection.emuc.addVideoInfoToPresence(mute);\n        self.connection.emuc.sendPresence();\n        return callback(mute)\n    };\n\n    if (mute == RTC.localVideo.isMuted())\n    {\n        // Even if no change occurs, the specified callback is to be executed.\n        // The specified callback may, optionally, return a successCallback\n        // which is to be executed as well.\n        var successCallback = localCallback(mute);\n\n        if (successCallback) {\n            successCallback();\n        }\n    } else {\n        RTC.localVideo.setMute(!mute);\n\n        this.hardMuteVideo(mute);\n\n        this.modifySources(localCallback(mute));\n    }\n};\n\n// SDP-based mute by going recvonly/sendrecv\n// FIXME: should probably black out the screen as well\nJingleSession.prototype.toggleVideoMute = function (callback) {\n    this.service.setVideoMute(RTC.localVideo.isMuted(), callback);\n};\n\nJingleSession.prototype.hardMuteVideo = function (muted) {\n    this.pendingop = muted ? 'mute' : 'unmute';\n};\n\nJingleSession.prototype.sendMute = function (muted, content) {\n    var info = $iq({to: this.peerjid,\n        type: 'set'})\n        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n            action: 'session-info',\n            initiator: this.initiator,\n            sid: this.sid });\n    info.c(muted ? 'mute' : 'unmute', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});\n    info.attrs({'creator': this.me == this.initiator ? 'creator' : 'responder'});\n    if (content) {\n        info.attrs({'name': content});\n    }\n    this.connection.send(info);\n};\n\nJingleSession.prototype.sendRinging = function () {\n    var info = $iq({to: this.peerjid,\n        type: 'set'})\n        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',\n            action: 'session-info',\n            initiator: this.initiator,\n            sid: this.sid });\n    info.c('ringing', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});\n    this.connection.send(info);\n};\n\nJingleSession.prototype.getStats = function (interval) {\n    var self = this;\n    var recv = {audio: 0, video: 0};\n    var lost = {audio: 0, video: 0};\n    var lastrecv = {audio: 0, video: 0};\n    var lastlost = {audio: 0, video: 0};\n    var loss = {audio: 0, video: 0};\n    var delta = {audio: 0, video: 0};\n    this.statsinterval = window.setInterval(function () {\n        if (self && self.peerconnection && self.peerconnection.getStats) {\n            self.peerconnection.getStats(function (stats) {\n                var results = stats.result();\n                // TODO: there are so much statistics you can get from this..\n                for (var i = 0; i < results.length; ++i) {\n                    if (results[i].type == 'ssrc') {\n                        var packetsrecv = results[i].stat('packetsReceived');\n                        var packetslost = results[i].stat('packetsLost');\n                        if (packetsrecv && packetslost) {\n                            packetsrecv = parseInt(packetsrecv, 10);\n                            packetslost = parseInt(packetslost, 10);\n\n                            if (results[i].stat('googFrameRateReceived')) {\n                                lastlost.video = lost.video;\n                                lastrecv.video = recv.video;\n                                recv.video = packetsrecv;\n                                lost.video = packetslost;\n                            } else {\n                                lastlost.audio = lost.audio;\n                                lastrecv.audio = recv.audio;\n                                recv.audio = packetsrecv;\n                                lost.audio = packetslost;\n                            }\n                        }\n                    }\n                }\n                delta.audio = recv.audio - lastrecv.audio;\n                delta.video = recv.video - lastrecv.video;\n                loss.audio = (delta.audio > 0) ? Math.ceil(100 * (lost.audio - lastlost.audio) / delta.audio) : 0;\n                loss.video = (delta.video > 0) ? Math.ceil(100 * (lost.video - lastlost.video) / delta.video) : 0;\n                $(document).trigger('packetloss.jingle', [self.sid, loss]);\n            });\n        }\n    }, interval || 3000);\n    return this.statsinterval;\n};\n\nJingleSession.onJingleError = function (session, error)\n{\n    console.error(\"Jingle error\", error);\n}\n\nJingleSession.onJingleFatalError = function (session, error)\n{\n    this.service.sessionTerminated = true;\n    connection.emuc.doLeave();\n    UI.messageHandler.showError(  \"Sorry\",\n        \"Internal application error[setRemoteDescription]\");\n}\n\nJingleSession.prototype.setLocalDescription = function () {\n    // put our ssrcs into presence so other clients can identify our stream\n    var newssrcs = [];\n    var media = simulcast.parseMedia(this.peerconnection.localDescription);\n    media.forEach(function (media) {\n\n        if(Object.keys(media.sources).length > 0) {\n            // TODO(gp) maybe exclude FID streams?\n            Object.keys(media.sources).forEach(function (ssrc) {\n                newssrcs.push({\n                    'ssrc': ssrc,\n                    'type': media.type,\n                    'direction': media.direction\n                });\n            });\n        }\n        else if(this.localStreamsSSRC && this.localStreamsSSRC[media.type])\n        {\n            newssrcs.push({\n                'ssrc': this.localStreamsSSRC[media.type],\n                'type': media.type,\n                'direction': media.direction\n            });\n        }\n\n    });\n\n    console.log('new ssrcs', newssrcs);\n\n    // Have to clear presence map to get rid of removed streams\n    this.connection.emuc.clearPresenceMedia();\n\n    if (newssrcs.length > 0) {\n        for (var i = 1; i <= newssrcs.length; i ++) {\n            // Change video type to screen\n            if (newssrcs[i-1].type === 'video' && desktopsharing.isUsingScreenStream()) {\n                newssrcs[i-1].type = 'screen';\n            }\n            this.connection.emuc.addMediaToPresence(i,\n                newssrcs[i-1].type, newssrcs[i-1].ssrc, newssrcs[i-1].direction);\n        }\n\n        this.connection.emuc.sendPresence();\n    }\n}\n\n// an attempt to work around https://github.com/jitsi/jitmeet/issues/32\nfunction sendKeyframe(pc) {\n    console.log('sendkeyframe', pc.iceConnectionState);\n    if (pc.iceConnectionState !== 'connected') return; // safe...\n    pc.setRemoteDescription(\n        pc.remoteDescription,\n        function () {\n            pc.createAnswer(\n                function (modifiedAnswer) {\n                    pc.setLocalDescription(\n                        modifiedAnswer,\n                        function () {\n                            // noop\n                        },\n                        function (error) {\n                            console.log('triggerKeyframe setLocalDescription failed', error);\n                            UI.messageHandler.showError();\n                        }\n                    );\n                },\n                function (error) {\n                    console.log('triggerKeyframe createAnswer failed', error);\n                    UI.messageHandler.showError();\n                }\n            );\n        },\n        function (error) {\n            console.log('triggerKeyframe setRemoteDescription failed', error);\n            UI.messageHandler.showError();\n        }\n    );\n}\n\n\nJingleSession.prototype.remoteStreamAdded = function (data) {\n    var self = this;\n    var thessrc;\n\n    // look up an associated JID for a stream id\n    if (data.stream.id && data.stream.id.indexOf('mixedmslabel') === -1) {\n        // look only at a=ssrc: and _not_ at a=ssrc-group: lines\n\n        var ssrclines\n            = SDPUtil.find_lines(this.peerconnection.remoteDescription.sdp, 'a=ssrc:');\n        ssrclines = ssrclines.filter(function (line) {\n            // NOTE(gp) previously we filtered on the mslabel, but that property\n            // is not always present.\n            // return line.indexOf('mslabel:' + data.stream.label) !== -1;\n\n            return ((line.indexOf('msid:' + data.stream.id) !== -1));\n        });\n        if (ssrclines.length) {\n            thessrc = ssrclines[0].substring(7).split(' ')[0];\n\n            // We signal our streams (through Jingle to the focus) before we set\n            // our presence (through which peers associate remote streams to\n            // jids). So, it might arrive that a remote stream is added but\n            // ssrc2jid is not yet updated and thus data.peerjid cannot be\n            // successfully set. Here we wait for up to a second for the\n            // presence to arrive.\n\n            if (!ssrc2jid[thessrc]) {\n                // TODO(gp) limit wait duration to 1 sec.\n                setTimeout(function(d) {\n                    return function() {\n                        self.remoteStreamAdded(d);\n                    }\n                }(data), 250);\n                return;\n            }\n\n            // ok to overwrite the one from focus? might save work in colibri.js\n            console.log('associated jid', ssrc2jid[thessrc], data.peerjid);\n            if (ssrc2jid[thessrc]) {\n                data.peerjid = ssrc2jid[thessrc];\n            }\n        }\n    }\n\n    //TODO: this code should be removed when firefox implement multistream support\n    if(RTC.getBrowserType() == RTCBrowserType.RTC_BROWSER_FIREFOX)\n    {\n        if((notReceivedSSRCs.length == 0) ||\n            !ssrc2jid[notReceivedSSRCs[notReceivedSSRCs.length - 1]])\n        {\n            // TODO(gp) limit wait duration to 1 sec.\n            setTimeout(function(d) {\n                return function() {\n                    self.remoteStreamAdded(d);\n                }\n            }(data), 250);\n            return;\n        }\n\n        thessrc = notReceivedSSRCs.pop();\n        if (ssrc2jid[thessrc]) {\n            data.peerjid = ssrc2jid[thessrc];\n        }\n    }\n\n    RTC.createRemoteStream(data, this.sid, thessrc);\n\n    var isVideo = data.stream.getVideoTracks().length > 0;\n    // an attempt to work around https://github.com/jitsi/jitmeet/issues/32\n    if (isVideo &&\n        data.peerjid && this.peerjid === data.peerjid &&\n        data.stream.getVideoTracks().length === 0 &&\n        RTC.localVideo.getTracks().length > 0) {\n        window.setTimeout(function () {\n            sendKeyframe(self.peerconnection);\n        }, 3000);\n    }\n}\n\nmodule.exports = JingleSession;","/* jshint -W117 */\nvar SDPUtil = require(\"./SDPUtil\");\n\n// SDP STUFF\nfunction SDP(sdp) {\n    this.media = sdp.split('\\r\\nm=');\n    for (var i = 1; i < this.media.length; i++) {\n        this.media[i] = 'm=' + this.media[i];\n        if (i != this.media.length - 1) {\n            this.media[i] += '\\r\\n';\n        }\n    }\n    this.session = this.media.shift() + '\\r\\n';\n    this.raw = this.session + this.media.join('');\n}\n/**\n * Returns map of MediaChannel mapped per channel idx.\n */\nSDP.prototype.getMediaSsrcMap = function() {\n    var self = this;\n    var media_ssrcs = {};\n    var tmp;\n    for (var mediaindex = 0; mediaindex < self.media.length; mediaindex++) {\n        tmp = SDPUtil.find_lines(self.media[mediaindex], 'a=ssrc:');\n        var mid = SDPUtil.parse_mid(SDPUtil.find_line(self.media[mediaindex], 'a=mid:'));\n        var media = {\n            mediaindex: mediaindex,\n            mid: mid,\n            ssrcs: {},\n            ssrcGroups: []\n        };\n        media_ssrcs[mediaindex] = media;\n        tmp.forEach(function (line) {\n            var linessrc = line.substring(7).split(' ')[0];\n            // allocate new ChannelSsrc\n            if(!media.ssrcs[linessrc]) {\n                media.ssrcs[linessrc] = {\n                    ssrc: linessrc,\n                    lines: []\n                };\n            }\n            media.ssrcs[linessrc].lines.push(line);\n        });\n        tmp = SDPUtil.find_lines(self.media[mediaindex], 'a=ssrc-group:');\n        tmp.forEach(function(line){\n            var semantics = line.substr(0, idx).substr(13);\n            var ssrcs = line.substr(14 + semantics.length).split(' ');\n            if (ssrcs.length != 0) {\n                media.ssrcGroups.push({\n                    semantics: semantics,\n                    ssrcs: ssrcs\n                });\n            }\n        });\n    }\n    return media_ssrcs;\n};\n/**\n * Returns <tt>true</tt> if this SDP contains given SSRC.\n * @param ssrc the ssrc to check.\n * @returns {boolean} <tt>true</tt> if this SDP contains given SSRC.\n */\nSDP.prototype.containsSSRC = function(ssrc) {\n    var medias = this.getMediaSsrcMap();\n    var contains = false;\n    Object.keys(medias).forEach(function(mediaindex){\n        var media = medias[mediaindex];\n        //console.log(\"Check\", channel, ssrc);\n        if(Object.keys(media.ssrcs).indexOf(ssrc) != -1){\n            contains = true;\n        }\n    });\n    return contains;\n};\n\n\n// remove iSAC and CN from SDP\nSDP.prototype.mangle = function () {\n    var i, j, mline, lines, rtpmap, newdesc;\n    for (i = 0; i < this.media.length; i++) {\n        lines = this.media[i].split('\\r\\n');\n        lines.pop(); // remove empty last element\n        mline = SDPUtil.parse_mline(lines.shift());\n        if (mline.media != 'audio')\n            continue;\n        newdesc = '';\n        mline.fmt.length = 0;\n        for (j = 0; j < lines.length; j++) {\n            if (lines[j].substr(0, 9) == 'a=rtpmap:') {\n                rtpmap = SDPUtil.parse_rtpmap(lines[j]);\n                if (rtpmap.name == 'CN' || rtpmap.name == 'ISAC')\n                    continue;\n                mline.fmt.push(rtpmap.id);\n                newdesc += lines[j] + '\\r\\n';\n            } else {\n                newdesc += lines[j] + '\\r\\n';\n            }\n        }\n        this.media[i] = SDPUtil.build_mline(mline) + '\\r\\n';\n        this.media[i] += newdesc;\n    }\n    this.raw = this.session + this.media.join('');\n};\n\n// remove lines matching prefix from session section\nSDP.prototype.removeSessionLines = function(prefix) {\n    var self = this;\n    var lines = SDPUtil.find_lines(this.session, prefix);\n    lines.forEach(function(line) {\n        self.session = self.session.replace(line + '\\r\\n', '');\n    });\n    this.raw = this.session + this.media.join('');\n    return lines;\n}\n// remove lines matching prefix from a media section specified by mediaindex\n// TODO: non-numeric mediaindex could match mid\nSDP.prototype.removeMediaLines = function(mediaindex, prefix) {\n    var self = this;\n    var lines = SDPUtil.find_lines(this.media[mediaindex], prefix);\n    lines.forEach(function(line) {\n        self.media[mediaindex] = self.media[mediaindex].replace(line + '\\r\\n', '');\n    });\n    this.raw = this.session + this.media.join('');\n    return lines;\n}\n\n// add content's to a jingle element\nSDP.prototype.toJingle = function (elem, thecreator, ssrcs) {\n//    console.log(\"SSRC\" + ssrcs[\"audio\"] + \" - \" + ssrcs[\"video\"]);\n    var i, j, k, mline, ssrc, rtpmap, tmp, line, lines;\n    var self = this;\n    // new bundle plan\n    if (SDPUtil.find_line(this.session, 'a=group:')) {\n        lines = SDPUtil.find_lines(this.session, 'a=group:');\n        for (i = 0; i < lines.length; i++) {\n            tmp = lines[i].split(' ');\n            var semantics = tmp.shift().substr(8);\n            elem.c('group', {xmlns: 'urn:xmpp:jingle:apps:grouping:0', semantics:semantics});\n            for (j = 0; j < tmp.length; j++) {\n                elem.c('content', {name: tmp[j]}).up();\n            }\n            elem.up();\n        }\n    }\n    for (i = 0; i < this.media.length; i++) {\n        mline = SDPUtil.parse_mline(this.media[i].split('\\r\\n')[0]);\n        if (!(mline.media === 'audio' ||\n              mline.media === 'video' ||\n              mline.media === 'application'))\n        {\n            continue;\n        }\n        if (SDPUtil.find_line(this.media[i], 'a=ssrc:')) {\n            ssrc = SDPUtil.find_line(this.media[i], 'a=ssrc:').substring(7).split(' ')[0]; // take the first\n        } else {\n            if(ssrcs && ssrcs[mline.media])\n            {\n                ssrc = ssrcs[mline.media];\n            }\n            else\n                ssrc = false;\n        }\n\n        elem.c('content', {creator: thecreator, name: mline.media});\n        if (SDPUtil.find_line(this.media[i], 'a=mid:')) {\n            // prefer identifier from a=mid if present\n            var mid = SDPUtil.parse_mid(SDPUtil.find_line(this.media[i], 'a=mid:'));\n            elem.attrs({ name: mid });\n        }\n\n        if (SDPUtil.find_line(this.media[i], 'a=rtpmap:').length)\n        {\n            elem.c('description',\n                {xmlns: 'urn:xmpp:jingle:apps:rtp:1',\n                    media: mline.media });\n            if (ssrc) {\n                elem.attrs({ssrc: ssrc});\n            }\n            for (j = 0; j < mline.fmt.length; j++) {\n                rtpmap = SDPUtil.find_line(this.media[i], 'a=rtpmap:' + mline.fmt[j]);\n                elem.c('payload-type', SDPUtil.parse_rtpmap(rtpmap));\n                // put any 'a=fmtp:' + mline.fmt[j] lines into <param name=foo value=bar/>\n                if (SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j])) {\n                    tmp = SDPUtil.parse_fmtp(SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j]));\n                    for (k = 0; k < tmp.length; k++) {\n                        elem.c('parameter', tmp[k]).up();\n                    }\n                }\n                this.RtcpFbToJingle(i, elem, mline.fmt[j]); // XEP-0293 -- map a=rtcp-fb\n\n                elem.up();\n            }\n            if (SDPUtil.find_line(this.media[i], 'a=crypto:', this.session)) {\n                elem.c('encryption', {required: 1});\n                var crypto = SDPUtil.find_lines(this.media[i], 'a=crypto:', this.session);\n                crypto.forEach(function(line) {\n                    elem.c('crypto', SDPUtil.parse_crypto(line)).up();\n                });\n                elem.up(); // end of encryption\n            }\n\n            if (ssrc) {\n                // new style mapping\n                elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });\n                // FIXME: group by ssrc and support multiple different ssrcs\n                var ssrclines = SDPUtil.find_lines(this.media[i], 'a=ssrc:');\n                if(ssrclines.length > 0) {\n                    ssrclines.forEach(function (line) {\n                        idx = line.indexOf(' ');\n                        var linessrc = line.substr(0, idx).substr(7);\n                        if (linessrc != ssrc) {\n                            elem.up();\n                            ssrc = linessrc;\n                            elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });\n                        }\n                        var kv = line.substr(idx + 1);\n                        elem.c('parameter');\n                        if (kv.indexOf(':') == -1) {\n                            elem.attrs({ name: kv });\n                        } else {\n                            elem.attrs({ name: kv.split(':', 2)[0] });\n                            elem.attrs({ value: kv.split(':', 2)[1] });\n                        }\n                        elem.up();\n                    });\n                    elem.up();\n                }\n                else\n                {\n                    elem.up();\n                    elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });\n                    elem.c('parameter');\n                    elem.attrs({name: \"cname\", value:Math.random().toString(36).substring(7)});\n                    elem.up();\n                    var msid = null;\n                    if(mline.media == \"audio\")\n                    {\n                        msid = RTC.localAudio.getId();\n                    }\n                    else\n                    {\n                        msid = RTC.localVideo.getId();\n                    }\n                    if(msid != null)\n                    {\n                        msid = msid.replace(/[\\{,\\}]/g,\"\");\n                        elem.c('parameter');\n                        elem.attrs({name: \"msid\", value:msid});\n                        elem.up();\n                        elem.c('parameter');\n                        elem.attrs({name: \"mslabel\", value:msid});\n                        elem.up();\n                        elem.c('parameter');\n                        elem.attrs({name: \"label\", value:msid});\n                        elem.up();\n                        elem.up();\n                    }\n\n\n                }\n\n                // XEP-0339 handle ssrc-group attributes\n                var ssrc_group_lines = SDPUtil.find_lines(this.media[i], 'a=ssrc-group:');\n                ssrc_group_lines.forEach(function(line) {\n                    idx = line.indexOf(' ');\n                    var semantics = line.substr(0, idx).substr(13);\n                    var ssrcs = line.substr(14 + semantics.length).split(' ');\n                    if (ssrcs.length != 0) {\n                        elem.c('ssrc-group', { semantics: semantics, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });\n                        ssrcs.forEach(function(ssrc) {\n                            elem.c('source', { ssrc: ssrc })\n                                .up();\n                        });\n                        elem.up();\n                    }\n                });\n            }\n\n            if (SDPUtil.find_line(this.media[i], 'a=rtcp-mux')) {\n                elem.c('rtcp-mux').up();\n            }\n\n            // XEP-0293 -- map a=rtcp-fb:*\n            this.RtcpFbToJingle(i, elem, '*');\n\n            // XEP-0294\n            if (SDPUtil.find_line(this.media[i], 'a=extmap:')) {\n                lines = SDPUtil.find_lines(this.media[i], 'a=extmap:');\n                for (j = 0; j < lines.length; j++) {\n                    tmp = SDPUtil.parse_extmap(lines[j]);\n                    elem.c('rtp-hdrext', { xmlns: 'urn:xmpp:jingle:apps:rtp:rtp-hdrext:0',\n                        uri: tmp.uri,\n                        id: tmp.value });\n                    if (tmp.hasOwnProperty('direction')) {\n                        switch (tmp.direction) {\n                            case 'sendonly':\n                                elem.attrs({senders: 'responder'});\n                                break;\n                            case 'recvonly':\n                                elem.attrs({senders: 'initiator'});\n                                break;\n                            case 'sendrecv':\n                                elem.attrs({senders: 'both'});\n                                break;\n                            case 'inactive':\n                                elem.attrs({senders: 'none'});\n                                break;\n                        }\n                    }\n                    // TODO: handle params\n                    elem.up();\n                }\n            }\n            elem.up(); // end of description\n        }\n\n        // map ice-ufrag/pwd, dtls fingerprint, candidates\n        this.TransportToJingle(i, elem);\n\n        if (SDPUtil.find_line(this.media[i], 'a=sendrecv', this.session)) {\n            elem.attrs({senders: 'both'});\n        } else if (SDPUtil.find_line(this.media[i], 'a=sendonly', this.session)) {\n            elem.attrs({senders: 'initiator'});\n        } else if (SDPUtil.find_line(this.media[i], 'a=recvonly', this.session)) {\n            elem.attrs({senders: 'responder'});\n        } else if (SDPUtil.find_line(this.media[i], 'a=inactive', this.session)) {\n            elem.attrs({senders: 'none'});\n        }\n        if (mline.port == '0') {\n            // estos hack to reject an m-line\n            elem.attrs({senders: 'rejected'});\n        }\n        elem.up(); // end of content\n    }\n    elem.up();\n    return elem;\n};\n\nSDP.prototype.TransportToJingle = function (mediaindex, elem) {\n    var i = mediaindex;\n    var tmp;\n    var self = this;\n    elem.c('transport');\n\n    // XEP-0343 DTLS/SCTP\n    if (SDPUtil.find_line(this.media[mediaindex], 'a=sctpmap:').length)\n    {\n        var sctpmap = SDPUtil.find_line(\n            this.media[i], 'a=sctpmap:', self.session);\n        if (sctpmap)\n        {\n            var sctpAttrs = SDPUtil.parse_sctpmap(sctpmap);\n            elem.c('sctpmap',\n                {\n                    xmlns: 'urn:xmpp:jingle:transports:dtls-sctp:1',\n                    number: sctpAttrs[0], /* SCTP port */\n                    protocol: sctpAttrs[1], /* protocol */\n                });\n            // Optional stream count attribute\n            if (sctpAttrs.length > 2)\n                elem.attrs({ streams: sctpAttrs[2]});\n            elem.up();\n        }\n    }\n    // XEP-0320\n    var fingerprints = SDPUtil.find_lines(this.media[mediaindex], 'a=fingerprint:', this.session);\n    fingerprints.forEach(function(line) {\n        tmp = SDPUtil.parse_fingerprint(line);\n        tmp.xmlns = 'urn:xmpp:jingle:apps:dtls:0';\n        elem.c('fingerprint').t(tmp.fingerprint);\n        delete tmp.fingerprint;\n        line = SDPUtil.find_line(self.media[mediaindex], 'a=setup:', self.session);\n        if (line) {\n            tmp.setup = line.substr(8);\n        }\n        elem.attrs(tmp);\n        elem.up(); // end of fingerprint\n    });\n    tmp = SDPUtil.iceparams(this.media[mediaindex], this.session);\n    if (tmp) {\n        tmp.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';\n        elem.attrs(tmp);\n        // XEP-0176\n        if (SDPUtil.find_line(this.media[mediaindex], 'a=candidate:', this.session)) { // add any a=candidate lines\n            var lines = SDPUtil.find_lines(this.media[mediaindex], 'a=candidate:', this.session);\n            lines.forEach(function (line) {\n                elem.c('candidate', SDPUtil.candidateToJingle(line)).up();\n            });\n        }\n    }\n    elem.up(); // end of transport\n}\n\nSDP.prototype.RtcpFbToJingle = function (mediaindex, elem, payloadtype) { // XEP-0293\n    var lines = SDPUtil.find_lines(this.media[mediaindex], 'a=rtcp-fb:' + payloadtype);\n    lines.forEach(function (line) {\n        var tmp = SDPUtil.parse_rtcpfb(line);\n        if (tmp.type == 'trr-int') {\n            elem.c('rtcp-fb-trr-int', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', value: tmp.params[0]});\n            elem.up();\n        } else {\n            elem.c('rtcp-fb', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', type: tmp.type});\n            if (tmp.params.length > 0) {\n                elem.attrs({'subtype': tmp.params[0]});\n            }\n            elem.up();\n        }\n    });\n};\n\nSDP.prototype.RtcpFbFromJingle = function (elem, payloadtype) { // XEP-0293\n    var media = '';\n    var tmp = elem.find('>rtcp-fb-trr-int[xmlns=\"urn:xmpp:jingle:apps:rtp:rtcp-fb:0\"]');\n    if (tmp.length) {\n        media += 'a=rtcp-fb:' + '*' + ' ' + 'trr-int' + ' ';\n        if (tmp.attr('value')) {\n            media += tmp.attr('value');\n        } else {\n            media += '0';\n        }\n        media += '\\r\\n';\n    }\n    tmp = elem.find('>rtcp-fb[xmlns=\"urn:xmpp:jingle:apps:rtp:rtcp-fb:0\"]');\n    tmp.each(function () {\n        media += 'a=rtcp-fb:' + payloadtype + ' ' + $(this).attr('type');\n        if ($(this).attr('subtype')) {\n            media += ' ' + $(this).attr('subtype');\n        }\n        media += '\\r\\n';\n    });\n    return media;\n};\n\n// construct an SDP from a jingle stanza\nSDP.prototype.fromJingle = function (jingle) {\n    var self = this;\n    this.raw = 'v=0\\r\\n' +\n        'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\\r\\n' +// FIXME\n        's=-\\r\\n' +\n        't=0 0\\r\\n';\n    // http://tools.ietf.org/html/draft-ietf-mmusic-sdp-bundle-negotiation-04#section-8\n    if ($(jingle).find('>group[xmlns=\"urn:xmpp:jingle:apps:grouping:0\"]').length) {\n        $(jingle).find('>group[xmlns=\"urn:xmpp:jingle:apps:grouping:0\"]').each(function (idx, group) {\n            var contents = $(group).find('>content').map(function (idx, content) {\n                return content.getAttribute('name');\n            }).get();\n            if (contents.length > 0) {\n                self.raw += 'a=group:' + (group.getAttribute('semantics') || group.getAttribute('type')) + ' ' + contents.join(' ') + '\\r\\n';\n            }\n        });\n    }\n\n    this.session = this.raw;\n    jingle.find('>content').each(function () {\n        var m = self.jingle2media($(this));\n        self.media.push(m);\n    });\n\n    // reconstruct msid-semantic -- apparently not necessary\n    /*\n     var msid = SDPUtil.parse_ssrc(this.raw);\n     if (msid.hasOwnProperty('mslabel')) {\n     this.session += \"a=msid-semantic: WMS \" + msid.mslabel + \"\\r\\n\";\n     }\n     */\n\n    this.raw = this.session + this.media.join('');\n};\n\n// translate a jingle content element into an an SDP media part\nSDP.prototype.jingle2media = function (content) {\n    var media = '',\n        desc = content.find('description'),\n        ssrc = desc.attr('ssrc'),\n        self = this,\n        tmp;\n    var sctp = content.find(\n        '>transport>sctpmap[xmlns=\"urn:xmpp:jingle:transports:dtls-sctp:1\"]');\n\n    tmp = { media: desc.attr('media') };\n    tmp.port = '1';\n    if (content.attr('senders') == 'rejected') {\n        // estos hack to reject an m-line.\n        tmp.port = '0';\n    }\n    if (content.find('>transport>fingerprint').length || desc.find('encryption').length) {\n        if (sctp.length)\n            tmp.proto = 'DTLS/SCTP';\n        else\n            tmp.proto = 'RTP/SAVPF';\n    } else {\n        tmp.proto = 'RTP/AVPF';\n    }\n    if (!sctp.length)\n    {\n        tmp.fmt = desc.find('payload-type').map(\n            function () { return this.getAttribute('id'); }).get();\n        media += SDPUtil.build_mline(tmp) + '\\r\\n';\n    }\n    else\n    {\n        media += 'm=application 1 DTLS/SCTP ' + sctp.attr('number') + '\\r\\n';\n        media += 'a=sctpmap:' + sctp.attr('number') +\n            ' ' + sctp.attr('protocol');\n\n        var streamCount = sctp.attr('streams');\n        if (streamCount)\n            media += ' ' + streamCount + '\\r\\n';\n        else\n            media += '\\r\\n';\n    }\n\n    media += 'c=IN IP4 0.0.0.0\\r\\n';\n    if (!sctp.length)\n        media += 'a=rtcp:1 IN IP4 0.0.0.0\\r\\n';\n    tmp = content.find('>transport[xmlns=\"urn:xmpp:jingle:transports:ice-udp:1\"]');\n    if (tmp.length) {\n        if (tmp.attr('ufrag')) {\n            media += SDPUtil.build_iceufrag(tmp.attr('ufrag')) + '\\r\\n';\n        }\n        if (tmp.attr('pwd')) {\n            media += SDPUtil.build_icepwd(tmp.attr('pwd')) + '\\r\\n';\n        }\n        tmp.find('>fingerprint').each(function () {\n            // FIXME: check namespace at some point\n            media += 'a=fingerprint:' + this.getAttribute('hash');\n            media += ' ' + $(this).text();\n            media += '\\r\\n';\n            if (this.getAttribute('setup')) {\n                media += 'a=setup:' + this.getAttribute('setup') + '\\r\\n';\n            }\n        });\n    }\n    switch (content.attr('senders')) {\n        case 'initiator':\n            media += 'a=sendonly\\r\\n';\n            break;\n        case 'responder':\n            media += 'a=recvonly\\r\\n';\n            break;\n        case 'none':\n            media += 'a=inactive\\r\\n';\n            break;\n        case 'both':\n            media += 'a=sendrecv\\r\\n';\n            break;\n    }\n    media += 'a=mid:' + content.attr('name') + '\\r\\n';\n\n    // <description><rtcp-mux/></description>\n    // see http://code.google.com/p/libjingle/issues/detail?id=309 -- no spec though\n    // and http://mail.jabber.org/pipermail/jingle/2011-December/001761.html\n    if (desc.find('rtcp-mux').length) {\n        media += 'a=rtcp-mux\\r\\n';\n    }\n\n    if (desc.find('encryption').length) {\n        desc.find('encryption>crypto').each(function () {\n            media += 'a=crypto:' + this.getAttribute('tag');\n            media += ' ' + this.getAttribute('crypto-suite');\n            media += ' ' + this.getAttribute('key-params');\n            if (this.getAttribute('session-params')) {\n                media += ' ' + this.getAttribute('session-params');\n            }\n            media += '\\r\\n';\n        });\n    }\n    desc.find('payload-type').each(function () {\n        media += SDPUtil.build_rtpmap(this) + '\\r\\n';\n        if ($(this).find('>parameter').length) {\n            media += 'a=fmtp:' + this.getAttribute('id') + ' ';\n            media += $(this).find('parameter').map(function () { return (this.getAttribute('name') ? (this.getAttribute('name') + '=') : '') + this.getAttribute('value'); }).get().join('; ');\n            media += '\\r\\n';\n        }\n        // xep-0293\n        media += self.RtcpFbFromJingle($(this), this.getAttribute('id'));\n    });\n\n    // xep-0293\n    media += self.RtcpFbFromJingle(desc, '*');\n\n    // xep-0294\n    tmp = desc.find('>rtp-hdrext[xmlns=\"urn:xmpp:jingle:apps:rtp:rtp-hdrext:0\"]');\n    tmp.each(function () {\n        media += 'a=extmap:' + this.getAttribute('id') + ' ' + this.getAttribute('uri') + '\\r\\n';\n    });\n\n    content.find('>transport[xmlns=\"urn:xmpp:jingle:transports:ice-udp:1\"]>candidate').each(function () {\n        media += SDPUtil.candidateFromJingle(this);\n    });\n\n    // XEP-0339 handle ssrc-group attributes\n    tmp = content.find('description>ssrc-group[xmlns=\"urn:xmpp:jingle:apps:rtp:ssma:0\"]').each(function() {\n        var semantics = this.getAttribute('semantics');\n        var ssrcs = $(this).find('>source').map(function() {\n            return this.getAttribute('ssrc');\n        }).get();\n\n        if (ssrcs.length != 0) {\n            media += 'a=ssrc-group:' + semantics + ' ' + ssrcs.join(' ') + '\\r\\n';\n        }\n    });\n\n    tmp = content.find('description>source[xmlns=\"urn:xmpp:jingle:apps:rtp:ssma:0\"]');\n    tmp.each(function () {\n        var ssrc = this.getAttribute('ssrc');\n        $(this).find('>parameter').each(function () {\n            media += 'a=ssrc:' + ssrc + ' ' + this.getAttribute('name');\n            if (this.getAttribute('value') && this.getAttribute('value').length)\n                media += ':' + this.getAttribute('value');\n            media += '\\r\\n';\n        });\n    });\n\n    return media;\n};\n\n\nmodule.exports = SDP;\n\n","function SDPDiffer(mySDP, otherSDP) {\n    this.mySDP = mySDP;\n    this.otherSDP = otherSDP;\n}\n\n/**\n * Returns map of MediaChannel that contains only media not contained in <tt>otherSdp</tt>. Mapped by channel idx.\n * @param otherSdp the other SDP to check ssrc with.\n */\nSDPDiffer.prototype.getNewMedia = function() {\n\n    // this could be useful in Array.prototype.\n    function arrayEquals(array) {\n        // if the other array is a falsy value, return\n        if (!array)\n            return false;\n\n        // compare lengths - can save a lot of time\n        if (this.length != array.length)\n            return false;\n\n        for (var i = 0, l=this.length; i < l; i++) {\n            // Check if we have nested arrays\n            if (this[i] instanceof Array && array[i] instanceof Array) {\n                // recurse into the nested arrays\n                if (!this[i].equals(array[i]))\n                    return false;\n            }\n            else if (this[i] != array[i]) {\n                // Warning - two different object instances will never be equal: {x:20} != {x:20}\n                return false;\n            }\n        }\n        return true;\n    }\n\n    var myMedias = this.mySDP.getMediaSsrcMap();\n    var othersMedias = this.otherSDP.getMediaSsrcMap();\n    var newMedia = {};\n    Object.keys(othersMedias).forEach(function(othersMediaIdx) {\n        var myMedia = myMedias[othersMediaIdx];\n        var othersMedia = othersMedias[othersMediaIdx];\n        if(!myMedia && othersMedia) {\n            // Add whole channel\n            newMedia[othersMediaIdx] = othersMedia;\n            return;\n        }\n        // Look for new ssrcs accross the channel\n        Object.keys(othersMedia.ssrcs).forEach(function(ssrc) {\n            if(Object.keys(myMedia.ssrcs).indexOf(ssrc) === -1) {\n                // Allocate channel if we've found ssrc that doesn't exist in our channel\n                if(!newMedia[othersMediaIdx]){\n                    newMedia[othersMediaIdx] = {\n                        mediaindex: othersMedia.mediaindex,\n                        mid: othersMedia.mid,\n                        ssrcs: {},\n                        ssrcGroups: []\n                    };\n                }\n                newMedia[othersMediaIdx].ssrcs[ssrc] = othersMedia.ssrcs[ssrc];\n            }\n        });\n\n        // Look for new ssrc groups across the channels\n        othersMedia.ssrcGroups.forEach(function(otherSsrcGroup){\n\n            // try to match the other ssrc-group with an ssrc-group of ours\n            var matched = false;\n            for (var i = 0; i < myMedia.ssrcGroups.length; i++) {\n                var mySsrcGroup = myMedia.ssrcGroups[i];\n                if (otherSsrcGroup.semantics == mySsrcGroup.semantics\n                    && arrayEquals.apply(otherSsrcGroup.ssrcs, [mySsrcGroup.ssrcs])) {\n\n                    matched = true;\n                    break;\n                }\n            }\n\n            if (!matched) {\n                // Allocate channel if we've found an ssrc-group that doesn't\n                // exist in our channel\n\n                if(!newMedia[othersMediaIdx]){\n                    newMedia[othersMediaIdx] = {\n                        mediaindex: othersMedia.mediaindex,\n                        mid: othersMedia.mid,\n                        ssrcs: {},\n                        ssrcGroups: []\n                    };\n                }\n                newMedia[othersMediaIdx].ssrcGroups.push(otherSsrcGroup);\n            }\n        });\n    });\n    return newMedia;\n};\n\n/**\n * Sends SSRC update IQ.\n * @param sdpMediaSsrcs SSRCs map obtained from SDP.getNewMedia. Cntains SSRCs to add/remove.\n * @param sid session identifier that will be put into the IQ.\n * @param initiator initiator identifier.\n * @param toJid destination Jid\n * @param isAdd indicates if this is remove or add operation.\n */\nSDPDiffer.prototype.toJingle = function(modify) {\n    var sdpMediaSsrcs = this.getNewMedia();\n    var self = this;\n\n    // FIXME: only announce video ssrcs since we mix audio and dont need\n    //      the audio ssrcs therefore\n    var modified = false;\n    Object.keys(sdpMediaSsrcs).forEach(function(mediaindex){\n        modified = true;\n        var media = sdpMediaSsrcs[mediaindex];\n        modify.c('content', {name: media.mid});\n\n        modify.c('description', {xmlns:'urn:xmpp:jingle:apps:rtp:1', media: media.mid});\n        // FIXME: not completly sure this operates on blocks and / or handles different ssrcs correctly\n        // generate sources from lines\n        Object.keys(media.ssrcs).forEach(function(ssrcNum) {\n            var mediaSsrc = media.ssrcs[ssrcNum];\n            modify.c('source', { xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });\n            modify.attrs({ssrc: mediaSsrc.ssrc});\n            // iterate over ssrc lines\n            mediaSsrc.lines.forEach(function (line) {\n                var idx = line.indexOf(' ');\n                var kv = line.substr(idx + 1);\n                modify.c('parameter');\n                if (kv.indexOf(':') == -1) {\n                    modify.attrs({ name: kv });\n                } else {\n                    modify.attrs({ name: kv.split(':', 2)[0] });\n                    modify.attrs({ value: kv.split(':', 2)[1] });\n                }\n                modify.up(); // end of parameter\n            });\n            modify.up(); // end of source\n        });\n\n        // generate source groups from lines\n        media.ssrcGroups.forEach(function(ssrcGroup) {\n            if (ssrcGroup.ssrcs.length != 0) {\n\n                modify.c('ssrc-group', {\n                    semantics: ssrcGroup.semantics,\n                    xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0'\n                });\n\n                ssrcGroup.ssrcs.forEach(function (ssrc) {\n                    modify.c('source', { ssrc: ssrc })\n                        .up(); // end of source\n                });\n                modify.up(); // end of ssrc-group\n            }\n        });\n\n        modify.up(); // end of description\n        modify.up(); // end of content\n    });\n\n    return modified;\n};\n\nmodule.exports = SDPDiffer;","SDPUtil = {\n    iceparams: function (mediadesc, sessiondesc) {\n        var data = null;\n        if (SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc) &&\n            SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc)) {\n            data = {\n                ufrag: SDPUtil.parse_iceufrag(SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc)),\n                pwd: SDPUtil.parse_icepwd(SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc))\n            };\n        }\n        return data;\n    },\n    parse_iceufrag: function (line) {\n        return line.substring(12);\n    },\n    build_iceufrag: function (frag) {\n        return 'a=ice-ufrag:' + frag;\n    },\n    parse_icepwd: function (line) {\n        return line.substring(10);\n    },\n    build_icepwd: function (pwd) {\n        return 'a=ice-pwd:' + pwd;\n    },\n    parse_mid: function (line) {\n        return line.substring(6);\n    },\n    parse_mline: function (line) {\n        var parts = line.substring(2).split(' '),\n            data = {};\n        data.media = parts.shift();\n        data.port = parts.shift();\n        data.proto = parts.shift();\n        if (parts[parts.length - 1] === '') { // trailing whitespace\n            parts.pop();\n        }\n        data.fmt = parts;\n        return data;\n    },\n    build_mline: function (mline) {\n        return 'm=' + mline.media + ' ' + mline.port + ' ' + mline.proto + ' ' + mline.fmt.join(' ');\n    },\n    parse_rtpmap: function (line) {\n        var parts = line.substring(9).split(' '),\n            data = {};\n        data.id = parts.shift();\n        parts = parts[0].split('/');\n        data.name = parts.shift();\n        data.clockrate = parts.shift();\n        data.channels = parts.length ? parts.shift() : '1';\n        return data;\n    },\n    /**\n     * Parses SDP line \"a=sctpmap:...\" and extracts SCTP port from it.\n     * @param line eg. \"a=sctpmap:5000 webrtc-datachannel\"\n     * @returns [SCTP port number, protocol, streams]\n     */\n    parse_sctpmap: function (line)\n    {\n        var parts = line.substring(10).split(' ');\n        var sctpPort = parts[0];\n        var protocol = parts[1];\n        // Stream count is optional\n        var streamCount = parts.length > 2 ? parts[2] : null;\n        return [sctpPort, protocol, streamCount];// SCTP port\n    },\n    build_rtpmap: function (el) {\n        var line = 'a=rtpmap:' + el.getAttribute('id') + ' ' + el.getAttribute('name') + '/' + el.getAttribute('clockrate');\n        if (el.getAttribute('channels') && el.getAttribute('channels') != '1') {\n            line += '/' + el.getAttribute('channels');\n        }\n        return line;\n    },\n    parse_crypto: function (line) {\n        var parts = line.substring(9).split(' '),\n            data = {};\n        data.tag = parts.shift();\n        data['crypto-suite'] = parts.shift();\n        data['key-params'] = parts.shift();\n        if (parts.length) {\n            data['session-params'] = parts.join(' ');\n        }\n        return data;\n    },\n    parse_fingerprint: function (line) { // RFC 4572\n        var parts = line.substring(14).split(' '),\n            data = {};\n        data.hash = parts.shift();\n        data.fingerprint = parts.shift();\n        // TODO assert that fingerprint satisfies 2UHEX *(\":\" 2UHEX) ?\n        return data;\n    },\n    parse_fmtp: function (line) {\n        var parts = line.split(' '),\n            i, key, value,\n            data = [];\n        parts.shift();\n        parts = parts.join(' ').split(';');\n        for (i = 0; i < parts.length; i++) {\n            key = parts[i].split('=')[0];\n            while (key.length && key[0] == ' ') {\n                key = key.substring(1);\n            }\n            value = parts[i].split('=')[1];\n            if (key && value) {\n                data.push({name: key, value: value});\n            } else if (key) {\n                // rfc 4733 (DTMF) style stuff\n                data.push({name: '', value: key});\n            }\n        }\n        return data;\n    },\n    parse_icecandidate: function (line) {\n        var candidate = {},\n            elems = line.split(' ');\n        candidate.foundation = elems[0].substring(12);\n        candidate.component = elems[1];\n        candidate.protocol = elems[2].toLowerCase();\n        candidate.priority = elems[3];\n        candidate.ip = elems[4];\n        candidate.port = elems[5];\n        // elems[6] => \"typ\"\n        candidate.type = elems[7];\n        candidate.generation = 0; // default value, may be overwritten below\n        for (var i = 8; i < elems.length; i += 2) {\n            switch (elems[i]) {\n                case 'raddr':\n                    candidate['rel-addr'] = elems[i + 1];\n                    break;\n                case 'rport':\n                    candidate['rel-port'] = elems[i + 1];\n                    break;\n                case 'generation':\n                    candidate.generation = elems[i + 1];\n                    break;\n                case 'tcptype':\n                    candidate.tcptype = elems[i + 1];\n                    break;\n                default: // TODO\n                    console.log('parse_icecandidate not translating \"' + elems[i] + '\" = \"' + elems[i + 1] + '\"');\n            }\n        }\n        candidate.network = '1';\n        candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random\n        return candidate;\n    },\n    build_icecandidate: function (cand) {\n        var line = ['a=candidate:' + cand.foundation, cand.component, cand.protocol, cand.priority, cand.ip, cand.port, 'typ', cand.type].join(' ');\n        line += ' ';\n        switch (cand.type) {\n            case 'srflx':\n            case 'prflx':\n            case 'relay':\n                if (cand.hasOwnAttribute('rel-addr') && cand.hasOwnAttribute('rel-port')) {\n                    line += 'raddr';\n                    line += ' ';\n                    line += cand['rel-addr'];\n                    line += ' ';\n                    line += 'rport';\n                    line += ' ';\n                    line += cand['rel-port'];\n                    line += ' ';\n                }\n                break;\n        }\n        if (cand.hasOwnAttribute('tcptype')) {\n            line += 'tcptype';\n            line += ' ';\n            line += cand.tcptype;\n            line += ' ';\n        }\n        line += 'generation';\n        line += ' ';\n        line += cand.hasOwnAttribute('generation') ? cand.generation : '0';\n        return line;\n    },\n    parse_ssrc: function (desc) {\n        // proprietary mapping of a=ssrc lines\n        // TODO: see \"Jingle RTP Source Description\" by Juberti and P. Thatcher on google docs\n        // and parse according to that\n        var lines = desc.split('\\r\\n'),\n            data = {};\n        for (var i = 0; i < lines.length; i++) {\n            if (lines[i].substring(0, 7) == 'a=ssrc:') {\n                var idx = lines[i].indexOf(' ');\n                data[lines[i].substr(idx + 1).split(':', 2)[0]] = lines[i].substr(idx + 1).split(':', 2)[1];\n            }\n        }\n        return data;\n    },\n    parse_rtcpfb: function (line) {\n        var parts = line.substr(10).split(' ');\n        var data = {};\n        data.pt = parts.shift();\n        data.type = parts.shift();\n        data.params = parts;\n        return data;\n    },\n    parse_extmap: function (line) {\n        var parts = line.substr(9).split(' ');\n        var data = {};\n        data.value = parts.shift();\n        if (data.value.indexOf('/') != -1) {\n            data.direction = data.value.substr(data.value.indexOf('/') + 1);\n            data.value = data.value.substr(0, data.value.indexOf('/'));\n        } else {\n            data.direction = 'both';\n        }\n        data.uri = parts.shift();\n        data.params = parts;\n        return data;\n    },\n    find_line: function (haystack, needle, sessionpart) {\n        var lines = haystack.split('\\r\\n');\n        for (var i = 0; i < lines.length; i++) {\n            if (lines[i].substring(0, needle.length) == needle) {\n                return lines[i];\n            }\n        }\n        if (!sessionpart) {\n            return false;\n        }\n        // search session part\n        lines = sessionpart.split('\\r\\n');\n        for (var j = 0; j < lines.length; j++) {\n            if (lines[j].substring(0, needle.length) == needle) {\n                return lines[j];\n            }\n        }\n        return false;\n    },\n    find_lines: function (haystack, needle, sessionpart) {\n        var lines = haystack.split('\\r\\n'),\n            needles = [];\n        for (var i = 0; i < lines.length; i++) {\n            if (lines[i].substring(0, needle.length) == needle)\n                needles.push(lines[i]);\n        }\n        if (needles.length || !sessionpart) {\n            return needles;\n        }\n        // search session part\n        lines = sessionpart.split('\\r\\n');\n        for (var j = 0; j < lines.length; j++) {\n            if (lines[j].substring(0, needle.length) == needle) {\n                needles.push(lines[j]);\n            }\n        }\n        return needles;\n    },\n    candidateToJingle: function (line) {\n        // a=candidate:2979166662 1 udp 2113937151 192.168.2.100 57698 typ host generation 0\n        //      <candidate component=... foundation=... generation=... id=... ip=... network=... port=... priority=... protocol=... type=.../>\n        if (line.indexOf('candidate:') === 0) {\n            line = 'a=' + line;\n        } else if (line.substring(0, 12) != 'a=candidate:') {\n            console.log('parseCandidate called with a line that is not a candidate line');\n            console.log(line);\n            return null;\n        }\n        if (line.substring(line.length - 2) == '\\r\\n') // chomp it\n            line = line.substring(0, line.length - 2);\n        var candidate = {},\n            elems = line.split(' '),\n            i;\n        if (elems[6] != 'typ') {\n            console.log('did not find typ in the right place');\n            console.log(line);\n            return null;\n        }\n        candidate.foundation = elems[0].substring(12);\n        candidate.component = elems[1];\n        candidate.protocol = elems[2].toLowerCase();\n        candidate.priority = elems[3];\n        candidate.ip = elems[4];\n        candidate.port = elems[5];\n        // elems[6] => \"typ\"\n        candidate.type = elems[7];\n\n        candidate.generation = '0'; // default, may be overwritten below\n        for (i = 8; i < elems.length; i += 2) {\n            switch (elems[i]) {\n                case 'raddr':\n                    candidate['rel-addr'] = elems[i + 1];\n                    break;\n                case 'rport':\n                    candidate['rel-port'] = elems[i + 1];\n                    break;\n                case 'generation':\n                    candidate.generation = elems[i + 1];\n                    break;\n                case 'tcptype':\n                    candidate.tcptype = elems[i + 1];\n                    break;\n                default: // TODO\n                    console.log('not translating \"' + elems[i] + '\" = \"' + elems[i + 1] + '\"');\n            }\n        }\n        candidate.network = '1';\n        candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random\n        return candidate;\n    },\n    candidateFromJingle: function (cand) {\n        var line = 'a=candidate:';\n        line += cand.getAttribute('foundation');\n        line += ' ';\n        line += cand.getAttribute('component');\n        line += ' ';\n        line += cand.getAttribute('protocol'); //.toUpperCase(); // chrome M23 doesn't like this\n        line += ' ';\n        line += cand.getAttribute('priority');\n        line += ' ';\n        line += cand.getAttribute('ip');\n        line += ' ';\n        line += cand.getAttribute('port');\n        line += ' ';\n        line += 'typ';\n        line += ' ' + cand.getAttribute('type');\n        line += ' ';\n        switch (cand.getAttribute('type')) {\n            case 'srflx':\n            case 'prflx':\n            case 'relay':\n                if (cand.getAttribute('rel-addr') && cand.getAttribute('rel-port')) {\n                    line += 'raddr';\n                    line += ' ';\n                    line += cand.getAttribute('rel-addr');\n                    line += ' ';\n                    line += 'rport';\n                    line += ' ';\n                    line += cand.getAttribute('rel-port');\n                    line += ' ';\n                }\n                break;\n        }\n        if (cand.getAttribute('protocol').toLowerCase() == 'tcp') {\n            line += 'tcptype';\n            line += ' ';\n            line += cand.getAttribute('tcptype');\n            line += ' ';\n        }\n        line += 'generation';\n        line += ' ';\n        line += cand.getAttribute('generation') || '0';\n        return line + '\\r\\n';\n    }\n};\nmodule.exports = SDPUtil;","function TraceablePeerConnection(ice_config, constraints) {\n    var self = this;\n    var RTCPeerconnection = navigator.mozGetUserMedia ? mozRTCPeerConnection : webkitRTCPeerConnection;\n    this.peerconnection = new RTCPeerconnection(ice_config, constraints);\n    this.updateLog = [];\n    this.stats = {};\n    this.statsinterval = null;\n    this.maxstats = 0; // limit to 300 values, i.e. 5 minutes; set to 0 to disable\n\n    // override as desired\n    this.trace = function (what, info) {\n        //console.warn('WTRACE', what, info);\n        self.updateLog.push({\n            time: new Date(),\n            type: what,\n            value: info || \"\"\n        });\n    };\n    this.onicecandidate = null;\n    this.peerconnection.onicecandidate = function (event) {\n        self.trace('onicecandidate', JSON.stringify(event.candidate, null, ' '));\n        if (self.onicecandidate !== null) {\n            self.onicecandidate(event);\n        }\n    };\n    this.onaddstream = null;\n    this.peerconnection.onaddstream = function (event) {\n        self.trace('onaddstream', event.stream.id);\n        if (self.onaddstream !== null) {\n            self.onaddstream(event);\n        }\n    };\n    this.onremovestream = null;\n    this.peerconnection.onremovestream = function (event) {\n        self.trace('onremovestream', event.stream.id);\n        if (self.onremovestream !== null) {\n            self.onremovestream(event);\n        }\n    };\n    this.onsignalingstatechange = null;\n    this.peerconnection.onsignalingstatechange = function (event) {\n        self.trace('onsignalingstatechange', self.signalingState);\n        if (self.onsignalingstatechange !== null) {\n            self.onsignalingstatechange(event);\n        }\n    };\n    this.oniceconnectionstatechange = null;\n    this.peerconnection.oniceconnectionstatechange = function (event) {\n        self.trace('oniceconnectionstatechange', self.iceConnectionState);\n        if (self.oniceconnectionstatechange !== null) {\n            self.oniceconnectionstatechange(event);\n        }\n    };\n    this.onnegotiationneeded = null;\n    this.peerconnection.onnegotiationneeded = function (event) {\n        self.trace('onnegotiationneeded');\n        if (self.onnegotiationneeded !== null) {\n            self.onnegotiationneeded(event);\n        }\n    };\n    self.ondatachannel = null;\n    this.peerconnection.ondatachannel = function (event) {\n        self.trace('ondatachannel', event);\n        if (self.ondatachannel !== null) {\n            self.ondatachannel(event);\n        }\n    };\n    if (!navigator.mozGetUserMedia && this.maxstats) {\n        this.statsinterval = window.setInterval(function() {\n            self.peerconnection.getStats(function(stats) {\n                var results = stats.result();\n                for (var i = 0; i < results.length; ++i) {\n                    //console.log(results[i].type, results[i].id, results[i].names())\n                    var now = new Date();\n                    results[i].names().forEach(function (name) {\n                        var id = results[i].id + '-' + name;\n                        if (!self.stats[id]) {\n                            self.stats[id] = {\n                                startTime: now,\n                                endTime: now,\n                                values: [],\n                                times: []\n                            };\n                        }\n                        self.stats[id].values.push(results[i].stat(name));\n                        self.stats[id].times.push(now.getTime());\n                        if (self.stats[id].values.length > self.maxstats) {\n                            self.stats[id].values.shift();\n                            self.stats[id].times.shift();\n                        }\n                        self.stats[id].endTime = now;\n                    });\n                }\n            });\n\n        }, 1000);\n    }\n};\n\ndumpSDP = function(description) {\n    return 'type: ' + description.type + '\\r\\n' + description.sdp;\n}\n\nif (TraceablePeerConnection.prototype.__defineGetter__ !== undefined) {\n    TraceablePeerConnection.prototype.__defineGetter__('signalingState', function() { return this.peerconnection.signalingState; });\n    TraceablePeerConnection.prototype.__defineGetter__('iceConnectionState', function() { return this.peerconnection.iceConnectionState; });\n    TraceablePeerConnection.prototype.__defineGetter__('localDescription', function() {\n        var publicLocalDescription = simulcast.reverseTransformLocalDescription(this.peerconnection.localDescription);\n        return publicLocalDescription;\n    });\n    TraceablePeerConnection.prototype.__defineGetter__('remoteDescription', function() {\n        var publicRemoteDescription = simulcast.reverseTransformRemoteDescription(this.peerconnection.remoteDescription);\n        return publicRemoteDescription;\n    });\n}\n\nTraceablePeerConnection.prototype.addStream = function (stream) {\n    this.trace('addStream', stream.id);\n    simulcast.resetSender();\n    try\n    {\n        this.peerconnection.addStream(stream);\n    }\n    catch (e)\n    {\n        console.error(e);\n        return;\n    }\n};\n\nTraceablePeerConnection.prototype.removeStream = function (stream, stopStreams) {\n    this.trace('removeStream', stream.id);\n    simulcast.resetSender();\n    if(stopStreams) {\n        stream.getAudioTracks().forEach(function (track) {\n            track.stop();\n        });\n        stream.getVideoTracks().forEach(function (track) {\n            track.stop();\n        });\n    }\n    this.peerconnection.removeStream(stream);\n};\n\nTraceablePeerConnection.prototype.createDataChannel = function (label, opts) {\n    this.trace('createDataChannel', label, opts);\n    return this.peerconnection.createDataChannel(label, opts);\n};\n\nTraceablePeerConnection.prototype.setLocalDescription = function (description, successCallback, failureCallback) {\n    var self = this;\n    description = simulcast.transformLocalDescription(description);\n    this.trace('setLocalDescription', dumpSDP(description));\n    this.peerconnection.setLocalDescription(description,\n        function () {\n            self.trace('setLocalDescriptionOnSuccess');\n            successCallback();\n        },\n        function (err) {\n            self.trace('setLocalDescriptionOnFailure', err);\n            failureCallback(err);\n        }\n    );\n    /*\n     if (this.statsinterval === null && this.maxstats > 0) {\n     // start gathering stats\n     }\n     */\n};\n\nTraceablePeerConnection.prototype.setRemoteDescription = function (description, successCallback, failureCallback) {\n    var self = this;\n    description = simulcast.transformRemoteDescription(description);\n    this.trace('setRemoteDescription', dumpSDP(description));\n    this.peerconnection.setRemoteDescription(description,\n        function () {\n            self.trace('setRemoteDescriptionOnSuccess');\n            successCallback();\n        },\n        function (err) {\n            self.trace('setRemoteDescriptionOnFailure', err);\n            failureCallback(err);\n        }\n    );\n    /*\n     if (this.statsinterval === null && this.maxstats > 0) {\n     // start gathering stats\n     }\n     */\n};\n\nTraceablePeerConnection.prototype.close = function () {\n    this.trace('stop');\n    if (this.statsinterval !== null) {\n        window.clearInterval(this.statsinterval);\n        this.statsinterval = null;\n    }\n    this.peerconnection.close();\n};\n\nTraceablePeerConnection.prototype.createOffer = function (successCallback, failureCallback, constraints) {\n    var self = this;\n    this.trace('createOffer', JSON.stringify(constraints, null, ' '));\n    this.peerconnection.createOffer(\n        function (offer) {\n            self.trace('createOfferOnSuccess', dumpSDP(offer));\n            successCallback(offer);\n        },\n        function(err) {\n            self.trace('createOfferOnFailure', err);\n            failureCallback(err);\n        },\n        constraints\n    );\n};\n\nTraceablePeerConnection.prototype.createAnswer = function (successCallback, failureCallback, constraints) {\n    var self = this;\n    this.trace('createAnswer', JSON.stringify(constraints, null, ' '));\n    this.peerconnection.createAnswer(\n        function (answer) {\n            answer = simulcast.transformAnswer(answer);\n            self.trace('createAnswerOnSuccess', dumpSDP(answer));\n            successCallback(answer);\n        },\n        function(err) {\n            self.trace('createAnswerOnFailure', err);\n            failureCallback(err);\n        },\n        constraints\n    );\n};\n\nTraceablePeerConnection.prototype.addIceCandidate = function (candidate, successCallback, failureCallback) {\n    var self = this;\n    this.trace('addIceCandidate', JSON.stringify(candidate, null, ' '));\n    this.peerconnection.addIceCandidate(candidate);\n    /* maybe later\n     this.peerconnection.addIceCandidate(candidate,\n     function () {\n     self.trace('addIceCandidateOnSuccess');\n     successCallback();\n     },\n     function (err) {\n     self.trace('addIceCandidateOnFailure', err);\n     failureCallback(err);\n     }\n     );\n     */\n};\n\nTraceablePeerConnection.prototype.getStats = function(callback, errback) {\n    if (navigator.mozGetUserMedia) {\n        // ignore for now...\n        if(!errback)\n            errback = function () {\n\n            }\n        this.peerconnection.getStats(null,callback,errback);\n    } else {\n        this.peerconnection.getStats(callback);\n    }\n};\n\nmodule.exports = TraceablePeerConnection;\n\n","/* global $, $iq, config, connection, UI, messageHandler,\n roomName, sessionTerminated, Strophe, Util */\n/**\n * Contains logic responsible for enabling/disabling functionality available\n * only to moderator users.\n */\nvar connection = null;\nvar focusUserJid;\nvar getNextTimeout = Util.createExpBackoffTimer(1000);\nvar getNextErrorTimeout = Util.createExpBackoffTimer(1000);\n// External authentication stuff\nvar externalAuthEnabled = false;\n// Sip gateway can be enabled by configuring Jigasi host in config.js or\n// it will be enabled automatically if focus detects the component through\n// service discovery.\nvar sipGatewayEnabled = config.hosts.call_control !== undefined;\n\nvar Moderator = {\n    isModerator: function () {\n        return connection && connection.emuc.isModerator();\n    },\n\n    isPeerModerator: function (peerJid) {\n        return connection &&\n            connection.emuc.getMemberRole(peerJid) === 'moderator';\n    },\n\n    isExternalAuthEnabled: function () {\n        return externalAuthEnabled;\n    },\n\n    isSipGatewayEnabled: function () {\n        return sipGatewayEnabled;\n    },\n\n    setConnection: function (con) {\n        connection = con;\n    },\n\n    init: function (xmpp) {\n        this.xmppService = xmpp;\n        this.onLocalRoleChange = function (from, member, pres) {\n            UI.onModeratorStatusChanged(Moderator.isModerator());\n        };\n    },\n\n    onMucLeft: function (jid) {\n        console.info(\"Someone left is it focus ? \" + jid);\n        var resource = Strophe.getResourceFromJid(jid);\n        if (resource === 'focus' && !this.xmppService.sessionTerminated) {\n            console.info(\n                \"Focus has left the room - leaving conference\");\n            //hangUp();\n            // We'd rather reload to have everything re-initialized\n            // FIXME: show some message before reload\n            location.reload();\n        }\n    },\n    \n    setFocusUserJid: function (focusJid) {\n        if (!focusUserJid) {\n            focusUserJid = focusJid;\n            console.info(\"Focus jid set to: \" + focusUserJid);\n        }\n    },\n\n    getFocusUserJid: function () {\n        return focusUserJid;\n    },\n\n    getFocusComponent: function () {\n        // Get focus component address\n        var focusComponent = config.hosts.focus;\n        // If not specified use default: 'focus.domain'\n        if (!focusComponent) {\n            focusComponent = 'focus.' + config.hosts.domain;\n        }\n        return focusComponent;\n    },\n\n    createConferenceIq: function (roomName) {\n        // Generate create conference IQ\n        var elem = $iq({to: Moderator.getFocusComponent(), type: 'set'});\n        elem.c('conference', {\n            xmlns: 'http://jitsi.org/protocol/focus',\n            room: roomName\n        });\n        if (config.hosts.bridge !== undefined) {\n            elem.c(\n                'property',\n                { name: 'bridge', value: config.hosts.bridge})\n                .up();\n        }\n        // Tell the focus we have Jigasi configured\n        if (config.hosts.call_control !== undefined) {\n            elem.c(\n                'property',\n                { name: 'call_control', value: config.hosts.call_control})\n                .up();\n        }\n        if (config.channelLastN !== undefined) {\n            elem.c(\n                'property',\n                { name: 'channelLastN', value: config.channelLastN})\n                .up();\n        }\n        if (config.adaptiveLastN !== undefined) {\n            elem.c(\n                'property',\n                { name: 'adaptiveLastN', value: config.adaptiveLastN})\n                .up();\n        }\n        if (config.adaptiveSimulcast !== undefined) {\n            elem.c(\n                'property',\n                { name: 'adaptiveSimulcast', value: config.adaptiveSimulcast})\n                .up();\n        }\n        if (config.openSctp !== undefined) {\n            elem.c(\n                'property',\n                { name: 'openSctp', value: config.openSctp})\n                .up();\n        }\n        if (config.enableFirefoxSupport !== undefined) {\n            elem.c(\n                'property',\n                { name: 'enableFirefoxHacks',\n                    value: config.enableFirefoxSupport})\n                .up();\n        }\n        elem.up();\n        return elem;\n    },\n\n    parseConfigOptions: function (resultIq) {\n    \n        Moderator.setFocusUserJid(\n            $(resultIq).find('conference').attr('focusjid'));\n    \n        var extAuthParam\n            = $(resultIq).find('>conference>property[name=\\'externalAuth\\']');\n        if (extAuthParam.length) {\n            externalAuthEnabled = extAuthParam.attr('value') === 'true';\n        }\n    \n        console.info(\"External authentication enabled: \" + externalAuthEnabled);\n    \n        // Check if focus has auto-detected Jigasi component(this will be also\n        // included if we have passed our host from the config)\n        if ($(resultIq).find(\n            '>conference>property[name=\\'sipGatewayEnabled\\']').length) {\n            sipGatewayEnabled = true;\n        }\n    \n        console.info(\"Sip gateway enabled: \" + sipGatewayEnabled);\n    },\n\n    // FIXME: we need to show the fact that we're waiting for the focus\n    // to the user(or that focus is not available)\n    allocateConferenceFocus: function (roomName, callback) {\n        // Try to use focus user JID from the config\n        Moderator.setFocusUserJid(config.focusUserJid);\n        // Send create conference IQ\n        var iq = Moderator.createConferenceIq(roomName);\n        connection.sendIQ(\n            iq,\n            function (result) {\n                if ('true' === $(result).find('conference').attr('ready')) {\n                    // Reset both timers\n                    getNextTimeout(true);\n                    getNextErrorTimeout(true);\n                    // Setup config options\n                    Moderator.parseConfigOptions(result);\n                    // Exec callback\n                    callback();\n                } else {\n                    var waitMs = getNextTimeout();\n                    console.info(\"Waiting for the focus... \" + waitMs);\n                    // Reset error timeout\n                    getNextErrorTimeout(true);\n                    window.setTimeout(\n                        function () {\n                            Moderator.allocateConferenceFocus(\n                                roomName, callback);\n                        }, waitMs);\n                }\n            },\n            function (error) {\n                // Not authorized to create new room\n                if ($(error).find('>error>not-authorized').length) {\n                    console.warn(\"Unauthorized to start the conference\");\n                    UI.onAuthenticationRequired(function () {\n                        Moderator.allocateConferenceFocus(roomName, callback);\n                    });\n                    return;\n                }\n                var waitMs = getNextErrorTimeout();\n                console.error(\"Focus error, retry after \" + waitMs, error);\n                // Show message\n                UI.messageHandler.notify(\n                    'Conference focus', 'disconnected',\n                        Moderator.getFocusComponent() +\n                        ' not available - retry in ' +\n                        (waitMs / 1000) + ' sec');\n                // Reset response timeout\n                getNextTimeout(true);\n                window.setTimeout(\n                    function () {\n                        Moderator.allocateConferenceFocus(roomName, callback);\n                    }, waitMs);\n            }\n        );\n    },\n\n    getAuthUrl: function (roomName, urlCallback) {\n        var iq = $iq({to: Moderator.getFocusComponent(), type: 'get'});\n        iq.c('auth-url', {\n            xmlns: 'http://jitsi.org/protocol/focus',\n            room: roomName\n        });\n        connection.sendIQ(\n            iq,\n            function (result) {\n                var url = $(result).find('auth-url').attr('url');\n                if (url) {\n                    console.info(\"Got auth url: \" + url);\n                    urlCallback(url);\n                } else {\n                    console.error(\n                        \"Failed to get auth url fro mthe focus\", result);\n                }\n            },\n            function (error) {\n                console.error(\"Get auth url error\", error);\n            }\n        );\n    }\n};\n\nmodule.exports = Moderator;\n\n\n\n","/* global $, $iq, config, connection, focusMucJid, messageHandler, Moderator,\n   Toolbar, Util */\nvar Moderator = require(\"./moderator\");\n\n\nvar recordingToken = null;\nvar recordingEnabled;\n\n/**\n * Whether to use a jirecon component for recording, or use the videobridge\n * through COLIBRI.\n */\nvar useJirecon = (typeof config.hosts.jirecon != \"undefined\");\n\n/**\n * The ID of the jirecon recording session. Jirecon generates it when we\n * initially start recording, and it needs to be used in subsequent requests\n * to jirecon.\n */\nvar jireconRid = null;\n\nfunction setRecordingToken(token) {\n    recordingToken = token;\n}\n\nfunction setRecording(state, token, callback) {\n    if (useJirecon){\n        this.setRecordingJirecon(state, token, callback);\n    } else {\n        this.setRecordingColibri(state, token, callback);\n    }\n}\n\nfunction setRecordingJirecon(state, token, callback) {\n    if (state == recordingEnabled){\n        return;\n    }\n\n    var iq = $iq({to: config.hosts.jirecon, type: 'set'})\n        .c('recording', {xmlns: 'http://jitsi.org/protocol/jirecon',\n            action: state ? 'start' : 'stop',\n            mucjid: connection.emuc.roomjid});\n    if (!state){\n        iq.attrs({rid: jireconRid});\n    }\n\n    console.log('Start recording');\n\n    connection.sendIQ(\n        iq,\n        function (result) {\n            // TODO wait for an IQ with the real status, since this is\n            // provisional?\n            jireconRid = $(result).find('recording').attr('rid');\n            console.log('Recording ' + (state ? 'started' : 'stopped') +\n                '(jirecon)' + result);\n            recordingEnabled = state;\n            if (!state){\n                jireconRid = null;\n            }\n\n            callback(state);\n        },\n        function (error) {\n            console.log('Failed to start recording, error: ', error);\n            callback(recordingEnabled);\n        });\n}\n\n// Sends a COLIBRI message which enables or disables (according to 'state')\n// the recording on the bridge. Waits for the result IQ and calls 'callback'\n// with the new recording state, according to the IQ.\nfunction setRecordingColibri(state, token, callback) {\n    var elem = $iq({to: focusMucJid, type: 'set'});\n    elem.c('conference', {\n        xmlns: 'http://jitsi.org/protocol/colibri'\n    });\n    elem.c('recording', {state: state, token: token});\n\n    connection.sendIQ(elem,\n        function (result) {\n            console.log('Set recording \"', state, '\". Result:', result);\n            var recordingElem = $(result).find('>conference>recording');\n            var newState = ('true' === recordingElem.attr('state'));\n\n            recordingEnabled = newState;\n            callback(newState);\n        },\n        function (error) {\n            console.warn(error);\n            callback(recordingEnabled);\n        }\n    );\n}\n\nvar Recording = {\n    toggleRecording: function (tokenEmptyCallback,\n                               startingCallback, startedCallback) {\n        if (!Moderator.isModerator()) {\n            console.log(\n                    'non-focus, or conference not yet organized:' +\n                    ' not enabling recording');\n            return;\n        }\n\n        // Jirecon does not (currently) support a token.\n        if (!recordingToken && !useJirecon) {\n            tokenEmptyCallback(function (value) {\n                setRecordingToken(value);\n                this.toggleRecording();\n            });\n\n            return;\n        }\n\n        var oldState = recordingEnabled;\n        startingCallback(!oldState);\n        setRecording(!oldState,\n            recordingToken,\n            function (state) {\n                console.log(\"New recording state: \", state);\n                if (state === oldState) {\n                    // FIXME: new focus:\n                    // this will not work when moderator changes\n                    // during active session. Then it will assume that\n                    // recording status has changed to true, but it might have\n                    // been already true(and we only received actual status from\n                    // the focus).\n                    //\n                    // SO we start with status null, so that it is initialized\n                    // here and will fail only after second click, so if invalid\n                    // token was used we have to press the button twice before\n                    // current status will be fetched and token will be reset.\n                    //\n                    // Reliable way would be to return authentication error.\n                    // Or status update when moderator connects.\n                    // Or we have to stop recording session when current\n                    // moderator leaves the room.\n\n                    // Failed to change, reset the token because it might\n                    // have been wrong\n                    setRecordingToken(null);\n                }\n                startedCallback(state);\n\n            }\n        );\n    }\n\n}\n\nmodule.exports = Recording;","/* jshint -W117 */\n/* a simple MUC connection plugin\n * can only handle a single MUC room\n */\n\nvar bridgeIsDown = false;\n\nvar Moderator = require(\"./moderator\");\n\nmodule.exports = function(XMPP, eventEmitter) {\n    Strophe.addConnectionPlugin('emuc', {\n        connection: null,\n        roomjid: null,\n        myroomjid: null,\n        members: {},\n        list_members: [], // so we can elect a new focus\n        presMap: {},\n        preziMap: {},\n        joined: false,\n        isOwner: false,\n        role: null,\n        init: function (conn) {\n            this.connection = conn;\n        },\n        initPresenceMap: function (myroomjid) {\n            this.presMap['to'] = myroomjid;\n            this.presMap['xns'] = 'http://jabber.org/protocol/muc';\n        },\n        doJoin: function (jid, password) {\n            this.myroomjid = jid;\n\n            console.info(\"Joined MUC as \" + this.myroomjid);\n\n            this.initPresenceMap(this.myroomjid);\n\n            if (!this.roomjid) {\n                this.roomjid = Strophe.getBareJidFromJid(jid);\n                // add handlers (just once)\n                this.connection.addHandler(this.onPresence.bind(this), null, 'presence', null, null, this.roomjid, {matchBare: true});\n                this.connection.addHandler(this.onPresenceUnavailable.bind(this), null, 'presence', 'unavailable', null, this.roomjid, {matchBare: true});\n                this.connection.addHandler(this.onPresenceError.bind(this), null, 'presence', 'error', null, this.roomjid, {matchBare: true});\n                this.connection.addHandler(this.onMessage.bind(this), null, 'message', null, null, this.roomjid, {matchBare: true});\n            }\n            if (password !== undefined) {\n                this.presMap['password'] = password;\n            }\n            this.sendPresence();\n        },\n        doLeave: function () {\n            console.log(\"do leave\", this.myroomjid);\n            var pres = $pres({to: this.myroomjid, type: 'unavailable' });\n            this.presMap.length = 0;\n            this.connection.send(pres);\n        },\n        createNonAnonymousRoom: function () {\n            // http://xmpp.org/extensions/xep-0045.html#createroom-reserved\n\n            var getForm = $iq({type: 'get', to: this.roomjid})\n                .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'})\n                .c('x', {xmlns: 'jabber:x:data', type: 'submit'});\n\n            this.connection.sendIQ(getForm, function (form) {\n\n                if (!$(form).find(\n                        '>query>x[xmlns=\"jabber:x:data\"]' +\n                        '>field[var=\"muc#roomconfig_whois\"]').length) {\n\n                    console.error('non-anonymous rooms not supported');\n                    return;\n                }\n\n                var formSubmit = $iq({to: this.roomjid, type: 'set'})\n                    .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});\n\n                formSubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});\n\n                formSubmit.c('field', {'var': 'FORM_TYPE'})\n                    .c('value')\n                    .t('http://jabber.org/protocol/muc#roomconfig').up().up();\n\n                formSubmit.c('field', {'var': 'muc#roomconfig_whois'})\n                    .c('value').t('anyone').up().up();\n\n                this.connection.sendIQ(formSubmit);\n\n            }, function (error) {\n                console.error(\"Error getting room configuration form\");\n            });\n        },\n        onPresence: function (pres) {\n            var from = pres.getAttribute('from');\n\n            // What is this for? A workaround for something?\n            if (pres.getAttribute('type')) {\n                return true;\n            }\n\n            // Parse etherpad tag.\n            var etherpad = $(pres).find('>etherpad');\n            if (etherpad.length) {\n                if (config.etherpad_base && !Moderator.isModerator()) {\n                    UI.initEtherpad(etherpad.text());\n                }\n            }\n\n            // Parse prezi tag.\n            var presentation = $(pres).find('>prezi');\n            if (presentation.length) {\n                var url = presentation.attr('url');\n                var current = presentation.find('>current').text();\n\n                console.log('presentation info received from', from, url);\n\n                if (this.preziMap[from] == null) {\n                    this.preziMap[from] = url;\n\n                    $(document).trigger('presentationadded.muc', [from, url, current]);\n                }\n                else {\n                    $(document).trigger('gotoslide.muc', [from, url, current]);\n                }\n            }\n            else if (this.preziMap[from] != null) {\n                var url = this.preziMap[from];\n                delete this.preziMap[from];\n                $(document).trigger('presentationremoved.muc', [from, url]);\n            }\n\n            // Parse audio info tag.\n            var audioMuted = $(pres).find('>audiomuted');\n            if (audioMuted.length) {\n                $(document).trigger('audiomuted.muc', [from, audioMuted.text()]);\n            }\n\n            // Parse video info tag.\n            var videoMuted = $(pres).find('>videomuted');\n            if (videoMuted.length) {\n                $(document).trigger('videomuted.muc', [from, videoMuted.text()]);\n            }\n\n            var stats = $(pres).find('>stats');\n            if (stats.length) {\n                var statsObj = {};\n                Strophe.forEachChild(stats[0], \"stat\", function (el) {\n                    statsObj[el.getAttribute(\"name\")] = el.getAttribute(\"value\");\n                });\n                connectionquality.updateRemoteStats(from, statsObj);\n            }\n\n            // Parse status.\n            if ($(pres).find('>x[xmlns=\"http://jabber.org/protocol/muc#user\"]>status[code=\"201\"]').length) {\n                this.isOwner = true;\n                this.createNonAnonymousRoom();\n            }\n\n            // Parse roles.\n            var member = {};\n            member.show = $(pres).find('>show').text();\n            member.status = $(pres).find('>status').text();\n            var tmp = $(pres).find('>x[xmlns=\"http://jabber.org/protocol/muc#user\"]>item');\n            member.affiliation = tmp.attr('affiliation');\n            member.role = tmp.attr('role');\n\n            // Focus recognition\n            member.jid = tmp.attr('jid');\n            member.isFocus = false;\n            if (member.jid\n                && member.jid.indexOf(Moderator.getFocusUserJid() + \"/\") == 0) {\n                member.isFocus = true;\n            }\n\n            var nicktag = $(pres).find('>nick[xmlns=\"http://jabber.org/protocol/nick\"]');\n            member.displayName = (nicktag.length > 0 ? nicktag.html() : null);\n\n            if (from == this.myroomjid) {\n                if (member.affiliation == 'owner') this.isOwner = true;\n                if (this.role !== member.role) {\n                    this.role = member.role;\n                    if (Moderator.onLocalRoleChange)\n                        Moderator.onLocalRoleChange(from, member, pres);\n                    UI.onLocalRoleChange(from, member, pres);\n                }\n                if (!this.joined) {\n                    this.joined = true;\n                    eventEmitter.emit(XMPPEvents.MUC_JOINED, from, member);\n                    this.list_members.push(from);\n                }\n            } else if (this.members[from] === undefined) {\n                // new participant\n                this.members[from] = member;\n                this.list_members.push(from);\n                console.log('entered', from, member);\n                if (member.isFocus) {\n                    focusMucJid = from;\n                    console.info(\"Ignore focus: \" + from + \", real JID: \" + member.jid);\n                }\n                else {\n                    var id = $(pres).find('>userID').text();\n                    var email = $(pres).find('>email');\n                    if (email.length > 0) {\n                        id = email.text();\n                    }\n                    UI.onMucEntered(from, id, member.displayName);\n                    API.triggerEvent(\"participantJoined\", {jid: from});\n                }\n            } else {\n                // Presence update for existing participant\n                // Watch role change:\n                if (this.members[from].role != member.role) {\n                    this.members[from].role = member.role;\n                    UI.onMucRoleChanged(member.role, member.displayName);\n                }\n            }\n\n            // Always trigger presence to update bindings\n            $(document).trigger('presence.muc', [from, member, pres]);\n            this.parsePresence(from, member, pres);\n\n            // Trigger status message update\n            if (member.status) {\n                UI.onMucPresenceStatus(from, member);\n            }\n\n            return true;\n        },\n        onPresenceUnavailable: function (pres) {\n            var from = pres.getAttribute('from');\n            // Status code 110 indicates that this notification is \"self-presence\".\n            if (!$(pres).find('>x[xmlns=\"http://jabber.org/protocol/muc#user\"]>status[code=\"110\"]').length) {\n                delete this.members[from];\n                this.list_members.splice(this.list_members.indexOf(from), 1);\n                this.onParticipantLeft(from);\n            }\n            // If the status code is 110 this means we're leaving and we would like\n            // to remove everyone else from our view, so we trigger the event.\n            else if (this.list_members.length > 1) {\n                for (var i = 0; i < this.list_members.length; i++) {\n                    var member = this.list_members[i];\n                    delete this.members[i];\n                    this.list_members.splice(i, 1);\n                    this.onParticipantLeft(member);\n                }\n            }\n            if ($(pres).find('>x[xmlns=\"http://jabber.org/protocol/muc#user\"]>status[code=\"307\"]').length) {\n                $(document).trigger('kicked.muc', [from]);\n                if (this.myroomjid === from) {\n                    XMPP.disposeConference(false);\n                    eventEmitter.emit(XMPPEvents.KICKED);\n                }\n            }\n            return true;\n        },\n        onPresenceError: function (pres) {\n            var from = pres.getAttribute('from');\n            if ($(pres).find('>error[type=\"auth\"]>not-authorized[xmlns=\"urn:ietf:params:xml:ns:xmpp-stanzas\"]').length) {\n                console.log('on password required', from);\n                var self = this;\n                UI.onPasswordReqiured(function (value) {\n                    self.doJoin(from, value);\n                });\n            } else if ($(pres).find(\n                '>error[type=\"cancel\"]>not-allowed[xmlns=\"urn:ietf:params:xml:ns:xmpp-stanzas\"]').length) {\n                var toDomain = Strophe.getDomainFromJid(pres.getAttribute('to'));\n                if (toDomain === config.hosts.anonymousdomain) {\n                    // we are connected with anonymous domain and only non anonymous users can create rooms\n                    // we must authorize the user\n                    XMPP.promptLogin();\n                } else {\n                    console.warn('onPresError ', pres);\n                    UI.messageHandler.openReportDialog(null,\n                        'Oops! Something went wrong and we couldn`t connect to the conference.',\n                        pres);\n                }\n            } else {\n                console.warn('onPresError ', pres);\n                UI.messageHandler.openReportDialog(null,\n                    'Oops! Something went wrong and we couldn`t connect to the conference.',\n                    pres);\n            }\n            return true;\n        },\n        sendMessage: function (body, nickname) {\n            var msg = $msg({to: this.roomjid, type: 'groupchat'});\n            msg.c('body', body).up();\n            if (nickname) {\n                msg.c('nick', {xmlns: 'http://jabber.org/protocol/nick'}).t(nickname).up().up();\n            }\n            this.connection.send(msg);\n            API.triggerEvent(\"outgoingMessage\", {\"message\": body});\n        },\n        setSubject: function (subject) {\n            var msg = $msg({to: this.roomjid, type: 'groupchat'});\n            msg.c('subject', subject);\n            this.connection.send(msg);\n            console.log(\"topic changed to \" + subject);\n        },\n        onMessage: function (msg) {\n            // FIXME: this is a hack. but jingle on muc makes nickchanges hard\n            var from = msg.getAttribute('from');\n            var nick = $(msg).find('>nick[xmlns=\"http://jabber.org/protocol/nick\"]').text() || Strophe.getResourceFromJid(from);\n\n            var txt = $(msg).find('>body').text();\n            var type = msg.getAttribute(\"type\");\n            if (type == \"error\") {\n                UI.chatAddError($(msg).find('>text').text(), txt);\n                return true;\n            }\n\n            var subject = $(msg).find('>subject');\n            if (subject.length) {\n                var subjectText = subject.text();\n                if (subjectText || subjectText == \"\") {\n                    UI.chatSetSubject(subjectText);\n                    console.log(\"Subject is changed to \" + subjectText);\n                }\n            }\n\n\n            if (txt) {\n                console.log('chat', nick, txt);\n                UI.updateChatConversation(from, nick, txt);\n                if (from != this.myroomjid)\n                    API.triggerEvent(\"incomingMessage\",\n                        {\"from\": from, \"nick\": nick, \"message\": txt});\n            }\n            return true;\n        },\n        lockRoom: function (key, onSuccess, onError, onNotSupported) {\n            //http://xmpp.org/extensions/xep-0045.html#roomconfig\n            var ob = this;\n            this.connection.sendIQ($iq({to: this.roomjid, type: 'get'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'}),\n                function (res) {\n                    if ($(res).find('>query>x[xmlns=\"jabber:x:data\"]>field[var=\"muc#roomconfig_roomsecret\"]').length) {\n                        var formsubmit = $iq({to: ob.roomjid, type: 'set'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});\n                        formsubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});\n                        formsubmit.c('field', {'var': 'FORM_TYPE'}).c('value').t('http://jabber.org/protocol/muc#roomconfig').up().up();\n                        formsubmit.c('field', {'var': 'muc#roomconfig_roomsecret'}).c('value').t(key).up().up();\n                        // Fixes a bug in prosody 0.9.+ https://code.google.com/p/lxmppd/issues/detail?id=373\n                        formsubmit.c('field', {'var': 'muc#roomconfig_whois'}).c('value').t('anyone').up().up();\n                        // FIXME: is muc#roomconfig_passwordprotectedroom required?\n                        this.connection.sendIQ(formsubmit,\n                            onSuccess,\n                            onError);\n                    } else {\n                        onNotSupported();\n                    }\n                }, onError);\n        },\n        kick: function (jid) {\n            var kickIQ = $iq({to: this.roomjid, type: 'set'})\n                .c('query', {xmlns: 'http://jabber.org/protocol/muc#admin'})\n                .c('item', {nick: Strophe.getResourceFromJid(jid), role: 'none'})\n                .c('reason').t('You have been kicked.').up().up().up();\n\n            this.connection.sendIQ(\n                kickIQ,\n                function (result) {\n                    console.log('Kick participant with jid: ', jid, result);\n                },\n                function (error) {\n                    console.log('Kick participant error: ', error);\n                });\n        },\n        sendPresence: function () {\n            var pres = $pres({to: this.presMap['to'] });\n            pres.c('x', {xmlns: this.presMap['xns']});\n\n            if (this.presMap['password']) {\n                pres.c('password').t(this.presMap['password']).up();\n            }\n\n            pres.up();\n\n            // Send XEP-0115 'c' stanza that contains our capabilities info\n            if (this.connection.caps) {\n                this.connection.caps.node = config.clientNode;\n                pres.c('c', this.connection.caps.generateCapsAttrs()).up();\n            }\n\n            pres.c('user-agent', {xmlns: 'http://jitsi.org/jitmeet/user-agent'})\n                .t(navigator.userAgent).up();\n\n            if (this.presMap['bridgeIsDown']) {\n                pres.c('bridgeIsDown').up();\n            }\n\n            if (this.presMap['email']) {\n                pres.c('email').t(this.presMap['email']).up();\n            }\n\n            if (this.presMap['userId']) {\n                pres.c('userId').t(this.presMap['userId']).up();\n            }\n\n            if (this.presMap['displayName']) {\n                // XEP-0172\n                pres.c('nick', {xmlns: 'http://jabber.org/protocol/nick'})\n                    .t(this.presMap['displayName']).up();\n            }\n\n            if (this.presMap['audions']) {\n                pres.c('audiomuted', {xmlns: this.presMap['audions']})\n                    .t(this.presMap['audiomuted']).up();\n            }\n\n            if (this.presMap['videons']) {\n                pres.c('videomuted', {xmlns: this.presMap['videons']})\n                    .t(this.presMap['videomuted']).up();\n            }\n\n            if (this.presMap['statsns']) {\n                var stats = pres.c('stats', {xmlns: this.presMap['statsns']});\n                for (var stat in this.presMap[\"stats\"])\n                    if (this.presMap[\"stats\"][stat] != null)\n                        stats.c(\"stat\", {name: stat, value: this.presMap[\"stats\"][stat]}).up();\n                pres.up();\n            }\n\n            if (this.presMap['prezins']) {\n                pres.c('prezi',\n                    {xmlns: this.presMap['prezins'],\n                        'url': this.presMap['preziurl']})\n                    .c('current').t(this.presMap['prezicurrent']).up().up();\n            }\n\n            if (this.presMap['etherpadns']) {\n                pres.c('etherpad', {xmlns: this.presMap['etherpadns']})\n                    .t(this.presMap['etherpadname']).up();\n            }\n\n            if (this.presMap['medians']) {\n                pres.c('media', {xmlns: this.presMap['medians']});\n                var sourceNumber = 0;\n                Object.keys(this.presMap).forEach(function (key) {\n                    if (key.indexOf('source') >= 0) {\n                        sourceNumber++;\n                    }\n                });\n                if (sourceNumber > 0)\n                    for (var i = 1; i <= sourceNumber / 3; i++) {\n                        pres.c('source',\n                            {type: this.presMap['source' + i + '_type'],\n                                ssrc: this.presMap['source' + i + '_ssrc'],\n                                direction: this.presMap['source' + i + '_direction']\n                                    || 'sendrecv' }\n                        ).up();\n                    }\n            }\n\n            pres.up();\n//        console.debug(pres.toString());\n            this.connection.send(pres);\n        },\n        addDisplayNameToPresence: function (displayName) {\n            this.presMap['displayName'] = displayName;\n        },\n        addMediaToPresence: function (sourceNumber, mtype, ssrcs, direction) {\n            if (!this.presMap['medians'])\n                this.presMap['medians'] = 'http://estos.de/ns/mjs';\n\n            this.presMap['source' + sourceNumber + '_type'] = mtype;\n            this.presMap['source' + sourceNumber + '_ssrc'] = ssrcs;\n            this.presMap['source' + sourceNumber + '_direction'] = direction;\n        },\n        clearPresenceMedia: function () {\n            var self = this;\n            Object.keys(this.presMap).forEach(function (key) {\n                if (key.indexOf('source') != -1) {\n                    delete self.presMap[key];\n                }\n            });\n        },\n        addPreziToPresence: function (url, currentSlide) {\n            this.presMap['prezins'] = 'http://jitsi.org/jitmeet/prezi';\n            this.presMap['preziurl'] = url;\n            this.presMap['prezicurrent'] = currentSlide;\n        },\n        removePreziFromPresence: function () {\n            delete this.presMap['prezins'];\n            delete this.presMap['preziurl'];\n            delete this.presMap['prezicurrent'];\n        },\n        addCurrentSlideToPresence: function (currentSlide) {\n            this.presMap['prezicurrent'] = currentSlide;\n        },\n        getPrezi: function (roomjid) {\n            return this.preziMap[roomjid];\n        },\n        addEtherpadToPresence: function (etherpadName) {\n            this.presMap['etherpadns'] = 'http://jitsi.org/jitmeet/etherpad';\n            this.presMap['etherpadname'] = etherpadName;\n        },\n        addAudioInfoToPresence: function (isMuted) {\n            this.presMap['audions'] = 'http://jitsi.org/jitmeet/audio';\n            this.presMap['audiomuted'] = isMuted.toString();\n        },\n        addVideoInfoToPresence: function (isMuted) {\n            this.presMap['videons'] = 'http://jitsi.org/jitmeet/video';\n            this.presMap['videomuted'] = isMuted.toString();\n        },\n        addConnectionInfoToPresence: function (stats) {\n            this.presMap['statsns'] = 'http://jitsi.org/jitmeet/stats';\n            this.presMap['stats'] = stats;\n        },\n        findJidFromResource: function (resourceJid) {\n            if (resourceJid &&\n                resourceJid === Strophe.getResourceFromJid(this.myroomjid)) {\n                return this.myroomjid;\n            }\n            var peerJid = null;\n            Object.keys(this.members).some(function (jid) {\n                peerJid = jid;\n                return Strophe.getResourceFromJid(jid) === resourceJid;\n            });\n            return peerJid;\n        },\n        addBridgeIsDownToPresence: function () {\n            this.presMap['bridgeIsDown'] = true;\n        },\n        addEmailToPresence: function (email) {\n            this.presMap['email'] = email;\n        },\n        addUserIdToPresence: function (userId) {\n            this.presMap['userId'] = userId;\n        },\n        isModerator: function () {\n            return this.role === 'moderator';\n        },\n        getMemberRole: function (peerJid) {\n            if (this.members[peerJid]) {\n                return this.members[peerJid].role;\n            }\n            return null;\n        },\n        onParticipantLeft: function (jid) {\n            UI.onMucLeft(jid);\n\n            API.triggerEvent(\"participantLeft\", {jid: jid});\n\n            delete jid2Ssrc[jid];\n\n            this.connection.jingle.terminateByJid(jid);\n\n            if (this.getPrezi(jid)) {\n                $(document).trigger('presentationremoved.muc',\n                    [jid, this.getPrezi(jid)]);\n            }\n\n            Moderator.onMucLeft(jid);\n        },\n        parsePresence: function (from, memeber, pres) {\n            if($(pres).find(\">bridgeIsDown\").length > 0 && !bridgeIsDown) {\n                bridgeIsDown = true;\n                eventEmitter.emit(XMPPEvents.BRIDGE_DOWN);\n            }\n\n            if(memeber.isFocus)\n                return;\n\n            // Remove old ssrcs coming from the jid\n            Object.keys(ssrc2jid).forEach(function (ssrc) {\n                if (ssrc2jid[ssrc] == jid) {\n                    delete ssrc2jid[ssrc];\n                    delete ssrc2videoType[ssrc];\n                }\n            });\n\n            var changedStreams = [];\n            $(pres).find('>media[xmlns=\"http://estos.de/ns/mjs\"]>source').each(function (idx, ssrc) {\n                //console.log(jid, 'assoc ssrc', ssrc.getAttribute('type'), ssrc.getAttribute('ssrc'));\n                var ssrcV = ssrc.getAttribute('ssrc');\n                ssrc2jid[ssrcV] = from;\n                notReceivedSSRCs.push(ssrcV);\n\n                var type = ssrc.getAttribute('type');\n                ssrc2videoType[ssrcV] = type;\n\n                var direction = ssrc.getAttribute('direction');\n\n                changedStreams.push({type: type, direction: direction});\n\n            });\n\n            eventEmitter.emit(XMPPEvents.CHANGED_STREAMS, from, changedStreams);\n\n            var displayName = !config.displayJids\n                ? memeber.displayName : Strophe.getResourceFromJid(from);\n\n            if (displayName && displayName.length > 0)\n            {\n//                $(document).trigger('displaynamechanged',\n//                    [jid, displayName]);\n                eventEmitter.emit(XMPPEvents.DISPLAY_NAME_CHANGED, from, displayName);\n            }\n\n\n            var id = $(pres).find('>userID').text();\n            var email = $(pres).find('>email');\n            if(email.length > 0) {\n                id = email.text();\n            }\n\n            eventEmitter.emit(XMPPEvents.USER_ID_CHANGED, from, id);\n        }\n    });\n};\n\n","/* jshint -W117 */\n\nvar JingleSession = require(\"./JingleSession\");\n\nfunction CallIncomingJingle(sid, connection) {\n    var sess = connection.jingle.sessions[sid];\n\n    // TODO: do we check activecall == null?\n    activecall = sess;\n\n    statistics.onConferenceCreated(sess);\n    RTC.onConferenceCreated(sess);\n\n    // TODO: check affiliation and/or role\n    console.log('emuc data for', sess.peerjid, connection.emuc.members[sess.peerjid]);\n    sess.usedrip = true; // not-so-naive trickle ice\n    sess.sendAnswer();\n    sess.accept();\n\n};\n\nmodule.exports = function(XMPP)\n{\n    Strophe.addConnectionPlugin('jingle', {\n        connection: null,\n        sessions: {},\n        jid2session: {},\n        ice_config: {iceServers: []},\n        pc_constraints: {},\n        media_constraints: {\n            mandatory: {\n                'OfferToReceiveAudio': true,\n                'OfferToReceiveVideo': true\n            }\n            // MozDontOfferDataChannel: true when this is firefox\n        },\n        init: function (conn) {\n            this.connection = conn;\n            if (this.connection.disco) {\n                // http://xmpp.org/extensions/xep-0167.html#support\n                // http://xmpp.org/extensions/xep-0176.html#support\n                this.connection.disco.addFeature('urn:xmpp:jingle:1');\n                this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:1');\n                this.connection.disco.addFeature('urn:xmpp:jingle:transports:ice-udp:1');\n                this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:audio');\n                this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:video');\n\n\n                // this is dealt with by SDP O/A so we don't need to annouce this\n                //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtcp-fb:0'); // XEP-0293\n                //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtp-hdrext:0'); // XEP-0294\n                if (config.useRtcpMux) {\n                    this.connection.disco.addFeature('urn:ietf:rfc:5761'); // rtcp-mux\n                }\n                if (config.useBundle) {\n                    this.connection.disco.addFeature('urn:ietf:rfc:5888'); // a=group, e.g. bundle\n                }\n                //this.connection.disco.addFeature('urn:ietf:rfc:5576'); // a=ssrc\n            }\n            this.connection.addHandler(this.onJingle.bind(this), 'urn:xmpp:jingle:1', 'iq', 'set', null, null);\n        },\n        onJingle: function (iq) {\n            var sid = $(iq).find('jingle').attr('sid');\n            var action = $(iq).find('jingle').attr('action');\n            var fromJid = iq.getAttribute('from');\n            // send ack first\n            var ack = $iq({type: 'result',\n                to: fromJid,\n                id: iq.getAttribute('id')\n            });\n            console.log('on jingle ' + action + ' from ' + fromJid, iq);\n            var sess = this.sessions[sid];\n            if ('session-initiate' != action) {\n                if (sess === null) {\n                    ack.type = 'error';\n                    ack.c('error', {type: 'cancel'})\n                        .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()\n                        .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});\n                    this.connection.send(ack);\n                    return true;\n                }\n                // compare from to sess.peerjid (bare jid comparison for later compat with message-mode)\n                // local jid is not checked\n                if (Strophe.getBareJidFromJid(fromJid) != Strophe.getBareJidFromJid(sess.peerjid)) {\n                    console.warn('jid mismatch for session id', sid, fromJid, sess.peerjid);\n                    ack.type = 'error';\n                    ack.c('error', {type: 'cancel'})\n                        .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()\n                        .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});\n                    this.connection.send(ack);\n                    return true;\n                }\n            } else if (sess !== undefined) {\n                // existing session with same session id\n                // this might be out-of-order if the sess.peerjid is the same as from\n                ack.type = 'error';\n                ack.c('error', {type: 'cancel'})\n                    .c('service-unavailable', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up();\n                console.warn('duplicate session id', sid);\n                this.connection.send(ack);\n                return true;\n            }\n            // FIXME: check for a defined action\n            this.connection.send(ack);\n            // see http://xmpp.org/extensions/xep-0166.html#concepts-session\n            switch (action) {\n                case 'session-initiate':\n                    sess = new JingleSession(\n                        $(iq).attr('to'), $(iq).find('jingle').attr('sid'),\n                        this.connection, XMPP);\n                    // configure session\n\n                    sess.media_constraints = this.media_constraints;\n                    sess.pc_constraints = this.pc_constraints;\n                    sess.ice_config = this.ice_config;\n\n                    sess.initiate(fromJid, false);\n                    // FIXME: setRemoteDescription should only be done when this call is to be accepted\n                    sess.setRemoteDescription($(iq).find('>jingle'), 'offer');\n\n                    this.sessions[sess.sid] = sess;\n                    this.jid2session[sess.peerjid] = sess;\n\n                    // the callback should either\n                    // .sendAnswer and .accept\n                    // or .sendTerminate -- not necessarily synchronus\n                    CallIncomingJingle(sess.sid, this.connection);\n                    break;\n                case 'session-accept':\n                    sess.setRemoteDescription($(iq).find('>jingle'), 'answer');\n                    sess.accept();\n                    $(document).trigger('callaccepted.jingle', [sess.sid]);\n                    break;\n                case 'session-terminate':\n                    // If this is not the focus sending the terminate, we have\n                    // nothing more to do here.\n                    if (Object.keys(this.sessions).length < 1\n                        || !(this.sessions[Object.keys(this.sessions)[0]]\n                            instanceof JingleSession))\n                    {\n                        break;\n                    }\n                    console.log('terminating...', sess.sid);\n                    sess.terminate();\n                    this.terminate(sess.sid);\n                    if ($(iq).find('>jingle>reason').length) {\n                        $(document).trigger('callterminated.jingle', [\n                            sess.sid,\n                            sess.peerjid,\n                            $(iq).find('>jingle>reason>:first')[0].tagName,\n                            $(iq).find('>jingle>reason>text').text()\n                        ]);\n                    } else {\n                        $(document).trigger('callterminated.jingle',\n                            [sess.sid, sess.peerjid]);\n                    }\n                    break;\n                case 'transport-info':\n                    sess.addIceCandidate($(iq).find('>jingle>content'));\n                    break;\n                case 'session-info':\n                    var affected;\n                    if ($(iq).find('>jingle>ringing[xmlns=\"urn:xmpp:jingle:apps:rtp:info:1\"]').length) {\n                        $(document).trigger('ringing.jingle', [sess.sid]);\n                    } else if ($(iq).find('>jingle>mute[xmlns=\"urn:xmpp:jingle:apps:rtp:info:1\"]').length) {\n                        affected = $(iq).find('>jingle>mute[xmlns=\"urn:xmpp:jingle:apps:rtp:info:1\"]').attr('name');\n                        $(document).trigger('mute.jingle', [sess.sid, affected]);\n                    } else if ($(iq).find('>jingle>unmute[xmlns=\"urn:xmpp:jingle:apps:rtp:info:1\"]').length) {\n                        affected = $(iq).find('>jingle>unmute[xmlns=\"urn:xmpp:jingle:apps:rtp:info:1\"]').attr('name');\n                        $(document).trigger('unmute.jingle', [sess.sid, affected]);\n                    }\n                    break;\n                case 'addsource': // FIXME: proprietary, un-jingleish\n                case 'source-add': // FIXME: proprietary\n                    sess.addSource($(iq).find('>jingle>content'), fromJid);\n                    break;\n                case 'removesource': // FIXME: proprietary, un-jingleish\n                case 'source-remove': // FIXME: proprietary\n                    sess.removeSource($(iq).find('>jingle>content'), fromJid);\n                    break;\n                default:\n                    console.warn('jingle action not implemented', action);\n                    break;\n            }\n            return true;\n        },\n        initiate: function (peerjid, myjid) { // initiate a new jinglesession to peerjid\n            var sess = new JingleSession(myjid || this.connection.jid,\n                Math.random().toString(36).substr(2, 12), // random string\n                this.connection, XMPP);\n            // configure session\n\n            sess.media_constraints = this.media_constraints;\n            sess.pc_constraints = this.pc_constraints;\n            sess.ice_config = this.ice_config;\n\n            sess.initiate(peerjid, true);\n            this.sessions[sess.sid] = sess;\n            this.jid2session[sess.peerjid] = sess;\n            sess.sendOffer();\n            return sess;\n        },\n        terminate: function (sid, reason, text) { // terminate by sessionid (or all sessions)\n            if (sid === null || sid === undefined) {\n                for (sid in this.sessions) {\n                    if (this.sessions[sid].state != 'ended') {\n                        this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);\n                        this.sessions[sid].terminate();\n                    }\n                    delete this.jid2session[this.sessions[sid].peerjid];\n                    delete this.sessions[sid];\n                }\n            } else if (this.sessions.hasOwnProperty(sid)) {\n                if (this.sessions[sid].state != 'ended') {\n                    this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);\n                    this.sessions[sid].terminate();\n                }\n                delete this.jid2session[this.sessions[sid].peerjid];\n                delete this.sessions[sid];\n            }\n        },\n        // Used to terminate a session when an unavailable presence is received.\n        terminateByJid: function (jid) {\n            if (this.jid2session.hasOwnProperty(jid)) {\n                var sess = this.jid2session[jid];\n                if (sess) {\n                    sess.terminate();\n                    console.log('peer went away silently', jid);\n                    delete this.sessions[sess.sid];\n                    delete this.jid2session[jid];\n                    $(document).trigger('callterminated.jingle',\n                        [sess.sid, jid], 'gone');\n                }\n            }\n        },\n        terminateRemoteByJid: function (jid, reason) {\n            if (this.jid2session.hasOwnProperty(jid)) {\n                var sess = this.jid2session[jid];\n                if (sess) {\n                    sess.sendTerminate(reason || (!sess.active()) ? 'kick' : null);\n                    sess.terminate();\n                    console.log('terminate peer with jid', sess.sid, jid);\n                    delete this.sessions[sess.sid];\n                    delete this.jid2session[jid];\n                    $(document).trigger('callterminated.jingle',\n                        [sess.sid, jid, 'kicked']);\n                }\n            }\n        },\n        getStunAndTurnCredentials: function () {\n            // get stun and turn configuration from server via xep-0215\n            // uses time-limited credentials as described in\n            // http://tools.ietf.org/html/draft-uberti-behave-turn-rest-00\n            //\n            // see https://code.google.com/p/prosody-modules/source/browse/mod_turncredentials/mod_turncredentials.lua\n            // for a prosody module which implements this\n            //\n            // currently, this doesn't work with updateIce and therefore credentials with a long\n            // validity have to be fetched before creating the peerconnection\n            // TODO: implement refresh via updateIce as described in\n            //      https://code.google.com/p/webrtc/issues/detail?id=1650\n            var self = this;\n            this.connection.sendIQ(\n                $iq({type: 'get', to: this.connection.domain})\n                    .c('services', {xmlns: 'urn:xmpp:extdisco:1'}).c('service', {host: 'turn.' + this.connection.domain}),\n                function (res) {\n                    var iceservers = [];\n                    $(res).find('>services>service').each(function (idx, el) {\n                        el = $(el);\n                        var dict = {};\n                        var type = el.attr('type');\n                        switch (type) {\n                            case 'stun':\n                                dict.url = 'stun:' + el.attr('host');\n                                if (el.attr('port')) {\n                                    dict.url += ':' + el.attr('port');\n                                }\n                                iceservers.push(dict);\n                                break;\n                            case 'turn':\n                            case 'turns':\n                                dict.url = type + ':';\n                                if (el.attr('username')) { // https://code.google.com/p/webrtc/issues/detail?id=1508\n                                    if (navigator.userAgent.match(/Chrom(e|ium)\\/([0-9]+)\\./) && parseInt(navigator.userAgent.match(/Chrom(e|ium)\\/([0-9]+)\\./)[2], 10) < 28) {\n                                        dict.url += el.attr('username') + '@';\n                                    } else {\n                                        dict.username = el.attr('username'); // only works in M28\n                                    }\n                                }\n                                dict.url += el.attr('host');\n                                if (el.attr('port') && el.attr('port') != '3478') {\n                                    dict.url += ':' + el.attr('port');\n                                }\n                                if (el.attr('transport') && el.attr('transport') != 'udp') {\n                                    dict.url += '?transport=' + el.attr('transport');\n                                }\n                                if (el.attr('password')) {\n                                    dict.credential = el.attr('password');\n                                }\n                                iceservers.push(dict);\n                                break;\n                        }\n                    });\n                    self.ice_config.iceServers = iceservers;\n                },\n                function (err) {\n                    console.warn('getting turn credentials failed', err);\n                    console.warn('is mod_turncredentials or similar installed?');\n                }\n            );\n            // implement push?\n        },\n\n        /**\n         * Populates the log data\n         */\n        populateData: function () {\n            var data = {};\n            Object.keys(this.sessions).forEach(function (sid) {\n                var session = this.sessions[sid];\n                if (session.peerconnection && session.peerconnection.updateLog) {\n                    // FIXME: should probably be a .dump call\n                    data[\"jingle_\" + session.sid] = {\n                        updateLog: session.peerconnection.updateLog,\n                        stats: session.peerconnection.stats,\n                        url: window.location.href\n                    };\n                }\n            });\n            return data;\n        }\n    });\n};\n\n","/* global Strophe */\nmodule.exports = function () {\n\n    Strophe.addConnectionPlugin('logger', {\n        // logs raw stanzas and makes them available for download as JSON\n        connection: null,\n        log: [],\n        init: function (conn) {\n            this.connection = conn;\n            this.connection.rawInput = this.log_incoming.bind(this);\n            this.connection.rawOutput = this.log_outgoing.bind(this);\n        },\n        log_incoming: function (stanza) {\n            this.log.push([new Date().getTime(), 'incoming', stanza]);\n        },\n        log_outgoing: function (stanza) {\n            this.log.push([new Date().getTime(), 'outgoing', stanza]);\n        }\n    });\n};","/* global $, $iq, config, connection, focusMucJid, forceMuted,\n   setAudioMuted, Strophe */\n/**\n * Moderate connection plugin.\n */\nmodule.exports = function (XMPP) {\n    Strophe.addConnectionPlugin('moderate', {\n        connection: null,\n        init: function (conn) {\n            this.connection = conn;\n\n            this.connection.addHandler(this.onMute.bind(this),\n                'http://jitsi.org/jitmeet/audio',\n                'iq',\n                'set',\n                null,\n                null);\n        },\n        setMute: function (jid, mute) {\n            console.info(\"set mute\", mute);\n            var iqToFocus = $iq({to: focusMucJid, type: 'set'})\n                .c('mute', {\n                    xmlns: 'http://jitsi.org/jitmeet/audio',\n                    jid: jid\n                })\n                .t(mute.toString())\n                .up();\n\n            this.connection.sendIQ(\n                iqToFocus,\n                function (result) {\n                    console.log('set mute', result);\n                },\n                function (error) {\n                    console.log('set mute error', error);\n                });\n        },\n        onMute: function (iq) {\n            var from = iq.getAttribute('from');\n            if (from !== focusMucJid) {\n                console.warn(\"Ignored mute from non focus peer\");\n                return false;\n            }\n            var mute = $(iq).find('mute');\n            if (mute.length) {\n                var doMuteAudio = mute.text() === \"true\";\n                UI.setAudioMuted(doMuteAudio);\n                XMPP.forceMuted = doMuteAudio;\n            }\n            return true;\n        },\n        eject: function (jid) {\n            // We're not the focus, so can't terminate\n            //connection.jingle.terminateRemoteByJid(jid, 'kick');\n            this.connection.emuc.kick(jid);\n        }\n    });\n}","/* jshint -W117 */\nmodule.exports = function() {\n    Strophe.addConnectionPlugin('rayo',\n        {\n            RAYO_XMLNS: 'urn:xmpp:rayo:1',\n            connection: null,\n            init: function (conn) {\n                this.connection = conn;\n                if (this.connection.disco) {\n                    this.connection.disco.addFeature('urn:xmpp:rayo:client:1');\n                }\n\n                this.connection.addHandler(\n                    this.onRayo.bind(this), this.RAYO_XMLNS, 'iq', 'set', null, null);\n            },\n            onRayo: function (iq) {\n                console.info(\"Rayo IQ\", iq);\n            },\n            dial: function (to, from, roomName, roomPass) {\n                var self = this;\n                var req = $iq(\n                    {\n                        type: 'set',\n                        to: focusMucJid\n                    }\n                );\n                req.c('dial',\n                    {\n                        xmlns: this.RAYO_XMLNS,\n                        to: to,\n                        from: from\n                    });\n                req.c('header',\n                    {\n                        name: 'JvbRoomName',\n                        value: roomName\n                    }).up();\n\n                if (roomPass !== null && roomPass.length) {\n\n                    req.c('header',\n                        {\n                            name: 'JvbRoomPassword',\n                            value: roomPass\n                        }).up();\n                }\n\n                this.connection.sendIQ(\n                    req,\n                    function (result) {\n                        console.info('Dial result ', result);\n\n                        var resource = $(result).find('ref').attr('uri');\n                        this.call_resource = resource.substr('xmpp:'.length);\n                        console.info(\n                                \"Received call resource: \" + this.call_resource);\n                    },\n                    function (error) {\n                        console.info('Dial error ', error);\n                    }\n                );\n            },\n            hang_up: function () {\n                if (!this.call_resource) {\n                    console.warn(\"No call in progress\");\n                    return;\n                }\n\n                var self = this;\n                var req = $iq(\n                    {\n                        type: 'set',\n                        to: this.call_resource\n                    }\n                );\n                req.c('hangup',\n                    {\n                        xmlns: this.RAYO_XMLNS\n                    });\n\n                this.connection.sendIQ(\n                    req,\n                    function (result) {\n                        console.info('Hangup result ', result);\n                        self.call_resource = null;\n                    },\n                    function (error) {\n                        console.info('Hangup error ', error);\n                        self.call_resource = null;\n                    }\n                );\n            }\n        }\n    );\n};\n","/**\n * Strophe logger implementation. Logs from level WARN and above.\n */\nmodule.exports = function () {\n\n    Strophe.log = function (level, msg) {\n        switch (level) {\n            case Strophe.LogLevel.WARN:\n                console.warn(\"Strophe: \" + msg);\n                break;\n            case Strophe.LogLevel.ERROR:\n            case Strophe.LogLevel.FATAL:\n                console.error(\"Strophe: \" + msg);\n                break;\n        }\n    };\n\n    Strophe.getStatusString = function (status) {\n        switch (status) {\n            case Strophe.Status.ERROR:\n                return \"ERROR\";\n            case Strophe.Status.CONNECTING:\n                return \"CONNECTING\";\n            case Strophe.Status.CONNFAIL:\n                return \"CONNFAIL\";\n            case Strophe.Status.AUTHENTICATING:\n                return \"AUTHENTICATING\";\n            case Strophe.Status.AUTHFAIL:\n                return \"AUTHFAIL\";\n            case Strophe.Status.CONNECTED:\n                return \"CONNECTED\";\n            case Strophe.Status.DISCONNECTED:\n                return \"DISCONNECTED\";\n            case Strophe.Status.DISCONNECTING:\n                return \"DISCONNECTING\";\n            case Strophe.Status.ATTACHED:\n                return \"ATTACHED\";\n            default:\n                return \"unknown\";\n        }\n    };\n};\n","var Moderator = require(\"./moderator\");\nvar EventEmitter = require(\"events\");\nvar Recording = require(\"./recording\");\nvar SDP = require(\"./SDP\");\n\nvar eventEmitter = new EventEmitter();\nvar connection = null;\nvar authenticatedUser = false;\nvar activecall = null;\n\nfunction connect(jid, password, uiCredentials) {\n    var bosh\n        = uiCredentials.bosh || config.bosh || '/http-bind';\n    connection = new Strophe.Connection(bosh);\n    Moderator.setConnection(connection);\n\n    var settings = UI.getSettings();\n    var email = settings.email;\n    var displayName = settings.displayName;\n    if(email) {\n        connection.emuc.addEmailToPresence(email);\n    } else {\n        connection.emuc.addUserIdToPresence(settings.uid);\n    }\n    if(displayName) {\n        connection.emuc.addDisplayNameToPresence(displayName);\n    }\n\n    if (connection.disco) {\n        // for chrome, add multistream cap\n    }\n    connection.jingle.pc_constraints = RTC.getPCConstraints();\n    if (config.useIPv6) {\n        // https://code.google.com/p/webrtc/issues/detail?id=2828\n        if (!connection.jingle.pc_constraints.optional)\n            connection.jingle.pc_constraints.optional = [];\n        connection.jingle.pc_constraints.optional.push({googIPv6: true});\n    }\n\n    if(!password)\n        password = uiCredentials.password;\n\n    var anonymousConnectionFailed = false;\n    connection.connect(jid, password, function (status, msg) {\n        console.log('Strophe status changed to',\n            Strophe.getStatusString(status));\n        if (status === Strophe.Status.CONNECTED) {\n            if (config.useStunTurn) {\n                connection.jingle.getStunAndTurnCredentials();\n            }\n            UI.disableConnect();\n\n            console.info(\"My Jabber ID: \" + connection.jid);\n\n            if(password)\n                authenticatedUser = true;\n            maybeDoJoin();\n        } else if (status === Strophe.Status.CONNFAIL) {\n            if(msg === 'x-strophe-bad-non-anon-jid') {\n                anonymousConnectionFailed = true;\n            }\n        } else if (status === Strophe.Status.DISCONNECTED) {\n            if(anonymousConnectionFailed) {\n                // prompt user for username and password\n                XMPP.promptLogin();\n            }\n        } else if (status === Strophe.Status.AUTHFAIL) {\n            // wrong password or username, prompt user\n            XMPP.promptLogin();\n\n        }\n    });\n}\n\n\n\nfunction maybeDoJoin() {\n    if (connection && connection.connected &&\n        Strophe.getResourceFromJid(connection.jid)\n        && (RTC.localAudio || RTC.localVideo)) {\n        // .connected is true while connecting?\n        doJoin();\n    }\n}\n\nfunction doJoin() {\n    var roomName = UI.generateRoomName();\n\n    Moderator.allocateConferenceFocus(\n        roomName, UI.checkForNicknameAndJoin);\n}\n\nfunction initStrophePlugins()\n{\n    require(\"./strophe.emuc\")(XMPP, eventEmitter);\n    require(\"./strophe.jingle\")();\n    require(\"./strophe.moderate\")(XMPP);\n    require(\"./strophe.util\")();\n    require(\"./strophe.rayo\")();\n    require(\"./strophe.logger\")();\n}\n\nfunction registerListeners() {\n    RTC.addStreamListener(maybeDoJoin,\n        StreamEventTypes.EVENT_TYPE_LOCAL_CREATED);\n}\n\nfunction setupEvents() {\n    $(window).bind('beforeunload', function () {\n        if (connection && connection.connected) {\n            // ensure signout\n            $.ajax({\n                type: 'POST',\n                url: config.bosh,\n                async: false,\n                cache: false,\n                contentType: 'application/xml',\n                data: \"<body rid='\" + (connection.rid || connection._proto.rid)\n                    + \"' xmlns='http://jabber.org/protocol/httpbind' sid='\"\n                    + (connection.sid || connection._proto.sid)\n                    + \"' type='terminate'>\" +\n                    \"<presence xmlns='jabber:client' type='unavailable'/>\" +\n                    \"</body>\",\n                success: function (data) {\n                    console.log('signed out');\n                    console.log(data);\n                },\n                error: function (XMLHttpRequest, textStatus, errorThrown) {\n                    console.log('signout error',\n                            textStatus + ' (' + errorThrown + ')');\n                }\n            });\n        }\n        XMPP.disposeConference(true);\n    });\n}\n\nvar XMPP = {\n    sessionTerminated: false,\n    /**\n     * Remembers if we were muted by the focus.\n     * @type {boolean}\n     */\n    forceMuted: false,\n    start: function (uiCredentials) {\n        setupEvents();\n        initStrophePlugins();\n        registerListeners();\n        Moderator.init();\n        var jid = uiCredentials.jid ||\n            config.hosts.anonymousdomain ||\n            config.hosts.domain ||\n            window.location.hostname;\n        connect(jid, null, uiCredentials);\n    },\n    promptLogin: function () {\n        UI.showLoginPopup(connect);\n    },\n    joinRooom: function(roomName, useNicks, nick)\n    {\n        var roomjid;\n        roomjid = roomName;\n\n        if (useNicks) {\n            if (nick) {\n                roomjid += '/' + nick;\n            } else {\n                roomjid += '/' + Strophe.getNodeFromJid(connection.jid);\n            }\n        } else {\n\n            var tmpJid = Strophe.getNodeFromJid(connection.jid);\n\n            if(!authenticatedUser)\n                tmpJid = tmpJid.substr(0, 8);\n\n            roomjid += '/' + tmpJid;\n        }\n        connection.emuc.doJoin(roomjid);\n    },\n    myJid: function () {\n        if(!connection)\n            return null;\n        return connection.emuc.myroomjid;\n    },\n    myResource: function () {\n        if(!connection || ! connection.emuc.myroomjid)\n            return null;\n        return Strophe.getResourceFromJid(connection.emuc.myroomjid);\n    },\n    disposeConference: function (onUnload) {\n        eventEmitter.emit(XMPPEvents.DISPOSE_CONFERENCE, onUnload);\n        var handler = activecall;\n        if (handler && handler.peerconnection) {\n            // FIXME: probably removing streams is not required and close() should\n            // be enough\n            if (RTC.localAudio) {\n                handler.peerconnection.removeStream(RTC.localAudio.getOriginalStream(), onUnload);\n            }\n            if (RTC.localVideo) {\n                handler.peerconnection.removeStream(RTC.localVideo.getOriginalStream(), onUnload);\n            }\n            handler.peerconnection.close();\n        }\n        activecall = null;\n        if(!onUnload)\n        {\n            this.sessionTerminated = true;\n            connection.emuc.doLeave();\n        }\n    },\n    addListener: function(type, listener)\n    {\n        eventEmitter.on(type, listener);\n    },\n    removeListener: function (type, listener) {\n        eventEmitter.removeListener(type, listener);\n    },\n    allocateConferenceFocus: function(roomName, callback) {\n        Moderator.allocateConferenceFocus(roomName, callback);\n    },\n    isModerator: function () {\n        return Moderator.isModerator();\n    },\n    isSipGatewayEnabled: function () {\n        return Moderator.isSipGatewayEnabled();\n    },\n    isExternalAuthEnabled: function () {\n        return Moderator.isExternalAuthEnabled();\n    },\n    switchStreams: function (stream, oldStream, callback) {\n        if (activecall) {\n            // FIXME: will block switchInProgress on true value in case of exception\n            activecall.switchStreams(stream, oldStream, callback);\n        } else {\n            // We are done immediately\n            console.error(\"No conference handler\");\n            UI.messageHandler.showError('Error',\n                'Unable to switch video stream.');\n            callback();\n        }\n    },\n    setVideoMute: function (mute, callback, options) {\n       if(activecall && connection && RTC.localVideo)\n       {\n           activecall.setVideoMute(mute, callback, options);\n       }\n    },\n    setAudioMute: function (mute, callback) {\n        if (!(connection && RTC.localAudio)) {\n            return false;\n        }\n\n\n        if (this.forceMuted && !mute) {\n            console.info(\"Asking focus for unmute\");\n            connection.moderate.setMute(connection.emuc.myroomjid, mute);\n            // FIXME: wait for result before resetting muted status\n            this.forceMuted = false;\n        }\n\n        if (mute == RTC.localAudio.isMuted()) {\n            // Nothing to do\n            return true;\n        }\n\n        // It is not clear what is the right way to handle multiple tracks.\n        // So at least make sure that they are all muted or all unmuted and\n        // that we send presence just once.\n        RTC.localAudio.mute();\n        // isMuted is the opposite of audioEnabled\n        connection.emuc.addAudioInfoToPresence(mute);\n        connection.emuc.sendPresence();\n        callback();\n        return true;\n    },\n    // Really mute video, i.e. dont even send black frames\n    muteVideo: function (pc, unmute) {\n        // FIXME: this probably needs another of those lovely state safeguards...\n        // which checks for iceconn == connected and sigstate == stable\n        pc.setRemoteDescription(pc.remoteDescription,\n            function () {\n                pc.createAnswer(\n                    function (answer) {\n                        var sdp = new SDP(answer.sdp);\n                        if (sdp.media.length > 1) {\n                            if (unmute)\n                                sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');\n                            else\n                                sdp.media[1] = sdp.media[1].replace('a=sendrecv', 'a=recvonly');\n                            sdp.raw = sdp.session + sdp.media.join('');\n                            answer.sdp = sdp.raw;\n                        }\n                        pc.setLocalDescription(answer,\n                            function () {\n                                console.log('mute SLD ok');\n                            },\n                            function (error) {\n                                console.log('mute SLD error');\n                                UI.messageHandler.showError('Error',\n                                        'Oops! Something went wrong and we failed to ' +\n                                        'mute! (SLD Failure)');\n                            }\n                        );\n                    },\n                    function (error) {\n                        console.log(error);\n                        UI.messageHandler.showError();\n                    }\n                );\n            },\n            function (error) {\n                console.log('muteVideo SRD error');\n                UI.messageHandler.showError('Error',\n                        'Oops! Something went wrong and we failed to stop video!' +\n                        '(SRD Failure)');\n\n            }\n        );\n    },\n    toggleRecording: function (tokenEmptyCallback,\n                               startingCallback, startedCallback) {\n        Recording.toggleRecording(tokenEmptyCallback,\n            startingCallback, startedCallback);\n    },\n    addToPresence: function (name, value, dontSend) {\n        switch (name)\n        {\n            case \"displayName\":\n                connection.emuc.addDisplayNameToPresence(value);\n                break;\n            case \"etherpad\":\n                connection.emuc.addEtherpadToPresence(value);\n                break;\n            case \"prezi\":\n                connection.emuc.addPreziToPresence(value, 0);\n                break;\n            case \"preziSlide\":\n                connection.emuc.addCurrentSlideToPresence(value);\n                break;\n            case \"connectionQuality\":\n                connection.emuc.addConnectionInfoToPresence(value);\n                break;\n            case \"email\":\n                connection.emuc.addEmailToPresence(value);\n            default :\n                console.log(\"Unknown tag for presence.\");\n                return;\n        }\n        if(!dontSend)\n            connection.emuc.sendPresence();\n    },\n    sendLogs: function (content) {\n        // XEP-0337-ish\n        var message = $msg({to: focusMucJid, type: 'normal'});\n        message.c('log', { xmlns: 'urn:xmpp:eventlog',\n            id: 'PeerConnectionStats'});\n        message.c('message').t(content).up();\n        if (deflate) {\n            message.c('tag', {name: \"deflated\", value: \"true\"}).up();\n        }\n        message.up();\n\n        connection.send(message);\n    },\n    populateData: function () {\n        var data = {};\n        if (connection.jingle) {\n            data = connection.jingle.populateData();\n        }\n        return data;\n    },\n    getLogger: function () {\n        if(connection.logger)\n            return connection.logger.log;\n        return null;\n    },\n    getPrezi: function () {\n        return connection.emuc.getPrezi(this.myJid());\n    },\n    removePreziFromPresence: function () {\n        connection.emuc.removePreziFromPresence();\n        connection.emuc.sendPresence();\n    },\n    sendChatMessage: function (message, nickname) {\n        connection.emuc.sendMessage(message, nickname);\n    },\n    setSubject: function (topic) {\n        connection.emuc.setSubject(topic);\n    },\n    lockRoom: function (key, onSuccess, onError, onNotSupported) {\n        connection.emuc.lockRoom(key, onSuccess, onError, onNotSupported);\n    },\n    dial: function (to, from, roomName,roomPass) {\n        connection.rayo.dial(to, from, roomName,roomPass);\n    },\n    setMute: function (jid, mute) {\n        connection.moderate.setMute(jid, mute);\n    },\n    eject: function (jid) {\n        connection.moderate.eject(jid);\n    },\n    findJidFromResource: function (resource) {\n        connection.emuc.findJidFromResource(resource);\n    },\n    getMembers: function () {\n        return connection.emuc.members;\n    }\n\n};\n\nmodule.exports = XMPP;","// Copyright Joyent, Inc. and other Node contributors.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a\n// copy of this software and associated documentation files (the\n// \"Software\"), to deal in the Software without restriction, including\n// without limitation the rights to use, copy, modify, merge, publish,\n// distribute, sublicense, and/or sell copies of the Software, and to permit\n// persons to whom the Software is furnished to do so, subject to the\n// following conditions:\n//\n// The above copyright notice and this permission notice shall be included\n// in all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS\n// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF\n// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN\n// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,\n// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR\n// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE\n// USE OR OTHER DEALINGS IN THE SOFTWARE.\n\nfunction EventEmitter() {\n  this._events = this._events || {};\n  this._maxListeners = this._maxListeners || undefined;\n}\nmodule.exports = EventEmitter;\n\n// Backwards-compat with node 0.10.x\nEventEmitter.EventEmitter = EventEmitter;\n\nEventEmitter.prototype._events = undefined;\nEventEmitter.prototype._maxListeners = undefined;\n\n// By default EventEmitters will print a warning if more than 10 listeners are\n// added to it. This is a useful default which helps finding memory leaks.\nEventEmitter.defaultMaxListeners = 10;\n\n// Obviously not all Emitters should be limited to 10. This function allows\n// that to be increased. Set to zero for unlimited.\nEventEmitter.prototype.setMaxListeners = function(n) {\n  if (!isNumber(n) || n < 0 || isNaN(n))\n    throw TypeError('n must be a positive number');\n  this._maxListeners = n;\n  return this;\n};\n\nEventEmitter.prototype.emit = function(type) {\n  var er, handler, len, args, i, listeners;\n\n  if (!this._events)\n    this._events = {};\n\n  // If there is no 'error' event listener then throw.\n  if (type === 'error') {\n    if (!this._events.error ||\n        (isObject(this._events.error) && !this._events.error.length)) {\n      er = arguments[1];\n      if (er instanceof Error) {\n        throw er; // Unhandled 'error' event\n      } else {\n        throw TypeError('Uncaught, unspecified \"error\" event.');\n      }\n      return false;\n    }\n  }\n\n  handler = this._events[type];\n\n  if (isUndefined(handler))\n    return false;\n\n  if (isFunction(handler)) {\n    switch (arguments.length) {\n      // fast cases\n      case 1:\n        handler.call(this);\n        break;\n      case 2:\n        handler.call(this, arguments[1]);\n        break;\n      case 3:\n        handler.call(this, arguments[1], arguments[2]);\n        break;\n      // slower\n      default:\n        len = arguments.length;\n        args = new Array(len - 1);\n        for (i = 1; i < len; i++)\n          args[i - 1] = arguments[i];\n        handler.apply(this, args);\n    }\n  } else if (isObject(handler)) {\n    len = arguments.length;\n    args = new Array(len - 1);\n    for (i = 1; i < len; i++)\n      args[i - 1] = arguments[i];\n\n    listeners = handler.slice();\n    len = listeners.length;\n    for (i = 0; i < len; i++)\n      listeners[i].apply(this, args);\n  }\n\n  return true;\n};\n\nEventEmitter.prototype.addListener = function(type, listener) {\n  var m;\n\n  if (!isFunction(listener))\n    throw TypeError('listener must be a function');\n\n  if (!this._events)\n    this._events = {};\n\n  // To avoid recursion in the case that type === \"newListener\"! Before\n  // adding it to the listeners, first emit \"newListener\".\n  if (this._events.newListener)\n    this.emit('newListener', type,\n              isFunction(listener.listener) ?\n              listener.listener : listener);\n\n  if (!this._events[type])\n    // Optimize the case of one listener. Don't need the extra array object.\n    this._events[type] = listener;\n  else if (isObject(this._events[type]))\n    // If we've already got an array, just append.\n    this._events[type].push(listener);\n  else\n    // Adding the second element, need to change to array.\n    this._events[type] = [this._events[type], listener];\n\n  // Check for listener leak\n  if (isObject(this._events[type]) && !this._events[type].warned) {\n    var m;\n    if (!isUndefined(this._maxListeners)) {\n      m = this._maxListeners;\n    } else {\n      m = EventEmitter.defaultMaxListeners;\n    }\n\n    if (m && m > 0 && this._events[type].length > m) {\n      this._events[type].warned = true;\n      console.error('(node) warning: possible EventEmitter memory ' +\n                    'leak detected. %d listeners added. ' +\n                    'Use emitter.setMaxListeners() to increase limit.',\n                    this._events[type].length);\n      if (typeof console.trace === 'function') {\n        // not supported in IE 10\n        console.trace();\n      }\n    }\n  }\n\n  return this;\n};\n\nEventEmitter.prototype.on = EventEmitter.prototype.addListener;\n\nEventEmitter.prototype.once = function(type, listener) {\n  if (!isFunction(listener))\n    throw TypeError('listener must be a function');\n\n  var fired = false;\n\n  function g() {\n    this.removeListener(type, g);\n\n    if (!fired) {\n      fired = true;\n      listener.apply(this, arguments);\n    }\n  }\n\n  g.listener = listener;\n  this.on(type, g);\n\n  return this;\n};\n\n// emits a 'removeListener' event iff the listener was removed\nEventEmitter.prototype.removeListener = function(type, listener) {\n  var list, position, length, i;\n\n  if (!isFunction(listener))\n    throw TypeError('listener must be a function');\n\n  if (!this._events || !this._events[type])\n    return this;\n\n  list = this._events[type];\n  length = list.length;\n  position = -1;\n\n  if (list === listener ||\n      (isFunction(list.listener) && list.listener === listener)) {\n    delete this._events[type];\n    if (this._events.removeListener)\n      this.emit('removeListener', type, listener);\n\n  } else if (isObject(list)) {\n    for (i = length; i-- > 0;) {\n      if (list[i] === listener ||\n          (list[i].listener && list[i].listener === listener)) {\n        position = i;\n        break;\n      }\n    }\n\n    if (position < 0)\n      return this;\n\n    if (list.length === 1) {\n      list.length = 0;\n      delete this._events[type];\n    } else {\n      list.splice(position, 1);\n    }\n\n    if (this._events.removeListener)\n      this.emit('removeListener', type, listener);\n  }\n\n  return this;\n};\n\nEventEmitter.prototype.removeAllListeners = function(type) {\n  var key, listeners;\n\n  if (!this._events)\n    return this;\n\n  // not listening for removeListener, no need to emit\n  if (!this._events.removeListener) {\n    if (arguments.length === 0)\n      this._events = {};\n    else if (this._events[type])\n      delete this._events[type];\n    return this;\n  }\n\n  // emit removeListener for all listeners on all events\n  if (arguments.length === 0) {\n    for (key in this._events) {\n      if (key === 'removeListener') continue;\n      this.removeAllListeners(key);\n    }\n    this.removeAllListeners('removeListener');\n    this._events = {};\n    return this;\n  }\n\n  listeners = this._events[type];\n\n  if (isFunction(listeners)) {\n    this.removeListener(type, listeners);\n  } else {\n    // LIFO order\n    while (listeners.length)\n      this.removeListener(type, listeners[listeners.length - 1]);\n  }\n  delete this._events[type];\n\n  return this;\n};\n\nEventEmitter.prototype.listeners = function(type) {\n  var ret;\n  if (!this._events || !this._events[type])\n    ret = [];\n  else if (isFunction(this._events[type]))\n    ret = [this._events[type]];\n  else\n    ret = this._events[type].slice();\n  return ret;\n};\n\nEventEmitter.listenerCount = function(emitter, type) {\n  var ret;\n  if (!emitter._events || !emitter._events[type])\n    ret = 0;\n  else if (isFunction(emitter._events[type]))\n    ret = 1;\n  else\n    ret = emitter._events[type].length;\n  return ret;\n};\n\nfunction isFunction(arg) {\n  return typeof arg === 'function';\n}\n\nfunction isNumber(arg) {\n  return typeof arg === 'number';\n}\n\nfunction isObject(arg) {\n  return typeof arg === 'object' && arg !== null;\n}\n\nfunction isUndefined(arg) {\n  return arg === void 0;\n}\n"]}
diff --git a/libs/rayo.js b/libs/rayo.js
deleted file mode 100644
index 3298093f8..000000000
--- a/libs/rayo.js
+++ /dev/null
@@ -1,103 +0,0 @@
-/* jshint -W117 */
-Strophe.addConnectionPlugin('rayo',
- {
- RAYO_XMLNS: 'urn:xmpp:rayo:1',
- connection: null,
- init: function (conn)
- {
- this.connection = conn;
- if (this.connection.disco)
- {
- this.connection.disco.addFeature('urn:xmpp:rayo:client:1');
- }
-
- this.connection.addHandler(
- this.onRayo.bind(this), this.RAYO_XMLNS, 'iq', 'set', null, null);
- },
- onRayo: function (iq)
- {
- console.info("Rayo IQ", iq);
- },
- dial: function (to, from, roomName, roomPass)
- {
- var self = this;
- var req = $iq(
- {
- type: 'set',
- to: focusMucJid
- }
- );
- req.c('dial',
- {
- xmlns: this.RAYO_XMLNS,
- to: to,
- from: from
- });
- req.c('header',
- {
- name: 'JvbRoomName',
- value: roomName
- }).up();
-
- if (roomPass !== null && roomPass.length) {
-
- req.c('header',
- {
- name: 'JvbRoomPassword',
- value: roomPass
- }).up();
- }
-
- this.connection.sendIQ(
- req,
- function (result)
- {
- console.info('Dial result ', result);
-
- var resource = $(result).find('ref').attr('uri');
- this.call_resource = resource.substr('xmpp:'.length);
- console.info(
- "Received call resource: " + this.call_resource);
- },
- function (error)
- {
- console.info('Dial error ', error);
- }
- );
- },
- hang_up: function ()
- {
- if (!this.call_resource)
- {
- console.warn("No call in progress");
- return;
- }
-
- var self = this;
- var req = $iq(
- {
- type: 'set',
- to: this.call_resource
- }
- );
- req.c('hangup',
- {
- xmlns: this.RAYO_XMLNS
- });
-
- this.connection.sendIQ(
- req,
- function (result)
- {
- console.info('Hangup result ', result);
- self.call_resource = null;
- },
- function (error)
- {
- console.info('Hangup error ', error);
- self.call_resource = null;
- }
- );
- }
- }
-);
\ No newline at end of file
diff --git a/libs/strophe/strophe.jingle.js b/libs/strophe/strophe.jingle.js
deleted file mode 100644
index cbc081798..000000000
--- a/libs/strophe/strophe.jingle.js
+++ /dev/null
@@ -1,327 +0,0 @@
-/* jshint -W117 */
-
-
-function CallIncomingJingle(sid) {
- var sess = connection.jingle.sessions[sid];
-
- // TODO: do we check activecall == null?
- activecall = sess;
-
- statistics.onConferenceCreated(sess);
- RTC.onConferenceCreated(sess);
-
- // TODO: check affiliation and/or role
- console.log('emuc data for', sess.peerjid, connection.emuc.members[sess.peerjid]);
- sess.usedrip = true; // not-so-naive trickle ice
- sess.sendAnswer();
- sess.accept();
-
-};
-
-Strophe.addConnectionPlugin('jingle', {
- connection: null,
- sessions: {},
- jid2session: {},
- ice_config: {iceServers: []},
- pc_constraints: {},
- media_constraints: {
- mandatory: {
- 'OfferToReceiveAudio': true,
- 'OfferToReceiveVideo': true
- }
- // MozDontOfferDataChannel: true when this is firefox
- },
- init: function (conn) {
- this.connection = conn;
- if (this.connection.disco) {
- // http://xmpp.org/extensions/xep-0167.html#support
- // http://xmpp.org/extensions/xep-0176.html#support
- this.connection.disco.addFeature('urn:xmpp:jingle:1');
- this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:1');
- this.connection.disco.addFeature('urn:xmpp:jingle:transports:ice-udp:1');
- this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:audio');
- this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:video');
-
-
- // this is dealt with by SDP O/A so we don't need to annouce this
- //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtcp-fb:0'); // XEP-0293
- //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtp-hdrext:0'); // XEP-0294
- if (config.useRtcpMux) {
- this.connection.disco.addFeature('urn:ietf:rfc:5761'); // rtcp-mux
- }
- if (config.useBundle) {
- this.connection.disco.addFeature('urn:ietf:rfc:5888'); // a=group, e.g. bundle
- }
- //this.connection.disco.addFeature('urn:ietf:rfc:5576'); // a=ssrc
- }
- this.connection.addHandler(this.onJingle.bind(this), 'urn:xmpp:jingle:1', 'iq', 'set', null, null);
- },
- 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: fromJid,
- id: iq.getAttribute('id')
- });
- console.log('on jingle ' + action + ' from ' + fromJid, iq);
- var sess = this.sessions[sid];
- if ('session-initiate' != action) {
- if (sess === null) {
- ack.type = 'error';
- ack.c('error', {type: 'cancel'})
- .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()
- .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
- this.connection.send(ack);
- return true;
- }
- // compare from to sess.peerjid (bare jid comparison for later compat with message-mode)
- // local jid is not checked
- 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()
- .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
- this.connection.send(ack);
- return true;
- }
- } else if (sess !== undefined) {
- // existing session with same session id
- // this might be out-of-order if the sess.peerjid is the same as from
- ack.type = 'error';
- ack.c('error', {type: 'cancel'})
- .c('service-unavailable', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up();
- console.warn('duplicate session id', sid);
- this.connection.send(ack);
- return true;
- }
- // FIXME: check for a defined action
- this.connection.send(ack);
- // see http://xmpp.org/extensions/xep-0166.html#concepts-session
- switch (action) {
- case 'session-initiate':
- sess = new JingleSession($(iq).attr('to'), $(iq).find('jingle').attr('sid'), this.connection);
- // configure session
-
- sess.media_constraints = this.media_constraints;
- sess.pc_constraints = this.pc_constraints;
- sess.ice_config = this.ice_config;
-
- sess.initiate(fromJid, false);
- // FIXME: setRemoteDescription should only be done when this call is to be accepted
- sess.setRemoteDescription($(iq).find('>jingle'), 'offer');
-
- this.sessions[sess.sid] = sess;
- this.jid2session[sess.peerjid] = sess;
-
- // the callback should either
- // .sendAnswer and .accept
- // or .sendTerminate -- not necessarily synchronus
- CallIncomingJingle(sess.sid);
- break;
- case 'session-accept':
- sess.setRemoteDescription($(iq).find('>jingle'), 'answer');
- sess.accept();
- $(document).trigger('callaccepted.jingle', [sess.sid]);
- break;
- case 'session-terminate':
- // If this is not the focus sending the terminate, we have
- // nothing more to do here.
- if (Object.keys(this.sessions).length < 1
- || !(this.sessions[Object.keys(this.sessions)[0]]
- instanceof JingleSession))
- {
- break;
- }
- console.log('terminating...', sess.sid);
- sess.terminate();
- this.terminate(sess.sid);
- if ($(iq).find('>jingle>reason').length) {
- $(document).trigger('callterminated.jingle', [
- sess.sid,
- sess.peerjid,
- $(iq).find('>jingle>reason>:first')[0].tagName,
- $(iq).find('>jingle>reason>text').text()
- ]);
- } else {
- $(document).trigger('callterminated.jingle',
- [sess.sid, sess.peerjid]);
- }
- break;
- case 'transport-info':
- sess.addIceCandidate($(iq).find('>jingle>content'));
- break;
- case 'session-info':
- var affected;
- if ($(iq).find('>jingle>ringing[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
- $(document).trigger('ringing.jingle', [sess.sid]);
- } else if ($(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
- affected = $(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
- $(document).trigger('mute.jingle', [sess.sid, affected]);
- } else if ($(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
- affected = $(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
- $(document).trigger('unmute.jingle', [sess.sid, affected]);
- }
- break;
- case 'addsource': // FIXME: proprietary, un-jingleish
- case 'source-add': // FIXME: proprietary
- sess.addSource($(iq).find('>jingle>content'), fromJid);
- break;
- case 'removesource': // FIXME: proprietary, un-jingleish
- case 'source-remove': // FIXME: proprietary
- sess.removeSource($(iq).find('>jingle>content'), fromJid);
- break;
- default:
- console.warn('jingle action not implemented', action);
- break;
- }
- return true;
- },
- initiate: function (peerjid, myjid) { // initiate a new jinglesession to peerjid
- var sess = new JingleSession(myjid || this.connection.jid,
- Math.random().toString(36).substr(2, 12), // random string
- this.connection);
- // configure session
-
- sess.media_constraints = this.media_constraints;
- sess.pc_constraints = this.pc_constraints;
- sess.ice_config = this.ice_config;
-
- sess.initiate(peerjid, true);
- this.sessions[sess.sid] = sess;
- this.jid2session[sess.peerjid] = sess;
- sess.sendOffer();
- return sess;
- },
- terminate: function (sid, reason, text) { // terminate by sessionid (or all sessions)
- if (sid === null || sid === undefined) {
- for (sid in this.sessions) {
- if (this.sessions[sid].state != 'ended') {
- this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
- this.sessions[sid].terminate();
- }
- delete this.jid2session[this.sessions[sid].peerjid];
- delete this.sessions[sid];
- }
- } else if (this.sessions.hasOwnProperty(sid)) {
- if (this.sessions[sid].state != 'ended') {
- this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
- this.sessions[sid].terminate();
- }
- delete this.jid2session[this.sessions[sid].peerjid];
- delete this.sessions[sid];
- }
- },
- // Used to terminate a session when an unavailable presence is received.
- terminateByJid: function (jid) {
- if (this.jid2session.hasOwnProperty(jid)) {
- var sess = this.jid2session[jid];
- if (sess) {
- sess.terminate();
- console.log('peer went away silently', jid);
- delete this.sessions[sess.sid];
- delete this.jid2session[jid];
- $(document).trigger('callterminated.jingle',
- [sess.sid, jid], 'gone');
- }
- }
- },
- terminateRemoteByJid: function (jid, reason) {
- if (this.jid2session.hasOwnProperty(jid)) {
- var sess = this.jid2session[jid];
- if (sess) {
- sess.sendTerminate(reason || (!sess.active()) ? 'kick' : null);
- sess.terminate();
- console.log('terminate peer with jid', sess.sid, jid);
- delete this.sessions[sess.sid];
- delete this.jid2session[jid];
- $(document).trigger('callterminated.jingle',
- [sess.sid, jid, 'kicked']);
- }
- }
- },
- getStunAndTurnCredentials: function () {
- // get stun and turn configuration from server via xep-0215
- // uses time-limited credentials as described in
- // http://tools.ietf.org/html/draft-uberti-behave-turn-rest-00
- //
- // see https://code.google.com/p/prosody-modules/source/browse/mod_turncredentials/mod_turncredentials.lua
- // for a prosody module which implements this
- //
- // currently, this doesn't work with updateIce and therefore credentials with a long
- // validity have to be fetched before creating the peerconnection
- // TODO: implement refresh via updateIce as described in
- // https://code.google.com/p/webrtc/issues/detail?id=1650
- var self = this;
- this.connection.sendIQ(
- $iq({type: 'get', to: this.connection.domain})
- .c('services', {xmlns: 'urn:xmpp:extdisco:1'}).c('service', {host: 'turn.' + this.connection.domain}),
- function (res) {
- var iceservers = [];
- $(res).find('>services>service').each(function (idx, el) {
- el = $(el);
- var dict = {};
- var type = el.attr('type');
- switch (type) {
- case 'stun':
- dict.url = 'stun:' + el.attr('host');
- if (el.attr('port')) {
- dict.url += ':' + el.attr('port');
- }
- iceservers.push(dict);
- break;
- case 'turn':
- case 'turns':
- dict.url = type + ':';
- if (el.attr('username')) { // https://code.google.com/p/webrtc/issues/detail?id=1508
- if (navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./) && parseInt(navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./)[2], 10) < 28) {
- dict.url += el.attr('username') + '@';
- } else {
- dict.username = el.attr('username'); // only works in M28
- }
- }
- dict.url += el.attr('host');
- if (el.attr('port') && el.attr('port') != '3478') {
- dict.url += ':' + el.attr('port');
- }
- if (el.attr('transport') && el.attr('transport') != 'udp') {
- dict.url += '?transport=' + el.attr('transport');
- }
- if (el.attr('password')) {
- dict.credential = el.attr('password');
- }
- iceservers.push(dict);
- break;
- }
- });
- self.ice_config.iceServers = iceservers;
- },
- function (err) {
- console.warn('getting turn credentials failed', err);
- console.warn('is mod_turncredentials or similar installed?');
- }
- );
- // implement push?
- },
-
- /**
- * Populates the log data
- */
- populateData: function () {
- var data = {};
- Object.keys(this.sessions).forEach(function (sid) {
- var session = this.sessions[sid];
- if (session.peerconnection && session.peerconnection.updateLog) {
- // FIXME: should probably be a .dump call
- data["jingle_" + session.sid] = {
- updateLog: session.peerconnection.updateLog,
- stats: session.peerconnection.stats,
- url: window.location.href
- };
- }
- });
- return data;
- }
-});
diff --git a/libs/strophe/strophe.util.js b/libs/strophe/strophe.util.js
deleted file mode 100644
index 126ecf633..000000000
--- a/libs/strophe/strophe.util.js
+++ /dev/null
@@ -1,41 +0,0 @@
-/**
- * Strophe logger implementation. Logs from level WARN and above.
- */
-Strophe.log = function (level, msg) {
- switch(level) {
- case Strophe.LogLevel.WARN:
- console.warn("Strophe: "+msg);
- break;
- case Strophe.LogLevel.ERROR:
- case Strophe.LogLevel.FATAL:
- console.error("Strophe: "+msg);
- break;
- }
-};
-
-Strophe.getStatusString = function(status)
-{
- switch (status)
- {
- case Strophe.Status.ERROR:
- return "ERROR";
- case Strophe.Status.CONNECTING:
- return "CONNECTING";
- case Strophe.Status.CONNFAIL:
- return "CONNFAIL";
- case Strophe.Status.AUTHENTICATING:
- return "AUTHENTICATING";
- case Strophe.Status.AUTHFAIL:
- return "AUTHFAIL";
- case Strophe.Status.CONNECTED:
- return "CONNECTED";
- case Strophe.Status.DISCONNECTED:
- return "DISCONNECTED";
- case Strophe.Status.DISCONNECTING:
- return "DISCONNECTING";
- case Strophe.Status.ATTACHED:
- return "ATTACHED";
- default:
- return "unknown";
- }
-};
diff --git a/moderatemuc.js b/moderatemuc.js
deleted file mode 100644
index e64821dd8..000000000
--- a/moderatemuc.js
+++ /dev/null
@@ -1,56 +0,0 @@
-/* global $, $iq, config, connection, focusMucJid, forceMuted,
- setAudioMuted, Strophe, toggleAudio */
-/**
- * Moderate connection plugin.
- */
-Strophe.addConnectionPlugin('moderate', {
- connection: null,
- init: function (conn) {
- this.connection = conn;
-
- this.connection.addHandler(this.onMute.bind(this),
- 'http://jitsi.org/jitmeet/audio',
- 'iq',
- 'set',
- null,
- null);
- },
- setMute: function (jid, mute) {
- console.info("set mute", mute);
- var iqToFocus = $iq({to: focusMucJid, type: 'set'})
- .c('mute', {
- xmlns: 'http://jitsi.org/jitmeet/audio',
- jid: jid
- })
- .t(mute.toString())
- .up();
-
- this.connection.sendIQ(
- iqToFocus,
- function (result) {
- console.log('set mute', result);
- },
- function (error) {
- console.log('set mute error', error);
- });
- },
- onMute: function (iq) {
- var from = iq.getAttribute('from');
- if (from !== focusMucJid) {
- console.warn("Ignored mute from non focus peer");
- return false;
- }
- var mute = $(iq).find('mute');
- if (mute.length) {
- var doMuteAudio = mute.text() === "true";
- setAudioMuted(doMuteAudio);
- forceMuted = doMuteAudio;
- }
- return true;
- },
- eject: function (jid) {
- // We're not the focus, so can't terminate
- //connection.jingle.terminateRemoteByJid(jid, 'kick');
- connection.emuc.kick(jid);
- }
-});
\ No newline at end of file
diff --git a/modules/API/API.js b/modules/API/API.js
index 934841629..b00a77e50 100644
--- a/modules/API/API.js
+++ b/modules/API/API.js
@@ -18,8 +18,8 @@
var commands =
{
displayName: UI.inputDisplayNameHandler,
- muteAudio: toggleAudio,
- muteVideo: toggleVideo,
+ muteAudio: UI.toggleAudio,
+ muteVideo: UI.toggleVideo,
toggleFilmStrip: UI.toggleFilmStrip,
toggleChat: UI.toggleChat,
toggleContactList: UI.toggleContactList
diff --git a/modules/RTC/DataChannels.js b/modules/RTC/DataChannels.js
index bf981af30..b9e74e994 100644
--- a/modules/RTC/DataChannels.js
+++ b/modules/RTC/DataChannels.js
@@ -1,4 +1,4 @@
-/* global connection, Strophe, updateLargeVideo, focusedVideoSrc*/
+/* global Strophe, updateLargeVideo, focusedVideoSrc*/
// cache datachannels to avoid garbage collection
// https://code.google.com/p/chromium/issues/detail?id=405545
@@ -91,7 +91,7 @@ var DataChannels =
newValue = new Boolean(newValue).valueOf();
}
}
- $(document).trigger('inlastnchanged', [oldValue, newValue]);
+ UI.onLastNChanged(oldValue, newValue);
}
else if ("LastNEndpointsChangeEvent" === colibriClass)
{
diff --git a/modules/RTC/RTC.js b/modules/RTC/RTC.js
index c65ef0b69..9b269c8a4 100644
--- a/modules/RTC/RTC.js
+++ b/modules/RTC/RTC.js
@@ -58,7 +58,7 @@ var RTC = {
createRemoteStream: function (data, sid, thessrc) {
var remoteStream = new MediaStream(data, sid, thessrc, eventEmitter,
this.getBrowserType());
- var jid = data.peerjid || connection.emuc.myroomjid;
+ var jid = data.peerjid || xmpp.myJid();
if(!this.remoteStreams[jid]) {
this.remoteStreams[jid] = {};
}
@@ -144,16 +144,7 @@ var RTC = {
RTC.localVideo = this.createLocalStream(stream, type, true);
// Stop the stream to trigger onended event for old stream
oldStream.stop();
- if (activecall) {
- // FIXME: will block switchInProgress on true value in case of exception
- activecall.switchStreams(stream, oldStream, callback);
- } else {
- // We are done immediately
- console.error("No conference handler");
- UI.messageHandler.showError('Error',
- 'Unable to switch video stream.');
- callback();
- }
+ xmpp.switchStreams(stream, oldStream,callback);
}
};
diff --git a/modules/UI/UI.js b/modules/UI/UI.js
index 5ef0deb48..0fa850941 100644
--- a/modules/UI/UI.js
+++ b/modules/UI/UI.js
@@ -17,9 +17,11 @@ var PanelToggler = require("./side_pannels/SidePanelToggler");
var RoomNameGenerator = require("./welcome_page/RoomnameGenerator");
UI.messageHandler = require("./util/MessageHandler");
var messageHandler = UI.messageHandler;
+var Authentication = require("./authentication/Authentication");
+var UIUtil = require("./util/UIUtil");
//var eventEmitter = new EventEmitter();
-
+var roomName = null;
function setupPrezi()
@@ -39,7 +41,7 @@ function setupChat()
}
function setupToolbars() {
- Toolbar.init();
+ Toolbar.init(UI);
Toolbar.setupButtonsFromConfig();
BottomToolbar.init();
}
@@ -62,6 +64,16 @@ function streamHandler(stream) {
}
}
+function onDisposeConference(unload) {
+ Toolbar.showAuthenticateButton(false);
+};
+
+function onDisplayNameChanged(jid, displayName) {
+ ContactList.onDisplayNameChange(jid, displayName);
+ SettingsMenu.onDisplayNameChange(jid, displayName);
+ VideoLayout.onDisplayNameChanged(jid, displayName);
+}
+
function registerListeners() {
RTC.addStreamListener(streamHandler, StreamEventTypes.EVENT_TYPE_LOCAL_CREATED);
@@ -70,14 +82,7 @@ function registerListeners() {
VideoLayout.onRemoteStreamAdded(stream);
}, StreamEventTypes.EVENT_TYPE_REMOTE_CREATED);
- // Listen for large video size updates
- document.getElementById('largeVideo')
- .addEventListener('loadedmetadata', function (e) {
- currentVideoWidth = this.videoWidth;
- currentVideoHeight = this.videoHeight;
- VideoLayout.positionLarge(currentVideoWidth, currentVideoHeight);
- });
-
+ VideoLayout.init();
statistics.addAudioLevelListener(function(jid, audioLevel)
{
@@ -104,8 +109,38 @@ function registerListeners() {
desktopsharing.addListener(
Toolbar.changeDesktopSharingButtonState,
DesktopSharingEventTypes.SWITCHING_DONE);
+ xmpp.addListener(XMPPEvents.DISPOSE_CONFERENCE, onDisposeConference);
+ xmpp.addListener(XMPPEvents.KICKED, function () {
+ messageHandler.openMessageDialog("Session Terminated",
+ "Ouch! You have been kicked out of the meet!");
+ });
+ xmpp.addListener(XMPPEvents.BRIDGE_DOWN, function () {
+ messageHandler.showError("Error",
+ "Jitsi Videobridge is currently unavailable. Please try again later!");
+ });
+ xmpp.addListener(XMPPEvents.USER_ID_CHANGED, Avatar.setUserAvatar);
+ xmpp.addListener(XMPPEvents.CHANGED_STREAMS, function (jid, changedStreams) {
+ for(stream in changedStreams)
+ {
+ // might need to update the direction if participant just went from sendrecv to recvonly
+ if (stream.type === 'video' || stream.type === 'screen') {
+ var el = $('#participant_' + Strophe.getResourceFromJid(jid) + '>video');
+ switch (stream.direction) {
+ case 'sendrecv':
+ el.show();
+ break;
+ case 'recvonly':
+ el.hide();
+ // FIXME: Check if we have to change large video
+ //VideoLayout.updateLargeVideo(el);
+ break;
+ }
+ }
+ }
-
+ });
+ xmpp.addListener(XMPPEvents.DISPLAY_NAME_CHANGED, onDisplayNameChanged);
+ xmpp.addListener(XMPPEvents.MUC_JOINED, onMucJoined);
}
function bindEvents()
@@ -117,10 +152,6 @@ function bindEvents()
function () {
VideoLayout.resizeLargeVideoContainer();
VideoLayout.positionLarge();
- isFullScreen = document.fullScreen ||
- document.mozFullScreen ||
- document.webkitIsFullScreen;
-
}
);
@@ -255,11 +286,6 @@ UI.start = function () {
};
-
-UI.setUserAvatar = function (jid, id) {
- Avatar.setUserAvatar(jid, id);
-};
-
UI.toggleSmileys = function () {
Chat.toggleSmileys();
};
@@ -278,7 +304,7 @@ UI.updateChatConversation = function (from, displayName, message) {
return Chat.updateChatConversation(from, displayName, message);
};
-UI.onMucJoined = function (jid, info) {
+function onMucJoined(jid, info) {
Toolbar.updateRoomUrl(window.location.href);
document.getElementById('localNick').appendChild(
document.createTextNode(Strophe.getResourceFromJid(jid) + ' (me)')
@@ -293,15 +319,14 @@ UI.onMucJoined = function (jid, info) {
// Show authenticate button if needed
Toolbar.showAuthenticateButton(
- Moderator.isExternalAuthEnabled() && !Moderator.isModerator());
+ xmpp.isExternalAuthEnabled() && !xmpp.isModerator());
var displayName = !config.displayJids
? info.displayName : Strophe.getResourceFromJid(jid);
if (displayName)
- $(document).trigger('displaynamechanged',
- ['localVideoContainer', displayName + ' (me)']);
-};
+ onDisplayNameChanged('localVideoContainer', displayName + ' (me)');
+}
UI.initEtherpad = function (name) {
Etherpad.init(name);
@@ -357,23 +382,19 @@ UI.toggleContactList = function () {
UI.onLocalRoleChange = function (jid, info, pres) {
console.info("My role changed, new role: " + info.role);
- var isModerator = Moderator.isModerator();
+ var isModerator = xmpp.isModerator();
VideoLayout.showModeratorIndicator();
Toolbar.showAuthenticateButton(
- Moderator.isExternalAuthEnabled() && !isModerator);
+ xmpp.isExternalAuthEnabled() && !isModerator);
if (isModerator) {
- Toolbar.closeAuthenticationWindow();
+ Authentication.closeAuthenticationWindow();
messageHandler.notify(
'Me', 'connected', 'Moderator rights granted !');
}
};
-UI.onDisposeConference = function (unload) {
- Toolbar.showAuthenticateButton(false);
-};
-
UI.onModeratorStatusChanged = function (isModerator) {
Toolbar.showSipCallButton(isModerator);
@@ -414,40 +435,11 @@ UI.onPasswordReqiured = function (callback) {
);
};
-UI.onAuthenticationRequired = function () {
- // This is the loop that will wait for the room to be created by
- // someone else. 'auth_required.moderator' will bring us back here.
- authRetryId = window.setTimeout(
- function () {
- Moderator.allocateConferenceFocus(roomName, doJoinAfterFocus);
- }, 5000);
- // Show prompt only if it's not open
- if (authDialog !== null) {
- return;
- }
- // extract room name from 'room@muc.server.net'
- var room = roomName.substr(0, roomName.indexOf('@'));
-
- authDialog = messageHandler.openDialog(
- 'Stop',
- 'Authentication is required to create room: ' + room +
- ' You can either authenticate to create the room or ' +
- 'just wait for someone else to do so.',
- true,
- {
- Authenticate: 'authNow'
- },
- function (onSubmitEvent, submitValue) {
-
- // Do not close the dialog yet
- onSubmitEvent.preventDefault();
-
- // Open login popup
- if (submitValue === 'authNow') {
- Toolbar.authenticateClicked();
- }
- }
- );
+UI.onAuthenticationRequired = function (intervalCallback) {
+ Authentication.openAuthenticationDialog(
+ roomName, intervalCallback, function () {
+ Toolbar.authenticateClicked();
+ });
};
UI.setRecordingButtonState = function (state) {
@@ -511,6 +503,8 @@ UI.showLocalAudioIndicator = function (mute) {
};
UI.generateRoomName = function() {
+ if(roomName)
+ return roomName;
var roomnode = null;
var path = window.location.pathname;
@@ -540,6 +534,7 @@ UI.generateRoomName = function() {
}
roomName = roomnode + '@' + config.hosts.muc;
+ return roomName;
};
@@ -556,25 +551,146 @@ UI.dockToolbar = function (isDock) {
return ToolbarToggler.dockToolbar(isDock);
};
+UI.getCreadentials = function () {
+ return {
+ bosh: document.getElementById('boshURL').value,
+ password: document.getElementById('password').value,
+ jid: document.getElementById('jid').value
+ };
+};
+
+UI.disableConnect = function () {
+ document.getElementById('connect').disabled = true;
+};
+
+UI.showLoginPopup = function(callback)
+{
+ console.log('password is required');
+
+ UI.messageHandler.openTwoButtonDialog(null,
+ '
Password required
' +
+ '' +
+ '',
+ true,
+ "Ok",
+ function (e, v, m, f) {
+ if (v) {
+ var username = document.getElementById('passwordrequired.username');
+ var password = document.getElementById('passwordrequired.password');
+
+ if (username.value !== null && password.value != null) {
+ callback(username.value, password.value);
+ }
+ }
+ },
+ function (event) {
+ document.getElementById('passwordrequired.username').focus();
+ }
+ );
+}
+
+UI.checkForNicknameAndJoin = function () {
+
+ Authentication.closeAuthenticationDialog();
+ Authentication.stopInterval();
+
+ var nick = null;
+ if (config.useNicks) {
+ nick = window.prompt('Your nickname (optional)');
+ }
+ xmpp.joinRooom(roomName, config.useNicks, nick);
+}
+
+
function dump(elem, filename) {
elem = elem.parentNode;
elem.download = filename || 'meetlog.json';
elem.href = 'data:application/json;charset=utf-8,\n';
- var data = {};
- if (connection.jingle) {
- data = connection.jingle.populateData();
- }
+ var data = xmpp.populateData();
var metadata = {};
metadata.time = new Date();
metadata.url = window.location.href;
metadata.ua = navigator.userAgent;
- if (connection.logger) {
- metadata.xmpp = connection.logger.log;
+ var log = xmpp.getLogger();
+ if (log) {
+ metadata.xmpp = log;
}
data.metadata = metadata;
elem.href += encodeURIComponent(JSON.stringify(data, null, ' '));
return false;
}
+UI.getRoomName = function () {
+ return roomName;
+}
+
+/**
+ * Mutes/unmutes the local video.
+ *
+ * @param mute true to mute the local video; otherwise, false
+ * @param options an object which specifies optional arguments such as the
+ * boolean key byUser with default value true which
+ * specifies whether the method was initiated in response to a user command (in
+ * contrast to an automatic decision taken by the application logic)
+ */
+function setVideoMute(mute, options) {
+ xmpp.setVideoMute(
+ mute,
+ function (mute) {
+ var video = $('#video');
+ var communicativeClass = "icon-camera";
+ var muteClass = "icon-camera icon-camera-disabled";
+
+ if (mute) {
+ video.removeClass(communicativeClass);
+ video.addClass(muteClass);
+ } else {
+ video.removeClass(muteClass);
+ video.addClass(communicativeClass);
+ }
+ },
+ options);
+}
+
+/**
+ * Mutes/unmutes the local video.
+ */
+UI.toggleVideo = function () {
+ UIUtil.buttonClick("#video", "icon-camera icon-camera-disabled");
+
+ setVideoMute(!RTC.localVideo.isMuted());
+};
+
+/**
+ * Mutes / unmutes audio for the local participant.
+ */
+UI.toggleAudio = function() {
+ UI.setAudioMuted(!RTC.localAudio.isMuted());
+};
+
+/**
+ * Sets muted audio state for the local participant.
+ */
+UI.setAudioMuted = function (mute) {
+
+ if(!xmpp.setAudioMute(mute, function () {
+ UI.showLocalAudioIndicator(mute);
+
+ UIUtil.buttonClick("#mute", "icon-microphone icon-mic-disabled");
+ }))
+ {
+ // We still click the button.
+ UIUtil.buttonClick("#mute", "icon-microphone icon-mic-disabled");
+ return;
+ }
+
+}
+
+UI.onLastNChanged = function (oldValue, newValue) {
+ if (config.muteLocalVideoIfNotInLastN) {
+ setVideoMute(!newValue, { 'byUser': false });
+ }
+}
+
module.exports = UI;
diff --git a/modules/UI/audio_levels/AudioLevels.js b/modules/UI/audio_levels/AudioLevels.js
index 350c353f0..115c54510 100644
--- a/modules/UI/audio_levels/AudioLevels.js
+++ b/modules/UI/audio_levels/AudioLevels.js
@@ -87,10 +87,10 @@ var AudioLevels = (function(my) {
drawContext.drawImage(canvasCache, 0, 0);
if(resourceJid === AudioLevels.LOCAL_LEVEL) {
- if(!connection.emuc.myroomjid) {
+ if(!xmpp.myJid()) {
return;
}
- resourceJid = Strophe.getResourceFromJid(connection.emuc.myroomjid);
+ resourceJid = xmpp.myResource();
}
if(resourceJid === largeVideoResourceJid) {
@@ -221,8 +221,8 @@ var AudioLevels = (function(my) {
function getVideoSpanId(resourceJid) {
var videoSpanId = null;
if (resourceJid === AudioLevels.LOCAL_LEVEL
- || (connection.emuc.myroomjid && resourceJid
- === Strophe.getResourceFromJid(connection.emuc.myroomjid)))
+ || (xmpp.myResource() && resourceJid
+ === xmpp.myResource()))
videoSpanId = 'localVideoContainer';
else
videoSpanId = 'participant_' + resourceJid;
diff --git a/modules/UI/authentication/Authentication.js b/modules/UI/authentication/Authentication.js
new file mode 100644
index 000000000..1a568d5ce
--- /dev/null
+++ b/modules/UI/authentication/Authentication.js
@@ -0,0 +1,84 @@
+/* Initial "authentication required" dialog */
+var authDialog = null;
+/* Loop retry ID that wits for other user to create the room */
+var authRetryId = null;
+var authenticationWindow = null;
+
+var Authentication = {
+ openAuthenticationDialog: function (roomName, intervalCallback, callback) {
+ // This is the loop that will wait for the room to be created by
+ // someone else. 'auth_required.moderator' will bring us back here.
+ authRetryId = window.setTimeout(intervalCallback , 5000);
+ // Show prompt only if it's not open
+ if (authDialog !== null) {
+ return;
+ }
+ // extract room name from 'room@muc.server.net'
+ var room = roomName.substr(0, roomName.indexOf('@'));
+
+ authDialog = messageHandler.openDialog(
+ 'Stop',
+ 'Authentication is required to create room: ' + room +
+ ' You can either authenticate to create the room or ' +
+ 'just wait for someone else to do so.',
+ true,
+ {
+ Authenticate: 'authNow'
+ },
+ function (onSubmitEvent, submitValue) {
+
+ // Do not close the dialog yet
+ onSubmitEvent.preventDefault();
+
+ // Open login popup
+ if (submitValue === 'authNow') {
+ callback();
+ }
+ }
+ );
+ },
+ closeAuthenticationWindow:function () {
+ if (authenticationWindow) {
+ authenticationWindow.close();
+ authenticationWindow = null;
+ }
+ },
+ focusAuthenticationWindow: function () {
+ // If auth window exists just bring it to the front
+ if (authenticationWindow) {
+ authenticationWindow.focus();
+ return;
+ }
+ },
+ closeAuthenticationDialog: function () {
+ // Close authentication dialog if opened
+ if (authDialog) {
+ UI.messageHandler.closeDialog();
+ authDialog = null;
+ }
+ },
+ createAuthenticationWindow: function (callback, url) {
+ authenticationWindow = messageHandler.openCenteredPopup(
+ url, 910, 660,
+ // On closed
+ function () {
+ // Close authentication dialog if opened
+ if (authDialog) {
+ messageHandler.closeDialog();
+ authDialog = null;
+ }
+ callback();
+ authenticationWindow = null;
+ });
+ return authenticationWindow;
+ },
+ stopInterval: function () {
+ // Clear retry interval, so that we don't call 'doJoinAfterFocus' twice
+ if (authRetryId) {
+ window.clearTimeout(authRetryId);
+ authRetryId = null;
+ }
+ }
+};
+
+module.exports = Authentication;
\ No newline at end of file
diff --git a/modules/UI/avatar/Avatar.js b/modules/UI/avatar/Avatar.js
index 60825ef7e..0e18477cd 100644
--- a/modules/UI/avatar/Avatar.js
+++ b/modules/UI/avatar/Avatar.js
@@ -12,7 +12,7 @@ function setVisibility(selector, show) {
function isUserMuted(jid) {
// XXX(gp) we may want to rename this method to something like
// isUserStreaming, for example.
- if (jid && jid != connection.emuc.myroomjid) {
+ if (jid && jid != xmpp.myJid()) {
var resource = Strophe.getResourceFromJid(jid);
if (!require("../videolayout/VideoLayout").isInLastN(resource)) {
return true;
@@ -26,7 +26,7 @@ function isUserMuted(jid) {
}
function getGravatarUrl(id, size) {
- if(id === connection.emuc.myroomjid || !id) {
+ if(id === xmpp.myJid() || !id) {
id = Settings.getSettings().uid;
}
return 'https://www.gravatar.com/avatar/' +
@@ -57,7 +57,7 @@ var Avatar = {
// set the avatar in the settings menu if it is local user and get the
// local video container
- if (jid === connection.emuc.myroomjid) {
+ if (jid === xmpp.myJid()) {
$('#avatar').get(0).src = thumbUrl;
thumbnail = $('#localVideoContainer');
}
@@ -100,7 +100,7 @@ var Avatar = {
var video = $('#participant_' + resourceJid + '>video');
var avatar = $('#avatar_' + resourceJid);
- if (jid === connection.emuc.myroomjid) {
+ if (jid === xmpp.myJid()) {
video = $('#localVideoWrapper>video');
}
if (show === undefined || show === null) {
@@ -130,7 +130,7 @@ var Avatar = {
*/
updateActiveSpeakerAvatarSrc: function (jid) {
if (!jid) {
- jid = connection.emuc.findJidFromResource(
+ jid = xmpp.findJidFromResource(
require("../videolayout/VideoLayout").getLargeVideoState().userResourceJid);
}
var avatar = $("#activeSpeakerAvatar")[0];
diff --git a/modules/UI/etherpad/Etherpad.js b/modules/UI/etherpad/Etherpad.js
index 8fc4a25f2..6dcfab458 100644
--- a/modules/UI/etherpad/Etherpad.js
+++ b/modules/UI/etherpad/Etherpad.js
@@ -1,4 +1,4 @@
-/* global $, config, connection, dockToolbar, Moderator,
+/* global $, config, dockToolbar,
setLargeVideoVisible, Util */
var VideoLayout = require("../videolayout/VideoLayout");
@@ -30,8 +30,7 @@ function resize() {
* Shares the Etherpad name with other participants.
*/
function shareEtherpad() {
- connection.emuc.addEtherpadToPresence(etherpadName);
- connection.emuc.sendPresence();
+ xmpp.addToPresence("etherpad", etherpadName);
}
/**
diff --git a/modules/UI/prezi/Prezi.js b/modules/UI/prezi/Prezi.js
index 03885c1c4..7ceb70c2e 100644
--- a/modules/UI/prezi/Prezi.js
+++ b/modules/UI/prezi/Prezi.js
@@ -30,7 +30,7 @@ var Prezi = {
* to load.
*/
openPreziDialog: function() {
- var myprezi = connection.emuc.getPrezi(connection.emuc.myroomjid);
+ var myprezi = xmpp.getPrezi();
if (myprezi) {
messageHandler.openTwoButtonDialog("Remove Prezi",
"Are you sure you would like to remove your Prezi?",
@@ -38,8 +38,7 @@ var Prezi = {
"Remove",
function(e,v,m,f) {
if(v) {
- connection.emuc.removePreziFromPresence();
- connection.emuc.sendPresence();
+ xmpp.removePreziFromPresence();
}
}
);
@@ -91,9 +90,7 @@ var Prezi = {
return false;
}
else {
- connection.emuc
- .addPreziToPresence(urlValue, 0);
- connection.emuc.sendPresence();
+ xmpp.addToPresence("prezi", urlValue);
$.prompt.close();
}
}
@@ -151,7 +148,7 @@ function presentationAdded(event, jid, presUrl, currentSlide) {
VideoLayout.resizeThumbnails();
var controlsEnabled = false;
- if (jid === connection.emuc.myroomjid)
+ if (jid === xmpp.myJid())
controlsEnabled = true;
setPresentationVisible(true);
@@ -191,15 +188,14 @@ function presentationAdded(event, jid, presUrl, currentSlide) {
preziPlayer.on(PreziPlayer.EVENT_STATUS, function(event) {
console.log("prezi status", event.value);
if (event.value == PreziPlayer.STATUS_CONTENT_READY) {
- if (jid != connection.emuc.myroomjid)
+ if (jid != xmpp.myJid())
preziPlayer.flyToStep(currentSlide);
}
});
preziPlayer.on(PreziPlayer.EVENT_CURRENT_STEP, function(event) {
console.log("event value", event.value);
- connection.emuc.addCurrentSlideToPresence(event.value);
- connection.emuc.sendPresence();
+ xmpp.addToPresence("preziSlide", event.value);
});
$("#" + elementId).css( 'background-image',
diff --git a/modules/UI/side_pannels/SidePanelToggler.js b/modules/UI/side_pannels/SidePanelToggler.js
index 44bbf042d..938e869d4 100644
--- a/modules/UI/side_pannels/SidePanelToggler.js
+++ b/modules/UI/side_pannels/SidePanelToggler.js
@@ -4,6 +4,7 @@ var Settings = require("./settings/Settings");
var SettingsMenu = require("./settings/SettingsMenu");
var VideoLayout = require("../videolayout/VideoLayout");
var ToolbarToggler = require("../toolbars/ToolbarToggler");
+var UIUtil = require("../util/UIUtil");
/**
* Toggler for the chat, contact list, settings menu, etc..
@@ -110,7 +111,7 @@ var PanelToggler = (function(my) {
* @param onClose function to be called if the window is going to be closed
*/
var toggle = function(object, selector, onOpenComplete, onOpen, onClose) {
- buttonClick(buttons[selector], "active");
+ UIUtil.buttonClick(buttons[selector], "active");
if (object.isVisible()) {
$("#toast-container").animate({
@@ -140,7 +141,7 @@ var PanelToggler = (function(my) {
if(currentlyOpen) {
var current = $(currentlyOpen);
- buttonClick(buttons[currentlyOpen], "active");
+ UIUtil.buttonClick(buttons[currentlyOpen], "active");
current.css('z-index', 4);
setTimeout(function () {
current.css('display', 'none');
diff --git a/modules/UI/side_pannels/chat/Chat.js b/modules/UI/side_pannels/chat/Chat.js
index 09683fea0..6ce82bf30 100644
--- a/modules/UI/side_pannels/chat/Chat.js
+++ b/modules/UI/side_pannels/chat/Chat.js
@@ -1,4 +1,4 @@
-/* global $, Util, connection, nickname:true, showToolbar */
+/* global $, Util, nickname:true, showToolbar */
var Replacement = require("./Replacement");
var CommandsProcessor = require("./Commands");
var ToolbarToggler = require("../../toolbars/ToolbarToggler");
@@ -184,8 +184,7 @@ var Chat = (function (my) {
nickname = val;
window.localStorage.displayname = nickname;
- connection.emuc.addDisplayNameToPresence(nickname);
- connection.emuc.sendPresence();
+ xmpp.addToPresence("displayName", nickname);
Chat.setChatConversationMode(true);
@@ -208,7 +207,7 @@ var Chat = (function (my) {
else
{
var message = Util.escapeHtml(value);
- connection.emuc.sendMessage(message, nickname);
+ xmpp.sendChatMessage(message, nickname);
}
}
});
@@ -234,7 +233,7 @@ var Chat = (function (my) {
my.updateChatConversation = function (from, displayName, message) {
var divClassName = '';
- if (connection.emuc.myroomjid === from) {
+ if (xmpp.myJid() === from) {
divClassName = "localuser";
}
else {
diff --git a/modules/UI/side_pannels/chat/Commands.js b/modules/UI/side_pannels/chat/Commands.js
index a9d926f83..6883268c3 100644
--- a/modules/UI/side_pannels/chat/Commands.js
+++ b/modules/UI/side_pannels/chat/Commands.js
@@ -32,7 +32,7 @@ function getCommand(message)
function processTopic(commandArguments)
{
var topic = Util.escapeHtml(commandArguments);
- connection.emuc.setSubject(topic);
+ xmpp.setSubject(topic);
}
/**
diff --git a/modules/UI/side_pannels/contactlist/ContactList.js b/modules/UI/side_pannels/contactlist/ContactList.js
index 8a1b8ec8c..b4f3feb7f 100644
--- a/modules/UI/side_pannels/contactlist/ContactList.js
+++ b/modules/UI/side_pannels/contactlist/ContactList.js
@@ -46,23 +46,6 @@ function createDisplayNameParagraph(displayName) {
}
-/**
- * Indicates that the display name has changed.
- */
-$(document).bind( 'displaynamechanged',
- function (event, peerJid, displayName) {
- if (peerJid === 'localVideoContainer')
- peerJid = connection.emuc.myroomjid;
-
- var resourceJid = Strophe.getResourceFromJid(peerJid);
-
- var contactName = $('#contactlist #' + resourceJid + '>p');
-
- if (contactName && displayName && displayName.length > 0)
- contactName.html(displayName);
- });
-
-
function stopGlowing(glower) {
window.clearInterval(notificationInterval);
notificationInterval = false;
@@ -127,7 +110,7 @@ var ContactList = {
var clElement = contactlist.get(0);
- if (resourceJid === Strophe.getResourceFromJid(connection.emuc.myroomjid)
+ if (resourceJid === xmpp.myResource()
&& $('#contactlist>ul .title')[0].nextSibling.nextSibling) {
clElement.insertBefore(newContact,
$('#contactlist>ul .title')[0].nextSibling.nextSibling);
@@ -182,6 +165,18 @@ var ContactList = {
} else {
contact.removeClass('clickable');
}
+ },
+
+ onDisplayNameChange: function (peerJid, displayName) {
+ if (peerJid === 'localVideoContainer')
+ peerJid = xmpp.myJid();
+
+ var resourceJid = Strophe.getResourceFromJid(peerJid);
+
+ var contactName = $('#contactlist #' + resourceJid + '>p');
+
+ if (contactName && displayName && displayName.length > 0)
+ contactName.html(displayName);
}
};
diff --git a/modules/UI/side_pannels/settings/SettingsMenu.js b/modules/UI/side_pannels/settings/SettingsMenu.js
index 82ee205f5..25fca6145 100644
--- a/modules/UI/side_pannels/settings/SettingsMenu.js
+++ b/modules/UI/side_pannels/settings/SettingsMenu.js
@@ -10,16 +10,15 @@ var SettingsMenu = {
if(newDisplayName) {
var displayName = Settings.setDisplayName(newDisplayName);
- connection.emuc.addDisplayNameToPresence(displayName);
+ xmpp.addToPresence("displayName", displayName, true);
}
- connection.emuc.addEmailToPresence(newEmail);
+ xmpp.addToPresence("email", newEmail);
var email = Settings.setEmail(newEmail);
- connection.emuc.sendPresence();
- Avatar.setUserAvatar(connection.emuc.myroomjid, email);
+ Avatar.setUserAvatar(xmpp.myJid(), email);
},
isVisible: function() {
@@ -29,14 +28,15 @@ var SettingsMenu = {
setDisplayName: function(newDisplayName) {
var displayName = Settings.setDisplayName(newDisplayName);
$('#setDisplayName').get(0).value = displayName;
+ },
+
+ onDisplayNameChange: function(peerJid, newDisplayName) {
+ if(peerJid === 'localVideoContainer' ||
+ peerJid === xmpp.myJid()) {
+ this.setDisplayName(newDisplayName);
+ }
}
};
-$(document).bind('displaynamechanged', function(event, peerJid, newDisplayName) {
- if(peerJid === 'localVideoContainer' ||
- peerJid === connection.emuc.myroomjid) {
- SettingsMenu.setDisplayName(newDisplayName);
- }
-});
module.exports = SettingsMenu;
\ No newline at end of file
diff --git a/modules/UI/toolbars/Toolbar.js b/modules/UI/toolbars/Toolbar.js
index 87597ed1b..55ba8949f 100644
--- a/modules/UI/toolbars/Toolbar.js
+++ b/modules/UI/toolbars/Toolbar.js
@@ -1,22 +1,24 @@
-/* global $, buttonClick, config, lockRoom, Moderator, roomName,
- setSharedKey, sharedKey, Util */
+/* global $, buttonClick, config, lockRoom,
+ setSharedKey, Util */
var messageHandler = require("../util/MessageHandler");
var BottomToolbar = require("./BottomToolbar");
var Prezi = require("../prezi/Prezi");
var Etherpad = require("../etherpad/Etherpad");
var PanelToggler = require("../side_pannels/SidePanelToggler");
+var Authentication = require("../authentication/Authentication");
+var UIUtil = require("../util/UIUtil");
var roomUrl = null;
var sharedKey = '';
-var authenticationWindow = null;
+var UI = null;
var buttonHandlers =
{
"toolbar_button_mute": function () {
- return toggleAudio();
+ return UI.toggleAudio();
},
"toolbar_button_camera": function () {
- return toggleVideo();
+ return UI.toggleVideo();
},
"toolbar_button_authentication": function () {
return Toolbar.authenticateClicked();
@@ -44,7 +46,7 @@ var buttonHandlers =
},
"toolbar_button_fullScreen": function()
{
- buttonClick("#fullScreen", "icon-full-screen icon-exit-full-screen");
+ UIUtil.buttonClick("#fullScreen", "icon-full-screen icon-exit-full-screen");
return Toolbar.toggleFullScreen();
},
"toolbar_button_sip": function () {
@@ -59,9 +61,7 @@ var buttonHandlers =
};
function hangup() {
- disposeConference();
- sessionTerminated = true;
- connection.emuc.doLeave();
+ xmpp.disposeConference();
if(config.enableWelcomePage)
{
setTimeout(function()
@@ -90,7 +90,29 @@ function hangup() {
*/
function toggleRecording() {
- Recording.toggleRecording();
+ xmpp.toggleRecording(function (callback) {
+ UI.messageHandler.openTwoButtonDialog(null,
+ '
Enter recording token
' +
+ '',
+ false,
+ "Save",
+ function (e, v, m, f) {
+ if (v) {
+ var token = document.getElementById('recordingToken');
+
+ if (token.value) {
+ callback(Util.escapeHtml(token.value));
+ }
+ }
+ },
+ function (event) {
+ document.getElementById('recordingToken').focus();
+ },
+ function () {
+ }
+ );
+ }, Toolbar.setRecordingButtonState, Toolbar.setRecordingButtonState);
}
/**
@@ -101,7 +123,7 @@ function lockRoom(lock) {
if (lock)
currentSharedKey = sharedKey;
- connection.emuc.lockRoom(currentSharedKey, function (res) {
+ xmpp.lockRoom(currentSharedKey, function (res) {
// password is required
if (sharedKey)
{
@@ -183,9 +205,8 @@ function callSipButtonClicked()
if (v) {
var numberInput = document.getElementById('sipNumber');
if (numberInput.value) {
- connection.rayo.dial(
- numberInput.value, 'fromnumber',
- roomName, sharedKey);
+ xmpp.dial(numberInput.value, 'fromnumber',
+ UI.getRoomName(), sharedKey);
}
}
},
@@ -197,9 +218,10 @@ function callSipButtonClicked()
var Toolbar = (function (my) {
- my.init = function () {
+ my.init = function (ui) {
for(var k in buttonHandlers)
$("#" + k).click(buttonHandlers[k]);
+ UI = ui;
}
/**
@@ -210,35 +232,15 @@ var Toolbar = (function (my) {
sharedKey = sKey;
};
- my.closeAuthenticationWindow = function () {
- if (authenticationWindow) {
- authenticationWindow.close();
- authenticationWindow = null;
- }
- }
-
my.authenticateClicked = function () {
- // If auth window exists just bring it to the front
- if (authenticationWindow) {
- authenticationWindow.focus();
- return;
- }
+ Authentication.focusAuthenticationWindow();
// Get authentication URL
- Moderator.getAuthUrl(function (url) {
+ xmpp.getAuthUrl(UI.getRoomName(), function (url) {
// Open popup with authentication URL
- authenticationWindow = messageHandler.openCenteredPopup(
- url, 910, 660,
- // On closed
- function () {
- // Close authentication dialog if opened
- if (authDialog) {
- messageHandler.closeDialog();
- authDialog = null;
- }
- // On popup closed - retry room allocation
- Moderator.allocateConferenceFocus(roomName, doJoinAfterFocus);
- authenticationWindow = null;
- });
+ var authenticationWindow = Authentication.createAuthenticationWindow(function () {
+ // On popup closed - retry room allocation
+ xmpp.allocateConferenceFocus(UI.getRoomName(), UI.checkForNicknameAndJoin);
+ }, url);
if (!authenticationWindow) {
Toolbar.showAuthenticateButton(true);
messageHandler.openMessageDialog(
@@ -279,7 +281,7 @@ var Toolbar = (function (my) {
*/
my.openLockDialog = function () {
// Only the focus is able to set a shared key.
- if (!Moderator.isModerator()) {
+ if (!xmpp.isModerator()) {
if (sharedKey) {
messageHandler.openMessageDialog(null,
"This conversation is currently protected by" +
@@ -436,14 +438,14 @@ var Toolbar = (function (my) {
*/
my.unlockLockButton = function () {
if ($("#lockIcon").hasClass("icon-security-locked"))
- buttonClick("#lockIcon", "icon-security icon-security-locked");
+ UIUtil.buttonClick("#lockIcon", "icon-security icon-security-locked");
};
/**
* Updates the lock button state to locked.
*/
my.lockLockButton = function () {
if ($("#lockIcon").hasClass("icon-security"))
- buttonClick("#lockIcon", "icon-security icon-security-locked");
+ UIUtil.buttonClick("#lockIcon", "icon-security icon-security-locked");
};
/**
@@ -486,7 +488,7 @@ var Toolbar = (function (my) {
// Shows or hides SIP calls button
my.showSipCallButton = function (show) {
- if (Moderator.isSipGatewayEnabled() && show) {
+ if (xmpp.isSipGatewayEnabled() && show) {
$('#sipCallButton').css({display: "inline"});
} else {
$('#sipCallButton').css({display: "none"});
diff --git a/modules/UI/toolbars/ToolbarToggler.js b/modules/UI/toolbars/ToolbarToggler.js
index 6a5655702..f64664684 100644
--- a/modules/UI/toolbars/ToolbarToggler.js
+++ b/modules/UI/toolbars/ToolbarToggler.js
@@ -67,7 +67,7 @@ var ToolbarToggler = {
toolbarTimeout = interfaceConfig.TOOLBAR_TIMEOUT;
}
- if (Moderator.isModerator())
+ if (xmpp.isModerator())
{
// TODO: Enable settings functionality.
// Need to uncomment the settings button in index.html.
diff --git a/modules/UI/util/UIUtil.js b/modules/UI/util/UIUtil.js
index fa0f4f87c..efb8a2b7b 100644
--- a/modules/UI/util/UIUtil.js
+++ b/modules/UI/util/UIUtil.js
@@ -11,6 +11,13 @@ module.exports = {
= PanelToggler.isVisible() ? PanelToggler.getPanelSize()[0] : 0;
return window.innerWidth - rightPanelWidth;
+ },
+ /**
+ * Changes the style class of the element given by id.
+ */
+ buttonClick: function(id, classname) {
+ $(id).toggleClass(classname); // add the class to the clicked element
}
+
};
\ No newline at end of file
diff --git a/modules/UI/videolayout/VideoLayout.js b/modules/UI/videolayout/VideoLayout.js
index 6061055a6..024755dd4 100644
--- a/modules/UI/videolayout/VideoLayout.js
+++ b/modules/UI/videolayout/VideoLayout.js
@@ -16,8 +16,99 @@ var largeVideoState = {
newSrc: ''
};
+/**
+ * Indicates if we have muted our audio before the conference has started.
+ * @type {boolean}
+ */
+var preMuted = false;
+
+var mutedAudios = {};
+
+var flipXLocalVideo = true;
+var currentVideoWidth = null;
+var currentVideoHeight = null;
+
+var localVideoSrc = null;
+
var defaultLocalDisplayName = "Me";
+function videoactive( videoelem) {
+ if (videoelem.attr('id').indexOf('mixedmslabel') === -1) {
+ // ignore mixedmslabela0 and v0
+
+ videoelem.show();
+ VideoLayout.resizeThumbnails();
+
+ var videoParent = videoelem.parent();
+ var parentResourceJid = null;
+ if (videoParent)
+ parentResourceJid
+ = VideoLayout.getPeerContainerResourceJid(videoParent[0]);
+
+ // Update the large video to the last added video only if there's no
+ // current dominant, focused speaker or prezi playing or update it to
+ // the current dominant speaker.
+ if ((!focusedVideoInfo &&
+ !VideoLayout.getDominantSpeakerResourceJid() &&
+ !require("../prezi/Prezi").isPresentationVisible()) ||
+ (parentResourceJid &&
+ VideoLayout.getDominantSpeakerResourceJid() === parentResourceJid)) {
+ VideoLayout.updateLargeVideo(
+ RTC.getVideoSrc(videoelem[0]),
+ 1,
+ parentResourceJid);
+ }
+
+ VideoLayout.showModeratorIndicator();
+ }
+}
+
+function waitForRemoteVideo(selector, ssrc, stream, jid) {
+ // XXX(gp) so, every call to this function is *always* preceded by a call
+ // to the RTC.attachMediaStream() function but that call is *not* followed
+ // by an update to the videoSrcToSsrc map!
+ //
+ // The above way of doing things results in video SRCs that don't correspond
+ // to any SSRC for a short period of time (to be more precise, for as long
+ // the waitForRemoteVideo takes to complete). This causes problems (see
+ // bellow).
+ //
+ // I'm wondering why we need to do that; i.e. why call RTC.attachMediaStream()
+ // a second time in here and only then update the videoSrcToSsrc map? Why
+ // not simply update the videoSrcToSsrc map when the RTC.attachMediaStream()
+ // is called the first time? I actually do that in the lastN changed event
+ // handler because the "orphan" video SRC is causing troubles there. The
+ // purpose of this method would then be to fire the "videoactive.jingle".
+ //
+ // Food for though I guess :-)
+
+ if (selector.removed || !selector.parent().is(":visible")) {
+ console.warn("Media removed before had started", selector);
+ return;
+ }
+
+ if (stream.id === 'mixedmslabel') return;
+
+ if (selector[0].currentTime > 0) {
+ var videoStream = simulcast.getReceivingVideoStream(stream);
+ RTC.attachMediaStream(selector, videoStream); // FIXME: why do i have to do this for FF?
+
+ // FIXME: add a class that will associate peer Jid, video.src, it's ssrc and video type
+ // in order to get rid of too many maps
+ if (ssrc && jid) {
+ jid2Ssrc[Strophe.getResourceFromJid(jid)] = ssrc;
+ } else {
+ console.warn("No ssrc given for jid", jid);
+ }
+
+ videoactive(selector);
+ } else {
+ setTimeout(function () {
+ waitForRemoteVideo(selector, ssrc, stream, jid);
+ }, 250);
+ }
+}
+
/**
* Returns an array of the video horizontal and vertical indents,
* so that if fits its parent.
@@ -194,7 +285,7 @@ function getParticipantContainer(resourceJid)
if (!resourceJid)
return null;
- if (resourceJid === Strophe.getResourceFromJid(connection.emuc.myroomjid))
+ if (resourceJid === xmpp.myResource())
return $("#localVideoContainer");
else
return $("#participant_" + resourceJid);
@@ -270,7 +361,8 @@ function addRemoteVideoMenu(jid, parentElement) {
event.preventDefault();
}
var isMute = mutedAudios[jid] == true;
- connection.moderate.setMute(jid, !isMute);
+ xmpp.setMute(jid, !isMute);
+
popupmenuElement.setAttribute('style', 'display:none;');
if (isMute) {
@@ -292,7 +384,7 @@ function addRemoteVideoMenu(jid, parentElement) {
var ejectLinkItem = document.createElement('a');
ejectLinkItem.innerHTML = ejectIndicator + ' Kick out';
ejectLinkItem.onclick = function(){
- connection.moderate.eject(jid);
+ xmpp.eject(jid);
popupmenuElement.setAttribute('style', 'display:none;');
};
@@ -400,6 +492,43 @@ function createModeratorIndicatorElement(parentElement) {
}
+/**
+ * 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}
+ */
+function isVideoSrcDesktop(jid) {
+ // FIXME: fix this mapping mess...
+ // figure out if large video is desktop stream or just a camera
+
+ if(!jid)
+ return false;
+ var isDesktop = false;
+ if (xmpp.myJid() &&
+ xmpp.myResource() === jid) {
+ // local video
+ isDesktop = desktopsharing.isUsingScreenStream();
+ } else {
+ // Do we have associations...
+ var videoSsrc = jid2Ssrc[jid];
+ if (videoSsrc) {
+ var videoType = ssrc2videoType[videoSsrc];
+ if (videoType) {
+ // Finally there...
+ isDesktop = videoType === 'screen';
+ } else {
+ console.error("No video type for ssrc: " + videoSsrc);
+ }
+ } else {
+ console.error("No ssrc for jid: " + jid);
+ }
+ }
+ return isDesktop;
+}
+
+
+
var VideoLayout = (function (my) {
my.connectionIndicators = {};
@@ -407,6 +536,16 @@ var VideoLayout = (function (my) {
my.getVideoSize = getCameraVideoSize;
my.getVideoPosition = getCameraVideoPosition;
+ my.init = function () {
+ // Listen for large video size updates
+ document.getElementById('largeVideo')
+ .addEventListener('loadedmetadata', function (e) {
+ currentVideoWidth = this.videoWidth;
+ currentVideoHeight = this.videoHeight;
+ VideoLayout.positionLarge(currentVideoWidth, currentVideoHeight);
+ });
+ };
+
my.isInLastN = function(resource) {
return lastNCount < 0 // lastN is disabled, return true
|| (lastNCount > 0 && lastNEndpointsCache.length == 0) // lastNEndpoints cache not built yet, return true
@@ -422,7 +561,10 @@ var VideoLayout = (function (my) {
document.getElementById('localAudio').autoplay = true;
document.getElementById('localAudio').volume = 0;
if (preMuted) {
- setAudioMuted(true);
+ if(!UI.setAudioMuted(true))
+ {
+ preMuted = mute;
+ }
preMuted = false;
}
};
@@ -459,14 +601,14 @@ var VideoLayout = (function (my) {
VideoLayout.handleVideoThumbClicked(
RTC.getVideoSrc(localVideo),
false,
- Strophe.getResourceFromJid(connection.emuc.myroomjid));
+ xmpp.myResource());
});
$('#localVideoContainer').click(function (event) {
event.stopPropagation();
VideoLayout.handleVideoThumbClicked(
RTC.getVideoSrc(localVideo),
false,
- Strophe.getResourceFromJid(connection.emuc.myroomjid));
+ xmpp.myResource());
});
// Add hover handler
@@ -496,11 +638,8 @@ var VideoLayout = (function (my) {
localVideoSrc = RTC.getVideoSrc(localVideo);
- var myResourceJid = null;
- if(connection.emuc.myroomjid)
- {
- myResourceJid = Strophe.getResourceFromJid(connection.emuc.myroomjid);
- }
+ var myResourceJid = xmpp.myResource();
+
VideoLayout.updateLargeVideo(localVideoSrc, 0,
myResourceJid);
@@ -539,7 +678,7 @@ var VideoLayout = (function (my) {
{
if(container.id == "localVideoWrapper")
{
- jid = Strophe.getResourceFromJid(connection.emuc.myroomjid);
+ jid = xmpp.myResource();
}
else
{
@@ -617,9 +756,9 @@ var VideoLayout = (function (my) {
largeVideoState.isVisible = $('#largeVideo').is(':visible');
largeVideoState.isDesktop = isVideoSrcDesktop(resourceJid);
if(jid2Ssrc[largeVideoState.userResourceJid] ||
- (connection && connection.emuc.myroomjid &&
+ (xmpp.myResource() &&
largeVideoState.userResourceJid ===
- Strophe.getResourceFromJid(connection.emuc.myroomjid))) {
+ xmpp.myResource())) {
largeVideoState.oldResourceJid = largeVideoState.userResourceJid;
} else {
largeVideoState.oldResourceJid = null;
@@ -643,7 +782,7 @@ var VideoLayout = (function (my) {
var doUpdate = function () {
Avatar.updateActiveSpeakerAvatarSrc(
- connection.emuc.findJidFromResource(
+ xmpp.findJidFromResource(
largeVideoState.userResourceJid));
if (!userChanged && largeVideoState.preload &&
@@ -723,7 +862,7 @@ var VideoLayout = (function (my) {
if(userChanged) {
Avatar.showUserAvatar(
- connection.emuc.findJidFromResource(
+ xmpp.findJidFromResource(
largeVideoState.oldResourceJid));
}
@@ -738,7 +877,7 @@ var VideoLayout = (function (my) {
}
} else {
Avatar.showUserAvatar(
- connection.emuc.findJidFromResource(
+ xmpp.findJidFromResource(
largeVideoState.userResourceJid));
}
@@ -877,7 +1016,7 @@ var VideoLayout = (function (my) {
focusedVideoInfo = null;
if(focusResourceJid) {
Avatar.showUserAvatar(
- connection.emuc.findJidFromResource(focusResourceJid));
+ xmpp.findJidFromResource(focusResourceJid));
}
}
}
@@ -949,7 +1088,7 @@ var VideoLayout = (function (my) {
// If the peerJid is null then this video span couldn't be directly
// associated with a participant (this could happen in the case of prezi).
- if (Moderator.isModerator() && peerJid !== null)
+ if (xmpp.isModerator() && peerJid !== null)
addRemoteVideoMenu(peerJid, container);
remotes.appendChild(container);
@@ -1134,13 +1273,13 @@ var VideoLayout = (function (my) {
if (state == 'show')
{
// peerContainer.css('-webkit-filter', '');
- var jid = connection.emuc.findJidFromResource(resourceJid);
+ var jid = xmpp.findJidFromResource(resourceJid);
Avatar.showUserAvatar(jid, false);
}
else // if (state == 'avatar')
{
// peerContainer.css('-webkit-filter', 'grayscale(100%)');
- var jid = connection.emuc.findJidFromResource(resourceJid);
+ var jid = xmpp.findJidFromResource(resourceJid);
Avatar.showUserAvatar(jid, true);
}
}
@@ -1166,8 +1305,7 @@ var VideoLayout = (function (my) {
if (name && nickname !== name) {
nickname = name;
window.localStorage.displayname = nickname;
- connection.emuc.addDisplayNameToPresence(nickname);
- connection.emuc.sendPresence();
+ xmpp.addToPresence("displayName", nickname);
Chat.setChatConversationMode(true);
}
@@ -1238,7 +1376,7 @@ var VideoLayout = (function (my) {
*/
my.showModeratorIndicator = function () {
- var isModerator = Moderator.isModerator();
+ var isModerator = xmpp.isModerator();
if (isModerator) {
var indicatorSpan = $('#localVideoContainer .focusindicator');
@@ -1247,7 +1385,10 @@ var VideoLayout = (function (my) {
createModeratorIndicatorElement(indicatorSpan[0]);
}
}
- Object.keys(connection.emuc.members).forEach(function (jid) {
+
+ var members = xmpp.getMembers();
+
+ Object.keys(members).forEach(function (jid) {
if (Strophe.getResourceFromJid(jid) === 'focus') {
// Skip server side focus
@@ -1263,7 +1404,7 @@ var VideoLayout = (function (my) {
return;
}
- var member = connection.emuc.members[jid];
+ var member = members[jid];
if (member.role === 'moderator') {
// Remove menu if peer is moderator
@@ -1435,7 +1576,7 @@ var VideoLayout = (function (my) {
var videoSpanId = null;
var videoContainerId = null;
if (resourceJid
- === Strophe.getResourceFromJid(connection.emuc.myroomjid)) {
+ === xmpp.myResource()) {
videoSpanId = 'localVideoWrapper';
videoContainerId = 'localVideoContainer';
}
@@ -1478,7 +1619,7 @@ var VideoLayout = (function (my) {
}
Avatar.showUserAvatar(
- connection.emuc.findJidFromResource(resourceJid));
+ xmpp.findJidFromResource(resourceJid));
}
};
@@ -1603,7 +1744,7 @@ var VideoLayout = (function (my) {
lastNPickupJid = jid;
$(document).trigger("pinnedendpointchanged", [jid]);
}
- } else if (jid == connection.emuc.myroomjid) {
+ } else if (jid == xmpp.myJid()) {
$("#localVideoContainer").click();
}
}
@@ -1615,13 +1756,13 @@ var VideoLayout = (function (my) {
$(document).bind('audiomuted.muc', function (event, jid, isMuted) {
/*
// FIXME: but focus can not mute in this case ? - check
- if (jid === connection.emuc.myroomjid) {
+ if (jid === xmpp.myJid()) {
// The local mute indicator is controlled locally
return;
}*/
var videoSpanId = null;
- if (jid === connection.emuc.myroomjid) {
+ if (jid === xmpp.myJid()) {
videoSpanId = 'localVideoContainer';
} else {
VideoLayout.ensurePeerContainerExists(jid);
@@ -1630,7 +1771,7 @@ var VideoLayout = (function (my) {
mutedAudios[jid] = isMuted;
- if (Moderator.isModerator()) {
+ if (xmpp.isModerator()) {
VideoLayout.updateRemoteVideoMenu(jid, isMuted);
}
@@ -1648,7 +1789,7 @@ var VideoLayout = (function (my) {
Avatar.showUserAvatar(jid, isMuted);
var videoSpanId = null;
- if (jid === connection.emuc.myroomjid) {
+ if (jid === xmpp.myJid()) {
videoSpanId = 'localVideoContainer';
} else {
VideoLayout.ensurePeerContainerExists(jid);
@@ -1662,11 +1803,11 @@ var VideoLayout = (function (my) {
/**
* Display name changed.
*/
- $(document).bind('displaynamechanged',
- function (event, jid, displayName, status) {
+ my.onDisplayNameChanged =
+ function (jid, displayName, status) {
var name = null;
if (jid === 'localVideoContainer'
- || jid === connection.emuc.myroomjid) {
+ || jid === xmpp.myJid()) {
name = nickname;
setDisplayName('localVideoContainer',
displayName);
@@ -1680,10 +1821,10 @@ var VideoLayout = (function (my) {
}
if(jid === 'localVideoContainer')
- jid = connection.emuc.myroomjid;
+ jid = xmpp.myJid();
if(!name || name != displayName)
API.triggerEvent("displayNameChange",{jid: jid, displayname: displayName});
- });
+ };
/**
* On dominant speaker changed event.
@@ -1691,7 +1832,7 @@ var VideoLayout = (function (my) {
$(document).bind('dominantspeakerchanged', function (event, resourceJid) {
// We ignore local user events.
if (resourceJid
- === Strophe.getResourceFromJid(connection.emuc.myroomjid))
+ === xmpp.myResource())
return;
// Update the current dominant speaker.
@@ -1822,7 +1963,7 @@ var VideoLayout = (function (my) {
if (!isVisible) {
console.log("Add to last N", resourceJid);
- var jid = connection.emuc.findJidFromResource(resourceJid);
+ var jid = xmpp.findJidFromResource(resourceJid);
var mediaStream = RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE];
var sel = $('#participant_' + resourceJid + '>video');
@@ -1855,7 +1996,7 @@ var VideoLayout = (function (my) {
var resource, container, src;
var myResource
- = Strophe.getResourceFromJid(connection.emuc.myroomjid);
+ = xmpp.myResource();
// Find out which endpoint to show in the large video.
for (var i = 0; i < lastNEndpoints.length; i++) {
@@ -1879,37 +2020,6 @@ var VideoLayout = (function (my) {
}
});
- $(document).bind('videoactive.jingle', function (event, videoelem) {
- if (videoelem.attr('id').indexOf('mixedmslabel') === -1) {
- // ignore mixedmslabela0 and v0
-
- videoelem.show();
- VideoLayout.resizeThumbnails();
-
- var videoParent = videoelem.parent();
- var parentResourceJid = null;
- if (videoParent)
- parentResourceJid
- = VideoLayout.getPeerContainerResourceJid(videoParent[0]);
-
- // Update the large video to the last added video only if there's no
- // current dominant, focused speaker or prezi playing or update it to
- // the current dominant speaker.
- if ((!focusedVideoInfo &&
- !VideoLayout.getDominantSpeakerResourceJid() &&
- !require("../prezi/Prezi").isPresentationVisible()) ||
- (parentResourceJid &&
- VideoLayout.getDominantSpeakerResourceJid() === parentResourceJid)) {
- VideoLayout.updateLargeVideo(
- RTC.getVideoSrc(videoelem[0]),
- 1,
- parentResourceJid);
- }
-
- VideoLayout.showModeratorIndicator();
- }
- });
-
$(document).bind('simulcastlayerschanging', function (event, endpointSimulcastLayers) {
endpointSimulcastLayers.forEach(function (esl) {
@@ -1930,13 +2040,13 @@ var VideoLayout = (function (my) {
// Get session and stream from primary ssrc.
var res = simulcast.getReceivingVideoStreamBySSRC(primarySSRC);
- var session = res.session;
+ var sid = res.sid;
var electedStream = res.stream;
- if (session && electedStream) {
+ if (sid && electedStream) {
var msid = simulcast.getRemoteVideoStreamIdBySSRC(primarySSRC);
- console.info([esl, primarySSRC, msid, session, electedStream]);
+ console.info([esl, primarySSRC, msid, sid, electedStream]);
var msidParts = msid.split(' ');
@@ -1956,7 +2066,7 @@ var VideoLayout = (function (my) {
}
} else {
- console.error('Could not find a stream or a session.', session, electedStream);
+ console.error('Could not find a stream or a session.', sid, electedStream);
}
});
});
@@ -1988,17 +2098,17 @@ var VideoLayout = (function (my) {
// Get session and stream from primary ssrc.
var res = simulcast.getReceivingVideoStreamBySSRC(primarySSRC);
- var session = res.session;
+ var sid = res.sid;
var electedStream = res.stream;
- if (session && electedStream) {
+ if (sid && electedStream) {
var msid = simulcast.getRemoteVideoStreamIdBySSRC(primarySSRC);
console.info('Switching simulcast substream.');
- console.info([esl, primarySSRC, msid, session, electedStream]);
+ console.info([esl, primarySSRC, msid, sid, electedStream]);
var msidParts = msid.split(' ');
- var selRemoteVideo = $(['#', 'remoteVideo_', session.sid, '_', msidParts[0]].join(''));
+ var selRemoteVideo = $(['#', 'remoteVideo_', sid, '_', msidParts[0]].join(''));
var updateLargeVideo = (Strophe.getResourceFromJid(ssrc2jid[primarySSRC])
== largeVideoState.userResourceJid);
@@ -2035,7 +2145,7 @@ var VideoLayout = (function (my) {
}
var videoId;
- if(resource == Strophe.getResourceFromJid(connection.emuc.myroomjid))
+ if(resource == xmpp.myResource())
{
videoId = "localVideoContainer";
}
@@ -2048,7 +2158,7 @@ var VideoLayout = (function (my) {
connectionIndicator.updatePopoverData();
} else {
- console.error('Could not find a stream or a session.', session, electedStream);
+ console.error('Could not find a stream or a sid.', sid, electedStream);
}
});
});
@@ -2063,8 +2173,8 @@ var VideoLayout = (function (my) {
if(object.resolution !== null)
{
resolution = object.resolution;
- object.resolution = resolution[connection.emuc.myroomjid];
- delete resolution[connection.emuc.myroomjid];
+ object.resolution = resolution[xmpp.myJid()];
+ delete resolution[xmpp.myJid()];
}
updateStatsIndicator("localVideoContainer", percent, object);
for(var jid in resolution)
diff --git a/modules/connectionquality/connectionquality.js b/modules/connectionquality/connectionquality.js
index 4a668a002..7652caec6 100644
--- a/modules/connectionquality/connectionquality.js
+++ b/modules/connectionquality/connectionquality.js
@@ -29,8 +29,7 @@ function startSendingStats() {
* Sends statistics to other participants
*/
function sendStats() {
- connection.emuc.addConnectionInfoToPresence(convertToMUCStats(stats));
- connection.emuc.sendPresence();
+ xmpp.addToPresence("connectionQuality", convertToMUCStats(stats));
}
/**
diff --git a/modules/desktopsharing/desktopsharing.js b/modules/desktopsharing/desktopsharing.js
index ebc943ae8..42b633f9d 100644
--- a/modules/desktopsharing/desktopsharing.js
+++ b/modules/desktopsharing/desktopsharing.js
@@ -1,4 +1,4 @@
-/* global $, alert, changeLocalVideo, chrome, config, connection, getConferenceHandler, getUserMediaWithConstraints */
+/* global $, alert, changeLocalVideo, chrome, config, getConferenceHandler, getUserMediaWithConstraints */
/**
* Indicates that desktop stream is currently in use(for toggle purpose).
* @type {boolean}
diff --git a/modules/simulcast/SimulcastReceiver.js b/modules/simulcast/SimulcastReceiver.js
index 863c5a826..c019a299f 100644
--- a/modules/simulcast/SimulcastReceiver.js
+++ b/modules/simulcast/SimulcastReceiver.js
@@ -159,43 +159,19 @@ SimulcastReceiver.prototype.getReceivingSSRC = function (jid) {
// If we haven't receiving a "changed" event yet, then we must be receiving
// low quality (that the sender always streams).
- if (!ssrc && connection.jingle) {
- var session;
- var i, j, k;
-
- var keys = Object.keys(connection.jingle.sessions);
- for (i = 0; i < keys.length; i++) {
- var sid = keys[i];
-
- if (ssrc) {
- // stream found, stop.
- break;
- }
-
- session = connection.jingle.sessions[sid];
- if (session.remoteStreams) {
- for (j = 0; j < session.remoteStreams.length; j++) {
- var remoteStream = session.remoteStreams[j];
-
- if (ssrc) {
- // stream found, stop.
- break;
- }
- var tracks = remoteStream.getVideoTracks();
- if (tracks) {
- for (k = 0; k < tracks.length; k++) {
- var track = tracks[k];
- var msid = [remoteStream.id, track.id].join(' ');
- var _ssrc = this._remoteMaps.msid2ssrc[msid];
- var _jid = ssrc2jid[_ssrc];
- var quality = this._remoteMaps.msid2Quality[msid];
- if (jid == _jid && quality == 0) {
- ssrc = _ssrc;
- // stream found, stop.
- break;
- }
- }
- }
+ if(!ssrc)
+ {
+ var remoteStreamObject = RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE];
+ var remoteStream = remoteStreamObject.getOriginalStream();
+ var tracks = remoteStream.getVideoTracks();
+ if (tracks) {
+ for (var k = 0; k < tracks.length; k++) {
+ var track = tracks[k];
+ var msid = [remoteStream.id, track.id].join(' ');
+ var _ssrc = this._remoteMaps.msid2ssrc[msid];
+ var quality = this._remoteMaps.msid2Quality[msid];
+ if (quality == 0) {
+ ssrc = _ssrc;
}
}
}
@@ -206,47 +182,32 @@ SimulcastReceiver.prototype.getReceivingSSRC = function (jid) {
SimulcastReceiver.prototype.getReceivingVideoStreamBySSRC = function (ssrc)
{
- var session, electedStream;
+ var sid, electedStream;
var i, j, k;
- if (connection.jingle) {
- var keys = Object.keys(connection.jingle.sessions);
- for (i = 0; i < keys.length; i++) {
- var sid = keys[i];
-
- if (electedStream) {
- // stream found, stop.
- break;
- }
-
- session = connection.jingle.sessions[sid];
- if (session.remoteStreams) {
- for (j = 0; j < session.remoteStreams.length; j++) {
- var remoteStream = session.remoteStreams[j];
-
- if (electedStream) {
- // stream found, stop.
- break;
- }
- var tracks = remoteStream.getVideoTracks();
- if (tracks) {
- for (k = 0; k < tracks.length; k++) {
- var track = tracks[k];
- var msid = [remoteStream.id, track.id].join(' ');
- var tmp = this._remoteMaps.msid2ssrc[msid];
- if (tmp == ssrc) {
- electedStream = new webkitMediaStream([track]);
- // stream found, stop.
- break;
- }
- }
- }
+ var jid = ssrc2jid[ssrc];
+ if(jid)
+ {
+ var remoteStreamObject = RTC.remoteStreams[jid][MediaStreamType.VIDEO_TYPE];
+ var remoteStream = remoteStreamObject.getOriginalStream();
+ var tracks = remoteStream.getVideoTracks();
+ if (tracks) {
+ for (k = 0; k < tracks.length; k++) {
+ var track = tracks[k];
+ var msid = [remoteStream.id, track.id].join(' ');
+ var tmp = this._remoteMaps.msid2ssrc[msid];
+ if (tmp == ssrc) {
+ electedStream = new webkitMediaStream([track]);
+ sid = remoteStreamObject.sid;
+ // stream found, stop.
+ break;
}
}
}
+
}
return {
- session: session,
+ sid: sid,
stream: electedStream
};
};
diff --git a/modules/statistics/RTPStatsCollector.js b/modules/statistics/RTPStatsCollector.js
index 3b070dda1..c7901a23b 100644
--- a/modules/statistics/RTPStatsCollector.js
+++ b/modules/statistics/RTPStatsCollector.js
@@ -329,30 +329,9 @@ StatsCollector.prototype.addStatsToBeLogged = function (reports) {
};
StatsCollector.prototype.logStats = function () {
- if (!focusMucJid) {
+
+ if(!xmpp.sendLogs(this.statsToBeLogged))
return;
- }
-
- var deflate = true;
-
- var content = JSON.stringify(this.statsToBeLogged);
- if (deflate) {
- content = String.fromCharCode.apply(null, Pako.deflateRaw(content));
- }
- content = Base64.encode(content);
-
- // XEP-0337-ish
- var message = $msg({to: focusMucJid, type: 'normal'});
- message.c('log', { xmlns: 'urn:xmpp:eventlog',
- id: 'PeerConnectionStats'});
- message.c('message').t(content).up();
- if (deflate) {
- message.c('tag', {name: "deflated", value: "true"}).up();
- }
- message.up();
-
- connection.send(message);
-
// Reset the stats
this.statsToBeLogged.stats = {};
this.statsToBeLogged.timestamps = [];
@@ -700,7 +679,7 @@ StatsCollector.prototype.processAudioLevelReport = function ()
// but it seems to vary between 0 and around 32k.
audioLevel = audioLevel / 32767;
jidStats.setSsrcAudioLevel(ssrc, audioLevel);
- if(jid != connection.emuc.myroomjid)
+ if(jid != xmpp.myJid())
this.eventEmitter.emit("statistics.audioLevel", jid, audioLevel);
}
diff --git a/modules/statistics/statistics.js b/modules/statistics/statistics.js
index 828c401e9..f0449cb7a 100644
--- a/modules/statistics/statistics.js
+++ b/modules/statistics/statistics.js
@@ -59,6 +59,14 @@ function onStreamCreated(stream)
localStats.start();
}
+function onDisposeConference(onUnload) {
+ stopRemote();
+ if(onUnload) {
+ stopLocal();
+ eventEmitter.removeAllListeners();
+ }
+}
+
var statistics =
{
@@ -117,19 +125,12 @@ var statistics =
startRemoteStats(event.peerconnection);
},
- onDisposeConference: function (onUnload) {
- stopRemote();
- if(onUnload) {
- stopLocal();
- eventEmitter.removeAllListeners();
- }
- },
-
start: function () {
this.addConnectionStatsListener(connectionquality.updateLocalStats);
this.addRemoteStatsStopListener(connectionquality.stopSendingStats);
RTC.addStreamListener(onStreamCreated,
StreamEventTypes.EVENT_TYPE_LOCAL_CREATED);
+ xmpp.addListener(XMPPEvents.DISPOSE_CONFERENCE, onDisposeConference);
}
};
diff --git a/libs/strophe/strophe.jingle.session.js b/modules/xmpp/JingleSession.js
similarity index 83%
rename from libs/strophe/strophe.jingle.session.js
rename to modules/xmpp/JingleSession.js
index 2517520e6..8ff3d0efa 100644
--- a/libs/strophe/strophe.jingle.session.js
+++ b/modules/xmpp/JingleSession.js
@@ -1,6 +1,11 @@
/* jshint -W117 */
+var TraceablePeerConnection = require("./TraceablePeerConnection");
+var SDPDiffer = require("./SDPDiffer");
+var SDPUtil = require("./SDPUtil");
+var SDP = require("./SDP");
+
// Jingle stuff
-function JingleSession(me, sid, connection) {
+function JingleSession(me, sid, connection, service) {
this.me = me;
this.sid = sid;
this.connection = connection;
@@ -12,13 +17,13 @@ function JingleSession(me, sid, connection) {
this.localSDP = null;
this.remoteSDP = null;
this.relayedStreams = [];
- this.remoteStreams = [];
this.startTime = null;
this.stopTime = null;
this.media_constraints = null;
this.pc_constraints = null;
this.ice_config = {};
this.drip_container = [];
+ this.service = service;
this.usetrickle = true;
this.usepranswer = false; // early transport warmup -- mind you, this might fail. depends on webrtc issue 1718
@@ -73,16 +78,11 @@ JingleSession.prototype.initiate = function (peerjid, isInitiator) {
self.sendIceCandidate(event.candidate);
};
this.peerconnection.onaddstream = function (event) {
- self.remoteStreams.push(event.stream);
console.log("REMOTE STREAM ADDED: " + event.stream + " - " + event.stream.id);
- $(document).trigger('remotestreamadded.jingle', [event, self.sid]);
+ self.remoteStreamAdded(event);
};
this.peerconnection.onremovestream = function (event) {
// Remove the stream from remoteStreams
- var streamIdx = self.remoteStreams.indexOf(event.stream);
- if(streamIdx !== -1){
- self.remoteStreams.splice(streamIdx, 1);
- }
// FIXME: remotestreamremoved.jingle not defined anywhere(unused)
$(document).trigger('remotestreamremoved.jingle', [event, self.sid]);
};
@@ -99,7 +99,7 @@ JingleSession.prototype.initiate = function (peerjid, isInitiator) {
this.stopTime = new Date();
break;
}
- $(document).trigger('iceconnectionstatechange.jingle', [self.sid, self]);
+ onIceConnectionStateChange(self.sid, self);
};
// add any local and relayed stream
RTC.localStreams.forEach(function(stream) {
@@ -110,6 +110,49 @@ JingleSession.prototype.initiate = function (peerjid, isInitiator) {
});
};
+function onIceConnectionStateChange(sid, session) {
+ switch (session.peerconnection.iceConnectionState) {
+ case 'checking':
+ session.timeChecking = (new Date()).getTime();
+ session.firstconnect = true;
+ break;
+ case 'completed': // on caller side
+ case 'connected':
+ if (session.firstconnect) {
+ session.firstconnect = false;
+ var metadata = {};
+ metadata.setupTime
+ = (new Date()).getTime() - session.timeChecking;
+ session.peerconnection.getStats(function (res) {
+ if(res && res.result) {
+ res.result().forEach(function (report) {
+ if (report.type == 'googCandidatePair' &&
+ report.stat('googActiveConnection') == 'true') {
+ metadata.localCandidateType
+ = report.stat('googLocalCandidateType');
+ metadata.remoteCandidateType
+ = report.stat('googRemoteCandidateType');
+
+ // log pair as well so we can get nice pie
+ // charts
+ metadata.candidatePair
+ = report.stat('googLocalCandidateType') +
+ ';' +
+ report.stat('googRemoteCandidateType');
+
+ if (report.stat('googRemoteAddress').indexOf('[') === 0)
+ {
+ metadata.ipv6 = true;
+ }
+ }
+ });
+ }
+ });
+ }
+ break;
+ }
+}
+
JingleSession.prototype.accept = function () {
var self = this;
this.state = 'active';
@@ -145,12 +188,13 @@ JingleSession.prototype.accept = function () {
// FIXME: change any inactive to sendrecv or whatever they were originally
sdp = sdp.replace('a=inactive', 'a=sendrecv');
}
+ var self = this;
this.peerconnection.setLocalDescription(new RTCSessionDescription({type: 'answer', sdp: sdp}),
function () {
//console.log('setLocalDescription success');
- $(document).trigger('setLocalDescription.jingle', [self.sid]);
+ self.setLocalDescription();
- this.connection.sendIQ(accept,
+ self.connection.sendIQ(accept,
function () {
var ack = {};
ack.source = 'answer';
@@ -347,8 +391,8 @@ JingleSession.prototype.createdOffer = function (sdp) {
action: 'session-initiate',
initiator: this.initiator,
sid: this.sid});
- this.localSDP.toJingle(init, this.initiator == this.me ? 'initiator' : 'responder', this.localStreamsSSRC);
- this.connection.sendIQ(init,
+ self.localSDP.toJingle(init, this.initiator == this.me ? 'initiator' : 'responder', this.localStreamsSSRC);
+ self.connection.sendIQ(init,
function () {
var ack = {};
ack.source = 'offer';
@@ -369,13 +413,11 @@ JingleSession.prototype.createdOffer = function (sdp) {
sdp.sdp = this.localSDP.raw;
this.peerconnection.setLocalDescription(sdp,
function () {
- if(this.usetrickle)
+ if(self.usetrickle)
{
sendJingle();
- $(document).trigger('setLocalDescription.jingle', [self.sid]);
}
- else
- $(document).trigger('setLocalDescription.jingle', [self.sid]);
+ self.setLocalDescription();
//console.log('setLocalDescription success');
},
function (e) {
@@ -587,7 +629,7 @@ JingleSession.prototype.createdAnswer = function (sdp, provisional) {
var publicLocalDesc = simulcast.reverseTransformLocalDescription(sdp);
var publicLocalSDP = new SDP(publicLocalDesc.sdp);
publicLocalSDP.toJingle(accept, self.initiator == self.me ? 'initiator' : 'responder', ssrcs);
- this.connection.sendIQ(accept,
+ self.connection.sendIQ(accept,
function () {
var ack = {};
ack.source = 'answer';
@@ -610,10 +652,8 @@ JingleSession.prototype.createdAnswer = function (sdp, provisional) {
//console.log('setLocalDescription success');
if (self.usetrickle && !self.usepranswer) {
sendJingle();
- $(document).trigger('setLocalDescription.jingle', [self.sid]);
}
- else
- $(document).trigger('setLocalDescription.jingle', [self.sid]);
+ self.setLocalDescription();
},
function (e) {
console.error('setLocalDescription failed', e);
@@ -799,7 +839,7 @@ JingleSession.prototype.modifySources = function (successCallback) {
if (this.peerconnection.signalingState == 'closed') return;
if (!(this.addssrc.length || this.removessrc.length || this.pendingop !== null || this.switchstreams)){
// There is nothing to do since scheduled job might have been executed by another succeeding call
- $(document).trigger('setLocalDescription.jingle', [self.sid]);
+ this.setLocalDescription();
if(successCallback){
successCallback();
}
@@ -889,7 +929,7 @@ JingleSession.prototype.modifySources = function (successCallback) {
self.peerconnection.setLocalDescription(modifiedAnswer,
function() {
//console.log('modified setLocalDescription ok');
- $(document).trigger('setLocalDescription.jingle', [self.sid]);
+ self.setLocalDescription();
if(successCallback){
successCallback();
}
@@ -1064,12 +1104,20 @@ JingleSession.prototype.setVideoMute = function (mute, callback, options) {
} else if (this.videoMuteByUser) {
return;
}
+
+ var self = this;
+ var localCallback = function (mute) {
+ self.connection.emuc.addVideoInfoToPresence(mute);
+ self.connection.emuc.sendPresence();
+ return callback(mute)
+ };
+
if (mute == RTC.localVideo.isMuted())
{
// Even if no change occurs, the specified callback is to be executed.
// The specified callback may, optionally, return a successCallback
// which is to be executed as well.
- var successCallback = callback(mute);
+ var successCallback = localCallback(mute);
if (successCallback) {
successCallback();
@@ -1079,14 +1127,14 @@ JingleSession.prototype.setVideoMute = function (mute, callback, options) {
this.hardMuteVideo(mute);
- this.modifySources(callback(mute));
+ this.modifySources(localCallback(mute));
}
};
// SDP-based mute by going recvonly/sendrecv
// FIXME: should probably black out the screen as well
JingleSession.prototype.toggleVideoMute = function (callback) {
- setVideoMute(RTC.localVideo.isMuted(), callback);
+ this.service.setVideoMute(RTC.localVideo.isMuted(), callback);
};
JingleSession.prototype.hardMuteVideo = function (muted) {
@@ -1172,8 +1220,170 @@ JingleSession.onJingleError = function (session, error)
JingleSession.onJingleFatalError = function (session, error)
{
- sessionTerminated = true;
+ this.service.sessionTerminated = true;
connection.emuc.doLeave();
UI.messageHandler.showError( "Sorry",
"Internal application error[setRemoteDescription]");
-}
\ No newline at end of file
+}
+
+JingleSession.prototype.setLocalDescription = function () {
+ // put our ssrcs into presence so other clients can identify our stream
+ var newssrcs = [];
+ var media = simulcast.parseMedia(this.peerconnection.localDescription);
+ media.forEach(function (media) {
+
+ if(Object.keys(media.sources).length > 0) {
+ // TODO(gp) maybe exclude FID streams?
+ Object.keys(media.sources).forEach(function (ssrc) {
+ newssrcs.push({
+ 'ssrc': ssrc,
+ 'type': media.type,
+ 'direction': media.direction
+ });
+ });
+ }
+ else if(this.localStreamsSSRC && this.localStreamsSSRC[media.type])
+ {
+ newssrcs.push({
+ 'ssrc': this.localStreamsSSRC[media.type],
+ 'type': media.type,
+ 'direction': media.direction
+ });
+ }
+
+ });
+
+ console.log('new ssrcs', newssrcs);
+
+ // Have to clear presence map to get rid of removed streams
+ this.connection.emuc.clearPresenceMedia();
+
+ if (newssrcs.length > 0) {
+ for (var i = 1; i <= newssrcs.length; i ++) {
+ // Change video type to screen
+ if (newssrcs[i-1].type === 'video' && desktopsharing.isUsingScreenStream()) {
+ newssrcs[i-1].type = 'screen';
+ }
+ this.connection.emuc.addMediaToPresence(i,
+ newssrcs[i-1].type, newssrcs[i-1].ssrc, newssrcs[i-1].direction);
+ }
+
+ this.connection.emuc.sendPresence();
+ }
+}
+
+// an attempt to work around https://github.com/jitsi/jitmeet/issues/32
+function sendKeyframe(pc) {
+ console.log('sendkeyframe', pc.iceConnectionState);
+ if (pc.iceConnectionState !== 'connected') return; // safe...
+ pc.setRemoteDescription(
+ pc.remoteDescription,
+ function () {
+ pc.createAnswer(
+ function (modifiedAnswer) {
+ pc.setLocalDescription(
+ modifiedAnswer,
+ function () {
+ // noop
+ },
+ function (error) {
+ console.log('triggerKeyframe setLocalDescription failed', error);
+ UI.messageHandler.showError();
+ }
+ );
+ },
+ function (error) {
+ console.log('triggerKeyframe createAnswer failed', error);
+ UI.messageHandler.showError();
+ }
+ );
+ },
+ function (error) {
+ console.log('triggerKeyframe setRemoteDescription failed', error);
+ UI.messageHandler.showError();
+ }
+ );
+}
+
+
+JingleSession.prototype.remoteStreamAdded = function (data) {
+ var self = this;
+ var thessrc;
+
+ // look up an associated JID for a stream id
+ if (data.stream.id && data.stream.id.indexOf('mixedmslabel') === -1) {
+ // look only at a=ssrc: and _not_ at a=ssrc-group: lines
+
+ var ssrclines
+ = SDPUtil.find_lines(this.peerconnection.remoteDescription.sdp, 'a=ssrc:');
+ ssrclines = ssrclines.filter(function (line) {
+ // NOTE(gp) previously we filtered on the mslabel, but that property
+ // is not always present.
+ // return line.indexOf('mslabel:' + data.stream.label) !== -1;
+
+ return ((line.indexOf('msid:' + data.stream.id) !== -1));
+ });
+ if (ssrclines.length) {
+ thessrc = ssrclines[0].substring(7).split(' ')[0];
+
+ // We signal our streams (through Jingle to the focus) before we set
+ // our presence (through which peers associate remote streams to
+ // jids). So, it might arrive that a remote stream is added but
+ // ssrc2jid is not yet updated and thus data.peerjid cannot be
+ // successfully set. Here we wait for up to a second for the
+ // presence to arrive.
+
+ if (!ssrc2jid[thessrc]) {
+ // TODO(gp) limit wait duration to 1 sec.
+ setTimeout(function(d) {
+ return function() {
+ self.remoteStreamAdded(d);
+ }
+ }(data), 250);
+ return;
+ }
+
+ // ok to overwrite the one from focus? might save work in colibri.js
+ console.log('associated jid', ssrc2jid[thessrc], data.peerjid);
+ if (ssrc2jid[thessrc]) {
+ data.peerjid = ssrc2jid[thessrc];
+ }
+ }
+ }
+
+ //TODO: this code should be removed when firefox implement multistream support
+ if(RTC.getBrowserType() == RTCBrowserType.RTC_BROWSER_FIREFOX)
+ {
+ if((notReceivedSSRCs.length == 0) ||
+ !ssrc2jid[notReceivedSSRCs[notReceivedSSRCs.length - 1]])
+ {
+ // TODO(gp) limit wait duration to 1 sec.
+ setTimeout(function(d) {
+ return function() {
+ self.remoteStreamAdded(d);
+ }
+ }(data), 250);
+ return;
+ }
+
+ thessrc = notReceivedSSRCs.pop();
+ if (ssrc2jid[thessrc]) {
+ data.peerjid = ssrc2jid[thessrc];
+ }
+ }
+
+ RTC.createRemoteStream(data, this.sid, thessrc);
+
+ var isVideo = data.stream.getVideoTracks().length > 0;
+ // an attempt to work around https://github.com/jitsi/jitmeet/issues/32
+ if (isVideo &&
+ data.peerjid && this.peerjid === data.peerjid &&
+ data.stream.getVideoTracks().length === 0 &&
+ RTC.localVideo.getTracks().length > 0) {
+ window.setTimeout(function () {
+ sendKeyframe(self.peerconnection);
+ }, 3000);
+ }
+}
+
+module.exports = JingleSession;
\ No newline at end of file
diff --git a/libs/strophe/strophe.jingle.sdp.js b/modules/xmpp/SDP.js
similarity index 55%
rename from libs/strophe/strophe.jingle.sdp.js
rename to modules/xmpp/SDP.js
index 03dbceb08..4103fbe13 100644
--- a/libs/strophe/strophe.jingle.sdp.js
+++ b/modules/xmpp/SDP.js
@@ -1,4 +1,6 @@
/* jshint -W117 */
+var SDPUtil = require("./SDPUtil");
+
// SDP STUFF
function SDP(sdp) {
this.media = sdp.split('\r\nm=');
@@ -71,169 +73,6 @@ SDP.prototype.containsSSRC = function(ssrc) {
return contains;
};
-function SDPDiffer(mySDP, otherSDP) {
- this.mySDP = mySDP;
- this.otherSDP = otherSDP;
-}
-
-/**
- * Returns map of MediaChannel that contains only media not contained in otherSdp. Mapped by channel idx.
- * @param otherSdp the other SDP to check ssrc with.
- */
-SDPDiffer.prototype.getNewMedia = function() {
-
- // this could be useful in Array.prototype.
- function arrayEquals(array) {
- // if the other array is a falsy value, return
- if (!array)
- return false;
-
- // compare lengths - can save a lot of time
- if (this.length != array.length)
- return false;
-
- for (var i = 0, l=this.length; i < l; i++) {
- // Check if we have nested arrays
- if (this[i] instanceof Array && array[i] instanceof Array) {
- // recurse into the nested arrays
- if (!this[i].equals(array[i]))
- return false;
- }
- else if (this[i] != array[i]) {
- // Warning - two different object instances will never be equal: {x:20} != {x:20}
- return false;
- }
- }
- return true;
- }
-
- var myMedias = this.mySDP.getMediaSsrcMap();
- var othersMedias = this.otherSDP.getMediaSsrcMap();
- var newMedia = {};
- Object.keys(othersMedias).forEach(function(othersMediaIdx) {
- var myMedia = myMedias[othersMediaIdx];
- var othersMedia = othersMedias[othersMediaIdx];
- if(!myMedia && othersMedia) {
- // Add whole channel
- newMedia[othersMediaIdx] = othersMedia;
- return;
- }
- // Look for new ssrcs accross the channel
- Object.keys(othersMedia.ssrcs).forEach(function(ssrc) {
- if(Object.keys(myMedia.ssrcs).indexOf(ssrc) === -1) {
- // Allocate channel if we've found ssrc that doesn't exist in our channel
- if(!newMedia[othersMediaIdx]){
- newMedia[othersMediaIdx] = {
- mediaindex: othersMedia.mediaindex,
- mid: othersMedia.mid,
- ssrcs: {},
- ssrcGroups: []
- };
- }
- newMedia[othersMediaIdx].ssrcs[ssrc] = othersMedia.ssrcs[ssrc];
- }
- });
-
- // Look for new ssrc groups across the channels
- othersMedia.ssrcGroups.forEach(function(otherSsrcGroup){
-
- // try to match the other ssrc-group with an ssrc-group of ours
- var matched = false;
- for (var i = 0; i < myMedia.ssrcGroups.length; i++) {
- var mySsrcGroup = myMedia.ssrcGroups[i];
- if (otherSsrcGroup.semantics == mySsrcGroup.semantics
- && arrayEquals.apply(otherSsrcGroup.ssrcs, [mySsrcGroup.ssrcs])) {
-
- matched = true;
- break;
- }
- }
-
- if (!matched) {
- // Allocate channel if we've found an ssrc-group that doesn't
- // exist in our channel
-
- if(!newMedia[othersMediaIdx]){
- newMedia[othersMediaIdx] = {
- mediaindex: othersMedia.mediaindex,
- mid: othersMedia.mid,
- ssrcs: {},
- ssrcGroups: []
- };
- }
- newMedia[othersMediaIdx].ssrcGroups.push(otherSsrcGroup);
- }
- });
- });
- return newMedia;
-};
-
-/**
- * Sends SSRC update IQ.
- * @param sdpMediaSsrcs SSRCs map obtained from SDP.getNewMedia. Cntains SSRCs to add/remove.
- * @param sid session identifier that will be put into the IQ.
- * @param initiator initiator identifier.
- * @param toJid destination Jid
- * @param isAdd indicates if this is remove or add operation.
- */
-SDPDiffer.prototype.toJingle = function(modify) {
- var sdpMediaSsrcs = this.getNewMedia();
- var self = this;
-
- // FIXME: only announce video ssrcs since we mix audio and dont need
- // the audio ssrcs therefore
- var modified = false;
- Object.keys(sdpMediaSsrcs).forEach(function(mediaindex){
- modified = true;
- var media = sdpMediaSsrcs[mediaindex];
- modify.c('content', {name: media.mid});
-
- modify.c('description', {xmlns:'urn:xmpp:jingle:apps:rtp:1', media: media.mid});
- // FIXME: not completly sure this operates on blocks and / or handles different ssrcs correctly
- // generate sources from lines
- Object.keys(media.ssrcs).forEach(function(ssrcNum) {
- var mediaSsrc = media.ssrcs[ssrcNum];
- modify.c('source', { xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
- modify.attrs({ssrc: mediaSsrc.ssrc});
- // iterate over ssrc lines
- mediaSsrc.lines.forEach(function (line) {
- var idx = line.indexOf(' ');
- var kv = line.substr(idx + 1);
- modify.c('parameter');
- if (kv.indexOf(':') == -1) {
- modify.attrs({ name: kv });
- } else {
- modify.attrs({ name: kv.split(':', 2)[0] });
- modify.attrs({ value: kv.split(':', 2)[1] });
- }
- modify.up(); // end of parameter
- });
- modify.up(); // end of source
- });
-
- // generate source groups from lines
- media.ssrcGroups.forEach(function(ssrcGroup) {
- if (ssrcGroup.ssrcs.length != 0) {
-
- modify.c('ssrc-group', {
- semantics: ssrcGroup.semantics,
- xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0'
- });
-
- ssrcGroup.ssrcs.forEach(function (ssrc) {
- modify.c('source', { ssrc: ssrc })
- .up(); // end of source
- });
- modify.up(); // end of ssrc-group
- }
- });
-
- modify.up(); // end of description
- modify.up(); // end of content
- });
-
- return modified;
-};
// remove iSAC and CN from SDP
SDP.prototype.mangle = function () {
@@ -776,352 +615,6 @@ SDP.prototype.jingle2media = function (content) {
return media;
};
-SDPUtil = {
- iceparams: function (mediadesc, sessiondesc) {
- var data = null;
- if (SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc) &&
- SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc)) {
- data = {
- ufrag: SDPUtil.parse_iceufrag(SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc)),
- pwd: SDPUtil.parse_icepwd(SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc))
- };
- }
- return data;
- },
- parse_iceufrag: function (line) {
- return line.substring(12);
- },
- build_iceufrag: function (frag) {
- return 'a=ice-ufrag:' + frag;
- },
- parse_icepwd: function (line) {
- return line.substring(10);
- },
- build_icepwd: function (pwd) {
- return 'a=ice-pwd:' + pwd;
- },
- parse_mid: function (line) {
- return line.substring(6);
- },
- parse_mline: function (line) {
- var parts = line.substring(2).split(' '),
- data = {};
- data.media = parts.shift();
- data.port = parts.shift();
- data.proto = parts.shift();
- if (parts[parts.length - 1] === '') { // trailing whitespace
- parts.pop();
- }
- data.fmt = parts;
- return data;
- },
- build_mline: function (mline) {
- return 'm=' + mline.media + ' ' + mline.port + ' ' + mline.proto + ' ' + mline.fmt.join(' ');
- },
- parse_rtpmap: function (line) {
- var parts = line.substring(9).split(' '),
- data = {};
- data.id = parts.shift();
- parts = parts[0].split('/');
- data.name = parts.shift();
- data.clockrate = parts.shift();
- data.channels = parts.length ? parts.shift() : '1';
- return data;
- },
- /**
- * Parses SDP line "a=sctpmap:..." and extracts SCTP port from it.
- * @param line eg. "a=sctpmap:5000 webrtc-datachannel"
- * @returns [SCTP port number, protocol, streams]
- */
- parse_sctpmap: function (line)
- {
- var parts = line.substring(10).split(' ');
- var sctpPort = parts[0];
- var protocol = parts[1];
- // Stream count is optional
- var streamCount = parts.length > 2 ? parts[2] : null;
- return [sctpPort, protocol, streamCount];// SCTP port
- },
- build_rtpmap: function (el) {
- var line = 'a=rtpmap:' + el.getAttribute('id') + ' ' + el.getAttribute('name') + '/' + el.getAttribute('clockrate');
- if (el.getAttribute('channels') && el.getAttribute('channels') != '1') {
- line += '/' + el.getAttribute('channels');
- }
- return line;
- },
- parse_crypto: function (line) {
- var parts = line.substring(9).split(' '),
- data = {};
- data.tag = parts.shift();
- data['crypto-suite'] = parts.shift();
- data['key-params'] = parts.shift();
- if (parts.length) {
- data['session-params'] = parts.join(' ');
- }
- return data;
- },
- parse_fingerprint: function (line) { // RFC 4572
- var parts = line.substring(14).split(' '),
- data = {};
- data.hash = parts.shift();
- data.fingerprint = parts.shift();
- // TODO assert that fingerprint satisfies 2UHEX *(":" 2UHEX) ?
- return data;
- },
- parse_fmtp: function (line) {
- var parts = line.split(' '),
- i, key, value,
- data = [];
- parts.shift();
- parts = parts.join(' ').split(';');
- for (i = 0; i < parts.length; i++) {
- key = parts[i].split('=')[0];
- while (key.length && key[0] == ' ') {
- key = key.substring(1);
- }
- value = parts[i].split('=')[1];
- if (key && value) {
- data.push({name: key, value: value});
- } else if (key) {
- // rfc 4733 (DTMF) style stuff
- data.push({name: '', value: key});
- }
- }
- return data;
- },
- parse_icecandidate: function (line) {
- var candidate = {},
- elems = line.split(' ');
- candidate.foundation = elems[0].substring(12);
- candidate.component = elems[1];
- candidate.protocol = elems[2].toLowerCase();
- candidate.priority = elems[3];
- candidate.ip = elems[4];
- candidate.port = elems[5];
- // elems[6] => "typ"
- candidate.type = elems[7];
- candidate.generation = 0; // default value, may be overwritten below
- for (var i = 8; i < elems.length; i += 2) {
- switch (elems[i]) {
- case 'raddr':
- candidate['rel-addr'] = elems[i + 1];
- break;
- case 'rport':
- candidate['rel-port'] = elems[i + 1];
- break;
- case 'generation':
- candidate.generation = elems[i + 1];
- break;
- case 'tcptype':
- candidate.tcptype = elems[i + 1];
- break;
- default: // TODO
- console.log('parse_icecandidate not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
- }
- }
- candidate.network = '1';
- candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
- return candidate;
- },
- build_icecandidate: function (cand) {
- var line = ['a=candidate:' + cand.foundation, cand.component, cand.protocol, cand.priority, cand.ip, cand.port, 'typ', cand.type].join(' ');
- line += ' ';
- switch (cand.type) {
- case 'srflx':
- case 'prflx':
- case 'relay':
- if (cand.hasOwnAttribute('rel-addr') && cand.hasOwnAttribute('rel-port')) {
- line += 'raddr';
- line += ' ';
- line += cand['rel-addr'];
- line += ' ';
- line += 'rport';
- line += ' ';
- line += cand['rel-port'];
- line += ' ';
- }
- break;
- }
- if (cand.hasOwnAttribute('tcptype')) {
- line += 'tcptype';
- line += ' ';
- line += cand.tcptype;
- line += ' ';
- }
- line += 'generation';
- line += ' ';
- line += cand.hasOwnAttribute('generation') ? cand.generation : '0';
- return line;
- },
- parse_ssrc: function (desc) {
- // proprietary mapping of a=ssrc lines
- // TODO: see "Jingle RTP Source Description" by Juberti and P. Thatcher on google docs
- // and parse according to that
- var lines = desc.split('\r\n'),
- data = {};
- for (var i = 0; i < lines.length; i++) {
- if (lines[i].substring(0, 7) == 'a=ssrc:') {
- var idx = lines[i].indexOf(' ');
- data[lines[i].substr(idx + 1).split(':', 2)[0]] = lines[i].substr(idx + 1).split(':', 2)[1];
- }
- }
- return data;
- },
- parse_rtcpfb: function (line) {
- var parts = line.substr(10).split(' ');
- var data = {};
- data.pt = parts.shift();
- data.type = parts.shift();
- data.params = parts;
- return data;
- },
- parse_extmap: function (line) {
- var parts = line.substr(9).split(' ');
- var data = {};
- data.value = parts.shift();
- if (data.value.indexOf('/') != -1) {
- data.direction = data.value.substr(data.value.indexOf('/') + 1);
- data.value = data.value.substr(0, data.value.indexOf('/'));
- } else {
- data.direction = 'both';
- }
- data.uri = parts.shift();
- data.params = parts;
- return data;
- },
- find_line: function (haystack, needle, sessionpart) {
- var lines = haystack.split('\r\n');
- for (var i = 0; i < lines.length; i++) {
- if (lines[i].substring(0, needle.length) == needle) {
- return lines[i];
- }
- }
- if (!sessionpart) {
- return false;
- }
- // search session part
- lines = sessionpart.split('\r\n');
- for (var j = 0; j < lines.length; j++) {
- if (lines[j].substring(0, needle.length) == needle) {
- return lines[j];
- }
- }
- return false;
- },
- find_lines: function (haystack, needle, sessionpart) {
- var lines = haystack.split('\r\n'),
- needles = [];
- for (var i = 0; i < lines.length; i++) {
- if (lines[i].substring(0, needle.length) == needle)
- needles.push(lines[i]);
- }
- if (needles.length || !sessionpart) {
- return needles;
- }
- // search session part
- lines = sessionpart.split('\r\n');
- for (var j = 0; j < lines.length; j++) {
- if (lines[j].substring(0, needle.length) == needle) {
- needles.push(lines[j]);
- }
- }
- return needles;
- },
- candidateToJingle: function (line) {
- // a=candidate:2979166662 1 udp 2113937151 192.168.2.100 57698 typ host generation 0
- //
- if (line.indexOf('candidate:') === 0) {
- line = 'a=' + line;
- } else if (line.substring(0, 12) != 'a=candidate:') {
- console.log('parseCandidate called with a line that is not a candidate line');
- console.log(line);
- return null;
- }
- if (line.substring(line.length - 2) == '\r\n') // chomp it
- line = line.substring(0, line.length - 2);
- var candidate = {},
- elems = line.split(' '),
- i;
- if (elems[6] != 'typ') {
- console.log('did not find typ in the right place');
- console.log(line);
- return null;
- }
- candidate.foundation = elems[0].substring(12);
- candidate.component = elems[1];
- candidate.protocol = elems[2].toLowerCase();
- candidate.priority = elems[3];
- candidate.ip = elems[4];
- candidate.port = elems[5];
- // elems[6] => "typ"
- candidate.type = elems[7];
- candidate.generation = '0'; // default, may be overwritten below
- for (i = 8; i < elems.length; i += 2) {
- switch (elems[i]) {
- case 'raddr':
- candidate['rel-addr'] = elems[i + 1];
- break;
- case 'rport':
- candidate['rel-port'] = elems[i + 1];
- break;
- case 'generation':
- candidate.generation = elems[i + 1];
- break;
- case 'tcptype':
- candidate.tcptype = elems[i + 1];
- break;
- default: // TODO
- console.log('not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
- }
- }
- candidate.network = '1';
- candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
- return candidate;
- },
- candidateFromJingle: function (cand) {
- var line = 'a=candidate:';
- line += cand.getAttribute('foundation');
- line += ' ';
- line += cand.getAttribute('component');
- line += ' ';
- line += cand.getAttribute('protocol'); //.toUpperCase(); // chrome M23 doesn't like this
- line += ' ';
- line += cand.getAttribute('priority');
- line += ' ';
- line += cand.getAttribute('ip');
- line += ' ';
- line += cand.getAttribute('port');
- line += ' ';
- line += 'typ';
- line += ' ' + cand.getAttribute('type');
- line += ' ';
- switch (cand.getAttribute('type')) {
- case 'srflx':
- case 'prflx':
- case 'relay':
- if (cand.getAttribute('rel-addr') && cand.getAttribute('rel-port')) {
- line += 'raddr';
- line += ' ';
- line += cand.getAttribute('rel-addr');
- line += ' ';
- line += 'rport';
- line += ' ';
- line += cand.getAttribute('rel-port');
- line += ' ';
- }
- break;
- }
- if (cand.getAttribute('protocol').toLowerCase() == 'tcp') {
- line += 'tcptype';
- line += ' ';
- line += cand.getAttribute('tcptype');
- line += ' ';
- }
- line += 'generation';
- line += ' ';
- line += cand.getAttribute('generation') || '0';
- return line + '\r\n';
- }
-};
+module.exports = SDP;
diff --git a/modules/xmpp/SDPDiffer.js b/modules/xmpp/SDPDiffer.js
new file mode 100644
index 000000000..ebaaadb13
--- /dev/null
+++ b/modules/xmpp/SDPDiffer.js
@@ -0,0 +1,165 @@
+function SDPDiffer(mySDP, otherSDP) {
+ this.mySDP = mySDP;
+ this.otherSDP = otherSDP;
+}
+
+/**
+ * Returns map of MediaChannel that contains only media not contained in otherSdp. Mapped by channel idx.
+ * @param otherSdp the other SDP to check ssrc with.
+ */
+SDPDiffer.prototype.getNewMedia = function() {
+
+ // this could be useful in Array.prototype.
+ function arrayEquals(array) {
+ // if the other array is a falsy value, return
+ if (!array)
+ return false;
+
+ // compare lengths - can save a lot of time
+ if (this.length != array.length)
+ return false;
+
+ for (var i = 0, l=this.length; i < l; i++) {
+ // Check if we have nested arrays
+ if (this[i] instanceof Array && array[i] instanceof Array) {
+ // recurse into the nested arrays
+ if (!this[i].equals(array[i]))
+ return false;
+ }
+ else if (this[i] != array[i]) {
+ // Warning - two different object instances will never be equal: {x:20} != {x:20}
+ return false;
+ }
+ }
+ return true;
+ }
+
+ var myMedias = this.mySDP.getMediaSsrcMap();
+ var othersMedias = this.otherSDP.getMediaSsrcMap();
+ var newMedia = {};
+ Object.keys(othersMedias).forEach(function(othersMediaIdx) {
+ var myMedia = myMedias[othersMediaIdx];
+ var othersMedia = othersMedias[othersMediaIdx];
+ if(!myMedia && othersMedia) {
+ // Add whole channel
+ newMedia[othersMediaIdx] = othersMedia;
+ return;
+ }
+ // Look for new ssrcs accross the channel
+ Object.keys(othersMedia.ssrcs).forEach(function(ssrc) {
+ if(Object.keys(myMedia.ssrcs).indexOf(ssrc) === -1) {
+ // Allocate channel if we've found ssrc that doesn't exist in our channel
+ if(!newMedia[othersMediaIdx]){
+ newMedia[othersMediaIdx] = {
+ mediaindex: othersMedia.mediaindex,
+ mid: othersMedia.mid,
+ ssrcs: {},
+ ssrcGroups: []
+ };
+ }
+ newMedia[othersMediaIdx].ssrcs[ssrc] = othersMedia.ssrcs[ssrc];
+ }
+ });
+
+ // Look for new ssrc groups across the channels
+ othersMedia.ssrcGroups.forEach(function(otherSsrcGroup){
+
+ // try to match the other ssrc-group with an ssrc-group of ours
+ var matched = false;
+ for (var i = 0; i < myMedia.ssrcGroups.length; i++) {
+ var mySsrcGroup = myMedia.ssrcGroups[i];
+ if (otherSsrcGroup.semantics == mySsrcGroup.semantics
+ && arrayEquals.apply(otherSsrcGroup.ssrcs, [mySsrcGroup.ssrcs])) {
+
+ matched = true;
+ break;
+ }
+ }
+
+ if (!matched) {
+ // Allocate channel if we've found an ssrc-group that doesn't
+ // exist in our channel
+
+ if(!newMedia[othersMediaIdx]){
+ newMedia[othersMediaIdx] = {
+ mediaindex: othersMedia.mediaindex,
+ mid: othersMedia.mid,
+ ssrcs: {},
+ ssrcGroups: []
+ };
+ }
+ newMedia[othersMediaIdx].ssrcGroups.push(otherSsrcGroup);
+ }
+ });
+ });
+ return newMedia;
+};
+
+/**
+ * Sends SSRC update IQ.
+ * @param sdpMediaSsrcs SSRCs map obtained from SDP.getNewMedia. Cntains SSRCs to add/remove.
+ * @param sid session identifier that will be put into the IQ.
+ * @param initiator initiator identifier.
+ * @param toJid destination Jid
+ * @param isAdd indicates if this is remove or add operation.
+ */
+SDPDiffer.prototype.toJingle = function(modify) {
+ var sdpMediaSsrcs = this.getNewMedia();
+ var self = this;
+
+ // FIXME: only announce video ssrcs since we mix audio and dont need
+ // the audio ssrcs therefore
+ var modified = false;
+ Object.keys(sdpMediaSsrcs).forEach(function(mediaindex){
+ modified = true;
+ var media = sdpMediaSsrcs[mediaindex];
+ modify.c('content', {name: media.mid});
+
+ modify.c('description', {xmlns:'urn:xmpp:jingle:apps:rtp:1', media: media.mid});
+ // FIXME: not completly sure this operates on blocks and / or handles different ssrcs correctly
+ // generate sources from lines
+ Object.keys(media.ssrcs).forEach(function(ssrcNum) {
+ var mediaSsrc = media.ssrcs[ssrcNum];
+ modify.c('source', { xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
+ modify.attrs({ssrc: mediaSsrc.ssrc});
+ // iterate over ssrc lines
+ mediaSsrc.lines.forEach(function (line) {
+ var idx = line.indexOf(' ');
+ var kv = line.substr(idx + 1);
+ modify.c('parameter');
+ if (kv.indexOf(':') == -1) {
+ modify.attrs({ name: kv });
+ } else {
+ modify.attrs({ name: kv.split(':', 2)[0] });
+ modify.attrs({ value: kv.split(':', 2)[1] });
+ }
+ modify.up(); // end of parameter
+ });
+ modify.up(); // end of source
+ });
+
+ // generate source groups from lines
+ media.ssrcGroups.forEach(function(ssrcGroup) {
+ if (ssrcGroup.ssrcs.length != 0) {
+
+ modify.c('ssrc-group', {
+ semantics: ssrcGroup.semantics,
+ xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0'
+ });
+
+ ssrcGroup.ssrcs.forEach(function (ssrc) {
+ modify.c('source', { ssrc: ssrc })
+ .up(); // end of source
+ });
+ modify.up(); // end of ssrc-group
+ }
+ });
+
+ modify.up(); // end of description
+ modify.up(); // end of content
+ });
+
+ return modified;
+};
+
+module.exports = SDPDiffer;
\ No newline at end of file
diff --git a/modules/xmpp/SDPUtil.js b/modules/xmpp/SDPUtil.js
new file mode 100644
index 000000000..d75d35f7a
--- /dev/null
+++ b/modules/xmpp/SDPUtil.js
@@ -0,0 +1,349 @@
+SDPUtil = {
+ iceparams: function (mediadesc, sessiondesc) {
+ var data = null;
+ if (SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc) &&
+ SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc)) {
+ data = {
+ ufrag: SDPUtil.parse_iceufrag(SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc)),
+ pwd: SDPUtil.parse_icepwd(SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc))
+ };
+ }
+ return data;
+ },
+ parse_iceufrag: function (line) {
+ return line.substring(12);
+ },
+ build_iceufrag: function (frag) {
+ return 'a=ice-ufrag:' + frag;
+ },
+ parse_icepwd: function (line) {
+ return line.substring(10);
+ },
+ build_icepwd: function (pwd) {
+ return 'a=ice-pwd:' + pwd;
+ },
+ parse_mid: function (line) {
+ return line.substring(6);
+ },
+ parse_mline: function (line) {
+ var parts = line.substring(2).split(' '),
+ data = {};
+ data.media = parts.shift();
+ data.port = parts.shift();
+ data.proto = parts.shift();
+ if (parts[parts.length - 1] === '') { // trailing whitespace
+ parts.pop();
+ }
+ data.fmt = parts;
+ return data;
+ },
+ build_mline: function (mline) {
+ return 'm=' + mline.media + ' ' + mline.port + ' ' + mline.proto + ' ' + mline.fmt.join(' ');
+ },
+ parse_rtpmap: function (line) {
+ var parts = line.substring(9).split(' '),
+ data = {};
+ data.id = parts.shift();
+ parts = parts[0].split('/');
+ data.name = parts.shift();
+ data.clockrate = parts.shift();
+ data.channels = parts.length ? parts.shift() : '1';
+ return data;
+ },
+ /**
+ * Parses SDP line "a=sctpmap:..." and extracts SCTP port from it.
+ * @param line eg. "a=sctpmap:5000 webrtc-datachannel"
+ * @returns [SCTP port number, protocol, streams]
+ */
+ parse_sctpmap: function (line)
+ {
+ var parts = line.substring(10).split(' ');
+ var sctpPort = parts[0];
+ var protocol = parts[1];
+ // Stream count is optional
+ var streamCount = parts.length > 2 ? parts[2] : null;
+ return [sctpPort, protocol, streamCount];// SCTP port
+ },
+ build_rtpmap: function (el) {
+ var line = 'a=rtpmap:' + el.getAttribute('id') + ' ' + el.getAttribute('name') + '/' + el.getAttribute('clockrate');
+ if (el.getAttribute('channels') && el.getAttribute('channels') != '1') {
+ line += '/' + el.getAttribute('channels');
+ }
+ return line;
+ },
+ parse_crypto: function (line) {
+ var parts = line.substring(9).split(' '),
+ data = {};
+ data.tag = parts.shift();
+ data['crypto-suite'] = parts.shift();
+ data['key-params'] = parts.shift();
+ if (parts.length) {
+ data['session-params'] = parts.join(' ');
+ }
+ return data;
+ },
+ parse_fingerprint: function (line) { // RFC 4572
+ var parts = line.substring(14).split(' '),
+ data = {};
+ data.hash = parts.shift();
+ data.fingerprint = parts.shift();
+ // TODO assert that fingerprint satisfies 2UHEX *(":" 2UHEX) ?
+ return data;
+ },
+ parse_fmtp: function (line) {
+ var parts = line.split(' '),
+ i, key, value,
+ data = [];
+ parts.shift();
+ parts = parts.join(' ').split(';');
+ for (i = 0; i < parts.length; i++) {
+ key = parts[i].split('=')[0];
+ while (key.length && key[0] == ' ') {
+ key = key.substring(1);
+ }
+ value = parts[i].split('=')[1];
+ if (key && value) {
+ data.push({name: key, value: value});
+ } else if (key) {
+ // rfc 4733 (DTMF) style stuff
+ data.push({name: '', value: key});
+ }
+ }
+ return data;
+ },
+ parse_icecandidate: function (line) {
+ var candidate = {},
+ elems = line.split(' ');
+ candidate.foundation = elems[0].substring(12);
+ candidate.component = elems[1];
+ candidate.protocol = elems[2].toLowerCase();
+ candidate.priority = elems[3];
+ candidate.ip = elems[4];
+ candidate.port = elems[5];
+ // elems[6] => "typ"
+ candidate.type = elems[7];
+ candidate.generation = 0; // default value, may be overwritten below
+ for (var i = 8; i < elems.length; i += 2) {
+ switch (elems[i]) {
+ case 'raddr':
+ candidate['rel-addr'] = elems[i + 1];
+ break;
+ case 'rport':
+ candidate['rel-port'] = elems[i + 1];
+ break;
+ case 'generation':
+ candidate.generation = elems[i + 1];
+ break;
+ case 'tcptype':
+ candidate.tcptype = elems[i + 1];
+ break;
+ default: // TODO
+ console.log('parse_icecandidate not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
+ }
+ }
+ candidate.network = '1';
+ candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
+ return candidate;
+ },
+ build_icecandidate: function (cand) {
+ var line = ['a=candidate:' + cand.foundation, cand.component, cand.protocol, cand.priority, cand.ip, cand.port, 'typ', cand.type].join(' ');
+ line += ' ';
+ switch (cand.type) {
+ case 'srflx':
+ case 'prflx':
+ case 'relay':
+ if (cand.hasOwnAttribute('rel-addr') && cand.hasOwnAttribute('rel-port')) {
+ line += 'raddr';
+ line += ' ';
+ line += cand['rel-addr'];
+ line += ' ';
+ line += 'rport';
+ line += ' ';
+ line += cand['rel-port'];
+ line += ' ';
+ }
+ break;
+ }
+ if (cand.hasOwnAttribute('tcptype')) {
+ line += 'tcptype';
+ line += ' ';
+ line += cand.tcptype;
+ line += ' ';
+ }
+ line += 'generation';
+ line += ' ';
+ line += cand.hasOwnAttribute('generation') ? cand.generation : '0';
+ return line;
+ },
+ parse_ssrc: function (desc) {
+ // proprietary mapping of a=ssrc lines
+ // TODO: see "Jingle RTP Source Description" by Juberti and P. Thatcher on google docs
+ // and parse according to that
+ var lines = desc.split('\r\n'),
+ data = {};
+ for (var i = 0; i < lines.length; i++) {
+ if (lines[i].substring(0, 7) == 'a=ssrc:') {
+ var idx = lines[i].indexOf(' ');
+ data[lines[i].substr(idx + 1).split(':', 2)[0]] = lines[i].substr(idx + 1).split(':', 2)[1];
+ }
+ }
+ return data;
+ },
+ parse_rtcpfb: function (line) {
+ var parts = line.substr(10).split(' ');
+ var data = {};
+ data.pt = parts.shift();
+ data.type = parts.shift();
+ data.params = parts;
+ return data;
+ },
+ parse_extmap: function (line) {
+ var parts = line.substr(9).split(' ');
+ var data = {};
+ data.value = parts.shift();
+ if (data.value.indexOf('/') != -1) {
+ data.direction = data.value.substr(data.value.indexOf('/') + 1);
+ data.value = data.value.substr(0, data.value.indexOf('/'));
+ } else {
+ data.direction = 'both';
+ }
+ data.uri = parts.shift();
+ data.params = parts;
+ return data;
+ },
+ find_line: function (haystack, needle, sessionpart) {
+ var lines = haystack.split('\r\n');
+ for (var i = 0; i < lines.length; i++) {
+ if (lines[i].substring(0, needle.length) == needle) {
+ return lines[i];
+ }
+ }
+ if (!sessionpart) {
+ return false;
+ }
+ // search session part
+ lines = sessionpart.split('\r\n');
+ for (var j = 0; j < lines.length; j++) {
+ if (lines[j].substring(0, needle.length) == needle) {
+ return lines[j];
+ }
+ }
+ return false;
+ },
+ find_lines: function (haystack, needle, sessionpart) {
+ var lines = haystack.split('\r\n'),
+ needles = [];
+ for (var i = 0; i < lines.length; i++) {
+ if (lines[i].substring(0, needle.length) == needle)
+ needles.push(lines[i]);
+ }
+ if (needles.length || !sessionpart) {
+ return needles;
+ }
+ // search session part
+ lines = sessionpart.split('\r\n');
+ for (var j = 0; j < lines.length; j++) {
+ if (lines[j].substring(0, needle.length) == needle) {
+ needles.push(lines[j]);
+ }
+ }
+ return needles;
+ },
+ candidateToJingle: function (line) {
+ // a=candidate:2979166662 1 udp 2113937151 192.168.2.100 57698 typ host generation 0
+ //
+ if (line.indexOf('candidate:') === 0) {
+ line = 'a=' + line;
+ } else if (line.substring(0, 12) != 'a=candidate:') {
+ console.log('parseCandidate called with a line that is not a candidate line');
+ console.log(line);
+ return null;
+ }
+ if (line.substring(line.length - 2) == '\r\n') // chomp it
+ line = line.substring(0, line.length - 2);
+ var candidate = {},
+ elems = line.split(' '),
+ i;
+ if (elems[6] != 'typ') {
+ console.log('did not find typ in the right place');
+ console.log(line);
+ return null;
+ }
+ candidate.foundation = elems[0].substring(12);
+ candidate.component = elems[1];
+ candidate.protocol = elems[2].toLowerCase();
+ candidate.priority = elems[3];
+ candidate.ip = elems[4];
+ candidate.port = elems[5];
+ // elems[6] => "typ"
+ candidate.type = elems[7];
+
+ candidate.generation = '0'; // default, may be overwritten below
+ for (i = 8; i < elems.length; i += 2) {
+ switch (elems[i]) {
+ case 'raddr':
+ candidate['rel-addr'] = elems[i + 1];
+ break;
+ case 'rport':
+ candidate['rel-port'] = elems[i + 1];
+ break;
+ case 'generation':
+ candidate.generation = elems[i + 1];
+ break;
+ case 'tcptype':
+ candidate.tcptype = elems[i + 1];
+ break;
+ default: // TODO
+ console.log('not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
+ }
+ }
+ candidate.network = '1';
+ candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
+ return candidate;
+ },
+ candidateFromJingle: function (cand) {
+ var line = 'a=candidate:';
+ line += cand.getAttribute('foundation');
+ line += ' ';
+ line += cand.getAttribute('component');
+ line += ' ';
+ line += cand.getAttribute('protocol'); //.toUpperCase(); // chrome M23 doesn't like this
+ line += ' ';
+ line += cand.getAttribute('priority');
+ line += ' ';
+ line += cand.getAttribute('ip');
+ line += ' ';
+ line += cand.getAttribute('port');
+ line += ' ';
+ line += 'typ';
+ line += ' ' + cand.getAttribute('type');
+ line += ' ';
+ switch (cand.getAttribute('type')) {
+ case 'srflx':
+ case 'prflx':
+ case 'relay':
+ if (cand.getAttribute('rel-addr') && cand.getAttribute('rel-port')) {
+ line += 'raddr';
+ line += ' ';
+ line += cand.getAttribute('rel-addr');
+ line += ' ';
+ line += 'rport';
+ line += ' ';
+ line += cand.getAttribute('rel-port');
+ line += ' ';
+ }
+ break;
+ }
+ if (cand.getAttribute('protocol').toLowerCase() == 'tcp') {
+ line += 'tcptype';
+ line += ' ';
+ line += cand.getAttribute('tcptype');
+ line += ' ';
+ }
+ line += 'generation';
+ line += ' ';
+ line += cand.getAttribute('generation') || '0';
+ return line + '\r\n';
+ }
+};
+module.exports = SDPUtil;
\ No newline at end of file
diff --git a/libs/strophe/strophe.jingle.adapter.js b/modules/xmpp/TraceablePeerConnection.js
similarity index 99%
rename from libs/strophe/strophe.jingle.adapter.js
rename to modules/xmpp/TraceablePeerConnection.js
index 035c43ab3..c8db15337 100644
--- a/libs/strophe/strophe.jingle.adapter.js
+++ b/modules/xmpp/TraceablePeerConnection.js
@@ -262,3 +262,5 @@ TraceablePeerConnection.prototype.getStats = function(callback, errback) {
}
};
+module.exports = TraceablePeerConnection;
+
diff --git a/moderator.js b/modules/xmpp/moderator.js
similarity index 70%
rename from moderator.js
rename to modules/xmpp/moderator.js
index 5f0ed3bd8..439d70311 100644
--- a/moderator.js
+++ b/modules/xmpp/moderator.js
@@ -1,47 +1,53 @@
-/* global $, $iq, config, connection, Etherpad, hangUp, messageHandler,
+/* global $, $iq, config, connection, UI, messageHandler,
roomName, sessionTerminated, Strophe, Util */
/**
* Contains logic responsible for enabling/disabling functionality available
* only to moderator users.
*/
-var Moderator = (function (my) {
+var connection = null;
+var focusUserJid;
+var getNextTimeout = Util.createExpBackoffTimer(1000);
+var getNextErrorTimeout = Util.createExpBackoffTimer(1000);
+// External authentication stuff
+var externalAuthEnabled = false;
+// Sip gateway can be enabled by configuring Jigasi host in config.js or
+// it will be enabled automatically if focus detects the component through
+// service discovery.
+var sipGatewayEnabled = config.hosts.call_control !== undefined;
- var focusUserJid;
- var getNextTimeout = Util.createExpBackoffTimer(1000);
- var getNextErrorTimeout = Util.createExpBackoffTimer(1000);
- // External authentication stuff
- var externalAuthEnabled = false;
- // Sip gateway can be enabled by configuring Jigasi host in config.js or
- // it will be enabled automatically if focus detects the component through
- // service discovery.
- var sipGatewayEnabled = config.hosts.call_control !== undefined;
-
- my.isModerator = function () {
+var Moderator = {
+ isModerator: function () {
return connection && connection.emuc.isModerator();
- };
+ },
- my.isPeerModerator = function (peerJid) {
- return connection && connection.emuc.getMemberRole(peerJid) === 'moderator';
- };
+ isPeerModerator: function (peerJid) {
+ return connection &&
+ connection.emuc.getMemberRole(peerJid) === 'moderator';
+ },
- my.isExternalAuthEnabled = function () {
+ isExternalAuthEnabled: function () {
return externalAuthEnabled;
- };
+ },
- my.isSipGatewayEnabled = function () {
+ isSipGatewayEnabled: function () {
return sipGatewayEnabled;
- };
+ },
- my.init = function () {
- Moderator.onLocalRoleChange = function (from, member, pres) {
+ setConnection: function (con) {
+ connection = con;
+ },
+
+ init: function (xmpp) {
+ this.xmppService = xmpp;
+ this.onLocalRoleChange = function (from, member, pres) {
UI.onModeratorStatusChanged(Moderator.isModerator());
};
- };
+ },
- my.onMucLeft = function (jid) {
+ onMucLeft: function (jid) {
console.info("Someone left is it focus ? " + jid);
var resource = Strophe.getResourceFromJid(jid);
- if (resource === 'focus' && !sessionTerminated) {
+ if (resource === 'focus' && !this.xmppService.sessionTerminated) {
console.info(
"Focus has left the room - leaving conference");
//hangUp();
@@ -49,20 +55,20 @@ var Moderator = (function (my) {
// FIXME: show some message before reload
location.reload();
}
- }
-
- my.setFocusUserJid = function (focusJid) {
+ },
+
+ setFocusUserJid: function (focusJid) {
if (!focusUserJid) {
focusUserJid = focusJid;
console.info("Focus jid set to: " + focusUserJid);
}
- };
+ },
- my.getFocusUserJid = function () {
+ getFocusUserJid: function () {
return focusUserJid;
- };
+ },
- my.getFocusComponent = function () {
+ getFocusComponent: function () {
// Get focus component address
var focusComponent = config.hosts.focus;
// If not specified use default: 'focus.domain'
@@ -70,99 +76,93 @@ var Moderator = (function (my) {
focusComponent = 'focus.' + config.hosts.domain;
}
return focusComponent;
- };
+ },
- my.createConferenceIq = function () {
+ createConferenceIq: function (roomName) {
// Generate create conference IQ
var elem = $iq({to: Moderator.getFocusComponent(), type: 'set'});
elem.c('conference', {
xmlns: 'http://jitsi.org/protocol/focus',
room: roomName
});
- if (config.hosts.bridge !== undefined)
- {
+ if (config.hosts.bridge !== undefined) {
elem.c(
'property',
{ name: 'bridge', value: config.hosts.bridge})
.up();
}
// Tell the focus we have Jigasi configured
- if (config.hosts.call_control !== undefined)
- {
+ if (config.hosts.call_control !== undefined) {
elem.c(
'property',
{ name: 'call_control', value: config.hosts.call_control})
.up();
}
- if (config.channelLastN !== undefined)
- {
+ if (config.channelLastN !== undefined) {
elem.c(
'property',
{ name: 'channelLastN', value: config.channelLastN})
.up();
}
- if (config.adaptiveLastN !== undefined)
- {
+ if (config.adaptiveLastN !== undefined) {
elem.c(
'property',
{ name: 'adaptiveLastN', value: config.adaptiveLastN})
.up();
}
- if (config.adaptiveSimulcast !== undefined)
- {
+ if (config.adaptiveSimulcast !== undefined) {
elem.c(
'property',
{ name: 'adaptiveSimulcast', value: config.adaptiveSimulcast})
.up();
}
- if (config.openSctp !== undefined)
- {
+ if (config.openSctp !== undefined) {
elem.c(
'property',
{ name: 'openSctp', value: config.openSctp})
.up();
}
- if (config.enableFirefoxSupport !== undefined)
- {
+ if (config.enableFirefoxSupport !== undefined) {
elem.c(
'property',
- { name: 'enableFirefoxHacks', value: config.enableFirefoxSupport})
+ { name: 'enableFirefoxHacks',
+ value: config.enableFirefoxSupport})
.up();
}
elem.up();
return elem;
- };
-
- my.parseConfigOptions = function (resultIq) {
+ },
+ parseConfigOptions: function (resultIq) {
+
Moderator.setFocusUserJid(
$(resultIq).find('conference').attr('focusjid'));
-
+
var extAuthParam
= $(resultIq).find('>conference>property[name=\'externalAuth\']');
if (extAuthParam.length) {
externalAuthEnabled = extAuthParam.attr('value') === 'true';
}
-
+
console.info("External authentication enabled: " + externalAuthEnabled);
-
+
// Check if focus has auto-detected Jigasi component(this will be also
// included if we have passed our host from the config)
if ($(resultIq).find(
- '>conference>property[name=\'sipGatewayEnabled\']').length) {
+ '>conference>property[name=\'sipGatewayEnabled\']').length) {
sipGatewayEnabled = true;
}
-
+
console.info("Sip gateway enabled: " + sipGatewayEnabled);
- };
+ },
// FIXME: we need to show the fact that we're waiting for the focus
// to the user(or that focus is not available)
- my.allocateConferenceFocus = function (roomName, callback) {
+ allocateConferenceFocus: function (roomName, callback) {
// Try to use focus user JID from the config
Moderator.setFocusUserJid(config.focusUserJid);
// Send create conference IQ
- var iq = Moderator.createConferenceIq();
+ var iq = Moderator.createConferenceIq(roomName);
connection.sendIQ(
iq,
function (result) {
@@ -190,7 +190,9 @@ var Moderator = (function (my) {
// Not authorized to create new room
if ($(error).find('>error>not-authorized').length) {
console.warn("Unauthorized to start the conference");
- UI.onAuthenticationRequired();
+ UI.onAuthenticationRequired(function () {
+ Moderator.allocateConferenceFocus(roomName, callback);
+ });
return;
}
var waitMs = getNextErrorTimeout();
@@ -198,8 +200,9 @@ var Moderator = (function (my) {
// Show message
UI.messageHandler.notify(
'Conference focus', 'disconnected',
- Moderator.getFocusComponent() +
- ' not available - retry in ' + (waitMs / 1000) + ' sec');
+ Moderator.getFocusComponent() +
+ ' not available - retry in ' +
+ (waitMs / 1000) + ' sec');
// Reset response timeout
getNextTimeout(true);
window.setTimeout(
@@ -208,9 +211,9 @@ var Moderator = (function (my) {
}, waitMs);
}
);
- };
+ },
- my.getAuthUrl = function (urlCallback) {
+ getAuthUrl: function (roomName, urlCallback) {
var iq = $iq({to: Moderator.getFocusComponent(), type: 'get'});
iq.c('auth-url', {
xmlns: 'http://jitsi.org/protocol/focus',
@@ -232,10 +235,10 @@ var Moderator = (function (my) {
console.error("Get auth url error", error);
}
);
- };
+ }
+};
- return my;
-}(Moderator || {}));
+module.exports = Moderator;
diff --git a/modules/xmpp/recording.js b/modules/xmpp/recording.js
new file mode 100644
index 000000000..245260c49
--- /dev/null
+++ b/modules/xmpp/recording.js
@@ -0,0 +1,152 @@
+/* global $, $iq, config, connection, focusMucJid, messageHandler, Moderator,
+ Toolbar, Util */
+var Moderator = require("./moderator");
+
+
+var recordingToken = null;
+var recordingEnabled;
+
+/**
+ * Whether to use a jirecon component for recording, or use the videobridge
+ * through COLIBRI.
+ */
+var useJirecon = (typeof config.hosts.jirecon != "undefined");
+
+/**
+ * The ID of the jirecon recording session. Jirecon generates it when we
+ * initially start recording, and it needs to be used in subsequent requests
+ * to jirecon.
+ */
+var jireconRid = null;
+
+function setRecordingToken(token) {
+ recordingToken = token;
+}
+
+function setRecording(state, token, callback) {
+ if (useJirecon){
+ this.setRecordingJirecon(state, token, callback);
+ } else {
+ this.setRecordingColibri(state, token, callback);
+ }
+}
+
+function setRecordingJirecon(state, token, callback) {
+ if (state == recordingEnabled){
+ return;
+ }
+
+ var iq = $iq({to: config.hosts.jirecon, type: 'set'})
+ .c('recording', {xmlns: 'http://jitsi.org/protocol/jirecon',
+ action: state ? 'start' : 'stop',
+ mucjid: connection.emuc.roomjid});
+ if (!state){
+ iq.attrs({rid: jireconRid});
+ }
+
+ console.log('Start recording');
+
+ connection.sendIQ(
+ iq,
+ function (result) {
+ // TODO wait for an IQ with the real status, since this is
+ // provisional?
+ jireconRid = $(result).find('recording').attr('rid');
+ console.log('Recording ' + (state ? 'started' : 'stopped') +
+ '(jirecon)' + result);
+ recordingEnabled = state;
+ if (!state){
+ jireconRid = null;
+ }
+
+ callback(state);
+ },
+ function (error) {
+ console.log('Failed to start recording, error: ', error);
+ callback(recordingEnabled);
+ });
+}
+
+// Sends a COLIBRI message which enables or disables (according to 'state')
+// the recording on the bridge. Waits for the result IQ and calls 'callback'
+// with the new recording state, according to the IQ.
+function setRecordingColibri(state, token, callback) {
+ var elem = $iq({to: focusMucJid, type: 'set'});
+ elem.c('conference', {
+ xmlns: 'http://jitsi.org/protocol/colibri'
+ });
+ elem.c('recording', {state: state, token: token});
+
+ connection.sendIQ(elem,
+ function (result) {
+ console.log('Set recording "', state, '". Result:', result);
+ var recordingElem = $(result).find('>conference>recording');
+ var newState = ('true' === recordingElem.attr('state'));
+
+ recordingEnabled = newState;
+ callback(newState);
+ },
+ function (error) {
+ console.warn(error);
+ callback(recordingEnabled);
+ }
+ );
+}
+
+var Recording = {
+ toggleRecording: function (tokenEmptyCallback,
+ startingCallback, startedCallback) {
+ if (!Moderator.isModerator()) {
+ console.log(
+ 'non-focus, or conference not yet organized:' +
+ ' not enabling recording');
+ return;
+ }
+
+ // Jirecon does not (currently) support a token.
+ if (!recordingToken && !useJirecon) {
+ tokenEmptyCallback(function (value) {
+ setRecordingToken(value);
+ this.toggleRecording();
+ });
+
+ return;
+ }
+
+ var oldState = recordingEnabled;
+ startingCallback(!oldState);
+ setRecording(!oldState,
+ recordingToken,
+ function (state) {
+ console.log("New recording state: ", state);
+ if (state === oldState) {
+ // FIXME: new focus:
+ // this will not work when moderator changes
+ // during active session. Then it will assume that
+ // recording status has changed to true, but it might have
+ // been already true(and we only received actual status from
+ // the focus).
+ //
+ // SO we start with status null, so that it is initialized
+ // here and will fail only after second click, so if invalid
+ // token was used we have to press the button twice before
+ // current status will be fetched and token will be reset.
+ //
+ // Reliable way would be to return authentication error.
+ // Or status update when moderator connects.
+ // Or we have to stop recording session when current
+ // moderator leaves the room.
+
+ // Failed to change, reset the token because it might
+ // have been wrong
+ setRecordingToken(null);
+ }
+ startedCallback(state);
+
+ }
+ );
+ }
+
+}
+
+module.exports = Recording;
\ No newline at end of file
diff --git a/modules/xmpp/strophe.emuc.js b/modules/xmpp/strophe.emuc.js
new file mode 100644
index 000000000..63bf14713
--- /dev/null
+++ b/modules/xmpp/strophe.emuc.js
@@ -0,0 +1,607 @@
+/* jshint -W117 */
+/* a simple MUC connection plugin
+ * can only handle a single MUC room
+ */
+
+var bridgeIsDown = false;
+
+var Moderator = require("./moderator");
+
+module.exports = function(XMPP, eventEmitter) {
+ Strophe.addConnectionPlugin('emuc', {
+ connection: null,
+ roomjid: null,
+ myroomjid: null,
+ members: {},
+ list_members: [], // so we can elect a new focus
+ presMap: {},
+ preziMap: {},
+ joined: false,
+ isOwner: false,
+ role: null,
+ init: function (conn) {
+ this.connection = conn;
+ },
+ initPresenceMap: function (myroomjid) {
+ this.presMap['to'] = myroomjid;
+ this.presMap['xns'] = 'http://jabber.org/protocol/muc';
+ },
+ doJoin: function (jid, password) {
+ this.myroomjid = jid;
+
+ console.info("Joined MUC as " + this.myroomjid);
+
+ this.initPresenceMap(this.myroomjid);
+
+ if (!this.roomjid) {
+ this.roomjid = Strophe.getBareJidFromJid(jid);
+ // add handlers (just once)
+ this.connection.addHandler(this.onPresence.bind(this), null, 'presence', null, null, this.roomjid, {matchBare: true});
+ this.connection.addHandler(this.onPresenceUnavailable.bind(this), null, 'presence', 'unavailable', null, this.roomjid, {matchBare: true});
+ this.connection.addHandler(this.onPresenceError.bind(this), null, 'presence', 'error', null, this.roomjid, {matchBare: true});
+ this.connection.addHandler(this.onMessage.bind(this), null, 'message', null, null, this.roomjid, {matchBare: true});
+ }
+ if (password !== undefined) {
+ this.presMap['password'] = password;
+ }
+ this.sendPresence();
+ },
+ doLeave: function () {
+ console.log("do leave", this.myroomjid);
+ var pres = $pres({to: this.myroomjid, type: 'unavailable' });
+ this.presMap.length = 0;
+ this.connection.send(pres);
+ },
+ createNonAnonymousRoom: function () {
+ // http://xmpp.org/extensions/xep-0045.html#createroom-reserved
+
+ var getForm = $iq({type: 'get', to: this.roomjid})
+ .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'})
+ .c('x', {xmlns: 'jabber:x:data', type: 'submit'});
+
+ this.connection.sendIQ(getForm, function (form) {
+
+ if (!$(form).find(
+ '>query>x[xmlns="jabber:x:data"]' +
+ '>field[var="muc#roomconfig_whois"]').length) {
+
+ console.error('non-anonymous rooms not supported');
+ return;
+ }
+
+ var formSubmit = $iq({to: this.roomjid, type: 'set'})
+ .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});
+
+ formSubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});
+
+ formSubmit.c('field', {'var': 'FORM_TYPE'})
+ .c('value')
+ .t('http://jabber.org/protocol/muc#roomconfig').up().up();
+
+ formSubmit.c('field', {'var': 'muc#roomconfig_whois'})
+ .c('value').t('anyone').up().up();
+
+ this.connection.sendIQ(formSubmit);
+
+ }, function (error) {
+ console.error("Error getting room configuration form");
+ });
+ },
+ onPresence: function (pres) {
+ var from = pres.getAttribute('from');
+
+ // What is this for? A workaround for something?
+ if (pres.getAttribute('type')) {
+ return true;
+ }
+
+ // Parse etherpad tag.
+ var etherpad = $(pres).find('>etherpad');
+ if (etherpad.length) {
+ if (config.etherpad_base && !Moderator.isModerator()) {
+ UI.initEtherpad(etherpad.text());
+ }
+ }
+
+ // Parse prezi tag.
+ var presentation = $(pres).find('>prezi');
+ if (presentation.length) {
+ var url = presentation.attr('url');
+ var current = presentation.find('>current').text();
+
+ console.log('presentation info received from', from, url);
+
+ if (this.preziMap[from] == null) {
+ this.preziMap[from] = url;
+
+ $(document).trigger('presentationadded.muc', [from, url, current]);
+ }
+ else {
+ $(document).trigger('gotoslide.muc', [from, url, current]);
+ }
+ }
+ else if (this.preziMap[from] != null) {
+ var url = this.preziMap[from];
+ delete this.preziMap[from];
+ $(document).trigger('presentationremoved.muc', [from, url]);
+ }
+
+ // Parse audio info tag.
+ var audioMuted = $(pres).find('>audiomuted');
+ if (audioMuted.length) {
+ $(document).trigger('audiomuted.muc', [from, audioMuted.text()]);
+ }
+
+ // Parse video info tag.
+ var videoMuted = $(pres).find('>videomuted');
+ if (videoMuted.length) {
+ $(document).trigger('videomuted.muc', [from, videoMuted.text()]);
+ }
+
+ var stats = $(pres).find('>stats');
+ if (stats.length) {
+ var statsObj = {};
+ Strophe.forEachChild(stats[0], "stat", function (el) {
+ statsObj[el.getAttribute("name")] = el.getAttribute("value");
+ });
+ connectionquality.updateRemoteStats(from, statsObj);
+ }
+
+ // Parse status.
+ if ($(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="201"]').length) {
+ this.isOwner = true;
+ this.createNonAnonymousRoom();
+ }
+
+ // Parse roles.
+ var member = {};
+ member.show = $(pres).find('>show').text();
+ member.status = $(pres).find('>status').text();
+ var tmp = $(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>item');
+ member.affiliation = tmp.attr('affiliation');
+ member.role = tmp.attr('role');
+
+ // Focus recognition
+ member.jid = tmp.attr('jid');
+ member.isFocus = false;
+ if (member.jid
+ && member.jid.indexOf(Moderator.getFocusUserJid() + "/") == 0) {
+ member.isFocus = true;
+ }
+
+ var nicktag = $(pres).find('>nick[xmlns="http://jabber.org/protocol/nick"]');
+ member.displayName = (nicktag.length > 0 ? nicktag.html() : null);
+
+ if (from == this.myroomjid) {
+ if (member.affiliation == 'owner') this.isOwner = true;
+ if (this.role !== member.role) {
+ this.role = member.role;
+ if (Moderator.onLocalRoleChange)
+ Moderator.onLocalRoleChange(from, member, pres);
+ UI.onLocalRoleChange(from, member, pres);
+ }
+ if (!this.joined) {
+ this.joined = true;
+ eventEmitter.emit(XMPPEvents.MUC_JOINED, from, member);
+ this.list_members.push(from);
+ }
+ } else if (this.members[from] === undefined) {
+ // new participant
+ this.members[from] = member;
+ this.list_members.push(from);
+ console.log('entered', from, member);
+ if (member.isFocus) {
+ focusMucJid = from;
+ console.info("Ignore focus: " + from + ", real JID: " + member.jid);
+ }
+ else {
+ var id = $(pres).find('>userID').text();
+ var email = $(pres).find('>email');
+ if (email.length > 0) {
+ id = email.text();
+ }
+ UI.onMucEntered(from, id, member.displayName);
+ API.triggerEvent("participantJoined", {jid: from});
+ }
+ } else {
+ // Presence update for existing participant
+ // Watch role change:
+ if (this.members[from].role != member.role) {
+ this.members[from].role = member.role;
+ UI.onMucRoleChanged(member.role, member.displayName);
+ }
+ }
+
+ // Always trigger presence to update bindings
+ $(document).trigger('presence.muc', [from, member, pres]);
+ this.parsePresence(from, member, pres);
+
+ // Trigger status message update
+ if (member.status) {
+ UI.onMucPresenceStatus(from, member);
+ }
+
+ return true;
+ },
+ onPresenceUnavailable: function (pres) {
+ var from = pres.getAttribute('from');
+ // Status code 110 indicates that this notification is "self-presence".
+ if (!$(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="110"]').length) {
+ delete this.members[from];
+ this.list_members.splice(this.list_members.indexOf(from), 1);
+ this.onParticipantLeft(from);
+ }
+ // If the status code is 110 this means we're leaving and we would like
+ // to remove everyone else from our view, so we trigger the event.
+ else if (this.list_members.length > 1) {
+ for (var i = 0; i < this.list_members.length; i++) {
+ var member = this.list_members[i];
+ delete this.members[i];
+ this.list_members.splice(i, 1);
+ this.onParticipantLeft(member);
+ }
+ }
+ if ($(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="307"]').length) {
+ $(document).trigger('kicked.muc', [from]);
+ if (this.myroomjid === from) {
+ XMPP.disposeConference(false);
+ eventEmitter.emit(XMPPEvents.KICKED);
+ }
+ }
+ return true;
+ },
+ onPresenceError: function (pres) {
+ var from = pres.getAttribute('from');
+ if ($(pres).find('>error[type="auth"]>not-authorized[xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"]').length) {
+ console.log('on password required', from);
+ var self = this;
+ UI.onPasswordReqiured(function (value) {
+ self.doJoin(from, value);
+ });
+ } else if ($(pres).find(
+ '>error[type="cancel"]>not-allowed[xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"]').length) {
+ var toDomain = Strophe.getDomainFromJid(pres.getAttribute('to'));
+ if (toDomain === config.hosts.anonymousdomain) {
+ // we are connected with anonymous domain and only non anonymous users can create rooms
+ // we must authorize the user
+ XMPP.promptLogin();
+ } else {
+ console.warn('onPresError ', pres);
+ UI.messageHandler.openReportDialog(null,
+ 'Oops! Something went wrong and we couldn`t connect to the conference.',
+ pres);
+ }
+ } else {
+ console.warn('onPresError ', pres);
+ UI.messageHandler.openReportDialog(null,
+ 'Oops! Something went wrong and we couldn`t connect to the conference.',
+ pres);
+ }
+ return true;
+ },
+ sendMessage: function (body, nickname) {
+ var msg = $msg({to: this.roomjid, type: 'groupchat'});
+ msg.c('body', body).up();
+ if (nickname) {
+ msg.c('nick', {xmlns: 'http://jabber.org/protocol/nick'}).t(nickname).up().up();
+ }
+ this.connection.send(msg);
+ API.triggerEvent("outgoingMessage", {"message": body});
+ },
+ setSubject: function (subject) {
+ var msg = $msg({to: this.roomjid, type: 'groupchat'});
+ msg.c('subject', subject);
+ this.connection.send(msg);
+ console.log("topic changed to " + subject);
+ },
+ onMessage: function (msg) {
+ // FIXME: this is a hack. but jingle on muc makes nickchanges hard
+ var from = msg.getAttribute('from');
+ var nick = $(msg).find('>nick[xmlns="http://jabber.org/protocol/nick"]').text() || Strophe.getResourceFromJid(from);
+
+ var txt = $(msg).find('>body').text();
+ var type = msg.getAttribute("type");
+ if (type == "error") {
+ UI.chatAddError($(msg).find('>text').text(), txt);
+ return true;
+ }
+
+ var subject = $(msg).find('>subject');
+ if (subject.length) {
+ var subjectText = subject.text();
+ if (subjectText || subjectText == "") {
+ UI.chatSetSubject(subjectText);
+ console.log("Subject is changed to " + subjectText);
+ }
+ }
+
+
+ if (txt) {
+ console.log('chat', nick, txt);
+ UI.updateChatConversation(from, nick, txt);
+ if (from != this.myroomjid)
+ API.triggerEvent("incomingMessage",
+ {"from": from, "nick": nick, "message": txt});
+ }
+ return true;
+ },
+ lockRoom: function (key, onSuccess, onError, onNotSupported) {
+ //http://xmpp.org/extensions/xep-0045.html#roomconfig
+ var ob = this;
+ this.connection.sendIQ($iq({to: this.roomjid, type: 'get'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'}),
+ function (res) {
+ if ($(res).find('>query>x[xmlns="jabber:x:data"]>field[var="muc#roomconfig_roomsecret"]').length) {
+ var formsubmit = $iq({to: ob.roomjid, type: 'set'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});
+ formsubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});
+ formsubmit.c('field', {'var': 'FORM_TYPE'}).c('value').t('http://jabber.org/protocol/muc#roomconfig').up().up();
+ formsubmit.c('field', {'var': 'muc#roomconfig_roomsecret'}).c('value').t(key).up().up();
+ // Fixes a bug in prosody 0.9.+ https://code.google.com/p/lxmppd/issues/detail?id=373
+ formsubmit.c('field', {'var': 'muc#roomconfig_whois'}).c('value').t('anyone').up().up();
+ // FIXME: is muc#roomconfig_passwordprotectedroom required?
+ this.connection.sendIQ(formsubmit,
+ onSuccess,
+ onError);
+ } else {
+ onNotSupported();
+ }
+ }, onError);
+ },
+ kick: function (jid) {
+ var kickIQ = $iq({to: this.roomjid, type: 'set'})
+ .c('query', {xmlns: 'http://jabber.org/protocol/muc#admin'})
+ .c('item', {nick: Strophe.getResourceFromJid(jid), role: 'none'})
+ .c('reason').t('You have been kicked.').up().up().up();
+
+ this.connection.sendIQ(
+ kickIQ,
+ function (result) {
+ console.log('Kick participant with jid: ', jid, result);
+ },
+ function (error) {
+ console.log('Kick participant error: ', error);
+ });
+ },
+ sendPresence: function () {
+ var pres = $pres({to: this.presMap['to'] });
+ pres.c('x', {xmlns: this.presMap['xns']});
+
+ if (this.presMap['password']) {
+ pres.c('password').t(this.presMap['password']).up();
+ }
+
+ pres.up();
+
+ // Send XEP-0115 'c' stanza that contains our capabilities info
+ if (this.connection.caps) {
+ this.connection.caps.node = config.clientNode;
+ pres.c('c', this.connection.caps.generateCapsAttrs()).up();
+ }
+
+ pres.c('user-agent', {xmlns: 'http://jitsi.org/jitmeet/user-agent'})
+ .t(navigator.userAgent).up();
+
+ if (this.presMap['bridgeIsDown']) {
+ pres.c('bridgeIsDown').up();
+ }
+
+ if (this.presMap['email']) {
+ pres.c('email').t(this.presMap['email']).up();
+ }
+
+ if (this.presMap['userId']) {
+ pres.c('userId').t(this.presMap['userId']).up();
+ }
+
+ if (this.presMap['displayName']) {
+ // XEP-0172
+ pres.c('nick', {xmlns: 'http://jabber.org/protocol/nick'})
+ .t(this.presMap['displayName']).up();
+ }
+
+ if (this.presMap['audions']) {
+ pres.c('audiomuted', {xmlns: this.presMap['audions']})
+ .t(this.presMap['audiomuted']).up();
+ }
+
+ if (this.presMap['videons']) {
+ pres.c('videomuted', {xmlns: this.presMap['videons']})
+ .t(this.presMap['videomuted']).up();
+ }
+
+ if (this.presMap['statsns']) {
+ var stats = pres.c('stats', {xmlns: this.presMap['statsns']});
+ for (var stat in this.presMap["stats"])
+ if (this.presMap["stats"][stat] != null)
+ stats.c("stat", {name: stat, value: this.presMap["stats"][stat]}).up();
+ pres.up();
+ }
+
+ if (this.presMap['prezins']) {
+ pres.c('prezi',
+ {xmlns: this.presMap['prezins'],
+ 'url': this.presMap['preziurl']})
+ .c('current').t(this.presMap['prezicurrent']).up().up();
+ }
+
+ if (this.presMap['etherpadns']) {
+ pres.c('etherpad', {xmlns: this.presMap['etherpadns']})
+ .t(this.presMap['etherpadname']).up();
+ }
+
+ if (this.presMap['medians']) {
+ pres.c('media', {xmlns: this.presMap['medians']});
+ var sourceNumber = 0;
+ Object.keys(this.presMap).forEach(function (key) {
+ if (key.indexOf('source') >= 0) {
+ sourceNumber++;
+ }
+ });
+ if (sourceNumber > 0)
+ for (var i = 1; i <= sourceNumber / 3; i++) {
+ pres.c('source',
+ {type: this.presMap['source' + i + '_type'],
+ ssrc: this.presMap['source' + i + '_ssrc'],
+ direction: this.presMap['source' + i + '_direction']
+ || 'sendrecv' }
+ ).up();
+ }
+ }
+
+ pres.up();
+// console.debug(pres.toString());
+ this.connection.send(pres);
+ },
+ addDisplayNameToPresence: function (displayName) {
+ this.presMap['displayName'] = displayName;
+ },
+ addMediaToPresence: function (sourceNumber, mtype, ssrcs, direction) {
+ if (!this.presMap['medians'])
+ this.presMap['medians'] = 'http://estos.de/ns/mjs';
+
+ this.presMap['source' + sourceNumber + '_type'] = mtype;
+ this.presMap['source' + sourceNumber + '_ssrc'] = ssrcs;
+ this.presMap['source' + sourceNumber + '_direction'] = direction;
+ },
+ clearPresenceMedia: function () {
+ var self = this;
+ Object.keys(this.presMap).forEach(function (key) {
+ if (key.indexOf('source') != -1) {
+ delete self.presMap[key];
+ }
+ });
+ },
+ addPreziToPresence: function (url, currentSlide) {
+ this.presMap['prezins'] = 'http://jitsi.org/jitmeet/prezi';
+ this.presMap['preziurl'] = url;
+ this.presMap['prezicurrent'] = currentSlide;
+ },
+ removePreziFromPresence: function () {
+ delete this.presMap['prezins'];
+ delete this.presMap['preziurl'];
+ delete this.presMap['prezicurrent'];
+ },
+ addCurrentSlideToPresence: function (currentSlide) {
+ this.presMap['prezicurrent'] = currentSlide;
+ },
+ getPrezi: function (roomjid) {
+ return this.preziMap[roomjid];
+ },
+ addEtherpadToPresence: function (etherpadName) {
+ this.presMap['etherpadns'] = 'http://jitsi.org/jitmeet/etherpad';
+ this.presMap['etherpadname'] = etherpadName;
+ },
+ addAudioInfoToPresence: function (isMuted) {
+ this.presMap['audions'] = 'http://jitsi.org/jitmeet/audio';
+ this.presMap['audiomuted'] = isMuted.toString();
+ },
+ addVideoInfoToPresence: function (isMuted) {
+ this.presMap['videons'] = 'http://jitsi.org/jitmeet/video';
+ this.presMap['videomuted'] = isMuted.toString();
+ },
+ addConnectionInfoToPresence: function (stats) {
+ this.presMap['statsns'] = 'http://jitsi.org/jitmeet/stats';
+ this.presMap['stats'] = stats;
+ },
+ findJidFromResource: function (resourceJid) {
+ if (resourceJid &&
+ resourceJid === Strophe.getResourceFromJid(this.myroomjid)) {
+ return this.myroomjid;
+ }
+ var peerJid = null;
+ Object.keys(this.members).some(function (jid) {
+ peerJid = jid;
+ return Strophe.getResourceFromJid(jid) === resourceJid;
+ });
+ return peerJid;
+ },
+ addBridgeIsDownToPresence: function () {
+ this.presMap['bridgeIsDown'] = true;
+ },
+ addEmailToPresence: function (email) {
+ this.presMap['email'] = email;
+ },
+ addUserIdToPresence: function (userId) {
+ this.presMap['userId'] = userId;
+ },
+ isModerator: function () {
+ return this.role === 'moderator';
+ },
+ getMemberRole: function (peerJid) {
+ if (this.members[peerJid]) {
+ return this.members[peerJid].role;
+ }
+ return null;
+ },
+ onParticipantLeft: function (jid) {
+ UI.onMucLeft(jid);
+
+ API.triggerEvent("participantLeft", {jid: jid});
+
+ delete jid2Ssrc[jid];
+
+ this.connection.jingle.terminateByJid(jid);
+
+ if (this.getPrezi(jid)) {
+ $(document).trigger('presentationremoved.muc',
+ [jid, this.getPrezi(jid)]);
+ }
+
+ Moderator.onMucLeft(jid);
+ },
+ parsePresence: function (from, memeber, pres) {
+ if($(pres).find(">bridgeIsDown").length > 0 && !bridgeIsDown) {
+ bridgeIsDown = true;
+ eventEmitter.emit(XMPPEvents.BRIDGE_DOWN);
+ }
+
+ if(memeber.isFocus)
+ return;
+
+ // Remove old ssrcs coming from the jid
+ Object.keys(ssrc2jid).forEach(function (ssrc) {
+ if (ssrc2jid[ssrc] == jid) {
+ delete ssrc2jid[ssrc];
+ delete ssrc2videoType[ssrc];
+ }
+ });
+
+ var changedStreams = [];
+ $(pres).find('>media[xmlns="http://estos.de/ns/mjs"]>source').each(function (idx, ssrc) {
+ //console.log(jid, 'assoc ssrc', ssrc.getAttribute('type'), ssrc.getAttribute('ssrc'));
+ var ssrcV = ssrc.getAttribute('ssrc');
+ ssrc2jid[ssrcV] = from;
+ notReceivedSSRCs.push(ssrcV);
+
+ var type = ssrc.getAttribute('type');
+ ssrc2videoType[ssrcV] = type;
+
+ var direction = ssrc.getAttribute('direction');
+
+ changedStreams.push({type: type, direction: direction});
+
+ });
+
+ eventEmitter.emit(XMPPEvents.CHANGED_STREAMS, from, changedStreams);
+
+ var displayName = !config.displayJids
+ ? memeber.displayName : Strophe.getResourceFromJid(from);
+
+ if (displayName && displayName.length > 0)
+ {
+// $(document).trigger('displaynamechanged',
+// [jid, displayName]);
+ eventEmitter.emit(XMPPEvents.DISPLAY_NAME_CHANGED, from, displayName);
+ }
+
+
+ var id = $(pres).find('>userID').text();
+ var email = $(pres).find('>email');
+ if(email.length > 0) {
+ id = email.text();
+ }
+
+ eventEmitter.emit(XMPPEvents.USER_ID_CHANGED, from, id);
+ }
+ });
+};
+
diff --git a/modules/xmpp/strophe.jingle.js b/modules/xmpp/strophe.jingle.js
new file mode 100644
index 000000000..4878d587c
--- /dev/null
+++ b/modules/xmpp/strophe.jingle.js
@@ -0,0 +1,334 @@
+/* jshint -W117 */
+
+var JingleSession = require("./JingleSession");
+
+function CallIncomingJingle(sid, connection) {
+ var sess = connection.jingle.sessions[sid];
+
+ // TODO: do we check activecall == null?
+ activecall = sess;
+
+ statistics.onConferenceCreated(sess);
+ RTC.onConferenceCreated(sess);
+
+ // TODO: check affiliation and/or role
+ console.log('emuc data for', sess.peerjid, connection.emuc.members[sess.peerjid]);
+ sess.usedrip = true; // not-so-naive trickle ice
+ sess.sendAnswer();
+ sess.accept();
+
+};
+
+module.exports = function(XMPP)
+{
+ Strophe.addConnectionPlugin('jingle', {
+ connection: null,
+ sessions: {},
+ jid2session: {},
+ ice_config: {iceServers: []},
+ pc_constraints: {},
+ media_constraints: {
+ mandatory: {
+ 'OfferToReceiveAudio': true,
+ 'OfferToReceiveVideo': true
+ }
+ // MozDontOfferDataChannel: true when this is firefox
+ },
+ init: function (conn) {
+ this.connection = conn;
+ if (this.connection.disco) {
+ // http://xmpp.org/extensions/xep-0167.html#support
+ // http://xmpp.org/extensions/xep-0176.html#support
+ this.connection.disco.addFeature('urn:xmpp:jingle:1');
+ this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:1');
+ this.connection.disco.addFeature('urn:xmpp:jingle:transports:ice-udp:1');
+ this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:audio');
+ this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:video');
+
+
+ // this is dealt with by SDP O/A so we don't need to annouce this
+ //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtcp-fb:0'); // XEP-0293
+ //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtp-hdrext:0'); // XEP-0294
+ if (config.useRtcpMux) {
+ this.connection.disco.addFeature('urn:ietf:rfc:5761'); // rtcp-mux
+ }
+ if (config.useBundle) {
+ this.connection.disco.addFeature('urn:ietf:rfc:5888'); // a=group, e.g. bundle
+ }
+ //this.connection.disco.addFeature('urn:ietf:rfc:5576'); // a=ssrc
+ }
+ this.connection.addHandler(this.onJingle.bind(this), 'urn:xmpp:jingle:1', 'iq', 'set', null, null);
+ },
+ 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: fromJid,
+ id: iq.getAttribute('id')
+ });
+ console.log('on jingle ' + action + ' from ' + fromJid, iq);
+ var sess = this.sessions[sid];
+ if ('session-initiate' != action) {
+ if (sess === null) {
+ ack.type = 'error';
+ ack.c('error', {type: 'cancel'})
+ .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()
+ .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
+ this.connection.send(ack);
+ return true;
+ }
+ // compare from to sess.peerjid (bare jid comparison for later compat with message-mode)
+ // local jid is not checked
+ 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()
+ .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
+ this.connection.send(ack);
+ return true;
+ }
+ } else if (sess !== undefined) {
+ // existing session with same session id
+ // this might be out-of-order if the sess.peerjid is the same as from
+ ack.type = 'error';
+ ack.c('error', {type: 'cancel'})
+ .c('service-unavailable', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up();
+ console.warn('duplicate session id', sid);
+ this.connection.send(ack);
+ return true;
+ }
+ // FIXME: check for a defined action
+ this.connection.send(ack);
+ // see http://xmpp.org/extensions/xep-0166.html#concepts-session
+ switch (action) {
+ case 'session-initiate':
+ sess = new JingleSession(
+ $(iq).attr('to'), $(iq).find('jingle').attr('sid'),
+ this.connection, XMPP);
+ // configure session
+
+ sess.media_constraints = this.media_constraints;
+ sess.pc_constraints = this.pc_constraints;
+ sess.ice_config = this.ice_config;
+
+ sess.initiate(fromJid, false);
+ // FIXME: setRemoteDescription should only be done when this call is to be accepted
+ sess.setRemoteDescription($(iq).find('>jingle'), 'offer');
+
+ this.sessions[sess.sid] = sess;
+ this.jid2session[sess.peerjid] = sess;
+
+ // the callback should either
+ // .sendAnswer and .accept
+ // or .sendTerminate -- not necessarily synchronus
+ CallIncomingJingle(sess.sid, this.connection);
+ break;
+ case 'session-accept':
+ sess.setRemoteDescription($(iq).find('>jingle'), 'answer');
+ sess.accept();
+ $(document).trigger('callaccepted.jingle', [sess.sid]);
+ break;
+ case 'session-terminate':
+ // If this is not the focus sending the terminate, we have
+ // nothing more to do here.
+ if (Object.keys(this.sessions).length < 1
+ || !(this.sessions[Object.keys(this.sessions)[0]]
+ instanceof JingleSession))
+ {
+ break;
+ }
+ console.log('terminating...', sess.sid);
+ sess.terminate();
+ this.terminate(sess.sid);
+ if ($(iq).find('>jingle>reason').length) {
+ $(document).trigger('callterminated.jingle', [
+ sess.sid,
+ sess.peerjid,
+ $(iq).find('>jingle>reason>:first')[0].tagName,
+ $(iq).find('>jingle>reason>text').text()
+ ]);
+ } else {
+ $(document).trigger('callterminated.jingle',
+ [sess.sid, sess.peerjid]);
+ }
+ break;
+ case 'transport-info':
+ sess.addIceCandidate($(iq).find('>jingle>content'));
+ break;
+ case 'session-info':
+ var affected;
+ if ($(iq).find('>jingle>ringing[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
+ $(document).trigger('ringing.jingle', [sess.sid]);
+ } else if ($(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
+ affected = $(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
+ $(document).trigger('mute.jingle', [sess.sid, affected]);
+ } else if ($(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
+ affected = $(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
+ $(document).trigger('unmute.jingle', [sess.sid, affected]);
+ }
+ break;
+ case 'addsource': // FIXME: proprietary, un-jingleish
+ case 'source-add': // FIXME: proprietary
+ sess.addSource($(iq).find('>jingle>content'), fromJid);
+ break;
+ case 'removesource': // FIXME: proprietary, un-jingleish
+ case 'source-remove': // FIXME: proprietary
+ sess.removeSource($(iq).find('>jingle>content'), fromJid);
+ break;
+ default:
+ console.warn('jingle action not implemented', action);
+ break;
+ }
+ return true;
+ },
+ initiate: function (peerjid, myjid) { // initiate a new jinglesession to peerjid
+ var sess = new JingleSession(myjid || this.connection.jid,
+ Math.random().toString(36).substr(2, 12), // random string
+ this.connection, XMPP);
+ // configure session
+
+ sess.media_constraints = this.media_constraints;
+ sess.pc_constraints = this.pc_constraints;
+ sess.ice_config = this.ice_config;
+
+ sess.initiate(peerjid, true);
+ this.sessions[sess.sid] = sess;
+ this.jid2session[sess.peerjid] = sess;
+ sess.sendOffer();
+ return sess;
+ },
+ terminate: function (sid, reason, text) { // terminate by sessionid (or all sessions)
+ if (sid === null || sid === undefined) {
+ for (sid in this.sessions) {
+ if (this.sessions[sid].state != 'ended') {
+ this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
+ this.sessions[sid].terminate();
+ }
+ delete this.jid2session[this.sessions[sid].peerjid];
+ delete this.sessions[sid];
+ }
+ } else if (this.sessions.hasOwnProperty(sid)) {
+ if (this.sessions[sid].state != 'ended') {
+ this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
+ this.sessions[sid].terminate();
+ }
+ delete this.jid2session[this.sessions[sid].peerjid];
+ delete this.sessions[sid];
+ }
+ },
+ // Used to terminate a session when an unavailable presence is received.
+ terminateByJid: function (jid) {
+ if (this.jid2session.hasOwnProperty(jid)) {
+ var sess = this.jid2session[jid];
+ if (sess) {
+ sess.terminate();
+ console.log('peer went away silently', jid);
+ delete this.sessions[sess.sid];
+ delete this.jid2session[jid];
+ $(document).trigger('callterminated.jingle',
+ [sess.sid, jid], 'gone');
+ }
+ }
+ },
+ terminateRemoteByJid: function (jid, reason) {
+ if (this.jid2session.hasOwnProperty(jid)) {
+ var sess = this.jid2session[jid];
+ if (sess) {
+ sess.sendTerminate(reason || (!sess.active()) ? 'kick' : null);
+ sess.terminate();
+ console.log('terminate peer with jid', sess.sid, jid);
+ delete this.sessions[sess.sid];
+ delete this.jid2session[jid];
+ $(document).trigger('callterminated.jingle',
+ [sess.sid, jid, 'kicked']);
+ }
+ }
+ },
+ getStunAndTurnCredentials: function () {
+ // get stun and turn configuration from server via xep-0215
+ // uses time-limited credentials as described in
+ // http://tools.ietf.org/html/draft-uberti-behave-turn-rest-00
+ //
+ // see https://code.google.com/p/prosody-modules/source/browse/mod_turncredentials/mod_turncredentials.lua
+ // for a prosody module which implements this
+ //
+ // currently, this doesn't work with updateIce and therefore credentials with a long
+ // validity have to be fetched before creating the peerconnection
+ // TODO: implement refresh via updateIce as described in
+ // https://code.google.com/p/webrtc/issues/detail?id=1650
+ var self = this;
+ this.connection.sendIQ(
+ $iq({type: 'get', to: this.connection.domain})
+ .c('services', {xmlns: 'urn:xmpp:extdisco:1'}).c('service', {host: 'turn.' + this.connection.domain}),
+ function (res) {
+ var iceservers = [];
+ $(res).find('>services>service').each(function (idx, el) {
+ el = $(el);
+ var dict = {};
+ var type = el.attr('type');
+ switch (type) {
+ case 'stun':
+ dict.url = 'stun:' + el.attr('host');
+ if (el.attr('port')) {
+ dict.url += ':' + el.attr('port');
+ }
+ iceservers.push(dict);
+ break;
+ case 'turn':
+ case 'turns':
+ dict.url = type + ':';
+ if (el.attr('username')) { // https://code.google.com/p/webrtc/issues/detail?id=1508
+ if (navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./) && parseInt(navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./)[2], 10) < 28) {
+ dict.url += el.attr('username') + '@';
+ } else {
+ dict.username = el.attr('username'); // only works in M28
+ }
+ }
+ dict.url += el.attr('host');
+ if (el.attr('port') && el.attr('port') != '3478') {
+ dict.url += ':' + el.attr('port');
+ }
+ if (el.attr('transport') && el.attr('transport') != 'udp') {
+ dict.url += '?transport=' + el.attr('transport');
+ }
+ if (el.attr('password')) {
+ dict.credential = el.attr('password');
+ }
+ iceservers.push(dict);
+ break;
+ }
+ });
+ self.ice_config.iceServers = iceservers;
+ },
+ function (err) {
+ console.warn('getting turn credentials failed', err);
+ console.warn('is mod_turncredentials or similar installed?');
+ }
+ );
+ // implement push?
+ },
+
+ /**
+ * Populates the log data
+ */
+ populateData: function () {
+ var data = {};
+ Object.keys(this.sessions).forEach(function (sid) {
+ var session = this.sessions[sid];
+ if (session.peerconnection && session.peerconnection.updateLog) {
+ // FIXME: should probably be a .dump call
+ data["jingle_" + session.sid] = {
+ updateLog: session.peerconnection.updateLog,
+ stats: session.peerconnection.stats,
+ url: window.location.href
+ };
+ }
+ });
+ return data;
+ }
+ });
+};
+
diff --git a/modules/xmpp/strophe.logger.js b/modules/xmpp/strophe.logger.js
new file mode 100644
index 000000000..4866ff89b
--- /dev/null
+++ b/modules/xmpp/strophe.logger.js
@@ -0,0 +1,20 @@
+/* global Strophe */
+module.exports = function () {
+
+ Strophe.addConnectionPlugin('logger', {
+ // logs raw stanzas and makes them available for download as JSON
+ connection: null,
+ log: [],
+ init: function (conn) {
+ this.connection = conn;
+ this.connection.rawInput = this.log_incoming.bind(this);
+ this.connection.rawOutput = this.log_outgoing.bind(this);
+ },
+ log_incoming: function (stanza) {
+ this.log.push([new Date().getTime(), 'incoming', stanza]);
+ },
+ log_outgoing: function (stanza) {
+ this.log.push([new Date().getTime(), 'outgoing', stanza]);
+ }
+ });
+};
\ No newline at end of file
diff --git a/modules/xmpp/strophe.moderate.js b/modules/xmpp/strophe.moderate.js
new file mode 100644
index 000000000..64a8bccfa
--- /dev/null
+++ b/modules/xmpp/strophe.moderate.js
@@ -0,0 +1,58 @@
+/* global $, $iq, config, connection, focusMucJid, forceMuted,
+ setAudioMuted, Strophe */
+/**
+ * Moderate connection plugin.
+ */
+module.exports = function (XMPP) {
+ Strophe.addConnectionPlugin('moderate', {
+ connection: null,
+ init: function (conn) {
+ this.connection = conn;
+
+ this.connection.addHandler(this.onMute.bind(this),
+ 'http://jitsi.org/jitmeet/audio',
+ 'iq',
+ 'set',
+ null,
+ null);
+ },
+ setMute: function (jid, mute) {
+ console.info("set mute", mute);
+ var iqToFocus = $iq({to: focusMucJid, type: 'set'})
+ .c('mute', {
+ xmlns: 'http://jitsi.org/jitmeet/audio',
+ jid: jid
+ })
+ .t(mute.toString())
+ .up();
+
+ this.connection.sendIQ(
+ iqToFocus,
+ function (result) {
+ console.log('set mute', result);
+ },
+ function (error) {
+ console.log('set mute error', error);
+ });
+ },
+ onMute: function (iq) {
+ var from = iq.getAttribute('from');
+ if (from !== focusMucJid) {
+ console.warn("Ignored mute from non focus peer");
+ return false;
+ }
+ var mute = $(iq).find('mute');
+ if (mute.length) {
+ var doMuteAudio = mute.text() === "true";
+ UI.setAudioMuted(doMuteAudio);
+ XMPP.forceMuted = doMuteAudio;
+ }
+ return true;
+ },
+ eject: function (jid) {
+ // We're not the focus, so can't terminate
+ //connection.jingle.terminateRemoteByJid(jid, 'kick');
+ this.connection.emuc.kick(jid);
+ }
+ });
+}
\ No newline at end of file
diff --git a/modules/xmpp/strophe.rayo.js b/modules/xmpp/strophe.rayo.js
new file mode 100644
index 000000000..9d0db5547
--- /dev/null
+++ b/modules/xmpp/strophe.rayo.js
@@ -0,0 +1,95 @@
+/* jshint -W117 */
+module.exports = function() {
+ Strophe.addConnectionPlugin('rayo',
+ {
+ RAYO_XMLNS: 'urn:xmpp:rayo:1',
+ connection: null,
+ init: function (conn) {
+ this.connection = conn;
+ if (this.connection.disco) {
+ this.connection.disco.addFeature('urn:xmpp:rayo:client:1');
+ }
+
+ this.connection.addHandler(
+ this.onRayo.bind(this), this.RAYO_XMLNS, 'iq', 'set', null, null);
+ },
+ onRayo: function (iq) {
+ console.info("Rayo IQ", iq);
+ },
+ dial: function (to, from, roomName, roomPass) {
+ var self = this;
+ var req = $iq(
+ {
+ type: 'set',
+ to: focusMucJid
+ }
+ );
+ req.c('dial',
+ {
+ xmlns: this.RAYO_XMLNS,
+ to: to,
+ from: from
+ });
+ req.c('header',
+ {
+ name: 'JvbRoomName',
+ value: roomName
+ }).up();
+
+ if (roomPass !== null && roomPass.length) {
+
+ req.c('header',
+ {
+ name: 'JvbRoomPassword',
+ value: roomPass
+ }).up();
+ }
+
+ this.connection.sendIQ(
+ req,
+ function (result) {
+ console.info('Dial result ', result);
+
+ var resource = $(result).find('ref').attr('uri');
+ this.call_resource = resource.substr('xmpp:'.length);
+ console.info(
+ "Received call resource: " + this.call_resource);
+ },
+ function (error) {
+ console.info('Dial error ', error);
+ }
+ );
+ },
+ hang_up: function () {
+ if (!this.call_resource) {
+ console.warn("No call in progress");
+ return;
+ }
+
+ var self = this;
+ var req = $iq(
+ {
+ type: 'set',
+ to: this.call_resource
+ }
+ );
+ req.c('hangup',
+ {
+ xmlns: this.RAYO_XMLNS
+ });
+
+ this.connection.sendIQ(
+ req,
+ function (result) {
+ console.info('Hangup result ', result);
+ self.call_resource = null;
+ },
+ function (error) {
+ console.info('Hangup error ', error);
+ self.call_resource = null;
+ }
+ );
+ }
+ }
+ );
+};
diff --git a/modules/xmpp/strophe.util.js b/modules/xmpp/strophe.util.js
new file mode 100644
index 000000000..b7d834828
--- /dev/null
+++ b/modules/xmpp/strophe.util.js
@@ -0,0 +1,42 @@
+/**
+ * Strophe logger implementation. Logs from level WARN and above.
+ */
+module.exports = function () {
+
+ Strophe.log = function (level, msg) {
+ switch (level) {
+ case Strophe.LogLevel.WARN:
+ console.warn("Strophe: " + msg);
+ break;
+ case Strophe.LogLevel.ERROR:
+ case Strophe.LogLevel.FATAL:
+ console.error("Strophe: " + msg);
+ break;
+ }
+ };
+
+ Strophe.getStatusString = function (status) {
+ switch (status) {
+ case Strophe.Status.ERROR:
+ return "ERROR";
+ case Strophe.Status.CONNECTING:
+ return "CONNECTING";
+ case Strophe.Status.CONNFAIL:
+ return "CONNFAIL";
+ case Strophe.Status.AUTHENTICATING:
+ return "AUTHENTICATING";
+ case Strophe.Status.AUTHFAIL:
+ return "AUTHFAIL";
+ case Strophe.Status.CONNECTED:
+ return "CONNECTED";
+ case Strophe.Status.DISCONNECTED:
+ return "DISCONNECTED";
+ case Strophe.Status.DISCONNECTING:
+ return "DISCONNECTING";
+ case Strophe.Status.ATTACHED:
+ return "ATTACHED";
+ default:
+ return "unknown";
+ }
+ };
+};
diff --git a/modules/xmpp/xmpp.js b/modules/xmpp/xmpp.js
new file mode 100644
index 000000000..d7ad4d2dc
--- /dev/null
+++ b/modules/xmpp/xmpp.js
@@ -0,0 +1,422 @@
+var Moderator = require("./moderator");
+var EventEmitter = require("events");
+var Recording = require("./recording");
+var SDP = require("./SDP");
+
+var eventEmitter = new EventEmitter();
+var connection = null;
+var authenticatedUser = false;
+var activecall = null;
+
+function connect(jid, password, uiCredentials) {
+ var bosh
+ = uiCredentials.bosh || config.bosh || '/http-bind';
+ connection = new Strophe.Connection(bosh);
+ Moderator.setConnection(connection);
+
+ var settings = UI.getSettings();
+ var email = settings.email;
+ var displayName = settings.displayName;
+ if(email) {
+ connection.emuc.addEmailToPresence(email);
+ } else {
+ connection.emuc.addUserIdToPresence(settings.uid);
+ }
+ if(displayName) {
+ connection.emuc.addDisplayNameToPresence(displayName);
+ }
+
+ if (connection.disco) {
+ // for chrome, add multistream cap
+ }
+ connection.jingle.pc_constraints = RTC.getPCConstraints();
+ if (config.useIPv6) {
+ // https://code.google.com/p/webrtc/issues/detail?id=2828
+ if (!connection.jingle.pc_constraints.optional)
+ connection.jingle.pc_constraints.optional = [];
+ connection.jingle.pc_constraints.optional.push({googIPv6: true});
+ }
+
+ if(!password)
+ password = uiCredentials.password;
+
+ var anonymousConnectionFailed = false;
+ connection.connect(jid, password, function (status, msg) {
+ console.log('Strophe status changed to',
+ Strophe.getStatusString(status));
+ if (status === Strophe.Status.CONNECTED) {
+ if (config.useStunTurn) {
+ connection.jingle.getStunAndTurnCredentials();
+ }
+ UI.disableConnect();
+
+ console.info("My Jabber ID: " + connection.jid);
+
+ if(password)
+ authenticatedUser = true;
+ maybeDoJoin();
+ } else if (status === Strophe.Status.CONNFAIL) {
+ if(msg === 'x-strophe-bad-non-anon-jid') {
+ anonymousConnectionFailed = true;
+ }
+ } else if (status === Strophe.Status.DISCONNECTED) {
+ if(anonymousConnectionFailed) {
+ // prompt user for username and password
+ XMPP.promptLogin();
+ }
+ } else if (status === Strophe.Status.AUTHFAIL) {
+ // wrong password or username, prompt user
+ XMPP.promptLogin();
+
+ }
+ });
+}
+
+
+
+function maybeDoJoin() {
+ if (connection && connection.connected &&
+ Strophe.getResourceFromJid(connection.jid)
+ && (RTC.localAudio || RTC.localVideo)) {
+ // .connected is true while connecting?
+ doJoin();
+ }
+}
+
+function doJoin() {
+ var roomName = UI.generateRoomName();
+
+ Moderator.allocateConferenceFocus(
+ roomName, UI.checkForNicknameAndJoin);
+}
+
+function initStrophePlugins()
+{
+ require("./strophe.emuc")(XMPP, eventEmitter);
+ require("./strophe.jingle")();
+ require("./strophe.moderate")(XMPP);
+ require("./strophe.util")();
+ require("./strophe.rayo")();
+ require("./strophe.logger")();
+}
+
+function registerListeners() {
+ RTC.addStreamListener(maybeDoJoin,
+ StreamEventTypes.EVENT_TYPE_LOCAL_CREATED);
+}
+
+function setupEvents() {
+ $(window).bind('beforeunload', function () {
+ if (connection && connection.connected) {
+ // ensure signout
+ $.ajax({
+ type: 'POST',
+ url: config.bosh,
+ async: false,
+ cache: false,
+ contentType: 'application/xml',
+ data: "" +
+ "" +
+ "",
+ success: function (data) {
+ console.log('signed out');
+ console.log(data);
+ },
+ error: function (XMLHttpRequest, textStatus, errorThrown) {
+ console.log('signout error',
+ textStatus + ' (' + errorThrown + ')');
+ }
+ });
+ }
+ XMPP.disposeConference(true);
+ });
+}
+
+var XMPP = {
+ sessionTerminated: false,
+ /**
+ * Remembers if we were muted by the focus.
+ * @type {boolean}
+ */
+ forceMuted: false,
+ start: function (uiCredentials) {
+ setupEvents();
+ initStrophePlugins();
+ registerListeners();
+ Moderator.init();
+ var jid = uiCredentials.jid ||
+ config.hosts.anonymousdomain ||
+ config.hosts.domain ||
+ window.location.hostname;
+ connect(jid, null, uiCredentials);
+ },
+ promptLogin: function () {
+ UI.showLoginPopup(connect);
+ },
+ joinRooom: function(roomName, useNicks, nick)
+ {
+ var roomjid;
+ roomjid = roomName;
+
+ if (useNicks) {
+ if (nick) {
+ roomjid += '/' + nick;
+ } else {
+ roomjid += '/' + Strophe.getNodeFromJid(connection.jid);
+ }
+ } else {
+
+ var tmpJid = Strophe.getNodeFromJid(connection.jid);
+
+ if(!authenticatedUser)
+ tmpJid = tmpJid.substr(0, 8);
+
+ roomjid += '/' + tmpJid;
+ }
+ connection.emuc.doJoin(roomjid);
+ },
+ myJid: function () {
+ if(!connection)
+ return null;
+ return connection.emuc.myroomjid;
+ },
+ myResource: function () {
+ if(!connection || ! connection.emuc.myroomjid)
+ return null;
+ return Strophe.getResourceFromJid(connection.emuc.myroomjid);
+ },
+ disposeConference: function (onUnload) {
+ eventEmitter.emit(XMPPEvents.DISPOSE_CONFERENCE, onUnload);
+ var handler = activecall;
+ if (handler && handler.peerconnection) {
+ // FIXME: probably removing streams is not required and close() should
+ // be enough
+ if (RTC.localAudio) {
+ handler.peerconnection.removeStream(RTC.localAudio.getOriginalStream(), onUnload);
+ }
+ if (RTC.localVideo) {
+ handler.peerconnection.removeStream(RTC.localVideo.getOriginalStream(), onUnload);
+ }
+ handler.peerconnection.close();
+ }
+ activecall = null;
+ if(!onUnload)
+ {
+ this.sessionTerminated = true;
+ connection.emuc.doLeave();
+ }
+ },
+ addListener: function(type, listener)
+ {
+ eventEmitter.on(type, listener);
+ },
+ removeListener: function (type, listener) {
+ eventEmitter.removeListener(type, listener);
+ },
+ allocateConferenceFocus: function(roomName, callback) {
+ Moderator.allocateConferenceFocus(roomName, callback);
+ },
+ isModerator: function () {
+ return Moderator.isModerator();
+ },
+ isSipGatewayEnabled: function () {
+ return Moderator.isSipGatewayEnabled();
+ },
+ isExternalAuthEnabled: function () {
+ return Moderator.isExternalAuthEnabled();
+ },
+ switchStreams: function (stream, oldStream, callback) {
+ if (activecall) {
+ // FIXME: will block switchInProgress on true value in case of exception
+ activecall.switchStreams(stream, oldStream, callback);
+ } else {
+ // We are done immediately
+ console.error("No conference handler");
+ UI.messageHandler.showError('Error',
+ 'Unable to switch video stream.');
+ callback();
+ }
+ },
+ setVideoMute: function (mute, callback, options) {
+ if(activecall && connection && RTC.localVideo)
+ {
+ activecall.setVideoMute(mute, callback, options);
+ }
+ },
+ setAudioMute: function (mute, callback) {
+ if (!(connection && RTC.localAudio)) {
+ return false;
+ }
+
+
+ if (this.forceMuted && !mute) {
+ console.info("Asking focus for unmute");
+ connection.moderate.setMute(connection.emuc.myroomjid, mute);
+ // FIXME: wait for result before resetting muted status
+ this.forceMuted = false;
+ }
+
+ if (mute == RTC.localAudio.isMuted()) {
+ // Nothing to do
+ return true;
+ }
+
+ // It is not clear what is the right way to handle multiple tracks.
+ // So at least make sure that they are all muted or all unmuted and
+ // that we send presence just once.
+ RTC.localAudio.mute();
+ // isMuted is the opposite of audioEnabled
+ connection.emuc.addAudioInfoToPresence(mute);
+ connection.emuc.sendPresence();
+ callback();
+ return true;
+ },
+ // Really mute video, i.e. dont even send black frames
+ muteVideo: function (pc, unmute) {
+ // FIXME: this probably needs another of those lovely state safeguards...
+ // which checks for iceconn == connected and sigstate == stable
+ pc.setRemoteDescription(pc.remoteDescription,
+ function () {
+ pc.createAnswer(
+ function (answer) {
+ var sdp = new SDP(answer.sdp);
+ if (sdp.media.length > 1) {
+ if (unmute)
+ sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');
+ else
+ sdp.media[1] = sdp.media[1].replace('a=sendrecv', 'a=recvonly');
+ sdp.raw = sdp.session + sdp.media.join('');
+ answer.sdp = sdp.raw;
+ }
+ pc.setLocalDescription(answer,
+ function () {
+ console.log('mute SLD ok');
+ },
+ function (error) {
+ console.log('mute SLD error');
+ UI.messageHandler.showError('Error',
+ 'Oops! Something went wrong and we failed to ' +
+ 'mute! (SLD Failure)');
+ }
+ );
+ },
+ function (error) {
+ console.log(error);
+ UI.messageHandler.showError();
+ }
+ );
+ },
+ function (error) {
+ console.log('muteVideo SRD error');
+ UI.messageHandler.showError('Error',
+ 'Oops! Something went wrong and we failed to stop video!' +
+ '(SRD Failure)');
+
+ }
+ );
+ },
+ toggleRecording: function (tokenEmptyCallback,
+ startingCallback, startedCallback) {
+ Recording.toggleRecording(tokenEmptyCallback,
+ startingCallback, startedCallback);
+ },
+ addToPresence: function (name, value, dontSend) {
+ switch (name)
+ {
+ case "displayName":
+ connection.emuc.addDisplayNameToPresence(value);
+ break;
+ case "etherpad":
+ connection.emuc.addEtherpadToPresence(value);
+ break;
+ case "prezi":
+ connection.emuc.addPreziToPresence(value, 0);
+ break;
+ case "preziSlide":
+ connection.emuc.addCurrentSlideToPresence(value);
+ break;
+ case "connectionQuality":
+ connection.emuc.addConnectionInfoToPresence(value);
+ break;
+ case "email":
+ connection.emuc.addEmailToPresence(value);
+ default :
+ console.log("Unknown tag for presence.");
+ return;
+ }
+ if(!dontSend)
+ connection.emuc.sendPresence();
+ },
+ sendLogs: function (data) {
+ if(!focusMucJid)
+ return;
+
+ var deflate = true;
+
+ var content = JSON.stringify(dataYes);
+ if (deflate) {
+ content = String.fromCharCode.apply(null, Pako.deflateRaw(content));
+ }
+ content = Base64.encode(content);
+ // XEP-0337-ish
+ var message = $msg({to: focusMucJid, type: 'normal'});
+ message.c('log', { xmlns: 'urn:xmpp:eventlog',
+ id: 'PeerConnectionStats'});
+ message.c('message').t(content).up();
+ if (deflate) {
+ message.c('tag', {name: "deflated", value: "true"}).up();
+ }
+ message.up();
+
+ connection.send(message);
+ },
+ populateData: function () {
+ var data = {};
+ if (connection.jingle) {
+ data = connection.jingle.populateData();
+ }
+ return data;
+ },
+ getLogger: function () {
+ if(connection.logger)
+ return connection.logger.log;
+ return null;
+ },
+ getPrezi: function () {
+ return connection.emuc.getPrezi(this.myJid());
+ },
+ removePreziFromPresence: function () {
+ connection.emuc.removePreziFromPresence();
+ connection.emuc.sendPresence();
+ },
+ sendChatMessage: function (message, nickname) {
+ connection.emuc.sendMessage(message, nickname);
+ },
+ setSubject: function (topic) {
+ connection.emuc.setSubject(topic);
+ },
+ lockRoom: function (key, onSuccess, onError, onNotSupported) {
+ connection.emuc.lockRoom(key, onSuccess, onError, onNotSupported);
+ },
+ dial: function (to, from, roomName,roomPass) {
+ connection.rayo.dial(to, from, roomName,roomPass);
+ },
+ setMute: function (jid, mute) {
+ connection.moderate.setMute(jid, mute);
+ },
+ eject: function (jid) {
+ connection.moderate.eject(jid);
+ },
+ findJidFromResource: function (resource) {
+ connection.emuc.findJidFromResource(resource);
+ },
+ getMembers: function () {
+ return connection.emuc.members;
+ }
+
+};
+
+module.exports = XMPP;
\ No newline at end of file
diff --git a/muc.js b/muc.js
deleted file mode 100644
index 98b347337..000000000
--- a/muc.js
+++ /dev/null
@@ -1,548 +0,0 @@
-/* jshint -W117 */
-/* a simple MUC connection plugin
- * can only handle a single MUC room
- */
-Strophe.addConnectionPlugin('emuc', {
- connection: null,
- roomjid: null,
- myroomjid: null,
- members: {},
- list_members: [], // so we can elect a new focus
- presMap: {},
- preziMap: {},
- joined: false,
- isOwner: false,
- role: null,
- init: function (conn) {
- this.connection = conn;
- },
- initPresenceMap: function (myroomjid) {
- this.presMap['to'] = myroomjid;
- this.presMap['xns'] = 'http://jabber.org/protocol/muc';
- },
- doJoin: function (jid, password) {
- this.myroomjid = jid;
-
- console.info("Joined MUC as " + this.myroomjid);
-
- this.initPresenceMap(this.myroomjid);
-
- if (!this.roomjid) {
- this.roomjid = Strophe.getBareJidFromJid(jid);
- // add handlers (just once)
- this.connection.addHandler(this.onPresence.bind(this), null, 'presence', null, null, this.roomjid, {matchBare: true});
- this.connection.addHandler(this.onPresenceUnavailable.bind(this), null, 'presence', 'unavailable', null, this.roomjid, {matchBare: true});
- this.connection.addHandler(this.onPresenceError.bind(this), null, 'presence', 'error', null, this.roomjid, {matchBare: true});
- this.connection.addHandler(this.onMessage.bind(this), null, 'message', null, null, this.roomjid, {matchBare: true});
- }
- if (password !== undefined) {
- this.presMap['password'] = password;
- }
- this.sendPresence();
- },
- doLeave: function() {
- console.log("do leave", this.myroomjid);
- var pres = $pres({to: this.myroomjid, type: 'unavailable' });
- this.presMap.length = 0;
- this.connection.send(pres);
- },
- createNonAnonymousRoom: function() {
- // http://xmpp.org/extensions/xep-0045.html#createroom-reserved
-
- var getForm = $iq({type: 'get', to: this.roomjid})
- .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'})
- .c('x', {xmlns: 'jabber:x:data', type: 'submit'});
-
- this.connection.sendIQ(getForm, function (form){
-
- if (!$(form).find(
- '>query>x[xmlns="jabber:x:data"]' +
- '>field[var="muc#roomconfig_whois"]').length) {
-
- console.error('non-anonymous rooms not supported');
- return;
- }
-
- var formSubmit = $iq({to: this.roomjid, type: 'set'})
- .c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});
-
- formSubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});
-
- formSubmit.c('field', {'var': 'FORM_TYPE'})
- .c('value')
- .t('http://jabber.org/protocol/muc#roomconfig').up().up();
-
- formSubmit.c('field', {'var': 'muc#roomconfig_whois'})
- .c('value').t('anyone').up().up();
-
- this.connection.sendIQ(formSubmit);
-
- }, function (error){
- console.error("Error getting room configuration form");
- });
- },
- onPresence: function (pres) {
- var from = pres.getAttribute('from');
-
- // What is this for? A workaround for something?
- if (pres.getAttribute('type')) {
- return true;
- }
-
- // Parse etherpad tag.
- var etherpad = $(pres).find('>etherpad');
- if (etherpad.length) {
- if (config.etherpad_base && !Moderator.isModerator()) {
- UI.initEtherpad(etherpad.text());
- }
- }
-
- // Parse prezi tag.
- var presentation = $(pres).find('>prezi');
- if (presentation.length)
- {
- var url = presentation.attr('url');
- var current = presentation.find('>current').text();
-
- console.log('presentation info received from', from, url);
-
- if (this.preziMap[from] == null) {
- this.preziMap[from] = url;
-
- $(document).trigger('presentationadded.muc', [from, url, current]);
- }
- else {
- $(document).trigger('gotoslide.muc', [from, url, current]);
- }
- }
- else if (this.preziMap[from] != null) {
- var url = this.preziMap[from];
- delete this.preziMap[from];
- $(document).trigger('presentationremoved.muc', [from, url]);
- }
-
- // Parse audio info tag.
- var audioMuted = $(pres).find('>audiomuted');
- if (audioMuted.length) {
- $(document).trigger('audiomuted.muc', [from, audioMuted.text()]);
- }
-
- // Parse video info tag.
- var videoMuted = $(pres).find('>videomuted');
- if (videoMuted.length) {
- $(document).trigger('videomuted.muc', [from, videoMuted.text()]);
- }
-
- var stats = $(pres).find('>stats');
- if(stats.length)
- {
- var statsObj = {};
- Strophe.forEachChild(stats[0], "stat", function (el) {
- statsObj[el.getAttribute("name")] = el.getAttribute("value");
- });
- connectionquality.updateRemoteStats(from, statsObj);
- }
-
- // Parse status.
- if ($(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="201"]').length) {
- this.isOwner = true;
- this.createNonAnonymousRoom();
- }
-
- // Parse roles.
- var member = {};
- member.show = $(pres).find('>show').text();
- member.status = $(pres).find('>status').text();
- var tmp = $(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>item');
- member.affiliation = tmp.attr('affiliation');
- member.role = tmp.attr('role');
-
- // Focus recognition
- member.jid = tmp.attr('jid');
- member.isFocus = false;
- if (member.jid
- && member.jid.indexOf(Moderator.getFocusUserJid() + "/") == 0) {
- member.isFocus = true;
- }
-
- var nicktag = $(pres).find('>nick[xmlns="http://jabber.org/protocol/nick"]');
- member.displayName = (nicktag.length > 0 ? nicktag.html() : null);
-
- if (from == this.myroomjid) {
- if (member.affiliation == 'owner') this.isOwner = true;
- if (this.role !== member.role) {
- this.role = member.role;
- if(Moderator.onLocalRoleChange)
- Moderator.onLocalRoleChange(from, member, pres);
- UI.onLocalRoleChange(from, member, pres);
- }
- if (!this.joined) {
- this.joined = true;
- $(document).trigger('joined.muc', [from, member]);
- UI.onMucJoined(from, member);
- this.list_members.push(from);
- }
- } else if (this.members[from] === undefined) {
- // new participant
- this.members[from] = member;
- this.list_members.push(from);
- console.log('entered', from, member);
- if (member.isFocus)
- {
- focusMucJid = from;
- console.info("Ignore focus: " + from +", real JID: " + member.jid);
- }
- else {
- var id = $(pres).find('>userID').text();
- var email = $(pres).find('>email');
- if (email.length > 0) {
- id = email.text();
- }
- UI.onMucEntered(from, id, member.displayName);
- API.triggerEvent("participantJoined",{jid: from});
- }
- } else {
- // Presence update for existing participant
- // Watch role change:
- if (this.members[from].role != member.role) {
- this.members[from].role = member.role;
- UI.onMucRoleChanged(member.role, member.displayName);
- }
- }
-
- // Always trigger presence to update bindings
- $(document).trigger('presence.muc', [from, member, pres]);
-
- // Trigger status message update
- if (member.status) {
- UI.onMucPresenceStatus(from, member);
- }
-
- return true;
- },
- onPresenceUnavailable: function (pres) {
- var from = pres.getAttribute('from');
- // Status code 110 indicates that this notification is "self-presence".
- if (!$(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="110"]').length) {
- delete this.members[from];
- this.list_members.splice(this.list_members.indexOf(from), 1);
- this.onParticipantLeft(from);
- }
- // If the status code is 110 this means we're leaving and we would like
- // to remove everyone else from our view, so we trigger the event.
- else if (this.list_members.length > 1) {
- for (var i = 0; i < this.list_members.length; i++) {
- var member = this.list_members[i];
- delete this.members[i];
- this.list_members.splice(i, 1);
- this.onParticipantLeft(member);
- }
- }
- if ($(pres).find('>x[xmlns="http://jabber.org/protocol/muc#user"]>status[code="307"]').length) {
- $(document).trigger('kicked.muc', [from]);
- }
- return true;
- },
- onPresenceError: function (pres) {
- var from = pres.getAttribute('from');
- if ($(pres).find('>error[type="auth"]>not-authorized[xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"]').length) {
- console.log('on password required', from);
-
- UI.onPasswordReqiured(function (value) {
- connection.emuc.doJoin(from, value);
- })
- } else if ($(pres).find(
- '>error[type="cancel"]>not-allowed[xmlns="urn:ietf:params:xml:ns:xmpp-stanzas"]').length) {
- var toDomain = Strophe.getDomainFromJid(pres.getAttribute('to'));
- if(toDomain === config.hosts.anonymousdomain) {
- // we are connected with anonymous domain and only non anonymous users can create rooms
- // we must authorize the user
- $(document).trigger('passwordrequired.main');
- } else {
- console.warn('onPresError ', pres);
- UI.messageHandler.openReportDialog(null,
- 'Oops! Something went wrong and we couldn`t connect to the conference.',
- pres);
- }
- } else {
- console.warn('onPresError ', pres);
- UI.messageHandler.openReportDialog(null,
- 'Oops! Something went wrong and we couldn`t connect to the conference.',
- pres);
- }
- return true;
- },
- sendMessage: function (body, nickname) {
- var msg = $msg({to: this.roomjid, type: 'groupchat'});
- msg.c('body', body).up();
- if (nickname) {
- msg.c('nick', {xmlns: 'http://jabber.org/protocol/nick'}).t(nickname).up().up();
- }
- this.connection.send(msg);
- API.triggerEvent("outgoingMessage", {"message": body});
- },
- setSubject: function (subject){
- var msg = $msg({to: this.roomjid, type: 'groupchat'});
- msg.c('subject', subject);
- this.connection.send(msg);
- console.log("topic changed to " + subject);
- },
- onMessage: function (msg) {
- // FIXME: this is a hack. but jingle on muc makes nickchanges hard
- var from = msg.getAttribute('from');
- var nick = $(msg).find('>nick[xmlns="http://jabber.org/protocol/nick"]').text() || Strophe.getResourceFromJid(from);
-
- var txt = $(msg).find('>body').text();
- var type = msg.getAttribute("type");
- if(type == "error")
- {
- UI.chatAddError($(msg).find('>text').text(), txt);
- return true;
- }
-
- var subject = $(msg).find('>subject');
- if(subject.length)
- {
- var subjectText = subject.text();
- if(subjectText || subjectText == "") {
- UI.chatSetSubject(subjectText);
- console.log("Subject is changed to " + subjectText);
- }
- }
-
-
- if (txt) {
- console.log('chat', nick, txt);
- UI.updateChatConversation(from, nick, txt);
- if(from != this.myroomjid)
- API.triggerEvent("incomingMessage",
- {"from": from, "nick": nick, "message": txt});
- }
- return true;
- },
- lockRoom: function (key, onSuccess, onError, onNotSupported) {
- //http://xmpp.org/extensions/xep-0045.html#roomconfig
- var ob = this;
- this.connection.sendIQ($iq({to: this.roomjid, type: 'get'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'}),
- function (res) {
- if ($(res).find('>query>x[xmlns="jabber:x:data"]>field[var="muc#roomconfig_roomsecret"]').length) {
- var formsubmit = $iq({to: ob.roomjid, type: 'set'}).c('query', {xmlns: 'http://jabber.org/protocol/muc#owner'});
- formsubmit.c('x', {xmlns: 'jabber:x:data', type: 'submit'});
- formsubmit.c('field', {'var': 'FORM_TYPE'}).c('value').t('http://jabber.org/protocol/muc#roomconfig').up().up();
- formsubmit.c('field', {'var': 'muc#roomconfig_roomsecret'}).c('value').t(key).up().up();
- // Fixes a bug in prosody 0.9.+ https://code.google.com/p/lxmppd/issues/detail?id=373
- formsubmit.c('field', {'var': 'muc#roomconfig_whois'}).c('value').t('anyone').up().up();
- // FIXME: is muc#roomconfig_passwordprotectedroom required?
- this.connection.sendIQ(formsubmit,
- onSuccess,
- onError);
- } else {
- onNotSupported();
- }
- }, onError);
- },
- kick: function (jid) {
- var kickIQ = $iq({to: this.roomjid, type: 'set'})
- .c('query', {xmlns: 'http://jabber.org/protocol/muc#admin'})
- .c('item', {nick: Strophe.getResourceFromJid(jid), role: 'none'})
- .c('reason').t('You have been kicked.').up().up().up();
-
- this.connection.sendIQ(
- kickIQ,
- function (result) {
- console.log('Kick participant with jid: ', jid, result);
- },
- function (error) {
- console.log('Kick participant error: ', error);
- });
- },
- sendPresence: function () {
- var pres = $pres({to: this.presMap['to'] });
- pres.c('x', {xmlns: this.presMap['xns']});
-
- if (this.presMap['password']) {
- pres.c('password').t(this.presMap['password']).up();
- }
-
- pres.up();
-
- // Send XEP-0115 'c' stanza that contains our capabilities info
- if (connection.caps) {
- connection.caps.node = config.clientNode;
- pres.c('c', connection.caps.generateCapsAttrs()).up();
- }
-
- pres.c('user-agent', {xmlns: 'http://jitsi.org/jitmeet/user-agent'})
- .t(navigator.userAgent).up();
-
- if(this.presMap['bridgeIsDown']) {
- pres.c('bridgeIsDown').up();
- }
-
- if(this.presMap['email']) {
- pres.c('email').t(this.presMap['email']).up();
- }
-
- if(this.presMap['userId']) {
- pres.c('userId').t(this.presMap['userId']).up();
- }
-
- if (this.presMap['displayName']) {
- // XEP-0172
- pres.c('nick', {xmlns: 'http://jabber.org/protocol/nick'})
- .t(this.presMap['displayName']).up();
- }
-
- if (this.presMap['audions']) {
- pres.c('audiomuted', {xmlns: this.presMap['audions']})
- .t(this.presMap['audiomuted']).up();
- }
-
- if (this.presMap['videons']) {
- pres.c('videomuted', {xmlns: this.presMap['videons']})
- .t(this.presMap['videomuted']).up();
- }
-
- if(this.presMap['statsns'])
- {
- var stats = pres.c('stats', {xmlns: this.presMap['statsns']});
- for(var stat in this.presMap["stats"])
- if(this.presMap["stats"][stat] != null)
- stats.c("stat",{name: stat, value: this.presMap["stats"][stat]}).up();
- pres.up();
- }
-
- if (this.presMap['prezins']) {
- pres.c('prezi',
- {xmlns: this.presMap['prezins'],
- 'url': this.presMap['preziurl']})
- .c('current').t(this.presMap['prezicurrent']).up().up();
- }
-
- if (this.presMap['etherpadns']) {
- pres.c('etherpad', {xmlns: this.presMap['etherpadns']})
- .t(this.presMap['etherpadname']).up();
- }
-
- if (this.presMap['medians'])
- {
- pres.c('media', {xmlns: this.presMap['medians']});
- var sourceNumber = 0;
- Object.keys(this.presMap).forEach(function (key) {
- if (key.indexOf('source') >= 0) {
- sourceNumber++;
- }
- });
- if (sourceNumber > 0)
- for (var i = 1; i <= sourceNumber/3; i ++) {
- pres.c('source',
- {type: this.presMap['source' + i + '_type'],
- ssrc: this.presMap['source' + i + '_ssrc'],
- direction: this.presMap['source'+ i + '_direction']
- || 'sendrecv' }
- ).up();
- }
- }
-
- pres.up();
-// console.debug(pres.toString());
- connection.send(pres);
- },
- addDisplayNameToPresence: function (displayName) {
- this.presMap['displayName'] = displayName;
- },
- addMediaToPresence: function (sourceNumber, mtype, ssrcs, direction) {
- if (!this.presMap['medians'])
- this.presMap['medians'] = 'http://estos.de/ns/mjs';
-
- this.presMap['source' + sourceNumber + '_type'] = mtype;
- this.presMap['source' + sourceNumber + '_ssrc'] = ssrcs;
- this.presMap['source' + sourceNumber + '_direction'] = direction;
- },
- clearPresenceMedia: function () {
- var self = this;
- Object.keys(this.presMap).forEach( function(key) {
- if(key.indexOf('source') != -1) {
- delete self.presMap[key];
- }
- });
- },
- addPreziToPresence: function (url, currentSlide) {
- this.presMap['prezins'] = 'http://jitsi.org/jitmeet/prezi';
- this.presMap['preziurl'] = url;
- this.presMap['prezicurrent'] = currentSlide;
- },
- removePreziFromPresence: function () {
- delete this.presMap['prezins'];
- delete this.presMap['preziurl'];
- delete this.presMap['prezicurrent'];
- },
- addCurrentSlideToPresence: function (currentSlide) {
- this.presMap['prezicurrent'] = currentSlide;
- },
- getPrezi: function (roomjid) {
- return this.preziMap[roomjid];
- },
- addEtherpadToPresence: function(etherpadName) {
- this.presMap['etherpadns'] = 'http://jitsi.org/jitmeet/etherpad';
- this.presMap['etherpadname'] = etherpadName;
- },
- addAudioInfoToPresence: function(isMuted) {
- this.presMap['audions'] = 'http://jitsi.org/jitmeet/audio';
- this.presMap['audiomuted'] = isMuted.toString();
- },
- addVideoInfoToPresence: function(isMuted) {
- this.presMap['videons'] = 'http://jitsi.org/jitmeet/video';
- this.presMap['videomuted'] = isMuted.toString();
- },
- addConnectionInfoToPresence: function(stats) {
- this.presMap['statsns'] = 'http://jitsi.org/jitmeet/stats';
- this.presMap['stats'] = stats;
- },
- findJidFromResource: function(resourceJid) {
- if(resourceJid &&
- resourceJid === Strophe.getResourceFromJid(connection.emuc.myroomjid)) {
- return connection.emuc.myroomjid;
- }
- var peerJid = null;
- Object.keys(this.members).some(function (jid) {
- peerJid = jid;
- return Strophe.getResourceFromJid(jid) === resourceJid;
- });
- return peerJid;
- },
- addBridgeIsDownToPresence: function() {
- this.presMap['bridgeIsDown'] = true;
- },
- addEmailToPresence: function(email) {
- this.presMap['email'] = email;
- },
- addUserIdToPresence: function(userId) {
- this.presMap['userId'] = userId;
- },
- isModerator: function() {
- return this.role === 'moderator';
- },
- getMemberRole: function(peerJid) {
- if (this.members[peerJid]) {
- return this.members[peerJid].role;
- }
- return null;
- },
- onParticipantLeft: function (jid) {
- UI.onMucLeft(jid);
-
- API.triggerEvent("participantLeft",{jid: jid});
-
- delete jid2Ssrc[jid];
-
- connection.jingle.terminateByJid(jid);
-
- if (connection.emuc.getPrezi(jid)) {
- $(document).trigger('presentationremoved.muc',
- [jid, connection.emuc.getPrezi(jid)]);
- }
-
- Moderator.onMucLeft(jid);
- }
-});
diff --git a/recording.js b/recording.js
deleted file mode 100644
index d60402582..000000000
--- a/recording.js
+++ /dev/null
@@ -1,167 +0,0 @@
-/* global $, $iq, config, connection, focusMucJid, messageHandler, Moderator,
- Toolbar, Util */
-var Recording = (function (my) {
- var recordingToken = null;
- var recordingEnabled;
-
- /**
- * Whether to use a jirecon component for recording, or use the videobridge
- * through COLIBRI.
- */
- var useJirecon = (typeof config.hosts.jirecon != "undefined");
-
- /**
- * The ID of the jirecon recording session. Jirecon generates it when we
- * initially start recording, and it needs to be used in subsequent requests
- * to jirecon.
- */
- var jireconRid = null;
-
- my.setRecordingToken = function (token) {
- recordingToken = token;
- };
-
- my.setRecording = function (state, token, callback) {
- if (useJirecon){
- this.setRecordingJirecon(state, token, callback);
- } else {
- this.setRecordingColibri(state, token, callback);
- }
- };
-
- my.setRecordingJirecon = function (state, token, callback) {
- if (state == recordingEnabled){
- return;
- }
-
- var iq = $iq({to: config.hosts.jirecon, type: 'set'})
- .c('recording', {xmlns: 'http://jitsi.org/protocol/jirecon',
- action: state ? 'start' : 'stop',
- mucjid: connection.emuc.roomjid});
- if (!state){
- iq.attrs({rid: jireconRid});
- }
-
- console.log('Start recording');
-
- connection.sendIQ(
- iq,
- function (result) {
- // TODO wait for an IQ with the real status, since this is
- // provisional?
- jireconRid = $(result).find('recording').attr('rid');
- console.log('Recording ' + (state ? 'started' : 'stopped') +
- '(jirecon)' + result);
- recordingEnabled = state;
- if (!state){
- jireconRid = null;
- }
-
- callback(state);
- },
- function (error) {
- console.log('Failed to start recording, error: ', error);
- callback(recordingEnabled);
- });
- };
-
- // Sends a COLIBRI message which enables or disables (according to 'state')
- // the recording on the bridge. Waits for the result IQ and calls 'callback'
- // with the new recording state, according to the IQ.
- my.setRecordingColibri = function (state, token, callback) {
- var elem = $iq({to: focusMucJid, type: 'set'});
- elem.c('conference', {
- xmlns: 'http://jitsi.org/protocol/colibri'
- });
- elem.c('recording', {state: state, token: token});
-
- connection.sendIQ(elem,
- function (result) {
- console.log('Set recording "', state, '". Result:', result);
- var recordingElem = $(result).find('>conference>recording');
- var newState = ('true' === recordingElem.attr('state'));
-
- recordingEnabled = newState;
- callback(newState);
- },
- function (error) {
- console.warn(error);
- callback(recordingEnabled);
- }
- );
- };
-
- my.toggleRecording = function () {
- if (!Moderator.isModerator()) {
- console.log(
- 'non-focus, or conference not yet organized:' +
- ' not enabling recording');
- return;
- }
-
- // Jirecon does not (currently) support a token.
- if (!recordingToken && !useJirecon)
- {
- UI.messageHandler.openTwoButtonDialog(null,
- '
Enter recording token
' +
- '',
- false,
- "Save",
- function (e, v, m, f) {
- if (v) {
- var token = document.getElementById('recordingToken');
-
- if (token.value) {
- my.setRecordingToken(
- Util.escapeHtml(token.value));
- my.toggleRecording();
- }
- }
- },
- function (event) {
- document.getElementById('recordingToken').focus();
- },
- function () {}
- );
-
- return;
- }
-
- var oldState = recordingEnabled;
- UI.setRecordingButtonState(!oldState);
- my.setRecording(!oldState,
- recordingToken,
- function (state) {
- console.log("New recording state: ", state);
- if (state === oldState)
- {
- // FIXME: new focus:
- // this will not work when moderator changes
- // during active session. Then it will assume that
- // recording status has changed to true, but it might have
- // been already true(and we only received actual status from
- // the focus).
- //
- // SO we start with status null, so that it is initialized
- // here and will fail only after second click, so if invalid
- // token was used we have to press the button twice before
- // current status will be fetched and token will be reset.
- //
- // Reliable way would be to return authentication error.
- // Or status update when moderator connects.
- // Or we have to stop recording session when current
- // moderator leaves the room.
-
- // Failed to change, reset the token because it might
- // have been wrong
- my.setRecordingToken(null);
- }
- // Update with returned status
- UI.setRecordingButtonState(state);
- }
- );
- };
-
- return my;
-}(Recording || {}));
diff --git a/service/xmpp/XMPPEvents.js b/service/xmpp/XMPPEvents.js
new file mode 100644
index 000000000..ccc2e1d1b
--- /dev/null
+++ b/service/xmpp/XMPPEvents.js
@@ -0,0 +1,14 @@
+var XMPPEvents = {
+ CONFERENCE_CERATED: "xmpp.conferenceCreated.jingle",
+ CALL_TERMINATED: "xmpp.callterminated.jingle",
+ CALL_INCOMING: "xmpp.callincoming.jingle",
+ DISPOSE_CONFERENCE: "xmpp.dispoce_confernce",
+ KICKED: "xmpp.kicked",
+ BRIDGE_DOWN: "xmpp.bridge_down",
+ USER_ID_CHANGED: "xmpp.user_id_changed",
+ CHANGED_STREAMS: "xmpp.changed_streams",
+ MUC_JOINED: "xmpp.muc_joined",
+ DISPLAY_NAME_CHANGED: "xmpp.display_name_changed",
+ REMOTE_STATS: "xmpp.remote_stats"
+};
+//module.exports = XMPPEvents;
\ No newline at end of file