/* global $, APP, interfaceConfig, JitsiMeetJS */ /* jshint -W101 */ import JitsiPopover from "../util/JitsiPopover"; import UIUtil from "../util/UIUtil"; const ParticipantConnectionStatus = JitsiMeetJS.constants.participantConnectionStatus; /** * Maps a connection quality value (in percent) to the width of the "full" icon. */ const qualityToWidth = [ // Full (5 bars) {percent: 80, width: "100%"}, // 4 bars {percent: 60, width: "80%"}, // 3 bars {percent: 40, width: "55%"}, // 2 bars {percent: 20, width: "40%"}, // 1 bar {percent: 0, width: "20%"} // Note: we never show 0 bars. ]; /** * Constructs new connection indicator. * @param videoContainer the video container associated with the indicator. * @param videoId the identifier of the video * @constructor */ function ConnectionIndicator(videoContainer, videoId) { this.videoContainer = videoContainer; this.bandwidth = null; this.packetLoss = null; this.bitrate = null; this.showMoreValue = false; this.resolution = null; this.transport = []; this.framerate = null; this.popover = null; this.id = videoId; this.create(); } ConnectionIndicator.getIP = function(value) { return value.substring(0, value.lastIndexOf(":")); }; ConnectionIndicator.getPort = function(value) { return value.substring(value.lastIndexOf(":") + 1, value.length); }; ConnectionIndicator.getStringFromArray = function (array) { var res = ""; for(var i = 0; i < array.length; i++) { res += (i === 0? "" : ", ") + array[i]; } return res; }; /** * Generates the html content. * @returns {string} the html content. */ ConnectionIndicator.prototype.generateText = function () { var downloadBitrate, uploadBitrate, packetLoss, i; if(!this.bitrate) { downloadBitrate = "N/A"; uploadBitrate = "N/A"; } else { downloadBitrate = this.bitrate.download? this.bitrate.download + " Kbps" : "N/A"; uploadBitrate = this.bitrate.upload? this.bitrate.upload + " Kbps" : "N/A"; } if(!this.packetLoss) { packetLoss = "N/A"; } else { packetLoss = "" + (this.packetLoss.download !== null ? this.packetLoss.download : "N/A") + "% " + (this.packetLoss.upload !== null? this.packetLoss.upload : "N/A") + "%"; } // GENERATE RESOLUTIONS STRING const resolutions = this.resolution || {}; const resolutionStr = Object.keys(resolutions).map(ssrc => { let {width, height} = resolutions[ssrc]; return `${width}x${height}`; }).join(', ') || 'N/A'; const framerates = this.framerate || {}; const frameRateStr = Object.keys(framerates).map(ssrc => framerates[ssrc] ).join(', ') || 'N/A'; let result = ( `
${downloadBitrate} ${uploadBitrate}
${packetLoss}
${resolutionStr}
${frameRateStr}
`); if(this.videoContainer.videoSpanId == "localVideoContainer") { result += ""; } if (this.showMoreValue) { var downloadBandwidth, uploadBandwidth, transport; if (!this.bandwidth) { downloadBandwidth = "N/A"; uploadBandwidth = "N/A"; } else { downloadBandwidth = this.bandwidth.download? this.bandwidth.download + " Kbps" : "N/A"; uploadBandwidth = this.bandwidth.upload? this.bandwidth.upload + " Kbps" : "N/A"; } if (!this.transport || this.transport.length === 0) { transport = "" + "" + " N/A"; } else { var data = { remoteIP: [], localIP:[], remotePort:[], localPort:[], transportType:[]}; for(i = 0; i < this.transport.length; i++) { var ip = ConnectionIndicator.getIP(this.transport[i].ip); var port = ConnectionIndicator.getPort(this.transport[i].ip); var localIP = ConnectionIndicator.getIP(this.transport[i].localip); var localPort = ConnectionIndicator.getPort(this.transport[i].localip); if(data.remoteIP.indexOf(ip) == -1) { data.remoteIP.push(ip); } if(data.remotePort.indexOf(port) == -1) { data.remotePort.push(port); } if(data.localIP.indexOf(localIP) == -1) { data.localIP.push(localIP); } if(data.localPort.indexOf(localPort) == -1) { data.localPort.push(localPort); } if(data.transportType.indexOf(this.transport[i].type) == -1) { data.transportType.push(this.transport[i].type); } } // All of the transports should be either P2P or JVB const isP2P = this.transport.length ? this.transport[0].p2p : false; var local_address_key = "connectionindicator.localaddress"; var remote_address_key = "connectionindicator.remoteaddress"; var localTransport = " " + ConnectionIndicator.getStringFromArray(data.localIP) + ""; transport = " " + ConnectionIndicator.getStringFromArray(data.remoteIP); // Append (p2p) to indicate the P2P type of transport if (isP2P) { transport += ""; } transport += ""; var key_remote = "connectionindicator.remoteport", key_local = "connectionindicator.localport"; transport += "" + "" + ""; localTransport += "" + "" + ""; transport += ConnectionIndicator.getStringFromArray(data.remotePort); localTransport += ConnectionIndicator.getStringFromArray(data.localPort); transport += ""; transport += localTransport + ""; transport +="" + "" + "" + ConnectionIndicator.getStringFromArray(data.transportType); + ""; } result += "" + "" + ""; result += transport + "
" + "" + "" + "" + downloadBandwidth + " " + uploadBandwidth + "
"; } return result; }; /** * Shows or hide the additional information. */ ConnectionIndicator.prototype.showMore = function () { this.showMoreValue = !this.showMoreValue; this.updatePopoverData(); }; function createIcon(classes, iconClass) { var icon = document.createElement("span"); for(var i in classes) { icon.classList.add(classes[i]); } icon.appendChild( document.createElement("i")).classList.add(iconClass); return icon; } /** * Creates the indicator */ ConnectionIndicator.prototype.create = function () { let indicatorId = 'connectionindicator'; let element = UIUtil.getVideoThumbnailIndicatorSpan({ videoSpanId: this.videoContainer.videoSpanId, indicatorId }); element.classList.add('show'); this.connectionIndicatorContainer = element; let popoverContent = ( `
` ); this.popover = new JitsiPopover($(element), { content: popoverContent, skin: "black", onBeforePosition: el => APP.translation.translateElement(el), position: interfaceConfig.VERTICAL_FILMSTRIP ? 'left' : 'top' }); // override popover show method to make sure we will update the content // before showing the popover var origShowFunc = this.popover.show; this.popover.show = function () { // update content by forcing it, to finish even if popover // is not visible this.updatePopoverData(true); // call the original show, passing its actual this origShowFunc.call(this.popover); }.bind(this); let connectionIconContainer = document.createElement('div'); connectionIconContainer.className = 'connection indicatoricon'; this.emptyIcon = connectionIconContainer.appendChild( createIcon(["connection_empty"], "icon-connection")); this.fullIcon = connectionIconContainer.appendChild( createIcon(["connection_full"], "icon-connection")); this.interruptedIndicator = connectionIconContainer.appendChild( createIcon(["connection_lost"],"icon-connection-lost")); this.ninjaIndicator = connectionIconContainer.appendChild( createIcon(["connection_ninja"],"icon-ninja")); $(this.interruptedIndicator).hide(); $(this.ninjaIndicator).hide(); this.connectionIndicatorContainer.appendChild(connectionIconContainer); }; /** * Removes the indicator */ ConnectionIndicator.prototype.remove = function() { if (this.connectionIndicatorContainer.parentNode) { this.connectionIndicatorContainer.parentNode.removeChild( this.connectionIndicatorContainer); } this.popover.forceHide(); }; /** * Updates the UI which displays or not a warning about user's connectivity * problems. * * @param {ParticipantConnectionStatus} connectionStatus */ ConnectionIndicator.prototype.updateConnectionStatusIndicator = function (connectionStatus) { this.connectionStatus = connectionStatus; if (connectionStatus === ParticipantConnectionStatus.INTERRUPTED) { $(this.interruptedIndicator).show(); $(this.emptyIcon).hide(); $(this.fullIcon).hide(); $(this.ninjaIndicator).hide(); } else if (connectionStatus === ParticipantConnectionStatus.INACTIVE) { $(this.interruptedIndicator).hide(); $(this.emptyIcon).hide(); $(this.fullIcon).hide(); $(this.ninjaIndicator).show(); } else { $(this.interruptedIndicator).hide(); $(this.emptyIcon).show(); $(this.fullIcon).show(); $(this.ninjaIndicator).hide(); } }; /** * Updates the data of the indicator * @param percent the percent of connection quality * @param object the statistics data. */ ConnectionIndicator.prototype.updateConnectionQuality = function (percent, object) { if (!percent) { this.connectionIndicatorContainer.style.display = "none"; this.popover.forceHide(); return; } else { if(this.connectionIndicatorContainer.style.display == "none") { this.connectionIndicatorContainer.style.display = "block"; } } if (object) { this.bandwidth = object.bandwidth; this.bitrate = object.bitrate; this.packetLoss = object.packetLoss; this.transport = object.transport; if (object.resolution) { this.resolution = object.resolution; } if (object.framerate) this.framerate = object.framerate; } let width = qualityToWidth.find(x => percent >= x.percent); this.fullIcon.style.width = width.width; this.updatePopoverData(); }; /** * Updates the resolution * @param resolution the new resolution */ ConnectionIndicator.prototype.updateResolution = function (resolution) { this.resolution = resolution; this.updatePopoverData(); }; /** * Updates the framerate * @param framerate the new resolution */ ConnectionIndicator.prototype.updateFramerate = function (framerate) { this.framerate = framerate; this.updatePopoverData(); }; /** * Updates the content of the popover if its visible * @param force to work even if popover is not visible */ ConnectionIndicator.prototype.updatePopoverData = function (force) { // generate content, translate it and add it to document only if // popover is visible or we force to do so. if(this.popover.popoverShown || force) { this.popover.updateContent( `
${this.generateText()}
` ); } }; /** * Hides the popover */ ConnectionIndicator.prototype.hide = function () { this.popover.forceHide(); }; /** * Hides the indicator */ ConnectionIndicator.prototype.hideIndicator = function () { this.connectionIndicatorContainer.style.display = "none"; if(this.popover) this.popover.forceHide(); }; /** * Adds a hover listener to the popover. */ ConnectionIndicator.prototype.addPopoverHoverListener = function (listener) { this.popover.addOnHoverPopover(listener); }; export default ConnectionIndicator;