From e4e66a03d7a9bcfad18bf4bf86c314a35ff68cfe Mon Sep 17 00:00:00 2001 From: hristoterezov Date: Mon, 19 Jan 2015 11:20:00 +0200 Subject: [PATCH] Creates initial version of xmpp module. --- app.js | 721 +-- estos_log.js | 17 - index.html | 16 +- keyboard_shortcut.js | 8 +- libs/modules/API.bundle.js | 7 +- libs/modules/RTC.bundle.js | 23 +- libs/modules/UI.bundle.js | 881 ++- libs/modules/connectionquality.bundle.js | 6 +- libs/modules/desktopsharing.bundle.js | 5 +- libs/modules/simulcast.bundle.js | 106 +- libs/modules/statistics.bundle.js | 35 +- libs/modules/xmpp.bundle.js | 5082 +++++++++++++++++ libs/rayo.js | 103 - libs/strophe/strophe.jingle.js | 327 -- libs/strophe/strophe.util.js | 41 - moderatemuc.js | 56 - modules/API/API.js | 4 +- modules/RTC/DataChannels.js | 4 +- modules/RTC/RTC.js | 13 +- modules/UI/UI.js | 260 +- modules/UI/audio_levels/AudioLevels.js | 8 +- modules/UI/authentication/Authentication.js | 84 + modules/UI/avatar/Avatar.js | 10 +- modules/UI/etherpad/Etherpad.js | 5 +- modules/UI/prezi/Prezi.js | 16 +- modules/UI/side_pannels/SidePanelToggler.js | 5 +- modules/UI/side_pannels/chat/Chat.js | 9 +- modules/UI/side_pannels/chat/Commands.js | 2 +- .../side_pannels/contactlist/ContactList.js | 31 +- .../UI/side_pannels/settings/SettingsMenu.js | 20 +- modules/UI/toolbars/Toolbar.js | 92 +- modules/UI/toolbars/ToolbarToggler.js | 2 +- modules/UI/util/UIUtil.js | 7 + modules/UI/videolayout/VideoLayout.js | 278 +- .../connectionquality/connectionquality.js | 3 +- modules/desktopsharing/desktopsharing.js | 2 +- modules/simulcast/SimulcastReceiver.js | 103 +- modules/statistics/RTPStatsCollector.js | 27 +- modules/statistics/statistics.js | 17 +- .../xmpp/JingleSession.js | 266 +- .../xmpp/SDP.js | 513 +- modules/xmpp/SDPDiffer.js | 165 + modules/xmpp/SDPUtil.js | 349 ++ .../xmpp/TraceablePeerConnection.js | 2 + moderator.js => modules/xmpp/moderator.js | 141 +- modules/xmpp/recording.js | 152 + modules/xmpp/strophe.emuc.js | 607 ++ modules/xmpp/strophe.jingle.js | 334 ++ modules/xmpp/strophe.logger.js | 20 + modules/xmpp/strophe.moderate.js | 58 + modules/xmpp/strophe.rayo.js | 95 + modules/xmpp/strophe.util.js | 42 + modules/xmpp/xmpp.js | 422 ++ muc.js | 548 -- recording.js | 167 - service/xmpp/XMPPEvents.js | 14 + 56 files changed, 8946 insertions(+), 3385 deletions(-) delete mode 100644 estos_log.js create mode 100644 libs/modules/xmpp.bundle.js delete mode 100644 libs/rayo.js delete mode 100644 libs/strophe/strophe.jingle.js delete mode 100644 libs/strophe/strophe.util.js delete mode 100644 moderatemuc.js create mode 100644 modules/UI/authentication/Authentication.js rename libs/strophe/strophe.jingle.session.js => modules/xmpp/JingleSession.js (83%) rename libs/strophe/strophe.jingle.sdp.js => modules/xmpp/SDP.js (55%) create mode 100644 modules/xmpp/SDPDiffer.js create mode 100644 modules/xmpp/SDPUtil.js rename libs/strophe/strophe.jingle.adapter.js => modules/xmpp/TraceablePeerConnection.js (99%) rename moderator.js => modules/xmpp/moderator.js (70%) create mode 100644 modules/xmpp/recording.js create mode 100644 modules/xmpp/strophe.emuc.js create mode 100644 modules/xmpp/strophe.jingle.js create mode 100644 modules/xmpp/strophe.logger.js create mode 100644 modules/xmpp/strophe.moderate.js create mode 100644 modules/xmpp/strophe.rayo.js create mode 100644 modules/xmpp/strophe.util.js create mode 100644 modules/xmpp/xmpp.js delete mode 100644 muc.js delete mode 100644 recording.js create mode 100644 service/xmpp/XMPPEvents.js 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