|
|
/******/ (() => { // webpackBootstrap
|
|
|
/******/ var __webpack_modules__ = ({
|
|
|
|
|
|
/***/ "./node_modules/freeice/index.js":
|
|
|
/*!***************************************!*\
|
|
|
!*** ./node_modules/freeice/index.js ***!
|
|
|
\***************************************/
|
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
|
|
"use strict";
|
|
|
/* jshint node: true */
|
|
|
|
|
|
|
|
|
var normalice = __webpack_require__(/*! normalice */ "./node_modules/normalice/index.js");
|
|
|
|
|
|
/**
|
|
|
# freeice
|
|
|
|
|
|
The `freeice` module is a simple way of getting random STUN or TURN server
|
|
|
for your WebRTC application. The list of servers (just STUN at this stage)
|
|
|
were sourced from this [gist](https://gist.github.com/zziuni/3741933).
|
|
|
|
|
|
## Example Use
|
|
|
|
|
|
The following demonstrates how you can use `freeice` with
|
|
|
[rtc-quickconnect](https://github.com/rtc-io/rtc-quickconnect):
|
|
|
|
|
|
<<< examples/quickconnect.js
|
|
|
|
|
|
As the `freeice` module generates ice servers in a list compliant with the
|
|
|
WebRTC spec you will be able to use it with raw `RTCPeerConnection`
|
|
|
constructors and other WebRTC libraries.
|
|
|
|
|
|
## Hey, don't use my STUN/TURN server!
|
|
|
|
|
|
If for some reason your free STUN or TURN server ends up in the
|
|
|
list of servers ([stun](https://github.com/DamonOehlman/freeice/blob/master/stun.json) or
|
|
|
[turn](https://github.com/DamonOehlman/freeice/blob/master/turn.json))
|
|
|
that is used in this module, you can feel
|
|
|
free to open an issue on this repository and those servers will be removed
|
|
|
within 24 hours (or sooner). This is the quickest and probably the most
|
|
|
polite way to have something removed (and provides us some visibility
|
|
|
if someone opens a pull request requesting that a server is added).
|
|
|
|
|
|
## Please add my server!
|
|
|
|
|
|
If you have a server that you wish to add to the list, that's awesome! I'm
|
|
|
sure I speak on behalf of a whole pile of WebRTC developers who say thanks.
|
|
|
To get it into the list, feel free to either open a pull request or if you
|
|
|
find that process a bit daunting then just create an issue requesting
|
|
|
the addition of the server (make sure you provide all the details, and if
|
|
|
you have a Terms of Service then including that in the PR/issue would be
|
|
|
awesome).
|
|
|
|
|
|
## I know of a free server, can I add it?
|
|
|
|
|
|
Sure, if you do your homework and make sure it is ok to use (I'm currently
|
|
|
in the process of reviewing the terms of those STUN servers included from
|
|
|
the original list). If it's ok to go, then please see the previous entry
|
|
|
for how to add it.
|
|
|
|
|
|
## Current List of Servers
|
|
|
|
|
|
* current as at the time of last `README.md` file generation
|
|
|
|
|
|
### STUN
|
|
|
|
|
|
<<< stun.json
|
|
|
|
|
|
### TURN
|
|
|
|
|
|
<<< turn.json
|
|
|
|
|
|
**/
|
|
|
|
|
|
var freeice = function(opts) {
|
|
|
// if a list of servers has been provided, then use it instead of defaults
|
|
|
var servers = {
|
|
|
stun: (opts || {}).stun || __webpack_require__(/*! ./stun.json */ "./node_modules/freeice/stun.json"),
|
|
|
turn: (opts || {}).turn || __webpack_require__(/*! ./turn.json */ "./node_modules/freeice/turn.json")
|
|
|
};
|
|
|
|
|
|
var stunCount = (opts || {}).stunCount || 2;
|
|
|
var turnCount = (opts || {}).turnCount || 0;
|
|
|
var selected;
|
|
|
|
|
|
function getServers(type, count) {
|
|
|
var out = [];
|
|
|
var input = [].concat(servers[type]);
|
|
|
var idx;
|
|
|
|
|
|
while (input.length && out.length < count) {
|
|
|
idx = (Math.random() * input.length) | 0;
|
|
|
out = out.concat(input.splice(idx, 1));
|
|
|
}
|
|
|
|
|
|
return out.map(function(url) {
|
|
|
//If it's a not a string, don't try to "normalice" it otherwise using type:url will screw it up
|
|
|
if ((typeof url !== 'string') && (! (url instanceof String))) {
|
|
|
return url;
|
|
|
} else {
|
|
|
return normalice(type + ':' + url);
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
// add stun servers
|
|
|
selected = [].concat(getServers('stun', stunCount));
|
|
|
|
|
|
if (turnCount) {
|
|
|
selected = selected.concat(getServers('turn', turnCount));
|
|
|
}
|
|
|
|
|
|
return selected;
|
|
|
};
|
|
|
|
|
|
module.exports = freeice;
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/normalice/index.js":
|
|
|
/*!*****************************************!*\
|
|
|
!*** ./node_modules/normalice/index.js ***!
|
|
|
\*****************************************/
|
|
|
/***/ ((module) => {
|
|
|
|
|
|
/**
|
|
|
# normalice
|
|
|
|
|
|
Normalize an ice server configuration object (or plain old string) into a format
|
|
|
that is usable in all browsers supporting WebRTC. Primarily this module is designed
|
|
|
to help with the transition of the `url` attribute of the configuration object to
|
|
|
the `urls` attribute.
|
|
|
|
|
|
## Example Usage
|
|
|
|
|
|
<<< examples/simple.js
|
|
|
|
|
|
**/
|
|
|
|
|
|
var protocols = [
|
|
|
'stun:',
|
|
|
'turn:'
|
|
|
];
|
|
|
|
|
|
module.exports = function(input) {
|
|
|
var url = (input || {}).url || input;
|
|
|
var protocol;
|
|
|
var parts;
|
|
|
var output = {};
|
|
|
|
|
|
// if we don't have a string url, then allow the input to passthrough
|
|
|
if (typeof url != 'string' && (! (url instanceof String))) {
|
|
|
return input;
|
|
|
}
|
|
|
|
|
|
// trim the url string, and convert to an array
|
|
|
url = url.trim();
|
|
|
|
|
|
// if the protocol is not known, then passthrough
|
|
|
protocol = protocols[protocols.indexOf(url.slice(0, 5))];
|
|
|
if (! protocol) {
|
|
|
return input;
|
|
|
}
|
|
|
|
|
|
// now let's attack the remaining url parts
|
|
|
url = url.slice(5);
|
|
|
parts = url.split('@');
|
|
|
|
|
|
output.username = input.username;
|
|
|
output.credential = input.credential;
|
|
|
// if we have an authentication part, then set the credentials
|
|
|
if (parts.length > 1) {
|
|
|
url = parts[1];
|
|
|
parts = parts[0].split(':');
|
|
|
|
|
|
// add the output credential and username
|
|
|
output.username = parts[0];
|
|
|
output.credential = (input || {}).credential || parts[1] || '';
|
|
|
}
|
|
|
|
|
|
output.url = protocol + url;
|
|
|
output.urls = [ output.url ];
|
|
|
|
|
|
return output;
|
|
|
};
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/sdp/sdp.js":
|
|
|
/*!*********************************!*\
|
|
|
!*** ./node_modules/sdp/sdp.js ***!
|
|
|
\*********************************/
|
|
|
/***/ ((module) => {
|
|
|
|
|
|
"use strict";
|
|
|
/* eslint-env node */
|
|
|
|
|
|
|
|
|
// SDP helpers.
|
|
|
const SDPUtils = {};
|
|
|
|
|
|
// Generate an alphanumeric identifier for cname or mids.
|
|
|
// TODO: use UUIDs instead? https://gist.github.com/jed/982883
|
|
|
SDPUtils.generateIdentifier = function() {
|
|
|
return Math.random().toString(36).substring(2, 12);
|
|
|
};
|
|
|
|
|
|
// The RTCP CNAME used by all peerconnections from the same JS.
|
|
|
SDPUtils.localCName = SDPUtils.generateIdentifier();
|
|
|
|
|
|
// Splits SDP into lines, dealing with both CRLF and LF.
|
|
|
SDPUtils.splitLines = function(blob) {
|
|
|
return blob.trim().split('\n').map(line => line.trim());
|
|
|
};
|
|
|
// Splits SDP into sessionpart and mediasections. Ensures CRLF.
|
|
|
SDPUtils.splitSections = function(blob) {
|
|
|
const parts = blob.split('\nm=');
|
|
|
return parts.map((part, index) => (index > 0 ?
|
|
|
'm=' + part : part).trim() + '\r\n');
|
|
|
};
|
|
|
|
|
|
// Returns the session description.
|
|
|
SDPUtils.getDescription = function(blob) {
|
|
|
const sections = SDPUtils.splitSections(blob);
|
|
|
return sections && sections[0];
|
|
|
};
|
|
|
|
|
|
// Returns the individual media sections.
|
|
|
SDPUtils.getMediaSections = function(blob) {
|
|
|
const sections = SDPUtils.splitSections(blob);
|
|
|
sections.shift();
|
|
|
return sections;
|
|
|
};
|
|
|
|
|
|
// Returns lines that start with a certain prefix.
|
|
|
SDPUtils.matchPrefix = function(blob, prefix) {
|
|
|
return SDPUtils.splitLines(blob).filter(line => line.indexOf(prefix) === 0);
|
|
|
};
|
|
|
|
|
|
// Parses an ICE candidate line. Sample input:
|
|
|
// candidate:702786350 2 udp 41819902 8.8.8.8 60769 typ relay raddr 8.8.8.8
|
|
|
// rport 55996"
|
|
|
// Input can be prefixed with a=.
|
|
|
SDPUtils.parseCandidate = function(line) {
|
|
|
let parts;
|
|
|
// Parse both variants.
|
|
|
if (line.indexOf('a=candidate:') === 0) {
|
|
|
parts = line.substring(12).split(' ');
|
|
|
} else {
|
|
|
parts = line.substring(10).split(' ');
|
|
|
}
|
|
|
|
|
|
const candidate = {
|
|
|
foundation: parts[0],
|
|
|
component: {1: 'rtp', 2: 'rtcp'}[parts[1]] || parts[1],
|
|
|
protocol: parts[2].toLowerCase(),
|
|
|
priority: parseInt(parts[3], 10),
|
|
|
ip: parts[4],
|
|
|
address: parts[4], // address is an alias for ip.
|
|
|
port: parseInt(parts[5], 10),
|
|
|
// skip parts[6] == 'typ'
|
|
|
type: parts[7],
|
|
|
};
|
|
|
|
|
|
for (let i = 8; i < parts.length; i += 2) {
|
|
|
switch (parts[i]) {
|
|
|
case 'raddr':
|
|
|
candidate.relatedAddress = parts[i + 1];
|
|
|
break;
|
|
|
case 'rport':
|
|
|
candidate.relatedPort = parseInt(parts[i + 1], 10);
|
|
|
break;
|
|
|
case 'tcptype':
|
|
|
candidate.tcpType = parts[i + 1];
|
|
|
break;
|
|
|
case 'ufrag':
|
|
|
candidate.ufrag = parts[i + 1]; // for backward compatibility.
|
|
|
candidate.usernameFragment = parts[i + 1];
|
|
|
break;
|
|
|
default: // extension handling, in particular ufrag. Don't overwrite.
|
|
|
if (candidate[parts[i]] === undefined) {
|
|
|
candidate[parts[i]] = parts[i + 1];
|
|
|
}
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
return candidate;
|
|
|
};
|
|
|
|
|
|
// Translates a candidate object into SDP candidate attribute.
|
|
|
// This does not include the a= prefix!
|
|
|
SDPUtils.writeCandidate = function(candidate) {
|
|
|
const sdp = [];
|
|
|
sdp.push(candidate.foundation);
|
|
|
|
|
|
const component = candidate.component;
|
|
|
if (component === 'rtp') {
|
|
|
sdp.push(1);
|
|
|
} else if (component === 'rtcp') {
|
|
|
sdp.push(2);
|
|
|
} else {
|
|
|
sdp.push(component);
|
|
|
}
|
|
|
sdp.push(candidate.protocol.toUpperCase());
|
|
|
sdp.push(candidate.priority);
|
|
|
sdp.push(candidate.address || candidate.ip);
|
|
|
sdp.push(candidate.port);
|
|
|
|
|
|
const type = candidate.type;
|
|
|
sdp.push('typ');
|
|
|
sdp.push(type);
|
|
|
if (type !== 'host' && candidate.relatedAddress &&
|
|
|
candidate.relatedPort) {
|
|
|
sdp.push('raddr');
|
|
|
sdp.push(candidate.relatedAddress);
|
|
|
sdp.push('rport');
|
|
|
sdp.push(candidate.relatedPort);
|
|
|
}
|
|
|
if (candidate.tcpType && candidate.protocol.toLowerCase() === 'tcp') {
|
|
|
sdp.push('tcptype');
|
|
|
sdp.push(candidate.tcpType);
|
|
|
}
|
|
|
if (candidate.usernameFragment || candidate.ufrag) {
|
|
|
sdp.push('ufrag');
|
|
|
sdp.push(candidate.usernameFragment || candidate.ufrag);
|
|
|
}
|
|
|
return 'candidate:' + sdp.join(' ');
|
|
|
};
|
|
|
|
|
|
// Parses an ice-options line, returns an array of option tags.
|
|
|
// Sample input:
|
|
|
// a=ice-options:foo bar
|
|
|
SDPUtils.parseIceOptions = function(line) {
|
|
|
return line.substring(14).split(' ');
|
|
|
};
|
|
|
|
|
|
// Parses a rtpmap line, returns RTCRtpCoddecParameters. Sample input:
|
|
|
// a=rtpmap:111 opus/48000/2
|
|
|
SDPUtils.parseRtpMap = function(line) {
|
|
|
let parts = line.substring(9).split(' ');
|
|
|
const parsed = {
|
|
|
payloadType: parseInt(parts.shift(), 10), // was: id
|
|
|
};
|
|
|
|
|
|
parts = parts[0].split('/');
|
|
|
|
|
|
parsed.name = parts[0];
|
|
|
parsed.clockRate = parseInt(parts[1], 10); // was: clockrate
|
|
|
parsed.channels = parts.length === 3 ? parseInt(parts[2], 10) : 1;
|
|
|
// legacy alias, got renamed back to channels in ORTC.
|
|
|
parsed.numChannels = parsed.channels;
|
|
|
return parsed;
|
|
|
};
|
|
|
|
|
|
// Generates a rtpmap line from RTCRtpCodecCapability or
|
|
|
// RTCRtpCodecParameters.
|
|
|
SDPUtils.writeRtpMap = function(codec) {
|
|
|
let pt = codec.payloadType;
|
|
|
if (codec.preferredPayloadType !== undefined) {
|
|
|
pt = codec.preferredPayloadType;
|
|
|
}
|
|
|
const channels = codec.channels || codec.numChannels || 1;
|
|
|
return 'a=rtpmap:' + pt + ' ' + codec.name + '/' + codec.clockRate +
|
|
|
(channels !== 1 ? '/' + channels : '') + '\r\n';
|
|
|
};
|
|
|
|
|
|
// Parses a extmap line (headerextension from RFC 5285). Sample input:
|
|
|
// a=extmap:2 urn:ietf:params:rtp-hdrext:toffset
|
|
|
// a=extmap:2/sendonly urn:ietf:params:rtp-hdrext:toffset
|
|
|
SDPUtils.parseExtmap = function(line) {
|
|
|
const parts = line.substring(9).split(' ');
|
|
|
return {
|
|
|
id: parseInt(parts[0], 10),
|
|
|
direction: parts[0].indexOf('/') > 0 ? parts[0].split('/')[1] : 'sendrecv',
|
|
|
uri: parts[1],
|
|
|
attributes: parts.slice(2).join(' '),
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Generates an extmap line from RTCRtpHeaderExtensionParameters or
|
|
|
// RTCRtpHeaderExtension.
|
|
|
SDPUtils.writeExtmap = function(headerExtension) {
|
|
|
return 'a=extmap:' + (headerExtension.id || headerExtension.preferredId) +
|
|
|
(headerExtension.direction && headerExtension.direction !== 'sendrecv'
|
|
|
? '/' + headerExtension.direction
|
|
|
: '') +
|
|
|
' ' + headerExtension.uri +
|
|
|
(headerExtension.attributes ? ' ' + headerExtension.attributes : '') +
|
|
|
'\r\n';
|
|
|
};
|
|
|
|
|
|
// Parses a fmtp line, returns dictionary. Sample input:
|
|
|
// a=fmtp:96 vbr=on;cng=on
|
|
|
// Also deals with vbr=on; cng=on
|
|
|
SDPUtils.parseFmtp = function(line) {
|
|
|
const parsed = {};
|
|
|
let kv;
|
|
|
const parts = line.substring(line.indexOf(' ') + 1).split(';');
|
|
|
for (let j = 0; j < parts.length; j++) {
|
|
|
kv = parts[j].trim().split('=');
|
|
|
parsed[kv[0].trim()] = kv[1];
|
|
|
}
|
|
|
return parsed;
|
|
|
};
|
|
|
|
|
|
// Generates a fmtp line from RTCRtpCodecCapability or RTCRtpCodecParameters.
|
|
|
SDPUtils.writeFmtp = function(codec) {
|
|
|
let line = '';
|
|
|
let pt = codec.payloadType;
|
|
|
if (codec.preferredPayloadType !== undefined) {
|
|
|
pt = codec.preferredPayloadType;
|
|
|
}
|
|
|
if (codec.parameters && Object.keys(codec.parameters).length) {
|
|
|
const params = [];
|
|
|
Object.keys(codec.parameters).forEach(param => {
|
|
|
if (codec.parameters[param] !== undefined) {
|
|
|
params.push(param + '=' + codec.parameters[param]);
|
|
|
} else {
|
|
|
params.push(param);
|
|
|
}
|
|
|
});
|
|
|
line += 'a=fmtp:' + pt + ' ' + params.join(';') + '\r\n';
|
|
|
}
|
|
|
return line;
|
|
|
};
|
|
|
|
|
|
// Parses a rtcp-fb line, returns RTCPRtcpFeedback object. Sample input:
|
|
|
// a=rtcp-fb:98 nack rpsi
|
|
|
SDPUtils.parseRtcpFb = function(line) {
|
|
|
const parts = line.substring(line.indexOf(' ') + 1).split(' ');
|
|
|
return {
|
|
|
type: parts.shift(),
|
|
|
parameter: parts.join(' '),
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Generate a=rtcp-fb lines from RTCRtpCodecCapability or RTCRtpCodecParameters.
|
|
|
SDPUtils.writeRtcpFb = function(codec) {
|
|
|
let lines = '';
|
|
|
let pt = codec.payloadType;
|
|
|
if (codec.preferredPayloadType !== undefined) {
|
|
|
pt = codec.preferredPayloadType;
|
|
|
}
|
|
|
if (codec.rtcpFeedback && codec.rtcpFeedback.length) {
|
|
|
// FIXME: special handling for trr-int?
|
|
|
codec.rtcpFeedback.forEach(fb => {
|
|
|
lines += 'a=rtcp-fb:' + pt + ' ' + fb.type +
|
|
|
(fb.parameter && fb.parameter.length ? ' ' + fb.parameter : '') +
|
|
|
'\r\n';
|
|
|
});
|
|
|
}
|
|
|
return lines;
|
|
|
};
|
|
|
|
|
|
// Parses a RFC 5576 ssrc media attribute. Sample input:
|
|
|
// a=ssrc:3735928559 cname:something
|
|
|
SDPUtils.parseSsrcMedia = function(line) {
|
|
|
const sp = line.indexOf(' ');
|
|
|
const parts = {
|
|
|
ssrc: parseInt(line.substring(7, sp), 10),
|
|
|
};
|
|
|
const colon = line.indexOf(':', sp);
|
|
|
if (colon > -1) {
|
|
|
parts.attribute = line.substring(sp + 1, colon);
|
|
|
parts.value = line.substring(colon + 1);
|
|
|
} else {
|
|
|
parts.attribute = line.substring(sp + 1);
|
|
|
}
|
|
|
return parts;
|
|
|
};
|
|
|
|
|
|
// Parse a ssrc-group line (see RFC 5576). Sample input:
|
|
|
// a=ssrc-group:semantics 12 34
|
|
|
SDPUtils.parseSsrcGroup = function(line) {
|
|
|
const parts = line.substring(13).split(' ');
|
|
|
return {
|
|
|
semantics: parts.shift(),
|
|
|
ssrcs: parts.map(ssrc => parseInt(ssrc, 10)),
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Extracts the MID (RFC 5888) from a media section.
|
|
|
// Returns the MID or undefined if no mid line was found.
|
|
|
SDPUtils.getMid = function(mediaSection) {
|
|
|
const mid = SDPUtils.matchPrefix(mediaSection, 'a=mid:')[0];
|
|
|
if (mid) {
|
|
|
return mid.substring(6);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// Parses a fingerprint line for DTLS-SRTP.
|
|
|
SDPUtils.parseFingerprint = function(line) {
|
|
|
const parts = line.substring(14).split(' ');
|
|
|
return {
|
|
|
algorithm: parts[0].toLowerCase(), // algorithm is case-sensitive in Edge.
|
|
|
value: parts[1].toUpperCase(), // the definition is upper-case in RFC 4572.
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Extracts DTLS parameters from SDP media section or sessionpart.
|
|
|
// FIXME: for consistency with other functions this should only
|
|
|
// get the fingerprint line as input. See also getIceParameters.
|
|
|
SDPUtils.getDtlsParameters = function(mediaSection, sessionpart) {
|
|
|
const lines = SDPUtils.matchPrefix(mediaSection + sessionpart,
|
|
|
'a=fingerprint:');
|
|
|
// Note: a=setup line is ignored since we use the 'auto' role in Edge.
|
|
|
return {
|
|
|
role: 'auto',
|
|
|
fingerprints: lines.map(SDPUtils.parseFingerprint),
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Serializes DTLS parameters to SDP.
|
|
|
SDPUtils.writeDtlsParameters = function(params, setupType) {
|
|
|
let sdp = 'a=setup:' + setupType + '\r\n';
|
|
|
params.fingerprints.forEach(fp => {
|
|
|
sdp += 'a=fingerprint:' + fp.algorithm + ' ' + fp.value + '\r\n';
|
|
|
});
|
|
|
return sdp;
|
|
|
};
|
|
|
|
|
|
// Parses a=crypto lines into
|
|
|
// https://rawgit.com/aboba/edgertc/master/msortc-rs4.html#dictionary-rtcsrtpsdesparameters-members
|
|
|
SDPUtils.parseCryptoLine = function(line) {
|
|
|
const parts = line.substring(9).split(' ');
|
|
|
return {
|
|
|
tag: parseInt(parts[0], 10),
|
|
|
cryptoSuite: parts[1],
|
|
|
keyParams: parts[2],
|
|
|
sessionParams: parts.slice(3),
|
|
|
};
|
|
|
};
|
|
|
|
|
|
SDPUtils.writeCryptoLine = function(parameters) {
|
|
|
return 'a=crypto:' + parameters.tag + ' ' +
|
|
|
parameters.cryptoSuite + ' ' +
|
|
|
(typeof parameters.keyParams === 'object'
|
|
|
? SDPUtils.writeCryptoKeyParams(parameters.keyParams)
|
|
|
: parameters.keyParams) +
|
|
|
(parameters.sessionParams ? ' ' + parameters.sessionParams.join(' ') : '') +
|
|
|
'\r\n';
|
|
|
};
|
|
|
|
|
|
// Parses the crypto key parameters into
|
|
|
// https://rawgit.com/aboba/edgertc/master/msortc-rs4.html#rtcsrtpkeyparam*
|
|
|
SDPUtils.parseCryptoKeyParams = function(keyParams) {
|
|
|
if (keyParams.indexOf('inline:') !== 0) {
|
|
|
return null;
|
|
|
}
|
|
|
const parts = keyParams.substring(7).split('|');
|
|
|
return {
|
|
|
keyMethod: 'inline',
|
|
|
keySalt: parts[0],
|
|
|
lifeTime: parts[1],
|
|
|
mkiValue: parts[2] ? parts[2].split(':')[0] : undefined,
|
|
|
mkiLength: parts[2] ? parts[2].split(':')[1] : undefined,
|
|
|
};
|
|
|
};
|
|
|
|
|
|
SDPUtils.writeCryptoKeyParams = function(keyParams) {
|
|
|
return keyParams.keyMethod + ':'
|
|
|
+ keyParams.keySalt +
|
|
|
(keyParams.lifeTime ? '|' + keyParams.lifeTime : '') +
|
|
|
(keyParams.mkiValue && keyParams.mkiLength
|
|
|
? '|' + keyParams.mkiValue + ':' + keyParams.mkiLength
|
|
|
: '');
|
|
|
};
|
|
|
|
|
|
// Extracts all SDES parameters.
|
|
|
SDPUtils.getCryptoParameters = function(mediaSection, sessionpart) {
|
|
|
const lines = SDPUtils.matchPrefix(mediaSection + sessionpart,
|
|
|
'a=crypto:');
|
|
|
return lines.map(SDPUtils.parseCryptoLine);
|
|
|
};
|
|
|
|
|
|
// Parses ICE information from SDP media section or sessionpart.
|
|
|
// FIXME: for consistency with other functions this should only
|
|
|
// get the ice-ufrag and ice-pwd lines as input.
|
|
|
SDPUtils.getIceParameters = function(mediaSection, sessionpart) {
|
|
|
const ufrag = SDPUtils.matchPrefix(mediaSection + sessionpart,
|
|
|
'a=ice-ufrag:')[0];
|
|
|
const pwd = SDPUtils.matchPrefix(mediaSection + sessionpart,
|
|
|
'a=ice-pwd:')[0];
|
|
|
if (!(ufrag && pwd)) {
|
|
|
return null;
|
|
|
}
|
|
|
return {
|
|
|
usernameFragment: ufrag.substring(12),
|
|
|
password: pwd.substring(10),
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// Serializes ICE parameters to SDP.
|
|
|
SDPUtils.writeIceParameters = function(params) {
|
|
|
let sdp = 'a=ice-ufrag:' + params.usernameFragment + '\r\n' +
|
|
|
'a=ice-pwd:' + params.password + '\r\n';
|
|
|
if (params.iceLite) {
|
|
|
sdp += 'a=ice-lite\r\n';
|
|
|
}
|
|
|
return sdp;
|
|
|
};
|
|
|
|
|
|
// Parses the SDP media section and returns RTCRtpParameters.
|
|
|
SDPUtils.parseRtpParameters = function(mediaSection) {
|
|
|
const description = {
|
|
|
codecs: [],
|
|
|
headerExtensions: [],
|
|
|
fecMechanisms: [],
|
|
|
rtcp: [],
|
|
|
};
|
|
|
const lines = SDPUtils.splitLines(mediaSection);
|
|
|
const mline = lines[0].split(' ');
|
|
|
description.profile = mline[2];
|
|
|
for (let i = 3; i < mline.length; i++) { // find all codecs from mline[3..]
|
|
|
const pt = mline[i];
|
|
|
const rtpmapline = SDPUtils.matchPrefix(
|
|
|
mediaSection, 'a=rtpmap:' + pt + ' ')[0];
|
|
|
if (rtpmapline) {
|
|
|
const codec = SDPUtils.parseRtpMap(rtpmapline);
|
|
|
const fmtps = SDPUtils.matchPrefix(
|
|
|
mediaSection, 'a=fmtp:' + pt + ' ');
|
|
|
// Only the first a=fmtp:<pt> is considered.
|
|
|
codec.parameters = fmtps.length ? SDPUtils.parseFmtp(fmtps[0]) : {};
|
|
|
codec.rtcpFeedback = SDPUtils.matchPrefix(
|
|
|
mediaSection, 'a=rtcp-fb:' + pt + ' ')
|
|
|
.map(SDPUtils.parseRtcpFb);
|
|
|
description.codecs.push(codec);
|
|
|
// parse FEC mechanisms from rtpmap lines.
|
|
|
switch (codec.name.toUpperCase()) {
|
|
|
case 'RED':
|
|
|
case 'ULPFEC':
|
|
|
description.fecMechanisms.push(codec.name.toUpperCase());
|
|
|
break;
|
|
|
default: // only RED and ULPFEC are recognized as FEC mechanisms.
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
SDPUtils.matchPrefix(mediaSection, 'a=extmap:').forEach(line => {
|
|
|
description.headerExtensions.push(SDPUtils.parseExtmap(line));
|
|
|
});
|
|
|
const wildcardRtcpFb = SDPUtils.matchPrefix(mediaSection, 'a=rtcp-fb:* ')
|
|
|
.map(SDPUtils.parseRtcpFb);
|
|
|
description.codecs.forEach(codec => {
|
|
|
wildcardRtcpFb.forEach(fb=> {
|
|
|
const duplicate = codec.rtcpFeedback.find(existingFeedback => {
|
|
|
return existingFeedback.type === fb.type &&
|
|
|
existingFeedback.parameter === fb.parameter;
|
|
|
});
|
|
|
if (!duplicate) {
|
|
|
codec.rtcpFeedback.push(fb);
|
|
|
}
|
|
|
});
|
|
|
});
|
|
|
// FIXME: parse rtcp.
|
|
|
return description;
|
|
|
};
|
|
|
|
|
|
// Generates parts of the SDP media section describing the capabilities /
|
|
|
// parameters.
|
|
|
SDPUtils.writeRtpDescription = function(kind, caps) {
|
|
|
let sdp = '';
|
|
|
|
|
|
// Build the mline.
|
|
|
sdp += 'm=' + kind + ' ';
|
|
|
sdp += caps.codecs.length > 0 ? '9' : '0'; // reject if no codecs.
|
|
|
sdp += ' ' + (caps.profile || 'UDP/TLS/RTP/SAVPF') + ' ';
|
|
|
sdp += caps.codecs.map(codec => {
|
|
|
if (codec.preferredPayloadType !== undefined) {
|
|
|
return codec.preferredPayloadType;
|
|
|
}
|
|
|
return codec.payloadType;
|
|
|
}).join(' ') + '\r\n';
|
|
|
|
|
|
sdp += 'c=IN IP4 0.0.0.0\r\n';
|
|
|
sdp += 'a=rtcp:9 IN IP4 0.0.0.0\r\n';
|
|
|
|
|
|
// Add a=rtpmap lines for each codec. Also fmtp and rtcp-fb.
|
|
|
caps.codecs.forEach(codec => {
|
|
|
sdp += SDPUtils.writeRtpMap(codec);
|
|
|
sdp += SDPUtils.writeFmtp(codec);
|
|
|
sdp += SDPUtils.writeRtcpFb(codec);
|
|
|
});
|
|
|
let maxptime = 0;
|
|
|
caps.codecs.forEach(codec => {
|
|
|
if (codec.maxptime > maxptime) {
|
|
|
maxptime = codec.maxptime;
|
|
|
}
|
|
|
});
|
|
|
if (maxptime > 0) {
|
|
|
sdp += 'a=maxptime:' + maxptime + '\r\n';
|
|
|
}
|
|
|
|
|
|
if (caps.headerExtensions) {
|
|
|
caps.headerExtensions.forEach(extension => {
|
|
|
sdp += SDPUtils.writeExtmap(extension);
|
|
|
});
|
|
|
}
|
|
|
// FIXME: write fecMechanisms.
|
|
|
return sdp;
|
|
|
};
|
|
|
|
|
|
// Parses the SDP media section and returns an array of
|
|
|
// RTCRtpEncodingParameters.
|
|
|
SDPUtils.parseRtpEncodingParameters = function(mediaSection) {
|
|
|
const encodingParameters = [];
|
|
|
const description = SDPUtils.parseRtpParameters(mediaSection);
|
|
|
const hasRed = description.fecMechanisms.indexOf('RED') !== -1;
|
|
|
const hasUlpfec = description.fecMechanisms.indexOf('ULPFEC') !== -1;
|
|
|
|
|
|
// filter a=ssrc:... cname:, ignore PlanB-msid
|
|
|
const ssrcs = SDPUtils.matchPrefix(mediaSection, 'a=ssrc:')
|
|
|
.map(line => SDPUtils.parseSsrcMedia(line))
|
|
|
.filter(parts => parts.attribute === 'cname');
|
|
|
const primarySsrc = ssrcs.length > 0 && ssrcs[0].ssrc;
|
|
|
let secondarySsrc;
|
|
|
|
|
|
const flows = SDPUtils.matchPrefix(mediaSection, 'a=ssrc-group:FID')
|
|
|
.map(line => {
|
|
|
const parts = line.substring(17).split(' ');
|
|
|
return parts.map(part => parseInt(part, 10));
|
|
|
});
|
|
|
if (flows.length > 0 && flows[0].length > 1 && flows[0][0] === primarySsrc) {
|
|
|
secondarySsrc = flows[0][1];
|
|
|
}
|
|
|
|
|
|
description.codecs.forEach(codec => {
|
|
|
if (codec.name.toUpperCase() === 'RTX' && codec.parameters.apt) {
|
|
|
let encParam = {
|
|
|
ssrc: primarySsrc,
|
|
|
codecPayloadType: parseInt(codec.parameters.apt, 10),
|
|
|
};
|
|
|
if (primarySsrc && secondarySsrc) {
|
|
|
encParam.rtx = {ssrc: secondarySsrc};
|
|
|
}
|
|
|
encodingParameters.push(encParam);
|
|
|
if (hasRed) {
|
|
|
encParam = JSON.parse(JSON.stringify(encParam));
|
|
|
encParam.fec = {
|
|
|
ssrc: primarySsrc,
|
|
|
mechanism: hasUlpfec ? 'red+ulpfec' : 'red',
|
|
|
};
|
|
|
encodingParameters.push(encParam);
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
if (encodingParameters.length === 0 && primarySsrc) {
|
|
|
encodingParameters.push({
|
|
|
ssrc: primarySsrc,
|
|
|
});
|
|
|
}
|
|
|
|
|
|
// we support both b=AS and b=TIAS but interpret AS as TIAS.
|
|
|
let bandwidth = SDPUtils.matchPrefix(mediaSection, 'b=');
|
|
|
if (bandwidth.length) {
|
|
|
if (bandwidth[0].indexOf('b=TIAS:') === 0) {
|
|
|
bandwidth = parseInt(bandwidth[0].substring(7), 10);
|
|
|
} else if (bandwidth[0].indexOf('b=AS:') === 0) {
|
|
|
// use formula from JSEP to convert b=AS to TIAS value.
|
|
|
bandwidth = parseInt(bandwidth[0].substring(5), 10) * 1000 * 0.95
|
|
|
- (50 * 40 * 8);
|
|
|
} else {
|
|
|
bandwidth = undefined;
|
|
|
}
|
|
|
encodingParameters.forEach(params => {
|
|
|
params.maxBitrate = bandwidth;
|
|
|
});
|
|
|
}
|
|
|
return encodingParameters;
|
|
|
};
|
|
|
|
|
|
// parses http://draft.ortc.org/#rtcrtcpparameters*
|
|
|
SDPUtils.parseRtcpParameters = function(mediaSection) {
|
|
|
const rtcpParameters = {};
|
|
|
|
|
|
// Gets the first SSRC. Note that with RTX there might be multiple
|
|
|
// SSRCs.
|
|
|
const remoteSsrc = SDPUtils.matchPrefix(mediaSection, 'a=ssrc:')
|
|
|
.map(line => SDPUtils.parseSsrcMedia(line))
|
|
|
.filter(obj => obj.attribute === 'cname')[0];
|
|
|
if (remoteSsrc) {
|
|
|
rtcpParameters.cname = remoteSsrc.value;
|
|
|
rtcpParameters.ssrc = remoteSsrc.ssrc;
|
|
|
}
|
|
|
|
|
|
// Edge uses the compound attribute instead of reducedSize
|
|
|
// compound is !reducedSize
|
|
|
const rsize = SDPUtils.matchPrefix(mediaSection, 'a=rtcp-rsize');
|
|
|
rtcpParameters.reducedSize = rsize.length > 0;
|
|
|
rtcpParameters.compound = rsize.length === 0;
|
|
|
|
|
|
// parses the rtcp-mux attrіbute.
|
|
|
// Note that Edge does not support unmuxed RTCP.
|
|
|
const mux = SDPUtils.matchPrefix(mediaSection, 'a=rtcp-mux');
|
|
|
rtcpParameters.mux = mux.length > 0;
|
|
|
|
|
|
return rtcpParameters;
|
|
|
};
|
|
|
|
|
|
SDPUtils.writeRtcpParameters = function(rtcpParameters) {
|
|
|
let sdp = '';
|
|
|
if (rtcpParameters.reducedSize) {
|
|
|
sdp += 'a=rtcp-rsize\r\n';
|
|
|
}
|
|
|
if (rtcpParameters.mux) {
|
|
|
sdp += 'a=rtcp-mux\r\n';
|
|
|
}
|
|
|
if (rtcpParameters.ssrc !== undefined && rtcpParameters.cname) {
|
|
|
sdp += 'a=ssrc:' + rtcpParameters.ssrc +
|
|
|
' cname:' + rtcpParameters.cname + '\r\n';
|
|
|
}
|
|
|
return sdp;
|
|
|
};
|
|
|
|
|
|
|
|
|
// parses either a=msid: or a=ssrc:... msid lines and returns
|
|
|
// the id of the MediaStream and MediaStreamTrack.
|
|
|
SDPUtils.parseMsid = function(mediaSection) {
|
|
|
let parts;
|
|
|
const spec = SDPUtils.matchPrefix(mediaSection, 'a=msid:');
|
|
|
if (spec.length === 1) {
|
|
|
parts = spec[0].substring(7).split(' ');
|
|
|
return {stream: parts[0], track: parts[1]};
|
|
|
}
|
|
|
const planB = SDPUtils.matchPrefix(mediaSection, 'a=ssrc:')
|
|
|
.map(line => SDPUtils.parseSsrcMedia(line))
|
|
|
.filter(msidParts => msidParts.attribute === 'msid');
|
|
|
if (planB.length > 0) {
|
|
|
parts = planB[0].value.split(' ');
|
|
|
return {stream: parts[0], track: parts[1]};
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// SCTP
|
|
|
// parses draft-ietf-mmusic-sctp-sdp-26 first and falls back
|
|
|
// to draft-ietf-mmusic-sctp-sdp-05
|
|
|
SDPUtils.parseSctpDescription = function(mediaSection) {
|
|
|
const mline = SDPUtils.parseMLine(mediaSection);
|
|
|
const maxSizeLine = SDPUtils.matchPrefix(mediaSection, 'a=max-message-size:');
|
|
|
let maxMessageSize;
|
|
|
if (maxSizeLine.length > 0) {
|
|
|
maxMessageSize = parseInt(maxSizeLine[0].substring(19), 10);
|
|
|
}
|
|
|
if (isNaN(maxMessageSize)) {
|
|
|
maxMessageSize = 65536;
|
|
|
}
|
|
|
const sctpPort = SDPUtils.matchPrefix(mediaSection, 'a=sctp-port:');
|
|
|
if (sctpPort.length > 0) {
|
|
|
return {
|
|
|
port: parseInt(sctpPort[0].substring(12), 10),
|
|
|
protocol: mline.fmt,
|
|
|
maxMessageSize,
|
|
|
};
|
|
|
}
|
|
|
const sctpMapLines = SDPUtils.matchPrefix(mediaSection, 'a=sctpmap:');
|
|
|
if (sctpMapLines.length > 0) {
|
|
|
const parts = sctpMapLines[0]
|
|
|
.substring(10)
|
|
|
.split(' ');
|
|
|
return {
|
|
|
port: parseInt(parts[0], 10),
|
|
|
protocol: parts[1],
|
|
|
maxMessageSize,
|
|
|
};
|
|
|
}
|
|
|
};
|
|
|
|
|
|
// SCTP
|
|
|
// outputs the draft-ietf-mmusic-sctp-sdp-26 version that all browsers
|
|
|
// support by now receiving in this format, unless we originally parsed
|
|
|
// as the draft-ietf-mmusic-sctp-sdp-05 format (indicated by the m-line
|
|
|
// protocol of DTLS/SCTP -- without UDP/ or TCP/)
|
|
|
SDPUtils.writeSctpDescription = function(media, sctp) {
|
|
|
let output = [];
|
|
|
if (media.protocol !== 'DTLS/SCTP') {
|
|
|
output = [
|
|
|
'm=' + media.kind + ' 9 ' + media.protocol + ' ' + sctp.protocol + '\r\n',
|
|
|
'c=IN IP4 0.0.0.0\r\n',
|
|
|
'a=sctp-port:' + sctp.port + '\r\n',
|
|
|
];
|
|
|
} else {
|
|
|
output = [
|
|
|
'm=' + media.kind + ' 9 ' + media.protocol + ' ' + sctp.port + '\r\n',
|
|
|
'c=IN IP4 0.0.0.0\r\n',
|
|
|
'a=sctpmap:' + sctp.port + ' ' + sctp.protocol + ' 65535\r\n',
|
|
|
];
|
|
|
}
|
|
|
if (sctp.maxMessageSize !== undefined) {
|
|
|
output.push('a=max-message-size:' + sctp.maxMessageSize + '\r\n');
|
|
|
}
|
|
|
return output.join('');
|
|
|
};
|
|
|
|
|
|
// Generate a session ID for SDP.
|
|
|
// https://tools.ietf.org/html/draft-ietf-rtcweb-jsep-20#section-5.2.1
|
|
|
// recommends using a cryptographically random +ve 64-bit value
|
|
|
// but right now this should be acceptable and within the right range
|
|
|
SDPUtils.generateSessionId = function() {
|
|
|
return Math.random().toString().substr(2, 22);
|
|
|
};
|
|
|
|
|
|
// Write boiler plate for start of SDP
|
|
|
// sessId argument is optional - if not supplied it will
|
|
|
// be generated randomly
|
|
|
// sessVersion is optional and defaults to 2
|
|
|
// sessUser is optional and defaults to 'thisisadapterortc'
|
|
|
SDPUtils.writeSessionBoilerplate = function(sessId, sessVer, sessUser) {
|
|
|
let sessionId;
|
|
|
const version = sessVer !== undefined ? sessVer : 2;
|
|
|
if (sessId) {
|
|
|
sessionId = sessId;
|
|
|
} else {
|
|
|
sessionId = SDPUtils.generateSessionId();
|
|
|
}
|
|
|
const user = sessUser || 'thisisadapterortc';
|
|
|
// FIXME: sess-id should be an NTP timestamp.
|
|
|
return 'v=0\r\n' +
|
|
|
'o=' + user + ' ' + sessionId + ' ' + version +
|
|
|
' IN IP4 127.0.0.1\r\n' +
|
|
|
's=-\r\n' +
|
|
|
't=0 0\r\n';
|
|
|
};
|
|
|
|
|
|
// Gets the direction from the mediaSection or the sessionpart.
|
|
|
SDPUtils.getDirection = function(mediaSection, sessionpart) {
|
|
|
// Look for sendrecv, sendonly, recvonly, inactive, default to sendrecv.
|
|
|
const lines = SDPUtils.splitLines(mediaSection);
|
|
|
for (let i = 0; i < lines.length; i++) {
|
|
|
switch (lines[i]) {
|
|
|
case 'a=sendrecv':
|
|
|
case 'a=sendonly':
|
|
|
case 'a=recvonly':
|
|
|
case 'a=inactive':
|
|
|
return lines[i].substring(2);
|
|
|
default:
|
|
|
// FIXME: What should happen here?
|
|
|
}
|
|
|
}
|
|
|
if (sessionpart) {
|
|
|
return SDPUtils.getDirection(sessionpart);
|
|
|
}
|
|
|
return 'sendrecv';
|
|
|
};
|
|
|
|
|
|
SDPUtils.getKind = function(mediaSection) {
|
|
|
const lines = SDPUtils.splitLines(mediaSection);
|
|
|
const mline = lines[0].split(' ');
|
|
|
return mline[0].substring(2);
|
|
|
};
|
|
|
|
|
|
SDPUtils.isRejected = function(mediaSection) {
|
|
|
return mediaSection.split(' ', 2)[1] === '0';
|
|
|
};
|
|
|
|
|
|
SDPUtils.parseMLine = function(mediaSection) {
|
|
|
const lines = SDPUtils.splitLines(mediaSection);
|
|
|
const parts = lines[0].substring(2).split(' ');
|
|
|
return {
|
|
|
kind: parts[0],
|
|
|
port: parseInt(parts[1], 10),
|
|
|
protocol: parts[2],
|
|
|
fmt: parts.slice(3).join(' '),
|
|
|
};
|
|
|
};
|
|
|
|
|
|
SDPUtils.parseOLine = function(mediaSection) {
|
|
|
const line = SDPUtils.matchPrefix(mediaSection, 'o=')[0];
|
|
|
const parts = line.substring(2).split(' ');
|
|
|
return {
|
|
|
username: parts[0],
|
|
|
sessionId: parts[1],
|
|
|
sessionVersion: parseInt(parts[2], 10),
|
|
|
netType: parts[3],
|
|
|
addressType: parts[4],
|
|
|
address: parts[5],
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// a very naive interpretation of a valid SDP.
|
|
|
SDPUtils.isValidSDP = function(blob) {
|
|
|
if (typeof blob !== 'string' || blob.length === 0) {
|
|
|
return false;
|
|
|
}
|
|
|
const lines = SDPUtils.splitLines(blob);
|
|
|
for (let i = 0; i < lines.length; i++) {
|
|
|
if (lines[i].length < 2 || lines[i].charAt(1) !== '=') {
|
|
|
return false;
|
|
|
}
|
|
|
// TODO: check the modifier a bit more.
|
|
|
}
|
|
|
return true;
|
|
|
};
|
|
|
|
|
|
// Expose public methods.
|
|
|
if (true) {
|
|
|
module.exports = SDPUtils;
|
|
|
}
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/ua-parser-js/src/ua-parser.js":
|
|
|
/*!****************************************************!*\
|
|
|
!*** ./node_modules/ua-parser-js/src/ua-parser.js ***!
|
|
|
\****************************************************/
|
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
|
|
var __WEBPACK_AMD_DEFINE_RESULT__;/////////////////////////////////////////////////////////////////////////////////
|
|
|
/* UAParser.js v1.0.37
|
|
|
Copyright © 2012-2021 Faisal Salman <f@faisalman.com>
|
|
|
MIT License *//*
|
|
|
Detect Browser, Engine, OS, CPU, and Device type/model from User-Agent data.
|
|
|
Supports browser & node.js environment.
|
|
|
Demo : https://faisalman.github.io/ua-parser-js
|
|
|
Source : https://github.com/faisalman/ua-parser-js */
|
|
|
/////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
(function (window, undefined) {
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
//////////////
|
|
|
// Constants
|
|
|
/////////////
|
|
|
|
|
|
|
|
|
var LIBVERSION = '1.0.37',
|
|
|
EMPTY = '',
|
|
|
UNKNOWN = '?',
|
|
|
FUNC_TYPE = 'function',
|
|
|
UNDEF_TYPE = 'undefined',
|
|
|
OBJ_TYPE = 'object',
|
|
|
STR_TYPE = 'string',
|
|
|
MAJOR = 'major',
|
|
|
MODEL = 'model',
|
|
|
NAME = 'name',
|
|
|
TYPE = 'type',
|
|
|
VENDOR = 'vendor',
|
|
|
VERSION = 'version',
|
|
|
ARCHITECTURE= 'architecture',
|
|
|
CONSOLE = 'console',
|
|
|
MOBILE = 'mobile',
|
|
|
TABLET = 'tablet',
|
|
|
SMARTTV = 'smarttv',
|
|
|
WEARABLE = 'wearable',
|
|
|
EMBEDDED = 'embedded',
|
|
|
UA_MAX_LENGTH = 500;
|
|
|
|
|
|
var AMAZON = 'Amazon',
|
|
|
APPLE = 'Apple',
|
|
|
ASUS = 'ASUS',
|
|
|
BLACKBERRY = 'BlackBerry',
|
|
|
BROWSER = 'Browser',
|
|
|
CHROME = 'Chrome',
|
|
|
EDGE = 'Edge',
|
|
|
FIREFOX = 'Firefox',
|
|
|
GOOGLE = 'Google',
|
|
|
HUAWEI = 'Huawei',
|
|
|
LG = 'LG',
|
|
|
MICROSOFT = 'Microsoft',
|
|
|
MOTOROLA = 'Motorola',
|
|
|
OPERA = 'Opera',
|
|
|
SAMSUNG = 'Samsung',
|
|
|
SHARP = 'Sharp',
|
|
|
SONY = 'Sony',
|
|
|
XIAOMI = 'Xiaomi',
|
|
|
ZEBRA = 'Zebra',
|
|
|
FACEBOOK = 'Facebook',
|
|
|
CHROMIUM_OS = 'Chromium OS',
|
|
|
MAC_OS = 'Mac OS';
|
|
|
|
|
|
///////////
|
|
|
// Helper
|
|
|
//////////
|
|
|
|
|
|
var extend = function (regexes, extensions) {
|
|
|
var mergedRegexes = {};
|
|
|
for (var i in regexes) {
|
|
|
if (extensions[i] && extensions[i].length % 2 === 0) {
|
|
|
mergedRegexes[i] = extensions[i].concat(regexes[i]);
|
|
|
} else {
|
|
|
mergedRegexes[i] = regexes[i];
|
|
|
}
|
|
|
}
|
|
|
return mergedRegexes;
|
|
|
},
|
|
|
enumerize = function (arr) {
|
|
|
var enums = {};
|
|
|
for (var i=0; i<arr.length; i++) {
|
|
|
enums[arr[i].toUpperCase()] = arr[i];
|
|
|
}
|
|
|
return enums;
|
|
|
},
|
|
|
has = function (str1, str2) {
|
|
|
return typeof str1 === STR_TYPE ? lowerize(str2).indexOf(lowerize(str1)) !== -1 : false;
|
|
|
},
|
|
|
lowerize = function (str) {
|
|
|
return str.toLowerCase();
|
|
|
},
|
|
|
majorize = function (version) {
|
|
|
return typeof(version) === STR_TYPE ? version.replace(/[^\d\.]/g, EMPTY).split('.')[0] : undefined;
|
|
|
},
|
|
|
trim = function (str, len) {
|
|
|
if (typeof(str) === STR_TYPE) {
|
|
|
str = str.replace(/^\s\s*/, EMPTY);
|
|
|
return typeof(len) === UNDEF_TYPE ? str : str.substring(0, UA_MAX_LENGTH);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
///////////////
|
|
|
// Map helper
|
|
|
//////////////
|
|
|
|
|
|
var rgxMapper = function (ua, arrays) {
|
|
|
|
|
|
var i = 0, j, k, p, q, matches, match;
|
|
|
|
|
|
// loop through all regexes maps
|
|
|
while (i < arrays.length && !matches) {
|
|
|
|
|
|
var regex = arrays[i], // even sequence (0,2,4,..)
|
|
|
props = arrays[i + 1]; // odd sequence (1,3,5,..)
|
|
|
j = k = 0;
|
|
|
|
|
|
// try matching uastring with regexes
|
|
|
while (j < regex.length && !matches) {
|
|
|
|
|
|
if (!regex[j]) { break; }
|
|
|
matches = regex[j++].exec(ua);
|
|
|
|
|
|
if (!!matches) {
|
|
|
for (p = 0; p < props.length; p++) {
|
|
|
match = matches[++k];
|
|
|
q = props[p];
|
|
|
// check if given property is actually array
|
|
|
if (typeof q === OBJ_TYPE && q.length > 0) {
|
|
|
if (q.length === 2) {
|
|
|
if (typeof q[1] == FUNC_TYPE) {
|
|
|
// assign modified match
|
|
|
this[q[0]] = q[1].call(this, match);
|
|
|
} else {
|
|
|
// assign given value, ignore regex match
|
|
|
this[q[0]] = q[1];
|
|
|
}
|
|
|
} else if (q.length === 3) {
|
|
|
// check whether function or regex
|
|
|
if (typeof q[1] === FUNC_TYPE && !(q[1].exec && q[1].test)) {
|
|
|
// call function (usually string mapper)
|
|
|
this[q[0]] = match ? q[1].call(this, match, q[2]) : undefined;
|
|
|
} else {
|
|
|
// sanitize match using given regex
|
|
|
this[q[0]] = match ? match.replace(q[1], q[2]) : undefined;
|
|
|
}
|
|
|
} else if (q.length === 4) {
|
|
|
this[q[0]] = match ? q[3].call(this, match.replace(q[1], q[2])) : undefined;
|
|
|
}
|
|
|
} else {
|
|
|
this[q] = match ? match : undefined;
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
i += 2;
|
|
|
}
|
|
|
},
|
|
|
|
|
|
strMapper = function (str, map) {
|
|
|
|
|
|
for (var i in map) {
|
|
|
// check if current value is array
|
|
|
if (typeof map[i] === OBJ_TYPE && map[i].length > 0) {
|
|
|
for (var j = 0; j < map[i].length; j++) {
|
|
|
if (has(map[i][j], str)) {
|
|
|
return (i === UNKNOWN) ? undefined : i;
|
|
|
}
|
|
|
}
|
|
|
} else if (has(map[i], str)) {
|
|
|
return (i === UNKNOWN) ? undefined : i;
|
|
|
}
|
|
|
}
|
|
|
return str;
|
|
|
};
|
|
|
|
|
|
///////////////
|
|
|
// String map
|
|
|
//////////////
|
|
|
|
|
|
// Safari < 3.0
|
|
|
var oldSafariMap = {
|
|
|
'1.0' : '/8',
|
|
|
'1.2' : '/1',
|
|
|
'1.3' : '/3',
|
|
|
'2.0' : '/412',
|
|
|
'2.0.2' : '/416',
|
|
|
'2.0.3' : '/417',
|
|
|
'2.0.4' : '/419',
|
|
|
'?' : '/'
|
|
|
},
|
|
|
windowsVersionMap = {
|
|
|
'ME' : '4.90',
|
|
|
'NT 3.11' : 'NT3.51',
|
|
|
'NT 4.0' : 'NT4.0',
|
|
|
'2000' : 'NT 5.0',
|
|
|
'XP' : ['NT 5.1', 'NT 5.2'],
|
|
|
'Vista' : 'NT 6.0',
|
|
|
'7' : 'NT 6.1',
|
|
|
'8' : 'NT 6.2',
|
|
|
'8.1' : 'NT 6.3',
|
|
|
'10' : ['NT 6.4', 'NT 10.0'],
|
|
|
'RT' : 'ARM'
|
|
|
};
|
|
|
|
|
|
//////////////
|
|
|
// Regex map
|
|
|
/////////////
|
|
|
|
|
|
var regexes = {
|
|
|
|
|
|
browser : [[
|
|
|
|
|
|
/\b(?:crmo|crios)\/([\w\.]+)/i // Chrome for Android/iOS
|
|
|
], [VERSION, [NAME, 'Chrome']], [
|
|
|
/edg(?:e|ios|a)?\/([\w\.]+)/i // Microsoft Edge
|
|
|
], [VERSION, [NAME, 'Edge']], [
|
|
|
|
|
|
// Presto based
|
|
|
/(opera mini)\/([-\w\.]+)/i, // Opera Mini
|
|
|
/(opera [mobiletab]{3,6})\b.+version\/([-\w\.]+)/i, // Opera Mobi/Tablet
|
|
|
/(opera)(?:.+version\/|[\/ ]+)([\w\.]+)/i // Opera
|
|
|
], [NAME, VERSION], [
|
|
|
/opios[\/ ]+([\w\.]+)/i // Opera mini on iphone >= 8.0
|
|
|
], [VERSION, [NAME, OPERA+' Mini']], [
|
|
|
/\bopr\/([\w\.]+)/i // Opera Webkit
|
|
|
], [VERSION, [NAME, OPERA]], [
|
|
|
|
|
|
// Mixed
|
|
|
/\bb[ai]*d(?:uhd|[ub]*[aekoprswx]{5,6})[\/ ]?([\w\.]+)/i // Baidu
|
|
|
], [VERSION, [NAME, 'Baidu']], [
|
|
|
/(kindle)\/([\w\.]+)/i, // Kindle
|
|
|
/(lunascape|maxthon|netfront|jasmine|blazer)[\/ ]?([\w\.]*)/i, // Lunascape/Maxthon/Netfront/Jasmine/Blazer
|
|
|
// Trident based
|
|
|
/(avant|iemobile|slim)\s?(?:browser)?[\/ ]?([\w\.]*)/i, // Avant/IEMobile/SlimBrowser
|
|
|
/(?:ms|\()(ie) ([\w\.]+)/i, // Internet Explorer
|
|
|
|
|
|
// Webkit/KHTML based // Flock/RockMelt/Midori/Epiphany/Silk/Skyfire/Bolt/Iron/Iridium/PhantomJS/Bowser/QupZilla/Falkon
|
|
|
/(flock|rockmelt|midori|epiphany|silk|skyfire|bolt|iron|vivaldi|iridium|phantomjs|bowser|quark|qupzilla|falkon|rekonq|puffin|brave|whale(?!.+naver)|qqbrowserlite|qq|duckduckgo)\/([-\w\.]+)/i,
|
|
|
// Rekonq/Puffin/Brave/Whale/QQBrowserLite/QQ, aka ShouQ
|
|
|
/(heytap|ovi)browser\/([\d\.]+)/i, // Heytap/Ovi
|
|
|
/(weibo)__([\d\.]+)/i // Weibo
|
|
|
], [NAME, VERSION], [
|
|
|
/(?:\buc? ?browser|(?:juc.+)ucweb)[\/ ]?([\w\.]+)/i // UCBrowser
|
|
|
], [VERSION, [NAME, 'UC'+BROWSER]], [
|
|
|
/microm.+\bqbcore\/([\w\.]+)/i, // WeChat Desktop for Windows Built-in Browser
|
|
|
/\bqbcore\/([\w\.]+).+microm/i,
|
|
|
/micromessenger\/([\w\.]+)/i // WeChat
|
|
|
], [VERSION, [NAME, 'WeChat']], [
|
|
|
/konqueror\/([\w\.]+)/i // Konqueror
|
|
|
], [VERSION, [NAME, 'Konqueror']], [
|
|
|
/trident.+rv[: ]([\w\.]{1,9})\b.+like gecko/i // IE11
|
|
|
], [VERSION, [NAME, 'IE']], [
|
|
|
/ya(?:search)?browser\/([\w\.]+)/i // Yandex
|
|
|
], [VERSION, [NAME, 'Yandex']], [
|
|
|
/slbrowser\/([\w\.]+)/i // Smart Lenovo Browser
|
|
|
], [VERSION, [NAME, 'Smart Lenovo '+BROWSER]], [
|
|
|
/(avast|avg)\/([\w\.]+)/i // Avast/AVG Secure Browser
|
|
|
], [[NAME, /(.+)/, '$1 Secure '+BROWSER], VERSION], [
|
|
|
/\bfocus\/([\w\.]+)/i // Firefox Focus
|
|
|
], [VERSION, [NAME, FIREFOX+' Focus']], [
|
|
|
/\bopt\/([\w\.]+)/i // Opera Touch
|
|
|
], [VERSION, [NAME, OPERA+' Touch']], [
|
|
|
/coc_coc\w+\/([\w\.]+)/i // Coc Coc Browser
|
|
|
], [VERSION, [NAME, 'Coc Coc']], [
|
|
|
/dolfin\/([\w\.]+)/i // Dolphin
|
|
|
], [VERSION, [NAME, 'Dolphin']], [
|
|
|
/coast\/([\w\.]+)/i // Opera Coast
|
|
|
], [VERSION, [NAME, OPERA+' Coast']], [
|
|
|
/miuibrowser\/([\w\.]+)/i // MIUI Browser
|
|
|
], [VERSION, [NAME, 'MIUI '+BROWSER]], [
|
|
|
/fxios\/([-\w\.]+)/i // Firefox for iOS
|
|
|
], [VERSION, [NAME, FIREFOX]], [
|
|
|
/\bqihu|(qi?ho?o?|360)browser/i // 360
|
|
|
], [[NAME, '360 ' + BROWSER]], [
|
|
|
/(oculus|sailfish|huawei|vivo)browser\/([\w\.]+)/i
|
|
|
], [[NAME, /(.+)/, '$1 ' + BROWSER], VERSION], [ // Oculus/Sailfish/HuaweiBrowser/VivoBrowser
|
|
|
/samsungbrowser\/([\w\.]+)/i // Samsung Internet
|
|
|
], [VERSION, [NAME, SAMSUNG + ' Internet']], [
|
|
|
/(comodo_dragon)\/([\w\.]+)/i // Comodo Dragon
|
|
|
], [[NAME, /_/g, ' '], VERSION], [
|
|
|
/metasr[\/ ]?([\d\.]+)/i // Sogou Explorer
|
|
|
], [VERSION, [NAME, 'Sogou Explorer']], [
|
|
|
/(sogou)mo\w+\/([\d\.]+)/i // Sogou Mobile
|
|
|
], [[NAME, 'Sogou Mobile'], VERSION], [
|
|
|
/(electron)\/([\w\.]+) safari/i, // Electron-based App
|
|
|
/(tesla)(?: qtcarbrowser|\/(20\d\d\.[-\w\.]+))/i, // Tesla
|
|
|
/m?(qqbrowser|2345Explorer)[\/ ]?([\w\.]+)/i // QQBrowser/2345 Browser
|
|
|
], [NAME, VERSION], [
|
|
|
/(lbbrowser)/i, // LieBao Browser
|
|
|
/\[(linkedin)app\]/i // LinkedIn App for iOS & Android
|
|
|
], [NAME], [
|
|
|
|
|
|
// WebView
|
|
|
/((?:fban\/fbios|fb_iab\/fb4a)(?!.+fbav)|;fbav\/([\w\.]+);)/i // Facebook App for iOS & Android
|
|
|
], [[NAME, FACEBOOK], VERSION], [
|
|
|
/(Klarna)\/([\w\.]+)/i, // Klarna Shopping Browser for iOS & Android
|
|
|
/(kakao(?:talk|story))[\/ ]([\w\.]+)/i, // Kakao App
|
|
|
/(naver)\(.*?(\d+\.[\w\.]+).*\)/i, // Naver InApp
|
|
|
/safari (line)\/([\w\.]+)/i, // Line App for iOS
|
|
|
/\b(line)\/([\w\.]+)\/iab/i, // Line App for Android
|
|
|
/(alipay)client\/([\w\.]+)/i, // Alipay
|
|
|
/(chromium|instagram|snapchat)[\/ ]([-\w\.]+)/i // Chromium/Instagram/Snapchat
|
|
|
], [NAME, VERSION], [
|
|
|
/\bgsa\/([\w\.]+) .*safari\//i // Google Search Appliance on iOS
|
|
|
], [VERSION, [NAME, 'GSA']], [
|
|
|
/musical_ly(?:.+app_?version\/|_)([\w\.]+)/i // TikTok
|
|
|
], [VERSION, [NAME, 'TikTok']], [
|
|
|
|
|
|
/headlesschrome(?:\/([\w\.]+)| )/i // Chrome Headless
|
|
|
], [VERSION, [NAME, CHROME+' Headless']], [
|
|
|
|
|
|
/ wv\).+(chrome)\/([\w\.]+)/i // Chrome WebView
|
|
|
], [[NAME, CHROME+' WebView'], VERSION], [
|
|
|
|
|
|
/droid.+ version\/([\w\.]+)\b.+(?:mobile safari|safari)/i // Android Browser
|
|
|
], [VERSION, [NAME, 'Android '+BROWSER]], [
|
|
|
|
|
|
/(chrome|omniweb|arora|[tizenoka]{5} ?browser)\/v?([\w\.]+)/i // Chrome/OmniWeb/Arora/Tizen/Nokia
|
|
|
], [NAME, VERSION], [
|
|
|
|
|
|
/version\/([\w\.\,]+) .*mobile\/\w+ (safari)/i // Mobile Safari
|
|
|
], [VERSION, [NAME, 'Mobile Safari']], [
|
|
|
/version\/([\w(\.|\,)]+) .*(mobile ?safari|safari)/i // Safari & Safari Mobile
|
|
|
], [VERSION, NAME], [
|
|
|
/webkit.+?(mobile ?safari|safari)(\/[\w\.]+)/i // Safari < 3.0
|
|
|
], [NAME, [VERSION, strMapper, oldSafariMap]], [
|
|
|
|
|
|
/(webkit|khtml)\/([\w\.]+)/i
|
|
|
], [NAME, VERSION], [
|
|
|
|
|
|
// Gecko based
|
|
|
/(navigator|netscape\d?)\/([-\w\.]+)/i // Netscape
|
|
|
], [[NAME, 'Netscape'], VERSION], [
|
|
|
/mobile vr; rv:([\w\.]+)\).+firefox/i // Firefox Reality
|
|
|
], [VERSION, [NAME, FIREFOX+' Reality']], [
|
|
|
/ekiohf.+(flow)\/([\w\.]+)/i, // Flow
|
|
|
/(swiftfox)/i, // Swiftfox
|
|
|
/(icedragon|iceweasel|camino|chimera|fennec|maemo browser|minimo|conkeror|klar)[\/ ]?([\w\.\+]+)/i,
|
|
|
// IceDragon/Iceweasel/Camino/Chimera/Fennec/Maemo/Minimo/Conkeror/Klar
|
|
|
/(seamonkey|k-meleon|icecat|iceape|firebird|phoenix|palemoon|basilisk|waterfox)\/([-\w\.]+)$/i,
|
|
|
// Firefox/SeaMonkey/K-Meleon/IceCat/IceApe/Firebird/Phoenix
|
|
|
/(firefox)\/([\w\.]+)/i, // Other Firefox-based
|
|
|
/(mozilla)\/([\w\.]+) .+rv\:.+gecko\/\d+/i, // Mozilla
|
|
|
|
|
|
// Other
|
|
|
/(polaris|lynx|dillo|icab|doris|amaya|w3m|netsurf|sleipnir|obigo|mosaic|(?:go|ice|up)[\. ]?browser)[-\/ ]?v?([\w\.]+)/i,
|
|
|
// Polaris/Lynx/Dillo/iCab/Doris/Amaya/w3m/NetSurf/Sleipnir/Obigo/Mosaic/Go/ICE/UP.Browser
|
|
|
/(links) \(([\w\.]+)/i, // Links
|
|
|
/panasonic;(viera)/i // Panasonic Viera
|
|
|
], [NAME, VERSION], [
|
|
|
|
|
|
/(cobalt)\/([\w\.]+)/i // Cobalt
|
|
|
], [NAME, [VERSION, /master.|lts./, ""]]
|
|
|
],
|
|
|
|
|
|
cpu : [[
|
|
|
|
|
|
/(?:(amd|x(?:(?:86|64)[-_])?|wow|win)64)[;\)]/i // AMD64 (x64)
|
|
|
], [[ARCHITECTURE, 'amd64']], [
|
|
|
|
|
|
/(ia32(?=;))/i // IA32 (quicktime)
|
|
|
], [[ARCHITECTURE, lowerize]], [
|
|
|
|
|
|
/((?:i[346]|x)86)[;\)]/i // IA32 (x86)
|
|
|
], [[ARCHITECTURE, 'ia32']], [
|
|
|
|
|
|
/\b(aarch64|arm(v?8e?l?|_?64))\b/i // ARM64
|
|
|
], [[ARCHITECTURE, 'arm64']], [
|
|
|
|
|
|
/\b(arm(?:v[67])?ht?n?[fl]p?)\b/i // ARMHF
|
|
|
], [[ARCHITECTURE, 'armhf']], [
|
|
|
|
|
|
// PocketPC mistakenly identified as PowerPC
|
|
|
/windows (ce|mobile); ppc;/i
|
|
|
], [[ARCHITECTURE, 'arm']], [
|
|
|
|
|
|
/((?:ppc|powerpc)(?:64)?)(?: mac|;|\))/i // PowerPC
|
|
|
], [[ARCHITECTURE, /ower/, EMPTY, lowerize]], [
|
|
|
|
|
|
/(sun4\w)[;\)]/i // SPARC
|
|
|
], [[ARCHITECTURE, 'sparc']], [
|
|
|
|
|
|
/((?:avr32|ia64(?=;))|68k(?=\))|\barm(?=v(?:[1-7]|[5-7]1)l?|;|eabi)|(?=atmel )avr|(?:irix|mips|sparc)(?:64)?\b|pa-risc)/i
|
|
|
// IA64, 68K, ARM/64, AVR/32, IRIX/64, MIPS/64, SPARC/64, PA-RISC
|
|
|
], [[ARCHITECTURE, lowerize]]
|
|
|
],
|
|
|
|
|
|
device : [[
|
|
|
|
|
|
//////////////////////////
|
|
|
// MOBILES & TABLETS
|
|
|
/////////////////////////
|
|
|
|
|
|
// Samsung
|
|
|
/\b(sch-i[89]0\d|shw-m380s|sm-[ptx]\w{2,4}|gt-[pn]\d{2,4}|sgh-t8[56]9|nexus 10)/i
|
|
|
], [MODEL, [VENDOR, SAMSUNG], [TYPE, TABLET]], [
|
|
|
/\b((?:s[cgp]h|gt|sm)-\w+|sc[g-]?[\d]+a?|galaxy nexus)/i,
|
|
|
/samsung[- ]([-\w]+)/i,
|
|
|
/sec-(sgh\w+)/i
|
|
|
], [MODEL, [VENDOR, SAMSUNG], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Apple
|
|
|
/(?:\/|\()(ip(?:hone|od)[\w, ]*)(?:\/|;)/i // iPod/iPhone
|
|
|
], [MODEL, [VENDOR, APPLE], [TYPE, MOBILE]], [
|
|
|
/\((ipad);[-\w\),; ]+apple/i, // iPad
|
|
|
/applecoremedia\/[\w\.]+ \((ipad)/i,
|
|
|
/\b(ipad)\d\d?,\d\d?[;\]].+ios/i
|
|
|
], [MODEL, [VENDOR, APPLE], [TYPE, TABLET]], [
|
|
|
/(macintosh);/i
|
|
|
], [MODEL, [VENDOR, APPLE]], [
|
|
|
|
|
|
// Sharp
|
|
|
/\b(sh-?[altvz]?\d\d[a-ekm]?)/i
|
|
|
], [MODEL, [VENDOR, SHARP], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Huawei
|
|
|
/\b((?:ag[rs][23]?|bah2?|sht?|btv)-a?[lw]\d{2})\b(?!.+d\/s)/i
|
|
|
], [MODEL, [VENDOR, HUAWEI], [TYPE, TABLET]], [
|
|
|
/(?:huawei|honor)([-\w ]+)[;\)]/i,
|
|
|
/\b(nexus 6p|\w{2,4}e?-[atu]?[ln][\dx][012359c][adn]?)\b(?!.+d\/s)/i
|
|
|
], [MODEL, [VENDOR, HUAWEI], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Xiaomi
|
|
|
/\b(poco[\w ]+|m2\d{3}j\d\d[a-z]{2})(?: bui|\))/i, // Xiaomi POCO
|
|
|
/\b; (\w+) build\/hm\1/i, // Xiaomi Hongmi 'numeric' models
|
|
|
/\b(hm[-_ ]?note?[_ ]?(?:\d\w)?) bui/i, // Xiaomi Hongmi
|
|
|
/\b(redmi[\-_ ]?(?:note|k)?[\w_ ]+)(?: bui|\))/i, // Xiaomi Redmi
|
|
|
/oid[^\)]+; (m?[12][0-389][01]\w{3,6}[c-y])( bui|; wv|\))/i, // Xiaomi Redmi 'numeric' models
|
|
|
/\b(mi[-_ ]?(?:a\d|one|one[_ ]plus|note lte|max|cc)?[_ ]?(?:\d?\w?)[_ ]?(?:plus|se|lite)?)(?: bui|\))/i // Xiaomi Mi
|
|
|
], [[MODEL, /_/g, ' '], [VENDOR, XIAOMI], [TYPE, MOBILE]], [
|
|
|
/oid[^\)]+; (2\d{4}(283|rpbf)[cgl])( bui|\))/i, // Redmi Pad
|
|
|
/\b(mi[-_ ]?(?:pad)(?:[\w_ ]+))(?: bui|\))/i // Mi Pad tablets
|
|
|
],[[MODEL, /_/g, ' '], [VENDOR, XIAOMI], [TYPE, TABLET]], [
|
|
|
|
|
|
// OPPO
|
|
|
/; (\w+) bui.+ oppo/i,
|
|
|
/\b(cph[12]\d{3}|p(?:af|c[al]|d\w|e[ar])[mt]\d0|x9007|a101op)\b/i
|
|
|
], [MODEL, [VENDOR, 'OPPO'], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Vivo
|
|
|
/vivo (\w+)(?: bui|\))/i,
|
|
|
/\b(v[12]\d{3}\w?[at])(?: bui|;)/i
|
|
|
], [MODEL, [VENDOR, 'Vivo'], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Realme
|
|
|
/\b(rmx[1-3]\d{3})(?: bui|;|\))/i
|
|
|
], [MODEL, [VENDOR, 'Realme'], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Motorola
|
|
|
/\b(milestone|droid(?:[2-4x]| (?:bionic|x2|pro|razr))?:?( 4g)?)\b[\w ]+build\//i,
|
|
|
/\bmot(?:orola)?[- ](\w*)/i,
|
|
|
/((?:moto[\w\(\) ]+|xt\d{3,4}|nexus 6)(?= bui|\)))/i
|
|
|
], [MODEL, [VENDOR, MOTOROLA], [TYPE, MOBILE]], [
|
|
|
/\b(mz60\d|xoom[2 ]{0,2}) build\//i
|
|
|
], [MODEL, [VENDOR, MOTOROLA], [TYPE, TABLET]], [
|
|
|
|
|
|
// LG
|
|
|
/((?=lg)?[vl]k\-?\d{3}) bui| 3\.[-\w; ]{10}lg?-([06cv9]{3,4})/i
|
|
|
], [MODEL, [VENDOR, LG], [TYPE, TABLET]], [
|
|
|
/(lm(?:-?f100[nv]?|-[\w\.]+)(?= bui|\))|nexus [45])/i,
|
|
|
/\blg[-e;\/ ]+((?!browser|netcast|android tv)\w+)/i,
|
|
|
/\blg-?([\d\w]+) bui/i
|
|
|
], [MODEL, [VENDOR, LG], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Lenovo
|
|
|
/(ideatab[-\w ]+)/i,
|
|
|
/lenovo ?(s[56]000[-\w]+|tab(?:[\w ]+)|yt[-\d\w]{6}|tb[-\d\w]{6})/i
|
|
|
], [MODEL, [VENDOR, 'Lenovo'], [TYPE, TABLET]], [
|
|
|
|
|
|
// Nokia
|
|
|
/(?:maemo|nokia).*(n900|lumia \d+)/i,
|
|
|
/nokia[-_ ]?([-\w\.]*)/i
|
|
|
], [[MODEL, /_/g, ' '], [VENDOR, 'Nokia'], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Google
|
|
|
/(pixel c)\b/i // Google Pixel C
|
|
|
], [MODEL, [VENDOR, GOOGLE], [TYPE, TABLET]], [
|
|
|
/droid.+; (pixel[\daxl ]{0,6})(?: bui|\))/i // Google Pixel
|
|
|
], [MODEL, [VENDOR, GOOGLE], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Sony
|
|
|
/droid.+ (a?\d[0-2]{2}so|[c-g]\d{4}|so[-gl]\w+|xq-a\w[4-7][12])(?= bui|\).+chrome\/(?![1-6]{0,1}\d\.))/i
|
|
|
], [MODEL, [VENDOR, SONY], [TYPE, MOBILE]], [
|
|
|
/sony tablet [ps]/i,
|
|
|
/\b(?:sony)?sgp\w+(?: bui|\))/i
|
|
|
], [[MODEL, 'Xperia Tablet'], [VENDOR, SONY], [TYPE, TABLET]], [
|
|
|
|
|
|
// OnePlus
|
|
|
/ (kb2005|in20[12]5|be20[12][59])\b/i,
|
|
|
/(?:one)?(?:plus)? (a\d0\d\d)(?: b|\))/i
|
|
|
], [MODEL, [VENDOR, 'OnePlus'], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Amazon
|
|
|
/(alexa)webm/i,
|
|
|
/(kf[a-z]{2}wi|aeo[c-r]{2})( bui|\))/i, // Kindle Fire without Silk / Echo Show
|
|
|
/(kf[a-z]+)( bui|\)).+silk\//i // Kindle Fire HD
|
|
|
], [MODEL, [VENDOR, AMAZON], [TYPE, TABLET]], [
|
|
|
/((?:sd|kf)[0349hijorstuw]+)( bui|\)).+silk\//i // Fire Phone
|
|
|
], [[MODEL, /(.+)/g, 'Fire Phone $1'], [VENDOR, AMAZON], [TYPE, MOBILE]], [
|
|
|
|
|
|
// BlackBerry
|
|
|
/(playbook);[-\w\),; ]+(rim)/i // BlackBerry PlayBook
|
|
|
], [MODEL, VENDOR, [TYPE, TABLET]], [
|
|
|
/\b((?:bb[a-f]|st[hv])100-\d)/i,
|
|
|
/\(bb10; (\w+)/i // BlackBerry 10
|
|
|
], [MODEL, [VENDOR, BLACKBERRY], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Asus
|
|
|
/(?:\b|asus_)(transfo[prime ]{4,10} \w+|eeepc|slider \w+|nexus 7|padfone|p00[cj])/i
|
|
|
], [MODEL, [VENDOR, ASUS], [TYPE, TABLET]], [
|
|
|
/ (z[bes]6[027][012][km][ls]|zenfone \d\w?)\b/i
|
|
|
], [MODEL, [VENDOR, ASUS], [TYPE, MOBILE]], [
|
|
|
|
|
|
// HTC
|
|
|
/(nexus 9)/i // HTC Nexus 9
|
|
|
], [MODEL, [VENDOR, 'HTC'], [TYPE, TABLET]], [
|
|
|
/(htc)[-;_ ]{1,2}([\w ]+(?=\)| bui)|\w+)/i, // HTC
|
|
|
|
|
|
// ZTE
|
|
|
/(zte)[- ]([\w ]+?)(?: bui|\/|\))/i,
|
|
|
/(alcatel|geeksphone|nexian|panasonic(?!(?:;|\.))|sony(?!-bra))[-_ ]?([-\w]*)/i // Alcatel/GeeksPhone/Nexian/Panasonic/Sony
|
|
|
], [VENDOR, [MODEL, /_/g, ' '], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Acer
|
|
|
/droid.+; ([ab][1-7]-?[0178a]\d\d?)/i
|
|
|
], [MODEL, [VENDOR, 'Acer'], [TYPE, TABLET]], [
|
|
|
|
|
|
// Meizu
|
|
|
/droid.+; (m[1-5] note) bui/i,
|
|
|
/\bmz-([-\w]{2,})/i
|
|
|
], [MODEL, [VENDOR, 'Meizu'], [TYPE, MOBILE]], [
|
|
|
|
|
|
// Ulefone
|
|
|
/; ((?:power )?armor(?:[\w ]{0,8}))(?: bui|\))/i
|
|
|
], [MODEL, [VENDOR, 'Ulefone'], [TYPE, MOBILE]], [
|
|
|
|
|
|
// MIXED
|
|
|
/(blackberry|benq|palm(?=\-)|sonyericsson|acer|asus|dell|meizu|motorola|polytron|infinix|tecno)[-_ ]?([-\w]*)/i,
|
|
|
// BlackBerry/BenQ/Palm/Sony-Ericsson/Acer/Asus/Dell/Meizu/Motorola/Polytron
|
|
|
/(hp) ([\w ]+\w)/i, // HP iPAQ
|
|
|
/(asus)-?(\w+)/i, // Asus
|
|
|
/(microsoft); (lumia[\w ]+)/i, // Microsoft Lumia
|
|
|
/(lenovo)[-_ ]?([-\w]+)/i, // Lenovo
|
|
|
/(jolla)/i, // Jolla
|
|
|
/(oppo) ?([\w ]+) bui/i // OPPO
|
|
|
], [VENDOR, MODEL, [TYPE, MOBILE]], [
|
|
|
|
|
|
/(kobo)\s(ereader|touch)/i, // Kobo
|
|
|
/(archos) (gamepad2?)/i, // Archos
|
|
|
/(hp).+(touchpad(?!.+tablet)|tablet)/i, // HP TouchPad
|
|
|
/(kindle)\/([\w\.]+)/i, // Kindle
|
|
|
/(nook)[\w ]+build\/(\w+)/i, // Nook
|
|
|
/(dell) (strea[kpr\d ]*[\dko])/i, // Dell Streak
|
|
|
/(le[- ]+pan)[- ]+(\w{1,9}) bui/i, // Le Pan Tablets
|
|
|
/(trinity)[- ]*(t\d{3}) bui/i, // Trinity Tablets
|
|
|
/(gigaset)[- ]+(q\w{1,9}) bui/i, // Gigaset Tablets
|
|
|
/(vodafone) ([\w ]+)(?:\)| bui)/i // Vodafone
|
|
|
], [VENDOR, MODEL, [TYPE, TABLET]], [
|
|
|
|
|
|
/(surface duo)/i // Surface Duo
|
|
|
], [MODEL, [VENDOR, MICROSOFT], [TYPE, TABLET]], [
|
|
|
/droid [\d\.]+; (fp\du?)(?: b|\))/i // Fairphone
|
|
|
], [MODEL, [VENDOR, 'Fairphone'], [TYPE, MOBILE]], [
|
|
|
/(u304aa)/i // AT&T
|
|
|
], [MODEL, [VENDOR, 'AT&T'], [TYPE, MOBILE]], [
|
|
|
/\bsie-(\w*)/i // Siemens
|
|
|
], [MODEL, [VENDOR, 'Siemens'], [TYPE, MOBILE]], [
|
|
|
/\b(rct\w+) b/i // RCA Tablets
|
|
|
], [MODEL, [VENDOR, 'RCA'], [TYPE, TABLET]], [
|
|
|
/\b(venue[\d ]{2,7}) b/i // Dell Venue Tablets
|
|
|
], [MODEL, [VENDOR, 'Dell'], [TYPE, TABLET]], [
|
|
|
/\b(q(?:mv|ta)\w+) b/i // Verizon Tablet
|
|
|
], [MODEL, [VENDOR, 'Verizon'], [TYPE, TABLET]], [
|
|
|
/\b(?:barnes[& ]+noble |bn[rt])([\w\+ ]*) b/i // Barnes & Noble Tablet
|
|
|
], [MODEL, [VENDOR, 'Barnes & Noble'], [TYPE, TABLET]], [
|
|
|
/\b(tm\d{3}\w+) b/i
|
|
|
], [MODEL, [VENDOR, 'NuVision'], [TYPE, TABLET]], [
|
|
|
/\b(k88) b/i // ZTE K Series Tablet
|
|
|
], [MODEL, [VENDOR, 'ZTE'], [TYPE, TABLET]], [
|
|
|
/\b(nx\d{3}j) b/i // ZTE Nubia
|
|
|
], [MODEL, [VENDOR, 'ZTE'], [TYPE, MOBILE]], [
|
|
|
/\b(gen\d{3}) b.+49h/i // Swiss GEN Mobile
|
|
|
], [MODEL, [VENDOR, 'Swiss'], [TYPE, MOBILE]], [
|
|
|
/\b(zur\d{3}) b/i // Swiss ZUR Tablet
|
|
|
], [MODEL, [VENDOR, 'Swiss'], [TYPE, TABLET]], [
|
|
|
/\b((zeki)?tb.*\b) b/i // Zeki Tablets
|
|
|
], [MODEL, [VENDOR, 'Zeki'], [TYPE, TABLET]], [
|
|
|
/\b([yr]\d{2}) b/i,
|
|
|
/\b(dragon[- ]+touch |dt)(\w{5}) b/i // Dragon Touch Tablet
|
|
|
], [[VENDOR, 'Dragon Touch'], MODEL, [TYPE, TABLET]], [
|
|
|
/\b(ns-?\w{0,9}) b/i // Insignia Tablets
|
|
|
], [MODEL, [VENDOR, 'Insignia'], [TYPE, TABLET]], [
|
|
|
/\b((nxa|next)-?\w{0,9}) b/i // NextBook Tablets
|
|
|
], [MODEL, [VENDOR, 'NextBook'], [TYPE, TABLET]], [
|
|
|
/\b(xtreme\_)?(v(1[045]|2[015]|[3469]0|7[05])) b/i // Voice Xtreme Phones
|
|
|
], [[VENDOR, 'Voice'], MODEL, [TYPE, MOBILE]], [
|
|
|
/\b(lvtel\-)?(v1[12]) b/i // LvTel Phones
|
|
|
], [[VENDOR, 'LvTel'], MODEL, [TYPE, MOBILE]], [
|
|
|
/\b(ph-1) /i // Essential PH-1
|
|
|
], [MODEL, [VENDOR, 'Essential'], [TYPE, MOBILE]], [
|
|
|
/\b(v(100md|700na|7011|917g).*\b) b/i // Envizen Tablets
|
|
|
], [MODEL, [VENDOR, 'Envizen'], [TYPE, TABLET]], [
|
|
|
/\b(trio[-\w\. ]+) b/i // MachSpeed Tablets
|
|
|
], [MODEL, [VENDOR, 'MachSpeed'], [TYPE, TABLET]], [
|
|
|
/\btu_(1491) b/i // Rotor Tablets
|
|
|
], [MODEL, [VENDOR, 'Rotor'], [TYPE, TABLET]], [
|
|
|
/(shield[\w ]+) b/i // Nvidia Shield Tablets
|
|
|
], [MODEL, [VENDOR, 'Nvidia'], [TYPE, TABLET]], [
|
|
|
/(sprint) (\w+)/i // Sprint Phones
|
|
|
], [VENDOR, MODEL, [TYPE, MOBILE]], [
|
|
|
/(kin\.[onetw]{3})/i // Microsoft Kin
|
|
|
], [[MODEL, /\./g, ' '], [VENDOR, MICROSOFT], [TYPE, MOBILE]], [
|
|
|
/droid.+; (cc6666?|et5[16]|mc[239][23]x?|vc8[03]x?)\)/i // Zebra
|
|
|
], [MODEL, [VENDOR, ZEBRA], [TYPE, TABLET]], [
|
|
|
/droid.+; (ec30|ps20|tc[2-8]\d[kx])\)/i
|
|
|
], [MODEL, [VENDOR, ZEBRA], [TYPE, MOBILE]], [
|
|
|
|
|
|
///////////////////
|
|
|
// SMARTTVS
|
|
|
///////////////////
|
|
|
|
|
|
/smart-tv.+(samsung)/i // Samsung
|
|
|
], [VENDOR, [TYPE, SMARTTV]], [
|
|
|
/hbbtv.+maple;(\d+)/i
|
|
|
], [[MODEL, /^/, 'SmartTV'], [VENDOR, SAMSUNG], [TYPE, SMARTTV]], [
|
|
|
/(nux; netcast.+smarttv|lg (netcast\.tv-201\d|android tv))/i // LG SmartTV
|
|
|
], [[VENDOR, LG], [TYPE, SMARTTV]], [
|
|
|
/(apple) ?tv/i // Apple TV
|
|
|
], [VENDOR, [MODEL, APPLE+' TV'], [TYPE, SMARTTV]], [
|
|
|
/crkey/i // Google Chromecast
|
|
|
], [[MODEL, CHROME+'cast'], [VENDOR, GOOGLE], [TYPE, SMARTTV]], [
|
|
|
/droid.+aft(\w+)( bui|\))/i // Fire TV
|
|
|
], [MODEL, [VENDOR, AMAZON], [TYPE, SMARTTV]], [
|
|
|
/\(dtv[\);].+(aquos)/i,
|
|
|
/(aquos-tv[\w ]+)\)/i // Sharp
|
|
|
], [MODEL, [VENDOR, SHARP], [TYPE, SMARTTV]],[
|
|
|
/(bravia[\w ]+)( bui|\))/i // Sony
|
|
|
], [MODEL, [VENDOR, SONY], [TYPE, SMARTTV]], [
|
|
|
/(mitv-\w{5}) bui/i // Xiaomi
|
|
|
], [MODEL, [VENDOR, XIAOMI], [TYPE, SMARTTV]], [
|
|
|
/Hbbtv.*(technisat) (.*);/i // TechniSAT
|
|
|
], [VENDOR, MODEL, [TYPE, SMARTTV]], [
|
|
|
/\b(roku)[\dx]*[\)\/]((?:dvp-)?[\d\.]*)/i, // Roku
|
|
|
/hbbtv\/\d+\.\d+\.\d+ +\([\w\+ ]*; *([\w\d][^;]*);([^;]*)/i // HbbTV devices
|
|
|
], [[VENDOR, trim], [MODEL, trim], [TYPE, SMARTTV]], [
|
|
|
/\b(android tv|smart[- ]?tv|opera tv|tv; rv:)\b/i // SmartTV from Unidentified Vendors
|
|
|
], [[TYPE, SMARTTV]], [
|
|
|
|
|
|
///////////////////
|
|
|
// CONSOLES
|
|
|
///////////////////
|
|
|
|
|
|
/(ouya)/i, // Ouya
|
|
|
/(nintendo) ([wids3utch]+)/i // Nintendo
|
|
|
], [VENDOR, MODEL, [TYPE, CONSOLE]], [
|
|
|
/droid.+; (shield) bui/i // Nvidia
|
|
|
], [MODEL, [VENDOR, 'Nvidia'], [TYPE, CONSOLE]], [
|
|
|
/(playstation [345portablevi]+)/i // Playstation
|
|
|
], [MODEL, [VENDOR, SONY], [TYPE, CONSOLE]], [
|
|
|
/\b(xbox(?: one)?(?!; xbox))[\); ]/i // Microsoft Xbox
|
|
|
], [MODEL, [VENDOR, MICROSOFT], [TYPE, CONSOLE]], [
|
|
|
|
|
|
///////////////////
|
|
|
// WEARABLES
|
|
|
///////////////////
|
|
|
|
|
|
/((pebble))app/i // Pebble
|
|
|
], [VENDOR, MODEL, [TYPE, WEARABLE]], [
|
|
|
/(watch)(?: ?os[,\/]|\d,\d\/)[\d\.]+/i // Apple Watch
|
|
|
], [MODEL, [VENDOR, APPLE], [TYPE, WEARABLE]], [
|
|
|
/droid.+; (glass) \d/i // Google Glass
|
|
|
], [MODEL, [VENDOR, GOOGLE], [TYPE, WEARABLE]], [
|
|
|
/droid.+; (wt63?0{2,3})\)/i
|
|
|
], [MODEL, [VENDOR, ZEBRA], [TYPE, WEARABLE]], [
|
|
|
/(quest( 2| pro)?)/i // Oculus Quest
|
|
|
], [MODEL, [VENDOR, FACEBOOK], [TYPE, WEARABLE]], [
|
|
|
|
|
|
///////////////////
|
|
|
// EMBEDDED
|
|
|
///////////////////
|
|
|
|
|
|
/(tesla)(?: qtcarbrowser|\/[-\w\.]+)/i // Tesla
|
|
|
], [VENDOR, [TYPE, EMBEDDED]], [
|
|
|
/(aeobc)\b/i // Echo Dot
|
|
|
], [MODEL, [VENDOR, AMAZON], [TYPE, EMBEDDED]], [
|
|
|
|
|
|
////////////////////
|
|
|
// MIXED (GENERIC)
|
|
|
///////////////////
|
|
|
|
|
|
/droid .+?; ([^;]+?)(?: bui|; wv\)|\) applew).+? mobile safari/i // Android Phones from Unidentified Vendors
|
|
|
], [MODEL, [TYPE, MOBILE]], [
|
|
|
/droid .+?; ([^;]+?)(?: bui|\) applew).+?(?! mobile) safari/i // Android Tablets from Unidentified Vendors
|
|
|
], [MODEL, [TYPE, TABLET]], [
|
|
|
/\b((tablet|tab)[;\/]|focus\/\d(?!.+mobile))/i // Unidentifiable Tablet
|
|
|
], [[TYPE, TABLET]], [
|
|
|
/(phone|mobile(?:[;\/]| [ \w\/\.]*safari)|pda(?=.+windows ce))/i // Unidentifiable Mobile
|
|
|
], [[TYPE, MOBILE]], [
|
|
|
/(android[-\w\. ]{0,9});.+buil/i // Generic Android Device
|
|
|
], [MODEL, [VENDOR, 'Generic']]
|
|
|
],
|
|
|
|
|
|
engine : [[
|
|
|
|
|
|
/windows.+ edge\/([\w\.]+)/i // EdgeHTML
|
|
|
], [VERSION, [NAME, EDGE+'HTML']], [
|
|
|
|
|
|
/webkit\/537\.36.+chrome\/(?!27)([\w\.]+)/i // Blink
|
|
|
], [VERSION, [NAME, 'Blink']], [
|
|
|
|
|
|
/(presto)\/([\w\.]+)/i, // Presto
|
|
|
/(webkit|trident|netfront|netsurf|amaya|lynx|w3m|goanna)\/([\w\.]+)/i, // WebKit/Trident/NetFront/NetSurf/Amaya/Lynx/w3m/Goanna
|
|
|
/ekioh(flow)\/([\w\.]+)/i, // Flow
|
|
|
/(khtml|tasman|links)[\/ ]\(?([\w\.]+)/i, // KHTML/Tasman/Links
|
|
|
/(icab)[\/ ]([23]\.[\d\.]+)/i, // iCab
|
|
|
/\b(libweb)/i
|
|
|
], [NAME, VERSION], [
|
|
|
|
|
|
/rv\:([\w\.]{1,9})\b.+(gecko)/i // Gecko
|
|
|
], [VERSION, NAME]
|
|
|
],
|
|
|
|
|
|
os : [[
|
|
|
|
|
|
// Windows
|
|
|
/microsoft (windows) (vista|xp)/i // Windows (iTunes)
|
|
|
], [NAME, VERSION], [
|
|
|
/(windows (?:phone(?: os)?|mobile))[\/ ]?([\d\.\w ]*)/i // Windows Phone
|
|
|
], [NAME, [VERSION, strMapper, windowsVersionMap]], [
|
|
|
/windows nt 6\.2; (arm)/i, // Windows RT
|
|
|
/windows[\/ ]?([ntce\d\. ]+\w)(?!.+xbox)/i,
|
|
|
/(?:win(?=3|9|n)|win 9x )([nt\d\.]+)/i
|
|
|
], [[VERSION, strMapper, windowsVersionMap], [NAME, 'Windows']], [
|
|
|
|
|
|
// iOS/macOS
|
|
|
/ip[honead]{2,4}\b(?:.*os ([\w]+) like mac|; opera)/i, // iOS
|
|
|
/(?:ios;fbsv\/|iphone.+ios[\/ ])([\d\.]+)/i,
|
|
|
/cfnetwork\/.+darwin/i
|
|
|
], [[VERSION, /_/g, '.'], [NAME, 'iOS']], [
|
|
|
/(mac os x) ?([\w\. ]*)/i,
|
|
|
/(macintosh|mac_powerpc\b)(?!.+haiku)/i // Mac OS
|
|
|
], [[NAME, MAC_OS], [VERSION, /_/g, '.']], [
|
|
|
|
|
|
// Mobile OSes
|
|
|
/droid ([\w\.]+)\b.+(android[- ]x86|harmonyos)/i // Android-x86/HarmonyOS
|
|
|
], [VERSION, NAME], [ // Android/WebOS/QNX/Bada/RIM/Maemo/MeeGo/Sailfish OS
|
|
|
/(android|webos|qnx|bada|rim tablet os|maemo|meego|sailfish)[-\/ ]?([\w\.]*)/i,
|
|
|
/(blackberry)\w*\/([\w\.]*)/i, // Blackberry
|
|
|
/(tizen|kaios)[\/ ]([\w\.]+)/i, // Tizen/KaiOS
|
|
|
/\((series40);/i // Series 40
|
|
|
], [NAME, VERSION], [
|
|
|
/\(bb(10);/i // BlackBerry 10
|
|
|
], [VERSION, [NAME, BLACKBERRY]], [
|
|
|
/(?:symbian ?os|symbos|s60(?=;)|series60)[-\/ ]?([\w\.]*)/i // Symbian
|
|
|
], [VERSION, [NAME, 'Symbian']], [
|
|
|
/mozilla\/[\d\.]+ \((?:mobile|tablet|tv|mobile; [\w ]+); rv:.+ gecko\/([\w\.]+)/i // Firefox OS
|
|
|
], [VERSION, [NAME, FIREFOX+' OS']], [
|
|
|
/web0s;.+rt(tv)/i,
|
|
|
/\b(?:hp)?wos(?:browser)?\/([\w\.]+)/i // WebOS
|
|
|
], [VERSION, [NAME, 'webOS']], [
|
|
|
/watch(?: ?os[,\/]|\d,\d\/)([\d\.]+)/i // watchOS
|
|
|
], [VERSION, [NAME, 'watchOS']], [
|
|
|
|
|
|
// Google Chromecast
|
|
|
/crkey\/([\d\.]+)/i // Google Chromecast
|
|
|
], [VERSION, [NAME, CHROME+'cast']], [
|
|
|
/(cros) [\w]+(?:\)| ([\w\.]+)\b)/i // Chromium OS
|
|
|
], [[NAME, CHROMIUM_OS], VERSION],[
|
|
|
|
|
|
// Smart TVs
|
|
|
/panasonic;(viera)/i, // Panasonic Viera
|
|
|
/(netrange)mmh/i, // Netrange
|
|
|
/(nettv)\/(\d+\.[\w\.]+)/i, // NetTV
|
|
|
|
|
|
// Console
|
|
|
/(nintendo|playstation) ([wids345portablevuch]+)/i, // Nintendo/Playstation
|
|
|
/(xbox); +xbox ([^\);]+)/i, // Microsoft Xbox (360, One, X, S, Series X, Series S)
|
|
|
|
|
|
// Other
|
|
|
/\b(joli|palm)\b ?(?:os)?\/?([\w\.]*)/i, // Joli/Palm
|
|
|
/(mint)[\/\(\) ]?(\w*)/i, // Mint
|
|
|
/(mageia|vectorlinux)[; ]/i, // Mageia/VectorLinux
|
|
|
/([kxln]?ubuntu|debian|suse|opensuse|gentoo|arch(?= linux)|slackware|fedora|mandriva|centos|pclinuxos|red ?hat|zenwalk|linpus|raspbian|plan 9|minix|risc os|contiki|deepin|manjaro|elementary os|sabayon|linspire)(?: gnu\/linux)?(?: enterprise)?(?:[- ]linux)?(?:-gnu)?[-\/ ]?(?!chrom|package)([-\w\.]*)/i,
|
|
|
// Ubuntu/Debian/SUSE/Gentoo/Arch/Slackware/Fedora/Mandriva/CentOS/PCLinuxOS/RedHat/Zenwalk/Linpus/Raspbian/Plan9/Minix/RISCOS/Contiki/Deepin/Manjaro/elementary/Sabayon/Linspire
|
|
|
/(hurd|linux) ?([\w\.]*)/i, // Hurd/Linux
|
|
|
/(gnu) ?([\w\.]*)/i, // GNU
|
|
|
/\b([-frentopcghs]{0,5}bsd|dragonfly)[\/ ]?(?!amd|[ix346]{1,2}86)([\w\.]*)/i, // FreeBSD/NetBSD/OpenBSD/PC-BSD/GhostBSD/DragonFly
|
|
|
/(haiku) (\w+)/i // Haiku
|
|
|
], [NAME, VERSION], [
|
|
|
/(sunos) ?([\w\.\d]*)/i // Solaris
|
|
|
], [[NAME, 'Solaris'], VERSION], [
|
|
|
/((?:open)?solaris)[-\/ ]?([\w\.]*)/i, // Solaris
|
|
|
/(aix) ((\d)(?=\.|\)| )[\w\.])*/i, // AIX
|
|
|
/\b(beos|os\/2|amigaos|morphos|openvms|fuchsia|hp-ux|serenityos)/i, // BeOS/OS2/AmigaOS/MorphOS/OpenVMS/Fuchsia/HP-UX/SerenityOS
|
|
|
/(unix) ?([\w\.]*)/i // UNIX
|
|
|
], [NAME, VERSION]
|
|
|
]
|
|
|
};
|
|
|
|
|
|
/////////////////
|
|
|
// Constructor
|
|
|
////////////////
|
|
|
|
|
|
var UAParser = function (ua, extensions) {
|
|
|
|
|
|
if (typeof ua === OBJ_TYPE) {
|
|
|
extensions = ua;
|
|
|
ua = undefined;
|
|
|
}
|
|
|
|
|
|
if (!(this instanceof UAParser)) {
|
|
|
return new UAParser(ua, extensions).getResult();
|
|
|
}
|
|
|
|
|
|
var _navigator = (typeof window !== UNDEF_TYPE && window.navigator) ? window.navigator : undefined;
|
|
|
var _ua = ua || ((_navigator && _navigator.userAgent) ? _navigator.userAgent : EMPTY);
|
|
|
var _uach = (_navigator && _navigator.userAgentData) ? _navigator.userAgentData : undefined;
|
|
|
var _rgxmap = extensions ? extend(regexes, extensions) : regexes;
|
|
|
var _isSelfNav = _navigator && _navigator.userAgent == _ua;
|
|
|
|
|
|
this.getBrowser = function () {
|
|
|
var _browser = {};
|
|
|
_browser[NAME] = undefined;
|
|
|
_browser[VERSION] = undefined;
|
|
|
rgxMapper.call(_browser, _ua, _rgxmap.browser);
|
|
|
_browser[MAJOR] = majorize(_browser[VERSION]);
|
|
|
// Brave-specific detection
|
|
|
if (_isSelfNav && _navigator && _navigator.brave && typeof _navigator.brave.isBrave == FUNC_TYPE) {
|
|
|
_browser[NAME] = 'Brave';
|
|
|
}
|
|
|
return _browser;
|
|
|
};
|
|
|
this.getCPU = function () {
|
|
|
var _cpu = {};
|
|
|
_cpu[ARCHITECTURE] = undefined;
|
|
|
rgxMapper.call(_cpu, _ua, _rgxmap.cpu);
|
|
|
return _cpu;
|
|
|
};
|
|
|
this.getDevice = function () {
|
|
|
var _device = {};
|
|
|
_device[VENDOR] = undefined;
|
|
|
_device[MODEL] = undefined;
|
|
|
_device[TYPE] = undefined;
|
|
|
rgxMapper.call(_device, _ua, _rgxmap.device);
|
|
|
if (_isSelfNav && !_device[TYPE] && _uach && _uach.mobile) {
|
|
|
_device[TYPE] = MOBILE;
|
|
|
}
|
|
|
// iPadOS-specific detection: identified as Mac, but has some iOS-only properties
|
|
|
if (_isSelfNav && _device[MODEL] == 'Macintosh' && _navigator && typeof _navigator.standalone !== UNDEF_TYPE && _navigator.maxTouchPoints && _navigator.maxTouchPoints > 2) {
|
|
|
_device[MODEL] = 'iPad';
|
|
|
_device[TYPE] = TABLET;
|
|
|
}
|
|
|
return _device;
|
|
|
};
|
|
|
this.getEngine = function () {
|
|
|
var _engine = {};
|
|
|
_engine[NAME] = undefined;
|
|
|
_engine[VERSION] = undefined;
|
|
|
rgxMapper.call(_engine, _ua, _rgxmap.engine);
|
|
|
return _engine;
|
|
|
};
|
|
|
this.getOS = function () {
|
|
|
var _os = {};
|
|
|
_os[NAME] = undefined;
|
|
|
_os[VERSION] = undefined;
|
|
|
rgxMapper.call(_os, _ua, _rgxmap.os);
|
|
|
if (_isSelfNav && !_os[NAME] && _uach && _uach.platform != 'Unknown') {
|
|
|
_os[NAME] = _uach.platform
|
|
|
.replace(/chrome os/i, CHROMIUM_OS)
|
|
|
.replace(/macos/i, MAC_OS); // backward compatibility
|
|
|
}
|
|
|
return _os;
|
|
|
};
|
|
|
this.getResult = function () {
|
|
|
return {
|
|
|
ua : this.getUA(),
|
|
|
browser : this.getBrowser(),
|
|
|
engine : this.getEngine(),
|
|
|
os : this.getOS(),
|
|
|
device : this.getDevice(),
|
|
|
cpu : this.getCPU()
|
|
|
};
|
|
|
};
|
|
|
this.getUA = function () {
|
|
|
return _ua;
|
|
|
};
|
|
|
this.setUA = function (ua) {
|
|
|
_ua = (typeof ua === STR_TYPE && ua.length > UA_MAX_LENGTH) ? trim(ua, UA_MAX_LENGTH) : ua;
|
|
|
return this;
|
|
|
};
|
|
|
this.setUA(_ua);
|
|
|
return this;
|
|
|
};
|
|
|
|
|
|
UAParser.VERSION = LIBVERSION;
|
|
|
UAParser.BROWSER = enumerize([NAME, VERSION, MAJOR]);
|
|
|
UAParser.CPU = enumerize([ARCHITECTURE]);
|
|
|
UAParser.DEVICE = enumerize([MODEL, VENDOR, TYPE, CONSOLE, MOBILE, SMARTTV, TABLET, WEARABLE, EMBEDDED]);
|
|
|
UAParser.ENGINE = UAParser.OS = enumerize([NAME, VERSION]);
|
|
|
|
|
|
///////////
|
|
|
// Export
|
|
|
//////////
|
|
|
|
|
|
// check js environment
|
|
|
if (typeof(exports) !== UNDEF_TYPE) {
|
|
|
// nodejs env
|
|
|
if ("object" !== UNDEF_TYPE && module.exports) {
|
|
|
exports = module.exports = UAParser;
|
|
|
}
|
|
|
exports.UAParser = UAParser;
|
|
|
} else {
|
|
|
// requirejs env (optional)
|
|
|
if ("function" === FUNC_TYPE && __webpack_require__.amdO) {
|
|
|
!(__WEBPACK_AMD_DEFINE_RESULT__ = (function () {
|
|
|
return UAParser;
|
|
|
}).call(exports, __webpack_require__, exports, module),
|
|
|
__WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));
|
|
|
} else if (typeof window !== UNDEF_TYPE) {
|
|
|
// browser env
|
|
|
window.UAParser = UAParser;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// jQuery/Zepto specific (optional)
|
|
|
// Note:
|
|
|
// In AMD env the global scope should be kept clean, but jQuery is an exception.
|
|
|
// jQuery always exports to global scope, unless jQuery.noConflict(true) is used,
|
|
|
// and we should catch that.
|
|
|
var $ = typeof window !== UNDEF_TYPE && (window.jQuery || window.Zepto);
|
|
|
if ($ && !$.ua) {
|
|
|
var parser = new UAParser();
|
|
|
$.ua = parser.getResult();
|
|
|
$.ua.get = function () {
|
|
|
return parser.getUA();
|
|
|
};
|
|
|
$.ua.set = function (ua) {
|
|
|
parser.setUA(ua);
|
|
|
var result = parser.getResult();
|
|
|
for (var prop in result) {
|
|
|
$.ua[prop] = result[prop];
|
|
|
}
|
|
|
};
|
|
|
}
|
|
|
|
|
|
})(typeof window === 'object' ? window : this);
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/webrtc-adapter/src/js/adapter_core.js":
|
|
|
/*!************************************************************!*\
|
|
|
!*** ./node_modules/webrtc-adapter/src/js/adapter_core.js ***!
|
|
|
\************************************************************/
|
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
|
|
"use strict";
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
|
/* harmony export */ "default": () => (__WEBPACK_DEFAULT_EXPORT__)
|
|
|
/* harmony export */ });
|
|
|
/* harmony import */ var _adapter_factory_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./adapter_factory.js */ "./node_modules/webrtc-adapter/src/js/adapter_factory.js");
|
|
|
/*
|
|
|
* Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.
|
|
|
*
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
* tree.
|
|
|
*/
|
|
|
/* eslint-env node */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const adapter =
|
|
|
(0,_adapter_factory_js__WEBPACK_IMPORTED_MODULE_0__.adapterFactory)({window: typeof window === 'undefined' ? undefined : window});
|
|
|
/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (adapter);
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/webrtc-adapter/src/js/adapter_factory.js":
|
|
|
/*!***************************************************************!*\
|
|
|
!*** ./node_modules/webrtc-adapter/src/js/adapter_factory.js ***!
|
|
|
\***************************************************************/
|
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
|
|
"use strict";
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
|
/* harmony export */ adapterFactory: () => (/* binding */ adapterFactory)
|
|
|
/* harmony export */ });
|
|
|
/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./utils */ "./node_modules/webrtc-adapter/src/js/utils.js");
|
|
|
/* harmony import */ var _chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./chrome/chrome_shim */ "./node_modules/webrtc-adapter/src/js/chrome/chrome_shim.js");
|
|
|
/* harmony import */ var _firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./firefox/firefox_shim */ "./node_modules/webrtc-adapter/src/js/firefox/firefox_shim.js");
|
|
|
/* harmony import */ var _safari_safari_shim__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./safari/safari_shim */ "./node_modules/webrtc-adapter/src/js/safari/safari_shim.js");
|
|
|
/* harmony import */ var _common_shim__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./common_shim */ "./node_modules/webrtc-adapter/src/js/common_shim.js");
|
|
|
/* harmony import */ var sdp__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! sdp */ "./node_modules/sdp/sdp.js");
|
|
|
/* harmony import */ var sdp__WEBPACK_IMPORTED_MODULE_5___default = /*#__PURE__*/__webpack_require__.n(sdp__WEBPACK_IMPORTED_MODULE_5__);
|
|
|
/*
|
|
|
* Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.
|
|
|
*
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
* tree.
|
|
|
*/
|
|
|
|
|
|
|
|
|
// Browser shims.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Shimming starts here.
|
|
|
function adapterFactory({window} = {}, options = {
|
|
|
shimChrome: true,
|
|
|
shimFirefox: true,
|
|
|
shimSafari: true,
|
|
|
}) {
|
|
|
// Utils.
|
|
|
const logging = _utils__WEBPACK_IMPORTED_MODULE_0__.log;
|
|
|
const browserDetails = _utils__WEBPACK_IMPORTED_MODULE_0__.detectBrowser(window);
|
|
|
|
|
|
const adapter = {
|
|
|
browserDetails,
|
|
|
commonShim: _common_shim__WEBPACK_IMPORTED_MODULE_4__,
|
|
|
extractVersion: _utils__WEBPACK_IMPORTED_MODULE_0__.extractVersion,
|
|
|
disableLog: _utils__WEBPACK_IMPORTED_MODULE_0__.disableLog,
|
|
|
disableWarnings: _utils__WEBPACK_IMPORTED_MODULE_0__.disableWarnings,
|
|
|
// Expose sdp as a convenience. For production apps include directly.
|
|
|
sdp: sdp__WEBPACK_IMPORTED_MODULE_5__,
|
|
|
};
|
|
|
|
|
|
// Shim browser if found.
|
|
|
switch (browserDetails.browser) {
|
|
|
case 'chrome':
|
|
|
if (!_chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__ || !_chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__.shimPeerConnection ||
|
|
|
!options.shimChrome) {
|
|
|
logging('Chrome shim is not included in this adapter release.');
|
|
|
return adapter;
|
|
|
}
|
|
|
if (browserDetails.version === null) {
|
|
|
logging('Chrome shim can not determine version, not shimming.');
|
|
|
return adapter;
|
|
|
}
|
|
|
logging('adapter.js shimming chrome.');
|
|
|
// Export to the adapter global object visible in the browser.
|
|
|
adapter.browserShim = _chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__;
|
|
|
|
|
|
// Must be called before shimPeerConnection.
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimAddIceCandidateNullOrEmpty(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimParameterlessSetLocalDescription(window, browserDetails);
|
|
|
|
|
|
_chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__.shimGetUserMedia(window, browserDetails);
|
|
|
_chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__.shimMediaStream(window, browserDetails);
|
|
|
_chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__.shimPeerConnection(window, browserDetails);
|
|
|
_chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__.shimOnTrack(window, browserDetails);
|
|
|
_chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__.shimAddTrackRemoveTrack(window, browserDetails);
|
|
|
_chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__.shimGetSendersWithDtmf(window, browserDetails);
|
|
|
_chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__.shimGetStats(window, browserDetails);
|
|
|
_chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__.shimSenderReceiverGetStats(window, browserDetails);
|
|
|
_chrome_chrome_shim__WEBPACK_IMPORTED_MODULE_1__.fixNegotiationNeeded(window, browserDetails);
|
|
|
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimRTCIceCandidate(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimRTCIceCandidateRelayProtocol(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimConnectionState(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimMaxMessageSize(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimSendThrowTypeError(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.removeExtmapAllowMixed(window, browserDetails);
|
|
|
break;
|
|
|
case 'firefox':
|
|
|
if (!_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__ || !_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__.shimPeerConnection ||
|
|
|
!options.shimFirefox) {
|
|
|
logging('Firefox shim is not included in this adapter release.');
|
|
|
return adapter;
|
|
|
}
|
|
|
logging('adapter.js shimming firefox.');
|
|
|
// Export to the adapter global object visible in the browser.
|
|
|
adapter.browserShim = _firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__;
|
|
|
|
|
|
// Must be called before shimPeerConnection.
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimAddIceCandidateNullOrEmpty(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimParameterlessSetLocalDescription(window, browserDetails);
|
|
|
|
|
|
_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__.shimGetUserMedia(window, browserDetails);
|
|
|
_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__.shimPeerConnection(window, browserDetails);
|
|
|
_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__.shimOnTrack(window, browserDetails);
|
|
|
_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__.shimRemoveStream(window, browserDetails);
|
|
|
_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__.shimSenderGetStats(window, browserDetails);
|
|
|
_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__.shimReceiverGetStats(window, browserDetails);
|
|
|
_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__.shimRTCDataChannel(window, browserDetails);
|
|
|
_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__.shimAddTransceiver(window, browserDetails);
|
|
|
_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__.shimGetParameters(window, browserDetails);
|
|
|
_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__.shimCreateOffer(window, browserDetails);
|
|
|
_firefox_firefox_shim__WEBPACK_IMPORTED_MODULE_2__.shimCreateAnswer(window, browserDetails);
|
|
|
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimRTCIceCandidate(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimConnectionState(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimMaxMessageSize(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimSendThrowTypeError(window, browserDetails);
|
|
|
break;
|
|
|
case 'safari':
|
|
|
if (!_safari_safari_shim__WEBPACK_IMPORTED_MODULE_3__ || !options.shimSafari) {
|
|
|
logging('Safari shim is not included in this adapter release.');
|
|
|
return adapter;
|
|
|
}
|
|
|
logging('adapter.js shimming safari.');
|
|
|
// Export to the adapter global object visible in the browser.
|
|
|
adapter.browserShim = _safari_safari_shim__WEBPACK_IMPORTED_MODULE_3__;
|
|
|
|
|
|
// Must be called before shimCallbackAPI.
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimAddIceCandidateNullOrEmpty(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimParameterlessSetLocalDescription(window, browserDetails);
|
|
|
|
|
|
_safari_safari_shim__WEBPACK_IMPORTED_MODULE_3__.shimRTCIceServerUrls(window, browserDetails);
|
|
|
_safari_safari_shim__WEBPACK_IMPORTED_MODULE_3__.shimCreateOfferLegacy(window, browserDetails);
|
|
|
_safari_safari_shim__WEBPACK_IMPORTED_MODULE_3__.shimCallbacksAPI(window, browserDetails);
|
|
|
_safari_safari_shim__WEBPACK_IMPORTED_MODULE_3__.shimLocalStreamsAPI(window, browserDetails);
|
|
|
_safari_safari_shim__WEBPACK_IMPORTED_MODULE_3__.shimRemoteStreamsAPI(window, browserDetails);
|
|
|
_safari_safari_shim__WEBPACK_IMPORTED_MODULE_3__.shimTrackEventTransceiver(window, browserDetails);
|
|
|
_safari_safari_shim__WEBPACK_IMPORTED_MODULE_3__.shimGetUserMedia(window, browserDetails);
|
|
|
_safari_safari_shim__WEBPACK_IMPORTED_MODULE_3__.shimAudioContext(window, browserDetails);
|
|
|
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimRTCIceCandidate(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimRTCIceCandidateRelayProtocol(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimMaxMessageSize(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.shimSendThrowTypeError(window, browserDetails);
|
|
|
_common_shim__WEBPACK_IMPORTED_MODULE_4__.removeExtmapAllowMixed(window, browserDetails);
|
|
|
break;
|
|
|
default:
|
|
|
logging('Unsupported browser!');
|
|
|
break;
|
|
|
}
|
|
|
|
|
|
return adapter;
|
|
|
}
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/webrtc-adapter/src/js/chrome/chrome_shim.js":
|
|
|
/*!******************************************************************!*\
|
|
|
!*** ./node_modules/webrtc-adapter/src/js/chrome/chrome_shim.js ***!
|
|
|
\******************************************************************/
|
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
|
|
"use strict";
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
|
/* harmony export */ fixNegotiationNeeded: () => (/* binding */ fixNegotiationNeeded),
|
|
|
/* harmony export */ shimAddTrackRemoveTrack: () => (/* binding */ shimAddTrackRemoveTrack),
|
|
|
/* harmony export */ shimAddTrackRemoveTrackWithNative: () => (/* binding */ shimAddTrackRemoveTrackWithNative),
|
|
|
/* harmony export */ shimGetDisplayMedia: () => (/* reexport safe */ _getdisplaymedia__WEBPACK_IMPORTED_MODULE_2__.shimGetDisplayMedia),
|
|
|
/* harmony export */ shimGetSendersWithDtmf: () => (/* binding */ shimGetSendersWithDtmf),
|
|
|
/* harmony export */ shimGetStats: () => (/* binding */ shimGetStats),
|
|
|
/* harmony export */ shimGetUserMedia: () => (/* reexport safe */ _getusermedia__WEBPACK_IMPORTED_MODULE_1__.shimGetUserMedia),
|
|
|
/* harmony export */ shimMediaStream: () => (/* binding */ shimMediaStream),
|
|
|
/* harmony export */ shimOnTrack: () => (/* binding */ shimOnTrack),
|
|
|
/* harmony export */ shimPeerConnection: () => (/* binding */ shimPeerConnection),
|
|
|
/* harmony export */ shimSenderReceiverGetStats: () => (/* binding */ shimSenderReceiverGetStats)
|
|
|
/* harmony export */ });
|
|
|
/* harmony import */ var _utils_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../utils.js */ "./node_modules/webrtc-adapter/src/js/utils.js");
|
|
|
/* harmony import */ var _getusermedia__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./getusermedia */ "./node_modules/webrtc-adapter/src/js/chrome/getusermedia.js");
|
|
|
/* harmony import */ var _getdisplaymedia__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./getdisplaymedia */ "./node_modules/webrtc-adapter/src/js/chrome/getdisplaymedia.js");
|
|
|
/*
|
|
|
* Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.
|
|
|
*
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
* tree.
|
|
|
*/
|
|
|
/* eslint-env node */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function shimMediaStream(window) {
|
|
|
window.MediaStream = window.MediaStream || window.webkitMediaStream;
|
|
|
}
|
|
|
|
|
|
function shimOnTrack(window) {
|
|
|
if (typeof window === 'object' && window.RTCPeerConnection && !('ontrack' in
|
|
|
window.RTCPeerConnection.prototype)) {
|
|
|
Object.defineProperty(window.RTCPeerConnection.prototype, 'ontrack', {
|
|
|
get() {
|
|
|
return this._ontrack;
|
|
|
},
|
|
|
set(f) {
|
|
|
if (this._ontrack) {
|
|
|
this.removeEventListener('track', this._ontrack);
|
|
|
}
|
|
|
this.addEventListener('track', this._ontrack = f);
|
|
|
},
|
|
|
enumerable: true,
|
|
|
configurable: true
|
|
|
});
|
|
|
const origSetRemoteDescription =
|
|
|
window.RTCPeerConnection.prototype.setRemoteDescription;
|
|
|
window.RTCPeerConnection.prototype.setRemoteDescription =
|
|
|
function setRemoteDescription() {
|
|
|
if (!this._ontrackpoly) {
|
|
|
this._ontrackpoly = (e) => {
|
|
|
// onaddstream does not fire when a track is added to an existing
|
|
|
// stream. But stream.onaddtrack is implemented so we use that.
|
|
|
e.stream.addEventListener('addtrack', te => {
|
|
|
let receiver;
|
|
|
if (window.RTCPeerConnection.prototype.getReceivers) {
|
|
|
receiver = this.getReceivers()
|
|
|
.find(r => r.track && r.track.id === te.track.id);
|
|
|
} else {
|
|
|
receiver = {track: te.track};
|
|
|
}
|
|
|
|
|
|
const event = new Event('track');
|
|
|
event.track = te.track;
|
|
|
event.receiver = receiver;
|
|
|
event.transceiver = {receiver};
|
|
|
event.streams = [e.stream];
|
|
|
this.dispatchEvent(event);
|
|
|
});
|
|
|
e.stream.getTracks().forEach(track => {
|
|
|
let receiver;
|
|
|
if (window.RTCPeerConnection.prototype.getReceivers) {
|
|
|
receiver = this.getReceivers()
|
|
|
.find(r => r.track && r.track.id === track.id);
|
|
|
} else {
|
|
|
receiver = {track};
|
|
|
}
|
|
|
const event = new Event('track');
|
|
|
event.track = track;
|
|
|
event.receiver = receiver;
|
|
|
event.transceiver = {receiver};
|
|
|
event.streams = [e.stream];
|
|
|
this.dispatchEvent(event);
|
|
|
});
|
|
|
};
|
|
|
this.addEventListener('addstream', this._ontrackpoly);
|
|
|
}
|
|
|
return origSetRemoteDescription.apply(this, arguments);
|
|
|
};
|
|
|
} else {
|
|
|
// even if RTCRtpTransceiver is in window, it is only used and
|
|
|
// emitted in unified-plan. Unfortunately this means we need
|
|
|
// to unconditionally wrap the event.
|
|
|
_utils_js__WEBPACK_IMPORTED_MODULE_0__.wrapPeerConnectionEvent(window, 'track', e => {
|
|
|
if (!e.transceiver) {
|
|
|
Object.defineProperty(e, 'transceiver',
|
|
|
{value: {receiver: e.receiver}});
|
|
|
}
|
|
|
return e;
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function shimGetSendersWithDtmf(window) {
|
|
|
// Overrides addTrack/removeTrack, depends on shimAddTrackRemoveTrack.
|
|
|
if (typeof window === 'object' && window.RTCPeerConnection &&
|
|
|
!('getSenders' in window.RTCPeerConnection.prototype) &&
|
|
|
'createDTMFSender' in window.RTCPeerConnection.prototype) {
|
|
|
const shimSenderWithDtmf = function(pc, track) {
|
|
|
return {
|
|
|
track,
|
|
|
get dtmf() {
|
|
|
if (this._dtmf === undefined) {
|
|
|
if (track.kind === 'audio') {
|
|
|
this._dtmf = pc.createDTMFSender(track);
|
|
|
} else {
|
|
|
this._dtmf = null;
|
|
|
}
|
|
|
}
|
|
|
return this._dtmf;
|
|
|
},
|
|
|
_pc: pc
|
|
|
};
|
|
|
};
|
|
|
|
|
|
// augment addTrack when getSenders is not available.
|
|
|
if (!window.RTCPeerConnection.prototype.getSenders) {
|
|
|
window.RTCPeerConnection.prototype.getSenders = function getSenders() {
|
|
|
this._senders = this._senders || [];
|
|
|
return this._senders.slice(); // return a copy of the internal state.
|
|
|
};
|
|
|
const origAddTrack = window.RTCPeerConnection.prototype.addTrack;
|
|
|
window.RTCPeerConnection.prototype.addTrack =
|
|
|
function addTrack(track, stream) {
|
|
|
let sender = origAddTrack.apply(this, arguments);
|
|
|
if (!sender) {
|
|
|
sender = shimSenderWithDtmf(this, track);
|
|
|
this._senders.push(sender);
|
|
|
}
|
|
|
return sender;
|
|
|
};
|
|
|
|
|
|
const origRemoveTrack = window.RTCPeerConnection.prototype.removeTrack;
|
|
|
window.RTCPeerConnection.prototype.removeTrack =
|
|
|
function removeTrack(sender) {
|
|
|
origRemoveTrack.apply(this, arguments);
|
|
|
const idx = this._senders.indexOf(sender);
|
|
|
if (idx !== -1) {
|
|
|
this._senders.splice(idx, 1);
|
|
|
}
|
|
|
};
|
|
|
}
|
|
|
const origAddStream = window.RTCPeerConnection.prototype.addStream;
|
|
|
window.RTCPeerConnection.prototype.addStream = function addStream(stream) {
|
|
|
this._senders = this._senders || [];
|
|
|
origAddStream.apply(this, [stream]);
|
|
|
stream.getTracks().forEach(track => {
|
|
|
this._senders.push(shimSenderWithDtmf(this, track));
|
|
|
});
|
|
|
};
|
|
|
|
|
|
const origRemoveStream = window.RTCPeerConnection.prototype.removeStream;
|
|
|
window.RTCPeerConnection.prototype.removeStream =
|
|
|
function removeStream(stream) {
|
|
|
this._senders = this._senders || [];
|
|
|
origRemoveStream.apply(this, [stream]);
|
|
|
|
|
|
stream.getTracks().forEach(track => {
|
|
|
const sender = this._senders.find(s => s.track === track);
|
|
|
if (sender) { // remove sender
|
|
|
this._senders.splice(this._senders.indexOf(sender), 1);
|
|
|
}
|
|
|
});
|
|
|
};
|
|
|
} else if (typeof window === 'object' && window.RTCPeerConnection &&
|
|
|
'getSenders' in window.RTCPeerConnection.prototype &&
|
|
|
'createDTMFSender' in window.RTCPeerConnection.prototype &&
|
|
|
window.RTCRtpSender &&
|
|
|
!('dtmf' in window.RTCRtpSender.prototype)) {
|
|
|
const origGetSenders = window.RTCPeerConnection.prototype.getSenders;
|
|
|
window.RTCPeerConnection.prototype.getSenders = function getSenders() {
|
|
|
const senders = origGetSenders.apply(this, []);
|
|
|
senders.forEach(sender => sender._pc = this);
|
|
|
return senders;
|
|
|
};
|
|
|
|
|
|
Object.defineProperty(window.RTCRtpSender.prototype, 'dtmf', {
|
|
|
get() {
|
|
|
if (this._dtmf === undefined) {
|
|
|
if (this.track.kind === 'audio') {
|
|
|
this._dtmf = this._pc.createDTMFSender(this.track);
|
|
|
} else {
|
|
|
this._dtmf = null;
|
|
|
}
|
|
|
}
|
|
|
return this._dtmf;
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function shimGetStats(window) {
|
|
|
if (!window.RTCPeerConnection) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
const origGetStats = window.RTCPeerConnection.prototype.getStats;
|
|
|
window.RTCPeerConnection.prototype.getStats = function getStats() {
|
|
|
const [selector, onSucc, onErr] = arguments;
|
|
|
|
|
|
// If selector is a function then we are in the old style stats so just
|
|
|
// pass back the original getStats format to avoid breaking old users.
|
|
|
if (arguments.length > 0 && typeof selector === 'function') {
|
|
|
return origGetStats.apply(this, arguments);
|
|
|
}
|
|
|
|
|
|
// When spec-style getStats is supported, return those when called with
|
|
|
// either no arguments or the selector argument is null.
|
|
|
if (origGetStats.length === 0 && (arguments.length === 0 ||
|
|
|
typeof selector !== 'function')) {
|
|
|
return origGetStats.apply(this, []);
|
|
|
}
|
|
|
|
|
|
const fixChromeStats_ = function(response) {
|
|
|
const standardReport = {};
|
|
|
const reports = response.result();
|
|
|
reports.forEach(report => {
|
|
|
const standardStats = {
|
|
|
id: report.id,
|
|
|
timestamp: report.timestamp,
|
|
|
type: {
|
|
|
localcandidate: 'local-candidate',
|
|
|
remotecandidate: 'remote-candidate'
|
|
|
}[report.type] || report.type
|
|
|
};
|
|
|
report.names().forEach(name => {
|
|
|
standardStats[name] = report.stat(name);
|
|
|
});
|
|
|
standardReport[standardStats.id] = standardStats;
|
|
|
});
|
|
|
|
|
|
return standardReport;
|
|
|
};
|
|
|
|
|
|
// shim getStats with maplike support
|
|
|
const makeMapStats = function(stats) {
|
|
|
return new Map(Object.keys(stats).map(key => [key, stats[key]]));
|
|
|
};
|
|
|
|
|
|
if (arguments.length >= 2) {
|
|
|
const successCallbackWrapper_ = function(response) {
|
|
|
onSucc(makeMapStats(fixChromeStats_(response)));
|
|
|
};
|
|
|
|
|
|
return origGetStats.apply(this, [successCallbackWrapper_,
|
|
|
selector]);
|
|
|
}
|
|
|
|
|
|
// promise-support
|
|
|
return new Promise((resolve, reject) => {
|
|
|
origGetStats.apply(this, [
|
|
|
function(response) {
|
|
|
resolve(makeMapStats(fixChromeStats_(response)));
|
|
|
}, reject]);
|
|
|
}).then(onSucc, onErr);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function shimSenderReceiverGetStats(window) {
|
|
|
if (!(typeof window === 'object' && window.RTCPeerConnection &&
|
|
|
window.RTCRtpSender && window.RTCRtpReceiver)) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// shim sender stats.
|
|
|
if (!('getStats' in window.RTCRtpSender.prototype)) {
|
|
|
const origGetSenders = window.RTCPeerConnection.prototype.getSenders;
|
|
|
if (origGetSenders) {
|
|
|
window.RTCPeerConnection.prototype.getSenders = function getSenders() {
|
|
|
const senders = origGetSenders.apply(this, []);
|
|
|
senders.forEach(sender => sender._pc = this);
|
|
|
return senders;
|
|
|
};
|
|
|
}
|
|
|
|
|
|
const origAddTrack = window.RTCPeerConnection.prototype.addTrack;
|
|
|
if (origAddTrack) {
|
|
|
window.RTCPeerConnection.prototype.addTrack = function addTrack() {
|
|
|
const sender = origAddTrack.apply(this, arguments);
|
|
|
sender._pc = this;
|
|
|
return sender;
|
|
|
};
|
|
|
}
|
|
|
window.RTCRtpSender.prototype.getStats = function getStats() {
|
|
|
const sender = this;
|
|
|
return this._pc.getStats().then(result =>
|
|
|
/* Note: this will include stats of all senders that
|
|
|
* send a track with the same id as sender.track as
|
|
|
* it is not possible to identify the RTCRtpSender.
|
|
|
*/
|
|
|
_utils_js__WEBPACK_IMPORTED_MODULE_0__.filterStats(result, sender.track, true));
|
|
|
};
|
|
|
}
|
|
|
|
|
|
// shim receiver stats.
|
|
|
if (!('getStats' in window.RTCRtpReceiver.prototype)) {
|
|
|
const origGetReceivers = window.RTCPeerConnection.prototype.getReceivers;
|
|
|
if (origGetReceivers) {
|
|
|
window.RTCPeerConnection.prototype.getReceivers =
|
|
|
function getReceivers() {
|
|
|
const receivers = origGetReceivers.apply(this, []);
|
|
|
receivers.forEach(receiver => receiver._pc = this);
|
|
|
return receivers;
|
|
|
};
|
|
|
}
|
|
|
_utils_js__WEBPACK_IMPORTED_MODULE_0__.wrapPeerConnectionEvent(window, 'track', e => {
|
|
|
e.receiver._pc = e.srcElement;
|
|
|
return e;
|
|
|
});
|
|
|
window.RTCRtpReceiver.prototype.getStats = function getStats() {
|
|
|
const receiver = this;
|
|
|
return this._pc.getStats().then(result =>
|
|
|
_utils_js__WEBPACK_IMPORTED_MODULE_0__.filterStats(result, receiver.track, false));
|
|
|
};
|
|
|
}
|
|
|
|
|
|
if (!('getStats' in window.RTCRtpSender.prototype &&
|
|
|
'getStats' in window.RTCRtpReceiver.prototype)) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// shim RTCPeerConnection.getStats(track).
|
|
|
const origGetStats = window.RTCPeerConnection.prototype.getStats;
|
|
|
window.RTCPeerConnection.prototype.getStats = function getStats() {
|
|
|
if (arguments.length > 0 &&
|
|
|
arguments[0] instanceof window.MediaStreamTrack) {
|
|
|
const track = arguments[0];
|
|
|
let sender;
|
|
|
let receiver;
|
|
|
let err;
|
|
|
this.getSenders().forEach(s => {
|
|
|
if (s.track === track) {
|
|
|
if (sender) {
|
|
|
err = true;
|
|
|
} else {
|
|
|
sender = s;
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
this.getReceivers().forEach(r => {
|
|
|
if (r.track === track) {
|
|
|
if (receiver) {
|
|
|
err = true;
|
|
|
} else {
|
|
|
receiver = r;
|
|
|
}
|
|
|
}
|
|
|
return r.track === track;
|
|
|
});
|
|
|
if (err || (sender && receiver)) {
|
|
|
return Promise.reject(new DOMException(
|
|
|
'There are more than one sender or receiver for the track.',
|
|
|
'InvalidAccessError'));
|
|
|
} else if (sender) {
|
|
|
return sender.getStats();
|
|
|
} else if (receiver) {
|
|
|
return receiver.getStats();
|
|
|
}
|
|
|
return Promise.reject(new DOMException(
|
|
|
'There is no sender or receiver for the track.',
|
|
|
'InvalidAccessError'));
|
|
|
}
|
|
|
return origGetStats.apply(this, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function shimAddTrackRemoveTrackWithNative(window) {
|
|
|
// shim addTrack/removeTrack with native variants in order to make
|
|
|
// the interactions with legacy getLocalStreams behave as in other browsers.
|
|
|
// Keeps a mapping stream.id => [stream, rtpsenders...]
|
|
|
window.RTCPeerConnection.prototype.getLocalStreams =
|
|
|
function getLocalStreams() {
|
|
|
this._shimmedLocalStreams = this._shimmedLocalStreams || {};
|
|
|
return Object.keys(this._shimmedLocalStreams)
|
|
|
.map(streamId => this._shimmedLocalStreams[streamId][0]);
|
|
|
};
|
|
|
|
|
|
const origAddTrack = window.RTCPeerConnection.prototype.addTrack;
|
|
|
window.RTCPeerConnection.prototype.addTrack =
|
|
|
function addTrack(track, stream) {
|
|
|
if (!stream) {
|
|
|
return origAddTrack.apply(this, arguments);
|
|
|
}
|
|
|
this._shimmedLocalStreams = this._shimmedLocalStreams || {};
|
|
|
|
|
|
const sender = origAddTrack.apply(this, arguments);
|
|
|
if (!this._shimmedLocalStreams[stream.id]) {
|
|
|
this._shimmedLocalStreams[stream.id] = [stream, sender];
|
|
|
} else if (this._shimmedLocalStreams[stream.id].indexOf(sender) === -1) {
|
|
|
this._shimmedLocalStreams[stream.id].push(sender);
|
|
|
}
|
|
|
return sender;
|
|
|
};
|
|
|
|
|
|
const origAddStream = window.RTCPeerConnection.prototype.addStream;
|
|
|
window.RTCPeerConnection.prototype.addStream = function addStream(stream) {
|
|
|
this._shimmedLocalStreams = this._shimmedLocalStreams || {};
|
|
|
|
|
|
stream.getTracks().forEach(track => {
|
|
|
const alreadyExists = this.getSenders().find(s => s.track === track);
|
|
|
if (alreadyExists) {
|
|
|
throw new DOMException('Track already exists.',
|
|
|
'InvalidAccessError');
|
|
|
}
|
|
|
});
|
|
|
const existingSenders = this.getSenders();
|
|
|
origAddStream.apply(this, arguments);
|
|
|
const newSenders = this.getSenders()
|
|
|
.filter(newSender => existingSenders.indexOf(newSender) === -1);
|
|
|
this._shimmedLocalStreams[stream.id] = [stream].concat(newSenders);
|
|
|
};
|
|
|
|
|
|
const origRemoveStream = window.RTCPeerConnection.prototype.removeStream;
|
|
|
window.RTCPeerConnection.prototype.removeStream =
|
|
|
function removeStream(stream) {
|
|
|
this._shimmedLocalStreams = this._shimmedLocalStreams || {};
|
|
|
delete this._shimmedLocalStreams[stream.id];
|
|
|
return origRemoveStream.apply(this, arguments);
|
|
|
};
|
|
|
|
|
|
const origRemoveTrack = window.RTCPeerConnection.prototype.removeTrack;
|
|
|
window.RTCPeerConnection.prototype.removeTrack =
|
|
|
function removeTrack(sender) {
|
|
|
this._shimmedLocalStreams = this._shimmedLocalStreams || {};
|
|
|
if (sender) {
|
|
|
Object.keys(this._shimmedLocalStreams).forEach(streamId => {
|
|
|
const idx = this._shimmedLocalStreams[streamId].indexOf(sender);
|
|
|
if (idx !== -1) {
|
|
|
this._shimmedLocalStreams[streamId].splice(idx, 1);
|
|
|
}
|
|
|
if (this._shimmedLocalStreams[streamId].length === 1) {
|
|
|
delete this._shimmedLocalStreams[streamId];
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
return origRemoveTrack.apply(this, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function shimAddTrackRemoveTrack(window, browserDetails) {
|
|
|
if (!window.RTCPeerConnection) {
|
|
|
return;
|
|
|
}
|
|
|
// shim addTrack and removeTrack.
|
|
|
if (window.RTCPeerConnection.prototype.addTrack &&
|
|
|
browserDetails.version >= 65) {
|
|
|
return shimAddTrackRemoveTrackWithNative(window);
|
|
|
}
|
|
|
|
|
|
// also shim pc.getLocalStreams when addTrack is shimmed
|
|
|
// to return the original streams.
|
|
|
const origGetLocalStreams = window.RTCPeerConnection.prototype
|
|
|
.getLocalStreams;
|
|
|
window.RTCPeerConnection.prototype.getLocalStreams =
|
|
|
function getLocalStreams() {
|
|
|
const nativeStreams = origGetLocalStreams.apply(this);
|
|
|
this._reverseStreams = this._reverseStreams || {};
|
|
|
return nativeStreams.map(stream => this._reverseStreams[stream.id]);
|
|
|
};
|
|
|
|
|
|
const origAddStream = window.RTCPeerConnection.prototype.addStream;
|
|
|
window.RTCPeerConnection.prototype.addStream = function addStream(stream) {
|
|
|
this._streams = this._streams || {};
|
|
|
this._reverseStreams = this._reverseStreams || {};
|
|
|
|
|
|
stream.getTracks().forEach(track => {
|
|
|
const alreadyExists = this.getSenders().find(s => s.track === track);
|
|
|
if (alreadyExists) {
|
|
|
throw new DOMException('Track already exists.',
|
|
|
'InvalidAccessError');
|
|
|
}
|
|
|
});
|
|
|
// Add identity mapping for consistency with addTrack.
|
|
|
// Unless this is being used with a stream from addTrack.
|
|
|
if (!this._reverseStreams[stream.id]) {
|
|
|
const newStream = new window.MediaStream(stream.getTracks());
|
|
|
this._streams[stream.id] = newStream;
|
|
|
this._reverseStreams[newStream.id] = stream;
|
|
|
stream = newStream;
|
|
|
}
|
|
|
origAddStream.apply(this, [stream]);
|
|
|
};
|
|
|
|
|
|
const origRemoveStream = window.RTCPeerConnection.prototype.removeStream;
|
|
|
window.RTCPeerConnection.prototype.removeStream =
|
|
|
function removeStream(stream) {
|
|
|
this._streams = this._streams || {};
|
|
|
this._reverseStreams = this._reverseStreams || {};
|
|
|
|
|
|
origRemoveStream.apply(this, [(this._streams[stream.id] || stream)]);
|
|
|
delete this._reverseStreams[(this._streams[stream.id] ?
|
|
|
this._streams[stream.id].id : stream.id)];
|
|
|
delete this._streams[stream.id];
|
|
|
};
|
|
|
|
|
|
window.RTCPeerConnection.prototype.addTrack =
|
|
|
function addTrack(track, stream) {
|
|
|
if (this.signalingState === 'closed') {
|
|
|
throw new DOMException(
|
|
|
'The RTCPeerConnection\'s signalingState is \'closed\'.',
|
|
|
'InvalidStateError');
|
|
|
}
|
|
|
const streams = [].slice.call(arguments, 1);
|
|
|
if (streams.length !== 1 ||
|
|
|
!streams[0].getTracks().find(t => t === track)) {
|
|
|
// this is not fully correct but all we can manage without
|
|
|
// [[associated MediaStreams]] internal slot.
|
|
|
throw new DOMException(
|
|
|
'The adapter.js addTrack polyfill only supports a single ' +
|
|
|
' stream which is associated with the specified track.',
|
|
|
'NotSupportedError');
|
|
|
}
|
|
|
|
|
|
const alreadyExists = this.getSenders().find(s => s.track === track);
|
|
|
if (alreadyExists) {
|
|
|
throw new DOMException('Track already exists.',
|
|
|
'InvalidAccessError');
|
|
|
}
|
|
|
|
|
|
this._streams = this._streams || {};
|
|
|
this._reverseStreams = this._reverseStreams || {};
|
|
|
const oldStream = this._streams[stream.id];
|
|
|
if (oldStream) {
|
|
|
// this is using odd Chrome behaviour, use with caution:
|
|
|
// https://bugs.chromium.org/p/webrtc/issues/detail?id=7815
|
|
|
// Note: we rely on the high-level addTrack/dtmf shim to
|
|
|
// create the sender with a dtmf sender.
|
|
|
oldStream.addTrack(track);
|
|
|
|
|
|
// Trigger ONN async.
|
|
|
Promise.resolve().then(() => {
|
|
|
this.dispatchEvent(new Event('negotiationneeded'));
|
|
|
});
|
|
|
} else {
|
|
|
const newStream = new window.MediaStream([track]);
|
|
|
this._streams[stream.id] = newStream;
|
|
|
this._reverseStreams[newStream.id] = stream;
|
|
|
this.addStream(newStream);
|
|
|
}
|
|
|
return this.getSenders().find(s => s.track === track);
|
|
|
};
|
|
|
|
|
|
// replace the internal stream id with the external one and
|
|
|
// vice versa.
|
|
|
function replaceInternalStreamId(pc, description) {
|
|
|
let sdp = description.sdp;
|
|
|
Object.keys(pc._reverseStreams || []).forEach(internalId => {
|
|
|
const externalStream = pc._reverseStreams[internalId];
|
|
|
const internalStream = pc._streams[externalStream.id];
|
|
|
sdp = sdp.replace(new RegExp(internalStream.id, 'g'),
|
|
|
externalStream.id);
|
|
|
});
|
|
|
return new RTCSessionDescription({
|
|
|
type: description.type,
|
|
|
sdp
|
|
|
});
|
|
|
}
|
|
|
function replaceExternalStreamId(pc, description) {
|
|
|
let sdp = description.sdp;
|
|
|
Object.keys(pc._reverseStreams || []).forEach(internalId => {
|
|
|
const externalStream = pc._reverseStreams[internalId];
|
|
|
const internalStream = pc._streams[externalStream.id];
|
|
|
sdp = sdp.replace(new RegExp(externalStream.id, 'g'),
|
|
|
internalStream.id);
|
|
|
});
|
|
|
return new RTCSessionDescription({
|
|
|
type: description.type,
|
|
|
sdp
|
|
|
});
|
|
|
}
|
|
|
['createOffer', 'createAnswer'].forEach(function(method) {
|
|
|
const nativeMethod = window.RTCPeerConnection.prototype[method];
|
|
|
const methodObj = {[method]() {
|
|
|
const args = arguments;
|
|
|
const isLegacyCall = arguments.length &&
|
|
|
typeof arguments[0] === 'function';
|
|
|
if (isLegacyCall) {
|
|
|
return nativeMethod.apply(this, [
|
|
|
(description) => {
|
|
|
const desc = replaceInternalStreamId(this, description);
|
|
|
args[0].apply(null, [desc]);
|
|
|
},
|
|
|
(err) => {
|
|
|
if (args[1]) {
|
|
|
args[1].apply(null, err);
|
|
|
}
|
|
|
}, arguments[2]
|
|
|
]);
|
|
|
}
|
|
|
return nativeMethod.apply(this, arguments)
|
|
|
.then(description => replaceInternalStreamId(this, description));
|
|
|
}};
|
|
|
window.RTCPeerConnection.prototype[method] = methodObj[method];
|
|
|
});
|
|
|
|
|
|
const origSetLocalDescription =
|
|
|
window.RTCPeerConnection.prototype.setLocalDescription;
|
|
|
window.RTCPeerConnection.prototype.setLocalDescription =
|
|
|
function setLocalDescription() {
|
|
|
if (!arguments.length || !arguments[0].type) {
|
|
|
return origSetLocalDescription.apply(this, arguments);
|
|
|
}
|
|
|
arguments[0] = replaceExternalStreamId(this, arguments[0]);
|
|
|
return origSetLocalDescription.apply(this, arguments);
|
|
|
};
|
|
|
|
|
|
// TODO: mangle getStats: https://w3c.github.io/webrtc-stats/#dom-rtcmediastreamstats-streamidentifier
|
|
|
|
|
|
const origLocalDescription = Object.getOwnPropertyDescriptor(
|
|
|
window.RTCPeerConnection.prototype, 'localDescription');
|
|
|
Object.defineProperty(window.RTCPeerConnection.prototype,
|
|
|
'localDescription', {
|
|
|
get() {
|
|
|
const description = origLocalDescription.get.apply(this);
|
|
|
if (description.type === '') {
|
|
|
return description;
|
|
|
}
|
|
|
return replaceInternalStreamId(this, description);
|
|
|
}
|
|
|
});
|
|
|
|
|
|
window.RTCPeerConnection.prototype.removeTrack =
|
|
|
function removeTrack(sender) {
|
|
|
if (this.signalingState === 'closed') {
|
|
|
throw new DOMException(
|
|
|
'The RTCPeerConnection\'s signalingState is \'closed\'.',
|
|
|
'InvalidStateError');
|
|
|
}
|
|
|
// We can not yet check for sender instanceof RTCRtpSender
|
|
|
// since we shim RTPSender. So we check if sender._pc is set.
|
|
|
if (!sender._pc) {
|
|
|
throw new DOMException('Argument 1 of RTCPeerConnection.removeTrack ' +
|
|
|
'does not implement interface RTCRtpSender.', 'TypeError');
|
|
|
}
|
|
|
const isLocal = sender._pc === this;
|
|
|
if (!isLocal) {
|
|
|
throw new DOMException('Sender was not created by this connection.',
|
|
|
'InvalidAccessError');
|
|
|
}
|
|
|
|
|
|
// Search for the native stream the senders track belongs to.
|
|
|
this._streams = this._streams || {};
|
|
|
let stream;
|
|
|
Object.keys(this._streams).forEach(streamid => {
|
|
|
const hasTrack = this._streams[streamid].getTracks()
|
|
|
.find(track => sender.track === track);
|
|
|
if (hasTrack) {
|
|
|
stream = this._streams[streamid];
|
|
|
}
|
|
|
});
|
|
|
|
|
|
if (stream) {
|
|
|
if (stream.getTracks().length === 1) {
|
|
|
// if this is the last track of the stream, remove the stream. This
|
|
|
// takes care of any shimmed _senders.
|
|
|
this.removeStream(this._reverseStreams[stream.id]);
|
|
|
} else {
|
|
|
// relying on the same odd chrome behaviour as above.
|
|
|
stream.removeTrack(sender.track);
|
|
|
}
|
|
|
this.dispatchEvent(new Event('negotiationneeded'));
|
|
|
}
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function shimPeerConnection(window, browserDetails) {
|
|
|
if (!window.RTCPeerConnection && window.webkitRTCPeerConnection) {
|
|
|
// very basic support for old versions.
|
|
|
window.RTCPeerConnection = window.webkitRTCPeerConnection;
|
|
|
}
|
|
|
if (!window.RTCPeerConnection) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// shim implicit creation of RTCSessionDescription/RTCIceCandidate
|
|
|
if (browserDetails.version < 53) {
|
|
|
['setLocalDescription', 'setRemoteDescription', 'addIceCandidate']
|
|
|
.forEach(function(method) {
|
|
|
const nativeMethod = window.RTCPeerConnection.prototype[method];
|
|
|
const methodObj = {[method]() {
|
|
|
arguments[0] = new ((method === 'addIceCandidate') ?
|
|
|
window.RTCIceCandidate :
|
|
|
window.RTCSessionDescription)(arguments[0]);
|
|
|
return nativeMethod.apply(this, arguments);
|
|
|
}};
|
|
|
window.RTCPeerConnection.prototype[method] = methodObj[method];
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Attempt to fix ONN in plan-b mode.
|
|
|
function fixNegotiationNeeded(window, browserDetails) {
|
|
|
_utils_js__WEBPACK_IMPORTED_MODULE_0__.wrapPeerConnectionEvent(window, 'negotiationneeded', e => {
|
|
|
const pc = e.target;
|
|
|
if (browserDetails.version < 72 || (pc.getConfiguration &&
|
|
|
pc.getConfiguration().sdpSemantics === 'plan-b')) {
|
|
|
if (pc.signalingState !== 'stable') {
|
|
|
return;
|
|
|
}
|
|
|
}
|
|
|
return e;
|
|
|
});
|
|
|
}
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/webrtc-adapter/src/js/chrome/getdisplaymedia.js":
|
|
|
/*!**********************************************************************!*\
|
|
|
!*** ./node_modules/webrtc-adapter/src/js/chrome/getdisplaymedia.js ***!
|
|
|
\**********************************************************************/
|
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
|
|
"use strict";
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
|
/* harmony export */ shimGetDisplayMedia: () => (/* binding */ shimGetDisplayMedia)
|
|
|
/* harmony export */ });
|
|
|
/*
|
|
|
* Copyright (c) 2018 The adapter.js project authors. All Rights Reserved.
|
|
|
*
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
* tree.
|
|
|
*/
|
|
|
/* eslint-env node */
|
|
|
|
|
|
function shimGetDisplayMedia(window, getSourceId) {
|
|
|
if (window.navigator.mediaDevices &&
|
|
|
'getDisplayMedia' in window.navigator.mediaDevices) {
|
|
|
return;
|
|
|
}
|
|
|
if (!(window.navigator.mediaDevices)) {
|
|
|
return;
|
|
|
}
|
|
|
// getSourceId is a function that returns a promise resolving with
|
|
|
// the sourceId of the screen/window/tab to be shared.
|
|
|
if (typeof getSourceId !== 'function') {
|
|
|
console.error('shimGetDisplayMedia: getSourceId argument is not ' +
|
|
|
'a function');
|
|
|
return;
|
|
|
}
|
|
|
window.navigator.mediaDevices.getDisplayMedia =
|
|
|
function getDisplayMedia(constraints) {
|
|
|
return getSourceId(constraints)
|
|
|
.then(sourceId => {
|
|
|
const widthSpecified = constraints.video && constraints.video.width;
|
|
|
const heightSpecified = constraints.video &&
|
|
|
constraints.video.height;
|
|
|
const frameRateSpecified = constraints.video &&
|
|
|
constraints.video.frameRate;
|
|
|
constraints.video = {
|
|
|
mandatory: {
|
|
|
chromeMediaSource: 'desktop',
|
|
|
chromeMediaSourceId: sourceId,
|
|
|
maxFrameRate: frameRateSpecified || 3
|
|
|
}
|
|
|
};
|
|
|
if (widthSpecified) {
|
|
|
constraints.video.mandatory.maxWidth = widthSpecified;
|
|
|
}
|
|
|
if (heightSpecified) {
|
|
|
constraints.video.mandatory.maxHeight = heightSpecified;
|
|
|
}
|
|
|
return window.navigator.mediaDevices.getUserMedia(constraints);
|
|
|
});
|
|
|
};
|
|
|
}
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/webrtc-adapter/src/js/chrome/getusermedia.js":
|
|
|
/*!*******************************************************************!*\
|
|
|
!*** ./node_modules/webrtc-adapter/src/js/chrome/getusermedia.js ***!
|
|
|
\*******************************************************************/
|
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
|
|
"use strict";
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
|
/* harmony export */ shimGetUserMedia: () => (/* binding */ shimGetUserMedia)
|
|
|
/* harmony export */ });
|
|
|
/* harmony import */ var _utils_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../utils.js */ "./node_modules/webrtc-adapter/src/js/utils.js");
|
|
|
/*
|
|
|
* Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.
|
|
|
*
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
* tree.
|
|
|
*/
|
|
|
/* eslint-env node */
|
|
|
|
|
|
|
|
|
const logging = _utils_js__WEBPACK_IMPORTED_MODULE_0__.log;
|
|
|
|
|
|
function shimGetUserMedia(window, browserDetails) {
|
|
|
const navigator = window && window.navigator;
|
|
|
|
|
|
if (!navigator.mediaDevices) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
const constraintsToChrome_ = function(c) {
|
|
|
if (typeof c !== 'object' || c.mandatory || c.optional) {
|
|
|
return c;
|
|
|
}
|
|
|
const cc = {};
|
|
|
Object.keys(c).forEach(key => {
|
|
|
if (key === 'require' || key === 'advanced' || key === 'mediaSource') {
|
|
|
return;
|
|
|
}
|
|
|
const r = (typeof c[key] === 'object') ? c[key] : {ideal: c[key]};
|
|
|
if (r.exact !== undefined && typeof r.exact === 'number') {
|
|
|
r.min = r.max = r.exact;
|
|
|
}
|
|
|
const oldname_ = function(prefix, name) {
|
|
|
if (prefix) {
|
|
|
return prefix + name.charAt(0).toUpperCase() + name.slice(1);
|
|
|
}
|
|
|
return (name === 'deviceId') ? 'sourceId' : name;
|
|
|
};
|
|
|
if (r.ideal !== undefined) {
|
|
|
cc.optional = cc.optional || [];
|
|
|
let oc = {};
|
|
|
if (typeof r.ideal === 'number') {
|
|
|
oc[oldname_('min', key)] = r.ideal;
|
|
|
cc.optional.push(oc);
|
|
|
oc = {};
|
|
|
oc[oldname_('max', key)] = r.ideal;
|
|
|
cc.optional.push(oc);
|
|
|
} else {
|
|
|
oc[oldname_('', key)] = r.ideal;
|
|
|
cc.optional.push(oc);
|
|
|
}
|
|
|
}
|
|
|
if (r.exact !== undefined && typeof r.exact !== 'number') {
|
|
|
cc.mandatory = cc.mandatory || {};
|
|
|
cc.mandatory[oldname_('', key)] = r.exact;
|
|
|
} else {
|
|
|
['min', 'max'].forEach(mix => {
|
|
|
if (r[mix] !== undefined) {
|
|
|
cc.mandatory = cc.mandatory || {};
|
|
|
cc.mandatory[oldname_(mix, key)] = r[mix];
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
});
|
|
|
if (c.advanced) {
|
|
|
cc.optional = (cc.optional || []).concat(c.advanced);
|
|
|
}
|
|
|
return cc;
|
|
|
};
|
|
|
|
|
|
const shimConstraints_ = function(constraints, func) {
|
|
|
if (browserDetails.version >= 61) {
|
|
|
return func(constraints);
|
|
|
}
|
|
|
constraints = JSON.parse(JSON.stringify(constraints));
|
|
|
if (constraints && typeof constraints.audio === 'object') {
|
|
|
const remap = function(obj, a, b) {
|
|
|
if (a in obj && !(b in obj)) {
|
|
|
obj[b] = obj[a];
|
|
|
delete obj[a];
|
|
|
}
|
|
|
};
|
|
|
constraints = JSON.parse(JSON.stringify(constraints));
|
|
|
remap(constraints.audio, 'autoGainControl', 'googAutoGainControl');
|
|
|
remap(constraints.audio, 'noiseSuppression', 'googNoiseSuppression');
|
|
|
constraints.audio = constraintsToChrome_(constraints.audio);
|
|
|
}
|
|
|
if (constraints && typeof constraints.video === 'object') {
|
|
|
// Shim facingMode for mobile & surface pro.
|
|
|
let face = constraints.video.facingMode;
|
|
|
face = face && ((typeof face === 'object') ? face : {ideal: face});
|
|
|
const getSupportedFacingModeLies = browserDetails.version < 66;
|
|
|
|
|
|
if ((face && (face.exact === 'user' || face.exact === 'environment' ||
|
|
|
face.ideal === 'user' || face.ideal === 'environment')) &&
|
|
|
!(navigator.mediaDevices.getSupportedConstraints &&
|
|
|
navigator.mediaDevices.getSupportedConstraints().facingMode &&
|
|
|
!getSupportedFacingModeLies)) {
|
|
|
delete constraints.video.facingMode;
|
|
|
let matches;
|
|
|
if (face.exact === 'environment' || face.ideal === 'environment') {
|
|
|
matches = ['back', 'rear'];
|
|
|
} else if (face.exact === 'user' || face.ideal === 'user') {
|
|
|
matches = ['front'];
|
|
|
}
|
|
|
if (matches) {
|
|
|
// Look for matches in label, or use last cam for back (typical).
|
|
|
return navigator.mediaDevices.enumerateDevices()
|
|
|
.then(devices => {
|
|
|
devices = devices.filter(d => d.kind === 'videoinput');
|
|
|
let dev = devices.find(d => matches.some(match =>
|
|
|
d.label.toLowerCase().includes(match)));
|
|
|
if (!dev && devices.length && matches.includes('back')) {
|
|
|
dev = devices[devices.length - 1]; // more likely the back cam
|
|
|
}
|
|
|
if (dev) {
|
|
|
constraints.video.deviceId = face.exact
|
|
|
? {exact: dev.deviceId}
|
|
|
: {ideal: dev.deviceId};
|
|
|
}
|
|
|
constraints.video = constraintsToChrome_(constraints.video);
|
|
|
logging('chrome: ' + JSON.stringify(constraints));
|
|
|
return func(constraints);
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
constraints.video = constraintsToChrome_(constraints.video);
|
|
|
}
|
|
|
logging('chrome: ' + JSON.stringify(constraints));
|
|
|
return func(constraints);
|
|
|
};
|
|
|
|
|
|
const shimError_ = function(e) {
|
|
|
if (browserDetails.version >= 64) {
|
|
|
return e;
|
|
|
}
|
|
|
return {
|
|
|
name: {
|
|
|
PermissionDeniedError: 'NotAllowedError',
|
|
|
PermissionDismissedError: 'NotAllowedError',
|
|
|
InvalidStateError: 'NotAllowedError',
|
|
|
DevicesNotFoundError: 'NotFoundError',
|
|
|
ConstraintNotSatisfiedError: 'OverconstrainedError',
|
|
|
TrackStartError: 'NotReadableError',
|
|
|
MediaDeviceFailedDueToShutdown: 'NotAllowedError',
|
|
|
MediaDeviceKillSwitchOn: 'NotAllowedError',
|
|
|
TabCaptureError: 'AbortError',
|
|
|
ScreenCaptureError: 'AbortError',
|
|
|
DeviceCaptureError: 'AbortError'
|
|
|
}[e.name] || e.name,
|
|
|
message: e.message,
|
|
|
constraint: e.constraint || e.constraintName,
|
|
|
toString() {
|
|
|
return this.name + (this.message && ': ') + this.message;
|
|
|
}
|
|
|
};
|
|
|
};
|
|
|
|
|
|
const getUserMedia_ = function(constraints, onSuccess, onError) {
|
|
|
shimConstraints_(constraints, c => {
|
|
|
navigator.webkitGetUserMedia(c, onSuccess, e => {
|
|
|
if (onError) {
|
|
|
onError(shimError_(e));
|
|
|
}
|
|
|
});
|
|
|
});
|
|
|
};
|
|
|
navigator.getUserMedia = getUserMedia_.bind(navigator);
|
|
|
|
|
|
// Even though Chrome 45 has navigator.mediaDevices and a getUserMedia
|
|
|
// function which returns a Promise, it does not accept spec-style
|
|
|
// constraints.
|
|
|
if (navigator.mediaDevices.getUserMedia) {
|
|
|
const origGetUserMedia = navigator.mediaDevices.getUserMedia.
|
|
|
bind(navigator.mediaDevices);
|
|
|
navigator.mediaDevices.getUserMedia = function(cs) {
|
|
|
return shimConstraints_(cs, c => origGetUserMedia(c).then(stream => {
|
|
|
if (c.audio && !stream.getAudioTracks().length ||
|
|
|
c.video && !stream.getVideoTracks().length) {
|
|
|
stream.getTracks().forEach(track => {
|
|
|
track.stop();
|
|
|
});
|
|
|
throw new DOMException('', 'NotFoundError');
|
|
|
}
|
|
|
return stream;
|
|
|
}, e => Promise.reject(shimError_(e))));
|
|
|
};
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/webrtc-adapter/src/js/common_shim.js":
|
|
|
/*!***********************************************************!*\
|
|
|
!*** ./node_modules/webrtc-adapter/src/js/common_shim.js ***!
|
|
|
\***********************************************************/
|
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
|
|
"use strict";
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
|
/* harmony export */ removeExtmapAllowMixed: () => (/* binding */ removeExtmapAllowMixed),
|
|
|
/* harmony export */ shimAddIceCandidateNullOrEmpty: () => (/* binding */ shimAddIceCandidateNullOrEmpty),
|
|
|
/* harmony export */ shimConnectionState: () => (/* binding */ shimConnectionState),
|
|
|
/* harmony export */ shimMaxMessageSize: () => (/* binding */ shimMaxMessageSize),
|
|
|
/* harmony export */ shimParameterlessSetLocalDescription: () => (/* binding */ shimParameterlessSetLocalDescription),
|
|
|
/* harmony export */ shimRTCIceCandidate: () => (/* binding */ shimRTCIceCandidate),
|
|
|
/* harmony export */ shimRTCIceCandidateRelayProtocol: () => (/* binding */ shimRTCIceCandidateRelayProtocol),
|
|
|
/* harmony export */ shimSendThrowTypeError: () => (/* binding */ shimSendThrowTypeError)
|
|
|
/* harmony export */ });
|
|
|
/* harmony import */ var sdp__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! sdp */ "./node_modules/sdp/sdp.js");
|
|
|
/* harmony import */ var sdp__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(sdp__WEBPACK_IMPORTED_MODULE_0__);
|
|
|
/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./utils */ "./node_modules/webrtc-adapter/src/js/utils.js");
|
|
|
/*
|
|
|
* Copyright (c) 2017 The WebRTC project authors. All Rights Reserved.
|
|
|
*
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
* tree.
|
|
|
*/
|
|
|
/* eslint-env node */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function shimRTCIceCandidate(window) {
|
|
|
// foundation is arbitrarily chosen as an indicator for full support for
|
|
|
// https://w3c.github.io/webrtc-pc/#rtcicecandidate-interface
|
|
|
if (!window.RTCIceCandidate || (window.RTCIceCandidate && 'foundation' in
|
|
|
window.RTCIceCandidate.prototype)) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
const NativeRTCIceCandidate = window.RTCIceCandidate;
|
|
|
window.RTCIceCandidate = function RTCIceCandidate(args) {
|
|
|
// Remove the a= which shouldn't be part of the candidate string.
|
|
|
if (typeof args === 'object' && args.candidate &&
|
|
|
args.candidate.indexOf('a=') === 0) {
|
|
|
args = JSON.parse(JSON.stringify(args));
|
|
|
args.candidate = args.candidate.substring(2);
|
|
|
}
|
|
|
|
|
|
if (args.candidate && args.candidate.length) {
|
|
|
// Augment the native candidate with the parsed fields.
|
|
|
const nativeCandidate = new NativeRTCIceCandidate(args);
|
|
|
const parsedCandidate = sdp__WEBPACK_IMPORTED_MODULE_0___default().parseCandidate(args.candidate);
|
|
|
for (const key in parsedCandidate) {
|
|
|
if (!(key in nativeCandidate)) {
|
|
|
Object.defineProperty(nativeCandidate, key,
|
|
|
{value: parsedCandidate[key]});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// Override serializer to not serialize the extra attributes.
|
|
|
nativeCandidate.toJSON = function toJSON() {
|
|
|
return {
|
|
|
candidate: nativeCandidate.candidate,
|
|
|
sdpMid: nativeCandidate.sdpMid,
|
|
|
sdpMLineIndex: nativeCandidate.sdpMLineIndex,
|
|
|
usernameFragment: nativeCandidate.usernameFragment,
|
|
|
};
|
|
|
};
|
|
|
return nativeCandidate;
|
|
|
}
|
|
|
return new NativeRTCIceCandidate(args);
|
|
|
};
|
|
|
window.RTCIceCandidate.prototype = NativeRTCIceCandidate.prototype;
|
|
|
|
|
|
// Hook up the augmented candidate in onicecandidate and
|
|
|
// addEventListener('icecandidate', ...)
|
|
|
_utils__WEBPACK_IMPORTED_MODULE_1__.wrapPeerConnectionEvent(window, 'icecandidate', e => {
|
|
|
if (e.candidate) {
|
|
|
Object.defineProperty(e, 'candidate', {
|
|
|
value: new window.RTCIceCandidate(e.candidate),
|
|
|
writable: 'false'
|
|
|
});
|
|
|
}
|
|
|
return e;
|
|
|
});
|
|
|
}
|
|
|
|
|
|
function shimRTCIceCandidateRelayProtocol(window) {
|
|
|
if (!window.RTCIceCandidate || (window.RTCIceCandidate && 'relayProtocol' in
|
|
|
window.RTCIceCandidate.prototype)) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// Hook up the augmented candidate in onicecandidate and
|
|
|
// addEventListener('icecandidate', ...)
|
|
|
_utils__WEBPACK_IMPORTED_MODULE_1__.wrapPeerConnectionEvent(window, 'icecandidate', e => {
|
|
|
if (e.candidate) {
|
|
|
const parsedCandidate = sdp__WEBPACK_IMPORTED_MODULE_0___default().parseCandidate(e.candidate.candidate);
|
|
|
if (parsedCandidate.type === 'relay') {
|
|
|
// This is a libwebrtc-specific mapping of local type preference
|
|
|
// to relayProtocol.
|
|
|
e.candidate.relayProtocol = {
|
|
|
0: 'tls',
|
|
|
1: 'tcp',
|
|
|
2: 'udp',
|
|
|
}[parsedCandidate.priority >> 24];
|
|
|
}
|
|
|
}
|
|
|
return e;
|
|
|
});
|
|
|
}
|
|
|
|
|
|
function shimMaxMessageSize(window, browserDetails) {
|
|
|
if (!window.RTCPeerConnection) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
if (!('sctp' in window.RTCPeerConnection.prototype)) {
|
|
|
Object.defineProperty(window.RTCPeerConnection.prototype, 'sctp', {
|
|
|
get() {
|
|
|
return typeof this._sctp === 'undefined' ? null : this._sctp;
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
const sctpInDescription = function(description) {
|
|
|
if (!description || !description.sdp) {
|
|
|
return false;
|
|
|
}
|
|
|
const sections = sdp__WEBPACK_IMPORTED_MODULE_0___default().splitSections(description.sdp);
|
|
|
sections.shift();
|
|
|
return sections.some(mediaSection => {
|
|
|
const mLine = sdp__WEBPACK_IMPORTED_MODULE_0___default().parseMLine(mediaSection);
|
|
|
return mLine && mLine.kind === 'application'
|
|
|
&& mLine.protocol.indexOf('SCTP') !== -1;
|
|
|
});
|
|
|
};
|
|
|
|
|
|
const getRemoteFirefoxVersion = function(description) {
|
|
|
// TODO: Is there a better solution for detecting Firefox?
|
|
|
const match = description.sdp.match(/mozilla...THIS_IS_SDPARTA-(\d+)/);
|
|
|
if (match === null || match.length < 2) {
|
|
|
return -1;
|
|
|
}
|
|
|
const version = parseInt(match[1], 10);
|
|
|
// Test for NaN (yes, this is ugly)
|
|
|
return version !== version ? -1 : version;
|
|
|
};
|
|
|
|
|
|
const getCanSendMaxMessageSize = function(remoteIsFirefox) {
|
|
|
// Every implementation we know can send at least 64 KiB.
|
|
|
// Note: Although Chrome is technically able to send up to 256 KiB, the
|
|
|
// data does not reach the other peer reliably.
|
|
|
// See: https://bugs.chromium.org/p/webrtc/issues/detail?id=8419
|
|
|
let canSendMaxMessageSize = 65536;
|
|
|
if (browserDetails.browser === 'firefox') {
|
|
|
if (browserDetails.version < 57) {
|
|
|
if (remoteIsFirefox === -1) {
|
|
|
// FF < 57 will send in 16 KiB chunks using the deprecated PPID
|
|
|
// fragmentation.
|
|
|
canSendMaxMessageSize = 16384;
|
|
|
} else {
|
|
|
// However, other FF (and RAWRTC) can reassemble PPID-fragmented
|
|
|
// messages. Thus, supporting ~2 GiB when sending.
|
|
|
canSendMaxMessageSize = 2147483637;
|
|
|
}
|
|
|
} else if (browserDetails.version < 60) {
|
|
|
// Currently, all FF >= 57 will reset the remote maximum message size
|
|
|
// to the default value when a data channel is created at a later
|
|
|
// stage. :(
|
|
|
// See: https://bugzilla.mozilla.org/show_bug.cgi?id=1426831
|
|
|
canSendMaxMessageSize =
|
|
|
browserDetails.version === 57 ? 65535 : 65536;
|
|
|
} else {
|
|
|
// FF >= 60 supports sending ~2 GiB
|
|
|
canSendMaxMessageSize = 2147483637;
|
|
|
}
|
|
|
}
|
|
|
return canSendMaxMessageSize;
|
|
|
};
|
|
|
|
|
|
const getMaxMessageSize = function(description, remoteIsFirefox) {
|
|
|
// Note: 65536 bytes is the default value from the SDP spec. Also,
|
|
|
// every implementation we know supports receiving 65536 bytes.
|
|
|
let maxMessageSize = 65536;
|
|
|
|
|
|
// FF 57 has a slightly incorrect default remote max message size, so
|
|
|
// we need to adjust it here to avoid a failure when sending.
|
|
|
// See: https://bugzilla.mozilla.org/show_bug.cgi?id=1425697
|
|
|
if (browserDetails.browser === 'firefox'
|
|
|
&& browserDetails.version === 57) {
|
|
|
maxMessageSize = 65535;
|
|
|
}
|
|
|
|
|
|
const match = sdp__WEBPACK_IMPORTED_MODULE_0___default().matchPrefix(description.sdp,
|
|
|
'a=max-message-size:');
|
|
|
if (match.length > 0) {
|
|
|
maxMessageSize = parseInt(match[0].substring(19), 10);
|
|
|
} else if (browserDetails.browser === 'firefox' &&
|
|
|
remoteIsFirefox !== -1) {
|
|
|
// If the maximum message size is not present in the remote SDP and
|
|
|
// both local and remote are Firefox, the remote peer can receive
|
|
|
// ~2 GiB.
|
|
|
maxMessageSize = 2147483637;
|
|
|
}
|
|
|
return maxMessageSize;
|
|
|
};
|
|
|
|
|
|
const origSetRemoteDescription =
|
|
|
window.RTCPeerConnection.prototype.setRemoteDescription;
|
|
|
window.RTCPeerConnection.prototype.setRemoteDescription =
|
|
|
function setRemoteDescription() {
|
|
|
this._sctp = null;
|
|
|
// Chrome decided to not expose .sctp in plan-b mode.
|
|
|
// As usual, adapter.js has to do an 'ugly worakaround'
|
|
|
// to cover up the mess.
|
|
|
if (browserDetails.browser === 'chrome' && browserDetails.version >= 76) {
|
|
|
const {sdpSemantics} = this.getConfiguration();
|
|
|
if (sdpSemantics === 'plan-b') {
|
|
|
Object.defineProperty(this, 'sctp', {
|
|
|
get() {
|
|
|
return typeof this._sctp === 'undefined' ? null : this._sctp;
|
|
|
},
|
|
|
enumerable: true,
|
|
|
configurable: true,
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (sctpInDescription(arguments[0])) {
|
|
|
// Check if the remote is FF.
|
|
|
const isFirefox = getRemoteFirefoxVersion(arguments[0]);
|
|
|
|
|
|
// Get the maximum message size the local peer is capable of sending
|
|
|
const canSendMMS = getCanSendMaxMessageSize(isFirefox);
|
|
|
|
|
|
// Get the maximum message size of the remote peer.
|
|
|
const remoteMMS = getMaxMessageSize(arguments[0], isFirefox);
|
|
|
|
|
|
// Determine final maximum message size
|
|
|
let maxMessageSize;
|
|
|
if (canSendMMS === 0 && remoteMMS === 0) {
|
|
|
maxMessageSize = Number.POSITIVE_INFINITY;
|
|
|
} else if (canSendMMS === 0 || remoteMMS === 0) {
|
|
|
maxMessageSize = Math.max(canSendMMS, remoteMMS);
|
|
|
} else {
|
|
|
maxMessageSize = Math.min(canSendMMS, remoteMMS);
|
|
|
}
|
|
|
|
|
|
// Create a dummy RTCSctpTransport object and the 'maxMessageSize'
|
|
|
// attribute.
|
|
|
const sctp = {};
|
|
|
Object.defineProperty(sctp, 'maxMessageSize', {
|
|
|
get() {
|
|
|
return maxMessageSize;
|
|
|
}
|
|
|
});
|
|
|
this._sctp = sctp;
|
|
|
}
|
|
|
|
|
|
return origSetRemoteDescription.apply(this, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function shimSendThrowTypeError(window) {
|
|
|
if (!(window.RTCPeerConnection &&
|
|
|
'createDataChannel' in window.RTCPeerConnection.prototype)) {
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// Note: Although Firefox >= 57 has a native implementation, the maximum
|
|
|
// message size can be reset for all data channels at a later stage.
|
|
|
// See: https://bugzilla.mozilla.org/show_bug.cgi?id=1426831
|
|
|
|
|
|
function wrapDcSend(dc, pc) {
|
|
|
const origDataChannelSend = dc.send;
|
|
|
dc.send = function send() {
|
|
|
const data = arguments[0];
|
|
|
const length = data.length || data.size || data.byteLength;
|
|
|
if (dc.readyState === 'open' &&
|
|
|
pc.sctp && length > pc.sctp.maxMessageSize) {
|
|
|
throw new TypeError('Message too large (can send a maximum of ' +
|
|
|
pc.sctp.maxMessageSize + ' bytes)');
|
|
|
}
|
|
|
return origDataChannelSend.apply(dc, arguments);
|
|
|
};
|
|
|
}
|
|
|
const origCreateDataChannel =
|
|
|
window.RTCPeerConnection.prototype.createDataChannel;
|
|
|
window.RTCPeerConnection.prototype.createDataChannel =
|
|
|
function createDataChannel() {
|
|
|
const dataChannel = origCreateDataChannel.apply(this, arguments);
|
|
|
wrapDcSend(dataChannel, this);
|
|
|
return dataChannel;
|
|
|
};
|
|
|
_utils__WEBPACK_IMPORTED_MODULE_1__.wrapPeerConnectionEvent(window, 'datachannel', e => {
|
|
|
wrapDcSend(e.channel, e.target);
|
|
|
return e;
|
|
|
});
|
|
|
}
|
|
|
|
|
|
|
|
|
/* shims RTCConnectionState by pretending it is the same as iceConnectionState.
|
|
|
* See https://bugs.chromium.org/p/webrtc/issues/detail?id=6145#c12
|
|
|
* for why this is a valid hack in Chrome. In Firefox it is slightly incorrect
|
|
|
* since DTLS failures would be hidden. See
|
|
|
* https://bugzilla.mozilla.org/show_bug.cgi?id=1265827
|
|
|
* for the Firefox tracking bug.
|
|
|
*/
|
|
|
function shimConnectionState(window) {
|
|
|
if (!window.RTCPeerConnection ||
|
|
|
'connectionState' in window.RTCPeerConnection.prototype) {
|
|
|
return;
|
|
|
}
|
|
|
const proto = window.RTCPeerConnection.prototype;
|
|
|
Object.defineProperty(proto, 'connectionState', {
|
|
|
get() {
|
|
|
return {
|
|
|
completed: 'connected',
|
|
|
checking: 'connecting'
|
|
|
}[this.iceConnectionState] || this.iceConnectionState;
|
|
|
},
|
|
|
enumerable: true,
|
|
|
configurable: true
|
|
|
});
|
|
|
Object.defineProperty(proto, 'onconnectionstatechange', {
|
|
|
get() {
|
|
|
return this._onconnectionstatechange || null;
|
|
|
},
|
|
|
set(cb) {
|
|
|
if (this._onconnectionstatechange) {
|
|
|
this.removeEventListener('connectionstatechange',
|
|
|
this._onconnectionstatechange);
|
|
|
delete this._onconnectionstatechange;
|
|
|
}
|
|
|
if (cb) {
|
|
|
this.addEventListener('connectionstatechange',
|
|
|
this._onconnectionstatechange = cb);
|
|
|
}
|
|
|
},
|
|
|
enumerable: true,
|
|
|
configurable: true
|
|
|
});
|
|
|
|
|
|
['setLocalDescription', 'setRemoteDescription'].forEach((method) => {
|
|
|
const origMethod = proto[method];
|
|
|
proto[method] = function() {
|
|
|
if (!this._connectionstatechangepoly) {
|
|
|
this._connectionstatechangepoly = e => {
|
|
|
const pc = e.target;
|
|
|
if (pc._lastConnectionState !== pc.connectionState) {
|
|
|
pc._lastConnectionState = pc.connectionState;
|
|
|
const newEvent = new Event('connectionstatechange', e);
|
|
|
pc.dispatchEvent(newEvent);
|
|
|
}
|
|
|
return e;
|
|
|
};
|
|
|
this.addEventListener('iceconnectionstatechange',
|
|
|
this._connectionstatechangepoly);
|
|
|
}
|
|
|
return origMethod.apply(this, arguments);
|
|
|
};
|
|
|
});
|
|
|
}
|
|
|
|
|
|
function removeExtmapAllowMixed(window, browserDetails) {
|
|
|
/* remove a=extmap-allow-mixed for webrtc.org < M71 */
|
|
|
if (!window.RTCPeerConnection) {
|
|
|
return;
|
|
|
}
|
|
|
if (browserDetails.browser === 'chrome' && browserDetails.version >= 71) {
|
|
|
return;
|
|
|
}
|
|
|
if (browserDetails.browser === 'safari' && browserDetails.version >= 605) {
|
|
|
return;
|
|
|
}
|
|
|
const nativeSRD = window.RTCPeerConnection.prototype.setRemoteDescription;
|
|
|
window.RTCPeerConnection.prototype.setRemoteDescription =
|
|
|
function setRemoteDescription(desc) {
|
|
|
if (desc && desc.sdp && desc.sdp.indexOf('\na=extmap-allow-mixed') !== -1) {
|
|
|
const sdp = desc.sdp.split('\n').filter((line) => {
|
|
|
return line.trim() !== 'a=extmap-allow-mixed';
|
|
|
}).join('\n');
|
|
|
// Safari enforces read-only-ness of RTCSessionDescription fields.
|
|
|
if (window.RTCSessionDescription &&
|
|
|
desc instanceof window.RTCSessionDescription) {
|
|
|
arguments[0] = new window.RTCSessionDescription({
|
|
|
type: desc.type,
|
|
|
sdp,
|
|
|
});
|
|
|
} else {
|
|
|
desc.sdp = sdp;
|
|
|
}
|
|
|
}
|
|
|
return nativeSRD.apply(this, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function shimAddIceCandidateNullOrEmpty(window, browserDetails) {
|
|
|
// Support for addIceCandidate(null or undefined)
|
|
|
// as well as addIceCandidate({candidate: "", ...})
|
|
|
// https://bugs.chromium.org/p/chromium/issues/detail?id=978582
|
|
|
// Note: must be called before other polyfills which change the signature.
|
|
|
if (!(window.RTCPeerConnection && window.RTCPeerConnection.prototype)) {
|
|
|
return;
|
|
|
}
|
|
|
const nativeAddIceCandidate =
|
|
|
window.RTCPeerConnection.prototype.addIceCandidate;
|
|
|
if (!nativeAddIceCandidate || nativeAddIceCandidate.length === 0) {
|
|
|
return;
|
|
|
}
|
|
|
window.RTCPeerConnection.prototype.addIceCandidate =
|
|
|
function addIceCandidate() {
|
|
|
if (!arguments[0]) {
|
|
|
if (arguments[1]) {
|
|
|
arguments[1].apply(null);
|
|
|
}
|
|
|
return Promise.resolve();
|
|
|
}
|
|
|
// Firefox 68+ emits and processes {candidate: "", ...}, ignore
|
|
|
// in older versions.
|
|
|
// Native support for ignoring exists for Chrome M77+.
|
|
|
// Safari ignores as well, exact version unknown but works in the same
|
|
|
// version that also ignores addIceCandidate(null).
|
|
|
if (((browserDetails.browser === 'chrome' && browserDetails.version < 78)
|
|
|
|| (browserDetails.browser === 'firefox'
|
|
|
&& browserDetails.version < 68)
|
|
|
|| (browserDetails.browser === 'safari'))
|
|
|
&& arguments[0] && arguments[0].candidate === '') {
|
|
|
return Promise.resolve();
|
|
|
}
|
|
|
return nativeAddIceCandidate.apply(this, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
// Note: Make sure to call this ahead of APIs that modify
|
|
|
// setLocalDescription.length
|
|
|
function shimParameterlessSetLocalDescription(window, browserDetails) {
|
|
|
if (!(window.RTCPeerConnection && window.RTCPeerConnection.prototype)) {
|
|
|
return;
|
|
|
}
|
|
|
const nativeSetLocalDescription =
|
|
|
window.RTCPeerConnection.prototype.setLocalDescription;
|
|
|
if (!nativeSetLocalDescription || nativeSetLocalDescription.length === 0) {
|
|
|
return;
|
|
|
}
|
|
|
window.RTCPeerConnection.prototype.setLocalDescription =
|
|
|
function setLocalDescription() {
|
|
|
let desc = arguments[0] || {};
|
|
|
if (typeof desc !== 'object' || (desc.type && desc.sdp)) {
|
|
|
return nativeSetLocalDescription.apply(this, arguments);
|
|
|
}
|
|
|
// The remaining steps should technically happen when SLD comes off the
|
|
|
// RTCPeerConnection's operations chain (not ahead of going on it), but
|
|
|
// this is too difficult to shim. Instead, this shim only covers the
|
|
|
// common case where the operations chain is empty. This is imperfect, but
|
|
|
// should cover many cases. Rationale: Even if we can't reduce the glare
|
|
|
// window to zero on imperfect implementations, there's value in tapping
|
|
|
// into the perfect negotiation pattern that several browsers support.
|
|
|
desc = {type: desc.type, sdp: desc.sdp};
|
|
|
if (!desc.type) {
|
|
|
switch (this.signalingState) {
|
|
|
case 'stable':
|
|
|
case 'have-local-offer':
|
|
|
case 'have-remote-pranswer':
|
|
|
desc.type = 'offer';
|
|
|
break;
|
|
|
default:
|
|
|
desc.type = 'answer';
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
if (desc.sdp || (desc.type !== 'offer' && desc.type !== 'answer')) {
|
|
|
return nativeSetLocalDescription.apply(this, [desc]);
|
|
|
}
|
|
|
const func = desc.type === 'offer' ? this.createOffer : this.createAnswer;
|
|
|
return func.apply(this)
|
|
|
.then(d => nativeSetLocalDescription.apply(this, [d]));
|
|
|
};
|
|
|
}
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/webrtc-adapter/src/js/firefox/firefox_shim.js":
|
|
|
/*!********************************************************************!*\
|
|
|
!*** ./node_modules/webrtc-adapter/src/js/firefox/firefox_shim.js ***!
|
|
|
\********************************************************************/
|
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
|
|
"use strict";
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
|
/* harmony export */ shimAddTransceiver: () => (/* binding */ shimAddTransceiver),
|
|
|
/* harmony export */ shimCreateAnswer: () => (/* binding */ shimCreateAnswer),
|
|
|
/* harmony export */ shimCreateOffer: () => (/* binding */ shimCreateOffer),
|
|
|
/* harmony export */ shimGetDisplayMedia: () => (/* reexport safe */ _getdisplaymedia__WEBPACK_IMPORTED_MODULE_2__.shimGetDisplayMedia),
|
|
|
/* harmony export */ shimGetParameters: () => (/* binding */ shimGetParameters),
|
|
|
/* harmony export */ shimGetUserMedia: () => (/* reexport safe */ _getusermedia__WEBPACK_IMPORTED_MODULE_1__.shimGetUserMedia),
|
|
|
/* harmony export */ shimOnTrack: () => (/* binding */ shimOnTrack),
|
|
|
/* harmony export */ shimPeerConnection: () => (/* binding */ shimPeerConnection),
|
|
|
/* harmony export */ shimRTCDataChannel: () => (/* binding */ shimRTCDataChannel),
|
|
|
/* harmony export */ shimReceiverGetStats: () => (/* binding */ shimReceiverGetStats),
|
|
|
/* harmony export */ shimRemoveStream: () => (/* binding */ shimRemoveStream),
|
|
|
/* harmony export */ shimSenderGetStats: () => (/* binding */ shimSenderGetStats)
|
|
|
/* harmony export */ });
|
|
|
/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../utils */ "./node_modules/webrtc-adapter/src/js/utils.js");
|
|
|
/* harmony import */ var _getusermedia__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./getusermedia */ "./node_modules/webrtc-adapter/src/js/firefox/getusermedia.js");
|
|
|
/* harmony import */ var _getdisplaymedia__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./getdisplaymedia */ "./node_modules/webrtc-adapter/src/js/firefox/getdisplaymedia.js");
|
|
|
/*
|
|
|
* Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.
|
|
|
*
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
* tree.
|
|
|
*/
|
|
|
/* eslint-env node */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function shimOnTrack(window) {
|
|
|
if (typeof window === 'object' && window.RTCTrackEvent &&
|
|
|
('receiver' in window.RTCTrackEvent.prototype) &&
|
|
|
!('transceiver' in window.RTCTrackEvent.prototype)) {
|
|
|
Object.defineProperty(window.RTCTrackEvent.prototype, 'transceiver', {
|
|
|
get() {
|
|
|
return {receiver: this.receiver};
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function shimPeerConnection(window, browserDetails) {
|
|
|
if (typeof window !== 'object' ||
|
|
|
!(window.RTCPeerConnection || window.mozRTCPeerConnection)) {
|
|
|
return; // probably media.peerconnection.enabled=false in about:config
|
|
|
}
|
|
|
if (!window.RTCPeerConnection && window.mozRTCPeerConnection) {
|
|
|
// very basic support for old versions.
|
|
|
window.RTCPeerConnection = window.mozRTCPeerConnection;
|
|
|
}
|
|
|
|
|
|
if (browserDetails.version < 53) {
|
|
|
// shim away need for obsolete RTCIceCandidate/RTCSessionDescription.
|
|
|
['setLocalDescription', 'setRemoteDescription', 'addIceCandidate']
|
|
|
.forEach(function(method) {
|
|
|
const nativeMethod = window.RTCPeerConnection.prototype[method];
|
|
|
const methodObj = {[method]() {
|
|
|
arguments[0] = new ((method === 'addIceCandidate') ?
|
|
|
window.RTCIceCandidate :
|
|
|
window.RTCSessionDescription)(arguments[0]);
|
|
|
return nativeMethod.apply(this, arguments);
|
|
|
}};
|
|
|
window.RTCPeerConnection.prototype[method] = methodObj[method];
|
|
|
});
|
|
|
}
|
|
|
|
|
|
const modernStatsTypes = {
|
|
|
inboundrtp: 'inbound-rtp',
|
|
|
outboundrtp: 'outbound-rtp',
|
|
|
candidatepair: 'candidate-pair',
|
|
|
localcandidate: 'local-candidate',
|
|
|
remotecandidate: 'remote-candidate'
|
|
|
};
|
|
|
|
|
|
const nativeGetStats = window.RTCPeerConnection.prototype.getStats;
|
|
|
window.RTCPeerConnection.prototype.getStats = function getStats() {
|
|
|
const [selector, onSucc, onErr] = arguments;
|
|
|
return nativeGetStats.apply(this, [selector || null])
|
|
|
.then(stats => {
|
|
|
if (browserDetails.version < 53 && !onSucc) {
|
|
|
// Shim only promise getStats with spec-hyphens in type names
|
|
|
// Leave callback version alone; misc old uses of forEach before Map
|
|
|
try {
|
|
|
stats.forEach(stat => {
|
|
|
stat.type = modernStatsTypes[stat.type] || stat.type;
|
|
|
});
|
|
|
} catch (e) {
|
|
|
if (e.name !== 'TypeError') {
|
|
|
throw e;
|
|
|
}
|
|
|
// Avoid TypeError: "type" is read-only, in old versions. 34-43ish
|
|
|
stats.forEach((stat, i) => {
|
|
|
stats.set(i, Object.assign({}, stat, {
|
|
|
type: modernStatsTypes[stat.type] || stat.type
|
|
|
}));
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
return stats;
|
|
|
})
|
|
|
.then(onSucc, onErr);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function shimSenderGetStats(window) {
|
|
|
if (!(typeof window === 'object' && window.RTCPeerConnection &&
|
|
|
window.RTCRtpSender)) {
|
|
|
return;
|
|
|
}
|
|
|
if (window.RTCRtpSender && 'getStats' in window.RTCRtpSender.prototype) {
|
|
|
return;
|
|
|
}
|
|
|
const origGetSenders = window.RTCPeerConnection.prototype.getSenders;
|
|
|
if (origGetSenders) {
|
|
|
window.RTCPeerConnection.prototype.getSenders = function getSenders() {
|
|
|
const senders = origGetSenders.apply(this, []);
|
|
|
senders.forEach(sender => sender._pc = this);
|
|
|
return senders;
|
|
|
};
|
|
|
}
|
|
|
|
|
|
const origAddTrack = window.RTCPeerConnection.prototype.addTrack;
|
|
|
if (origAddTrack) {
|
|
|
window.RTCPeerConnection.prototype.addTrack = function addTrack() {
|
|
|
const sender = origAddTrack.apply(this, arguments);
|
|
|
sender._pc = this;
|
|
|
return sender;
|
|
|
};
|
|
|
}
|
|
|
window.RTCRtpSender.prototype.getStats = function getStats() {
|
|
|
return this.track ? this._pc.getStats(this.track) :
|
|
|
Promise.resolve(new Map());
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function shimReceiverGetStats(window) {
|
|
|
if (!(typeof window === 'object' && window.RTCPeerConnection &&
|
|
|
window.RTCRtpSender)) {
|
|
|
return;
|
|
|
}
|
|
|
if (window.RTCRtpSender && 'getStats' in window.RTCRtpReceiver.prototype) {
|
|
|
return;
|
|
|
}
|
|
|
const origGetReceivers = window.RTCPeerConnection.prototype.getReceivers;
|
|
|
if (origGetReceivers) {
|
|
|
window.RTCPeerConnection.prototype.getReceivers = function getReceivers() {
|
|
|
const receivers = origGetReceivers.apply(this, []);
|
|
|
receivers.forEach(receiver => receiver._pc = this);
|
|
|
return receivers;
|
|
|
};
|
|
|
}
|
|
|
_utils__WEBPACK_IMPORTED_MODULE_0__.wrapPeerConnectionEvent(window, 'track', e => {
|
|
|
e.receiver._pc = e.srcElement;
|
|
|
return e;
|
|
|
});
|
|
|
window.RTCRtpReceiver.prototype.getStats = function getStats() {
|
|
|
return this._pc.getStats(this.track);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function shimRemoveStream(window) {
|
|
|
if (!window.RTCPeerConnection ||
|
|
|
'removeStream' in window.RTCPeerConnection.prototype) {
|
|
|
return;
|
|
|
}
|
|
|
window.RTCPeerConnection.prototype.removeStream =
|
|
|
function removeStream(stream) {
|
|
|
_utils__WEBPACK_IMPORTED_MODULE_0__.deprecated('removeStream', 'removeTrack');
|
|
|
this.getSenders().forEach(sender => {
|
|
|
if (sender.track && stream.getTracks().includes(sender.track)) {
|
|
|
this.removeTrack(sender);
|
|
|
}
|
|
|
});
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function shimRTCDataChannel(window) {
|
|
|
// rename DataChannel to RTCDataChannel (native fix in FF60):
|
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=1173851
|
|
|
if (window.DataChannel && !window.RTCDataChannel) {
|
|
|
window.RTCDataChannel = window.DataChannel;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function shimAddTransceiver(window) {
|
|
|
// https://github.com/webrtcHacks/adapter/issues/998#issuecomment-516921647
|
|
|
// Firefox ignores the init sendEncodings options passed to addTransceiver
|
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=1396918
|
|
|
if (!(typeof window === 'object' && window.RTCPeerConnection)) {
|
|
|
return;
|
|
|
}
|
|
|
const origAddTransceiver = window.RTCPeerConnection.prototype.addTransceiver;
|
|
|
if (origAddTransceiver) {
|
|
|
window.RTCPeerConnection.prototype.addTransceiver =
|
|
|
function addTransceiver() {
|
|
|
this.setParametersPromises = [];
|
|
|
// WebIDL input coercion and validation
|
|
|
let sendEncodings = arguments[1] && arguments[1].sendEncodings;
|
|
|
if (sendEncodings === undefined) {
|
|
|
sendEncodings = [];
|
|
|
}
|
|
|
sendEncodings = [...sendEncodings];
|
|
|
const shouldPerformCheck = sendEncodings.length > 0;
|
|
|
if (shouldPerformCheck) {
|
|
|
// If sendEncodings params are provided, validate grammar
|
|
|
sendEncodings.forEach((encodingParam) => {
|
|
|
if ('rid' in encodingParam) {
|
|
|
const ridRegex = /^[a-z0-9]{0,16}$/i;
|
|
|
if (!ridRegex.test(encodingParam.rid)) {
|
|
|
throw new TypeError('Invalid RID value provided.');
|
|
|
}
|
|
|
}
|
|
|
if ('scaleResolutionDownBy' in encodingParam) {
|
|
|
if (!(parseFloat(encodingParam.scaleResolutionDownBy) >= 1.0)) {
|
|
|
throw new RangeError('scale_resolution_down_by must be >= 1.0');
|
|
|
}
|
|
|
}
|
|
|
if ('maxFramerate' in encodingParam) {
|
|
|
if (!(parseFloat(encodingParam.maxFramerate) >= 0)) {
|
|
|
throw new RangeError('max_framerate must be >= 0.0');
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
const transceiver = origAddTransceiver.apply(this, arguments);
|
|
|
if (shouldPerformCheck) {
|
|
|
// Check if the init options were applied. If not we do this in an
|
|
|
// asynchronous way and save the promise reference in a global object.
|
|
|
// This is an ugly hack, but at the same time is way more robust than
|
|
|
// checking the sender parameters before and after the createOffer
|
|
|
// Also note that after the createoffer we are not 100% sure that
|
|
|
// the params were asynchronously applied so we might miss the
|
|
|
// opportunity to recreate offer.
|
|
|
const {sender} = transceiver;
|
|
|
const params = sender.getParameters();
|
|
|
if (!('encodings' in params) ||
|
|
|
// Avoid being fooled by patched getParameters() below.
|
|
|
(params.encodings.length === 1 &&
|
|
|
Object.keys(params.encodings[0]).length === 0)) {
|
|
|
params.encodings = sendEncodings;
|
|
|
sender.sendEncodings = sendEncodings;
|
|
|
this.setParametersPromises.push(sender.setParameters(params)
|
|
|
.then(() => {
|
|
|
delete sender.sendEncodings;
|
|
|
}).catch(() => {
|
|
|
delete sender.sendEncodings;
|
|
|
})
|
|
|
);
|
|
|
}
|
|
|
}
|
|
|
return transceiver;
|
|
|
};
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function shimGetParameters(window) {
|
|
|
if (!(typeof window === 'object' && window.RTCRtpSender)) {
|
|
|
return;
|
|
|
}
|
|
|
const origGetParameters = window.RTCRtpSender.prototype.getParameters;
|
|
|
if (origGetParameters) {
|
|
|
window.RTCRtpSender.prototype.getParameters =
|
|
|
function getParameters() {
|
|
|
const params = origGetParameters.apply(this, arguments);
|
|
|
if (!('encodings' in params)) {
|
|
|
params.encodings = [].concat(this.sendEncodings || [{}]);
|
|
|
}
|
|
|
return params;
|
|
|
};
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function shimCreateOffer(window) {
|
|
|
// https://github.com/webrtcHacks/adapter/issues/998#issuecomment-516921647
|
|
|
// Firefox ignores the init sendEncodings options passed to addTransceiver
|
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=1396918
|
|
|
if (!(typeof window === 'object' && window.RTCPeerConnection)) {
|
|
|
return;
|
|
|
}
|
|
|
const origCreateOffer = window.RTCPeerConnection.prototype.createOffer;
|
|
|
window.RTCPeerConnection.prototype.createOffer = function createOffer() {
|
|
|
if (this.setParametersPromises && this.setParametersPromises.length) {
|
|
|
return Promise.all(this.setParametersPromises)
|
|
|
.then(() => {
|
|
|
return origCreateOffer.apply(this, arguments);
|
|
|
})
|
|
|
.finally(() => {
|
|
|
this.setParametersPromises = [];
|
|
|
});
|
|
|
}
|
|
|
return origCreateOffer.apply(this, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function shimCreateAnswer(window) {
|
|
|
// https://github.com/webrtcHacks/adapter/issues/998#issuecomment-516921647
|
|
|
// Firefox ignores the init sendEncodings options passed to addTransceiver
|
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=1396918
|
|
|
if (!(typeof window === 'object' && window.RTCPeerConnection)) {
|
|
|
return;
|
|
|
}
|
|
|
const origCreateAnswer = window.RTCPeerConnection.prototype.createAnswer;
|
|
|
window.RTCPeerConnection.prototype.createAnswer = function createAnswer() {
|
|
|
if (this.setParametersPromises && this.setParametersPromises.length) {
|
|
|
return Promise.all(this.setParametersPromises)
|
|
|
.then(() => {
|
|
|
return origCreateAnswer.apply(this, arguments);
|
|
|
})
|
|
|
.finally(() => {
|
|
|
this.setParametersPromises = [];
|
|
|
});
|
|
|
}
|
|
|
return origCreateAnswer.apply(this, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/webrtc-adapter/src/js/firefox/getdisplaymedia.js":
|
|
|
/*!***********************************************************************!*\
|
|
|
!*** ./node_modules/webrtc-adapter/src/js/firefox/getdisplaymedia.js ***!
|
|
|
\***********************************************************************/
|
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
|
|
"use strict";
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
|
/* harmony export */ shimGetDisplayMedia: () => (/* binding */ shimGetDisplayMedia)
|
|
|
/* harmony export */ });
|
|
|
/*
|
|
|
* Copyright (c) 2018 The adapter.js project authors. All Rights Reserved.
|
|
|
*
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
* tree.
|
|
|
*/
|
|
|
/* eslint-env node */
|
|
|
|
|
|
|
|
|
function shimGetDisplayMedia(window, preferredMediaSource) {
|
|
|
if (window.navigator.mediaDevices &&
|
|
|
'getDisplayMedia' in window.navigator.mediaDevices) {
|
|
|
return;
|
|
|
}
|
|
|
if (!(window.navigator.mediaDevices)) {
|
|
|
return;
|
|
|
}
|
|
|
window.navigator.mediaDevices.getDisplayMedia =
|
|
|
function getDisplayMedia(constraints) {
|
|
|
if (!(constraints && constraints.video)) {
|
|
|
const err = new DOMException('getDisplayMedia without video ' +
|
|
|
'constraints is undefined');
|
|
|
err.name = 'NotFoundError';
|
|
|
// from https://heycam.github.io/webidl/#idl-DOMException-error-names
|
|
|
err.code = 8;
|
|
|
return Promise.reject(err);
|
|
|
}
|
|
|
if (constraints.video === true) {
|
|
|
constraints.video = {mediaSource: preferredMediaSource};
|
|
|
} else {
|
|
|
constraints.video.mediaSource = preferredMediaSource;
|
|
|
}
|
|
|
return window.navigator.mediaDevices.getUserMedia(constraints);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/webrtc-adapter/src/js/firefox/getusermedia.js":
|
|
|
/*!********************************************************************!*\
|
|
|
!*** ./node_modules/webrtc-adapter/src/js/firefox/getusermedia.js ***!
|
|
|
\********************************************************************/
|
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
|
|
"use strict";
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
|
/* harmony export */ shimGetUserMedia: () => (/* binding */ shimGetUserMedia)
|
|
|
/* harmony export */ });
|
|
|
/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../utils */ "./node_modules/webrtc-adapter/src/js/utils.js");
|
|
|
/*
|
|
|
* Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.
|
|
|
*
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
* tree.
|
|
|
*/
|
|
|
/* eslint-env node */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function shimGetUserMedia(window, browserDetails) {
|
|
|
const navigator = window && window.navigator;
|
|
|
const MediaStreamTrack = window && window.MediaStreamTrack;
|
|
|
|
|
|
navigator.getUserMedia = function(constraints, onSuccess, onError) {
|
|
|
// Replace Firefox 44+'s deprecation warning with unprefixed version.
|
|
|
_utils__WEBPACK_IMPORTED_MODULE_0__.deprecated('navigator.getUserMedia',
|
|
|
'navigator.mediaDevices.getUserMedia');
|
|
|
navigator.mediaDevices.getUserMedia(constraints).then(onSuccess, onError);
|
|
|
};
|
|
|
|
|
|
if (!(browserDetails.version > 55 &&
|
|
|
'autoGainControl' in navigator.mediaDevices.getSupportedConstraints())) {
|
|
|
const remap = function(obj, a, b) {
|
|
|
if (a in obj && !(b in obj)) {
|
|
|
obj[b] = obj[a];
|
|
|
delete obj[a];
|
|
|
}
|
|
|
};
|
|
|
|
|
|
const nativeGetUserMedia = navigator.mediaDevices.getUserMedia.
|
|
|
bind(navigator.mediaDevices);
|
|
|
navigator.mediaDevices.getUserMedia = function(c) {
|
|
|
if (typeof c === 'object' && typeof c.audio === 'object') {
|
|
|
c = JSON.parse(JSON.stringify(c));
|
|
|
remap(c.audio, 'autoGainControl', 'mozAutoGainControl');
|
|
|
remap(c.audio, 'noiseSuppression', 'mozNoiseSuppression');
|
|
|
}
|
|
|
return nativeGetUserMedia(c);
|
|
|
};
|
|
|
|
|
|
if (MediaStreamTrack && MediaStreamTrack.prototype.getSettings) {
|
|
|
const nativeGetSettings = MediaStreamTrack.prototype.getSettings;
|
|
|
MediaStreamTrack.prototype.getSettings = function() {
|
|
|
const obj = nativeGetSettings.apply(this, arguments);
|
|
|
remap(obj, 'mozAutoGainControl', 'autoGainControl');
|
|
|
remap(obj, 'mozNoiseSuppression', 'noiseSuppression');
|
|
|
return obj;
|
|
|
};
|
|
|
}
|
|
|
|
|
|
if (MediaStreamTrack && MediaStreamTrack.prototype.applyConstraints) {
|
|
|
const nativeApplyConstraints =
|
|
|
MediaStreamTrack.prototype.applyConstraints;
|
|
|
MediaStreamTrack.prototype.applyConstraints = function(c) {
|
|
|
if (this.kind === 'audio' && typeof c === 'object') {
|
|
|
c = JSON.parse(JSON.stringify(c));
|
|
|
remap(c, 'autoGainControl', 'mozAutoGainControl');
|
|
|
remap(c, 'noiseSuppression', 'mozNoiseSuppression');
|
|
|
}
|
|
|
return nativeApplyConstraints.apply(this, [c]);
|
|
|
};
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/webrtc-adapter/src/js/safari/safari_shim.js":
|
|
|
/*!******************************************************************!*\
|
|
|
!*** ./node_modules/webrtc-adapter/src/js/safari/safari_shim.js ***!
|
|
|
\******************************************************************/
|
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
|
|
"use strict";
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
|
/* harmony export */ shimAudioContext: () => (/* binding */ shimAudioContext),
|
|
|
/* harmony export */ shimCallbacksAPI: () => (/* binding */ shimCallbacksAPI),
|
|
|
/* harmony export */ shimConstraints: () => (/* binding */ shimConstraints),
|
|
|
/* harmony export */ shimCreateOfferLegacy: () => (/* binding */ shimCreateOfferLegacy),
|
|
|
/* harmony export */ shimGetUserMedia: () => (/* binding */ shimGetUserMedia),
|
|
|
/* harmony export */ shimLocalStreamsAPI: () => (/* binding */ shimLocalStreamsAPI),
|
|
|
/* harmony export */ shimRTCIceServerUrls: () => (/* binding */ shimRTCIceServerUrls),
|
|
|
/* harmony export */ shimRemoteStreamsAPI: () => (/* binding */ shimRemoteStreamsAPI),
|
|
|
/* harmony export */ shimTrackEventTransceiver: () => (/* binding */ shimTrackEventTransceiver)
|
|
|
/* harmony export */ });
|
|
|
/* harmony import */ var _utils__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../utils */ "./node_modules/webrtc-adapter/src/js/utils.js");
|
|
|
/*
|
|
|
* Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.
|
|
|
*
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
* tree.
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
function shimLocalStreamsAPI(window) {
|
|
|
if (typeof window !== 'object' || !window.RTCPeerConnection) {
|
|
|
return;
|
|
|
}
|
|
|
if (!('getLocalStreams' in window.RTCPeerConnection.prototype)) {
|
|
|
window.RTCPeerConnection.prototype.getLocalStreams =
|
|
|
function getLocalStreams() {
|
|
|
if (!this._localStreams) {
|
|
|
this._localStreams = [];
|
|
|
}
|
|
|
return this._localStreams;
|
|
|
};
|
|
|
}
|
|
|
if (!('addStream' in window.RTCPeerConnection.prototype)) {
|
|
|
const _addTrack = window.RTCPeerConnection.prototype.addTrack;
|
|
|
window.RTCPeerConnection.prototype.addStream = function addStream(stream) {
|
|
|
if (!this._localStreams) {
|
|
|
this._localStreams = [];
|
|
|
}
|
|
|
if (!this._localStreams.includes(stream)) {
|
|
|
this._localStreams.push(stream);
|
|
|
}
|
|
|
// Try to emulate Chrome's behaviour of adding in audio-video order.
|
|
|
// Safari orders by track id.
|
|
|
stream.getAudioTracks().forEach(track => _addTrack.call(this, track,
|
|
|
stream));
|
|
|
stream.getVideoTracks().forEach(track => _addTrack.call(this, track,
|
|
|
stream));
|
|
|
};
|
|
|
|
|
|
window.RTCPeerConnection.prototype.addTrack =
|
|
|
function addTrack(track, ...streams) {
|
|
|
if (streams) {
|
|
|
streams.forEach((stream) => {
|
|
|
if (!this._localStreams) {
|
|
|
this._localStreams = [stream];
|
|
|
} else if (!this._localStreams.includes(stream)) {
|
|
|
this._localStreams.push(stream);
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
return _addTrack.apply(this, arguments);
|
|
|
};
|
|
|
}
|
|
|
if (!('removeStream' in window.RTCPeerConnection.prototype)) {
|
|
|
window.RTCPeerConnection.prototype.removeStream =
|
|
|
function removeStream(stream) {
|
|
|
if (!this._localStreams) {
|
|
|
this._localStreams = [];
|
|
|
}
|
|
|
const index = this._localStreams.indexOf(stream);
|
|
|
if (index === -1) {
|
|
|
return;
|
|
|
}
|
|
|
this._localStreams.splice(index, 1);
|
|
|
const tracks = stream.getTracks();
|
|
|
this.getSenders().forEach(sender => {
|
|
|
if (tracks.includes(sender.track)) {
|
|
|
this.removeTrack(sender);
|
|
|
}
|
|
|
});
|
|
|
};
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function shimRemoteStreamsAPI(window) {
|
|
|
if (typeof window !== 'object' || !window.RTCPeerConnection) {
|
|
|
return;
|
|
|
}
|
|
|
if (!('getRemoteStreams' in window.RTCPeerConnection.prototype)) {
|
|
|
window.RTCPeerConnection.prototype.getRemoteStreams =
|
|
|
function getRemoteStreams() {
|
|
|
return this._remoteStreams ? this._remoteStreams : [];
|
|
|
};
|
|
|
}
|
|
|
if (!('onaddstream' in window.RTCPeerConnection.prototype)) {
|
|
|
Object.defineProperty(window.RTCPeerConnection.prototype, 'onaddstream', {
|
|
|
get() {
|
|
|
return this._onaddstream;
|
|
|
},
|
|
|
set(f) {
|
|
|
if (this._onaddstream) {
|
|
|
this.removeEventListener('addstream', this._onaddstream);
|
|
|
this.removeEventListener('track', this._onaddstreampoly);
|
|
|
}
|
|
|
this.addEventListener('addstream', this._onaddstream = f);
|
|
|
this.addEventListener('track', this._onaddstreampoly = (e) => {
|
|
|
e.streams.forEach(stream => {
|
|
|
if (!this._remoteStreams) {
|
|
|
this._remoteStreams = [];
|
|
|
}
|
|
|
if (this._remoteStreams.includes(stream)) {
|
|
|
return;
|
|
|
}
|
|
|
this._remoteStreams.push(stream);
|
|
|
const event = new Event('addstream');
|
|
|
event.stream = stream;
|
|
|
this.dispatchEvent(event);
|
|
|
});
|
|
|
});
|
|
|
}
|
|
|
});
|
|
|
const origSetRemoteDescription =
|
|
|
window.RTCPeerConnection.prototype.setRemoteDescription;
|
|
|
window.RTCPeerConnection.prototype.setRemoteDescription =
|
|
|
function setRemoteDescription() {
|
|
|
const pc = this;
|
|
|
if (!this._onaddstreampoly) {
|
|
|
this.addEventListener('track', this._onaddstreampoly = function(e) {
|
|
|
e.streams.forEach(stream => {
|
|
|
if (!pc._remoteStreams) {
|
|
|
pc._remoteStreams = [];
|
|
|
}
|
|
|
if (pc._remoteStreams.indexOf(stream) >= 0) {
|
|
|
return;
|
|
|
}
|
|
|
pc._remoteStreams.push(stream);
|
|
|
const event = new Event('addstream');
|
|
|
event.stream = stream;
|
|
|
pc.dispatchEvent(event);
|
|
|
});
|
|
|
});
|
|
|
}
|
|
|
return origSetRemoteDescription.apply(pc, arguments);
|
|
|
};
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function shimCallbacksAPI(window) {
|
|
|
if (typeof window !== 'object' || !window.RTCPeerConnection) {
|
|
|
return;
|
|
|
}
|
|
|
const prototype = window.RTCPeerConnection.prototype;
|
|
|
const origCreateOffer = prototype.createOffer;
|
|
|
const origCreateAnswer = prototype.createAnswer;
|
|
|
const setLocalDescription = prototype.setLocalDescription;
|
|
|
const setRemoteDescription = prototype.setRemoteDescription;
|
|
|
const addIceCandidate = prototype.addIceCandidate;
|
|
|
|
|
|
prototype.createOffer =
|
|
|
function createOffer(successCallback, failureCallback) {
|
|
|
const options = (arguments.length >= 2) ? arguments[2] : arguments[0];
|
|
|
const promise = origCreateOffer.apply(this, [options]);
|
|
|
if (!failureCallback) {
|
|
|
return promise;
|
|
|
}
|
|
|
promise.then(successCallback, failureCallback);
|
|
|
return Promise.resolve();
|
|
|
};
|
|
|
|
|
|
prototype.createAnswer =
|
|
|
function createAnswer(successCallback, failureCallback) {
|
|
|
const options = (arguments.length >= 2) ? arguments[2] : arguments[0];
|
|
|
const promise = origCreateAnswer.apply(this, [options]);
|
|
|
if (!failureCallback) {
|
|
|
return promise;
|
|
|
}
|
|
|
promise.then(successCallback, failureCallback);
|
|
|
return Promise.resolve();
|
|
|
};
|
|
|
|
|
|
let withCallback = function(description, successCallback, failureCallback) {
|
|
|
const promise = setLocalDescription.apply(this, [description]);
|
|
|
if (!failureCallback) {
|
|
|
return promise;
|
|
|
}
|
|
|
promise.then(successCallback, failureCallback);
|
|
|
return Promise.resolve();
|
|
|
};
|
|
|
prototype.setLocalDescription = withCallback;
|
|
|
|
|
|
withCallback = function(description, successCallback, failureCallback) {
|
|
|
const promise = setRemoteDescription.apply(this, [description]);
|
|
|
if (!failureCallback) {
|
|
|
return promise;
|
|
|
}
|
|
|
promise.then(successCallback, failureCallback);
|
|
|
return Promise.resolve();
|
|
|
};
|
|
|
prototype.setRemoteDescription = withCallback;
|
|
|
|
|
|
withCallback = function(candidate, successCallback, failureCallback) {
|
|
|
const promise = addIceCandidate.apply(this, [candidate]);
|
|
|
if (!failureCallback) {
|
|
|
return promise;
|
|
|
}
|
|
|
promise.then(successCallback, failureCallback);
|
|
|
return Promise.resolve();
|
|
|
};
|
|
|
prototype.addIceCandidate = withCallback;
|
|
|
}
|
|
|
|
|
|
function shimGetUserMedia(window) {
|
|
|
const navigator = window && window.navigator;
|
|
|
|
|
|
if (navigator.mediaDevices && navigator.mediaDevices.getUserMedia) {
|
|
|
// shim not needed in Safari 12.1
|
|
|
const mediaDevices = navigator.mediaDevices;
|
|
|
const _getUserMedia = mediaDevices.getUserMedia.bind(mediaDevices);
|
|
|
navigator.mediaDevices.getUserMedia = (constraints) => {
|
|
|
return _getUserMedia(shimConstraints(constraints));
|
|
|
};
|
|
|
}
|
|
|
|
|
|
if (!navigator.getUserMedia && navigator.mediaDevices &&
|
|
|
navigator.mediaDevices.getUserMedia) {
|
|
|
navigator.getUserMedia = function getUserMedia(constraints, cb, errcb) {
|
|
|
navigator.mediaDevices.getUserMedia(constraints)
|
|
|
.then(cb, errcb);
|
|
|
}.bind(navigator);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function shimConstraints(constraints) {
|
|
|
if (constraints && constraints.video !== undefined) {
|
|
|
return Object.assign({},
|
|
|
constraints,
|
|
|
{video: _utils__WEBPACK_IMPORTED_MODULE_0__.compactObject(constraints.video)}
|
|
|
);
|
|
|
}
|
|
|
|
|
|
return constraints;
|
|
|
}
|
|
|
|
|
|
function shimRTCIceServerUrls(window) {
|
|
|
if (!window.RTCPeerConnection) {
|
|
|
return;
|
|
|
}
|
|
|
// migrate from non-spec RTCIceServer.url to RTCIceServer.urls
|
|
|
const OrigPeerConnection = window.RTCPeerConnection;
|
|
|
window.RTCPeerConnection =
|
|
|
function RTCPeerConnection(pcConfig, pcConstraints) {
|
|
|
if (pcConfig && pcConfig.iceServers) {
|
|
|
const newIceServers = [];
|
|
|
for (let i = 0; i < pcConfig.iceServers.length; i++) {
|
|
|
let server = pcConfig.iceServers[i];
|
|
|
if (server.urls === undefined && server.url) {
|
|
|
_utils__WEBPACK_IMPORTED_MODULE_0__.deprecated('RTCIceServer.url', 'RTCIceServer.urls');
|
|
|
server = JSON.parse(JSON.stringify(server));
|
|
|
server.urls = server.url;
|
|
|
delete server.url;
|
|
|
newIceServers.push(server);
|
|
|
} else {
|
|
|
newIceServers.push(pcConfig.iceServers[i]);
|
|
|
}
|
|
|
}
|
|
|
pcConfig.iceServers = newIceServers;
|
|
|
}
|
|
|
return new OrigPeerConnection(pcConfig, pcConstraints);
|
|
|
};
|
|
|
window.RTCPeerConnection.prototype = OrigPeerConnection.prototype;
|
|
|
// wrap static methods. Currently just generateCertificate.
|
|
|
if ('generateCertificate' in OrigPeerConnection) {
|
|
|
Object.defineProperty(window.RTCPeerConnection, 'generateCertificate', {
|
|
|
get() {
|
|
|
return OrigPeerConnection.generateCertificate;
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function shimTrackEventTransceiver(window) {
|
|
|
// Add event.transceiver member over deprecated event.receiver
|
|
|
if (typeof window === 'object' && window.RTCTrackEvent &&
|
|
|
'receiver' in window.RTCTrackEvent.prototype &&
|
|
|
!('transceiver' in window.RTCTrackEvent.prototype)) {
|
|
|
Object.defineProperty(window.RTCTrackEvent.prototype, 'transceiver', {
|
|
|
get() {
|
|
|
return {receiver: this.receiver};
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
function shimCreateOfferLegacy(window) {
|
|
|
const origCreateOffer = window.RTCPeerConnection.prototype.createOffer;
|
|
|
window.RTCPeerConnection.prototype.createOffer =
|
|
|
function createOffer(offerOptions) {
|
|
|
if (offerOptions) {
|
|
|
if (typeof offerOptions.offerToReceiveAudio !== 'undefined') {
|
|
|
// support bit values
|
|
|
offerOptions.offerToReceiveAudio =
|
|
|
!!offerOptions.offerToReceiveAudio;
|
|
|
}
|
|
|
const audioTransceiver = this.getTransceivers().find(transceiver =>
|
|
|
transceiver.receiver.track.kind === 'audio');
|
|
|
if (offerOptions.offerToReceiveAudio === false && audioTransceiver) {
|
|
|
if (audioTransceiver.direction === 'sendrecv') {
|
|
|
if (audioTransceiver.setDirection) {
|
|
|
audioTransceiver.setDirection('sendonly');
|
|
|
} else {
|
|
|
audioTransceiver.direction = 'sendonly';
|
|
|
}
|
|
|
} else if (audioTransceiver.direction === 'recvonly') {
|
|
|
if (audioTransceiver.setDirection) {
|
|
|
audioTransceiver.setDirection('inactive');
|
|
|
} else {
|
|
|
audioTransceiver.direction = 'inactive';
|
|
|
}
|
|
|
}
|
|
|
} else if (offerOptions.offerToReceiveAudio === true &&
|
|
|
!audioTransceiver) {
|
|
|
this.addTransceiver('audio', {direction: 'recvonly'});
|
|
|
}
|
|
|
|
|
|
if (typeof offerOptions.offerToReceiveVideo !== 'undefined') {
|
|
|
// support bit values
|
|
|
offerOptions.offerToReceiveVideo =
|
|
|
!!offerOptions.offerToReceiveVideo;
|
|
|
}
|
|
|
const videoTransceiver = this.getTransceivers().find(transceiver =>
|
|
|
transceiver.receiver.track.kind === 'video');
|
|
|
if (offerOptions.offerToReceiveVideo === false && videoTransceiver) {
|
|
|
if (videoTransceiver.direction === 'sendrecv') {
|
|
|
if (videoTransceiver.setDirection) {
|
|
|
videoTransceiver.setDirection('sendonly');
|
|
|
} else {
|
|
|
videoTransceiver.direction = 'sendonly';
|
|
|
}
|
|
|
} else if (videoTransceiver.direction === 'recvonly') {
|
|
|
if (videoTransceiver.setDirection) {
|
|
|
videoTransceiver.setDirection('inactive');
|
|
|
} else {
|
|
|
videoTransceiver.direction = 'inactive';
|
|
|
}
|
|
|
}
|
|
|
} else if (offerOptions.offerToReceiveVideo === true &&
|
|
|
!videoTransceiver) {
|
|
|
this.addTransceiver('video', {direction: 'recvonly'});
|
|
|
}
|
|
|
}
|
|
|
return origCreateOffer.apply(this, arguments);
|
|
|
};
|
|
|
}
|
|
|
|
|
|
function shimAudioContext(window) {
|
|
|
if (typeof window !== 'object' || window.AudioContext) {
|
|
|
return;
|
|
|
}
|
|
|
window.AudioContext = window.webkitAudioContext;
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/webrtc-adapter/src/js/utils.js":
|
|
|
/*!*****************************************************!*\
|
|
|
!*** ./node_modules/webrtc-adapter/src/js/utils.js ***!
|
|
|
\*****************************************************/
|
|
|
/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => {
|
|
|
|
|
|
"use strict";
|
|
|
__webpack_require__.r(__webpack_exports__);
|
|
|
/* harmony export */ __webpack_require__.d(__webpack_exports__, {
|
|
|
/* harmony export */ compactObject: () => (/* binding */ compactObject),
|
|
|
/* harmony export */ deprecated: () => (/* binding */ deprecated),
|
|
|
/* harmony export */ detectBrowser: () => (/* binding */ detectBrowser),
|
|
|
/* harmony export */ disableLog: () => (/* binding */ disableLog),
|
|
|
/* harmony export */ disableWarnings: () => (/* binding */ disableWarnings),
|
|
|
/* harmony export */ extractVersion: () => (/* binding */ extractVersion),
|
|
|
/* harmony export */ filterStats: () => (/* binding */ filterStats),
|
|
|
/* harmony export */ log: () => (/* binding */ log),
|
|
|
/* harmony export */ walkStats: () => (/* binding */ walkStats),
|
|
|
/* harmony export */ wrapPeerConnectionEvent: () => (/* binding */ wrapPeerConnectionEvent)
|
|
|
/* harmony export */ });
|
|
|
/*
|
|
|
* Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.
|
|
|
*
|
|
|
* Use of this source code is governed by a BSD-style license
|
|
|
* that can be found in the LICENSE file in the root of the source
|
|
|
* tree.
|
|
|
*/
|
|
|
/* eslint-env node */
|
|
|
|
|
|
|
|
|
let logDisabled_ = true;
|
|
|
let deprecationWarnings_ = true;
|
|
|
|
|
|
/**
|
|
|
* Extract browser version out of the provided user agent string.
|
|
|
*
|
|
|
* @param {!string} uastring userAgent string.
|
|
|
* @param {!string} expr Regular expression used as match criteria.
|
|
|
* @param {!number} pos position in the version string to be returned.
|
|
|
* @return {!number} browser version.
|
|
|
*/
|
|
|
function extractVersion(uastring, expr, pos) {
|
|
|
const match = uastring.match(expr);
|
|
|
return match && match.length >= pos && parseInt(match[pos], 10);
|
|
|
}
|
|
|
|
|
|
// Wraps the peerconnection event eventNameToWrap in a function
|
|
|
// which returns the modified event object (or false to prevent
|
|
|
// the event).
|
|
|
function wrapPeerConnectionEvent(window, eventNameToWrap, wrapper) {
|
|
|
if (!window.RTCPeerConnection) {
|
|
|
return;
|
|
|
}
|
|
|
const proto = window.RTCPeerConnection.prototype;
|
|
|
const nativeAddEventListener = proto.addEventListener;
|
|
|
proto.addEventListener = function(nativeEventName, cb) {
|
|
|
if (nativeEventName !== eventNameToWrap) {
|
|
|
return nativeAddEventListener.apply(this, arguments);
|
|
|
}
|
|
|
const wrappedCallback = (e) => {
|
|
|
const modifiedEvent = wrapper(e);
|
|
|
if (modifiedEvent) {
|
|
|
if (cb.handleEvent) {
|
|
|
cb.handleEvent(modifiedEvent);
|
|
|
} else {
|
|
|
cb(modifiedEvent);
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
this._eventMap = this._eventMap || {};
|
|
|
if (!this._eventMap[eventNameToWrap]) {
|
|
|
this._eventMap[eventNameToWrap] = new Map();
|
|
|
}
|
|
|
this._eventMap[eventNameToWrap].set(cb, wrappedCallback);
|
|
|
return nativeAddEventListener.apply(this, [nativeEventName,
|
|
|
wrappedCallback]);
|
|
|
};
|
|
|
|
|
|
const nativeRemoveEventListener = proto.removeEventListener;
|
|
|
proto.removeEventListener = function(nativeEventName, cb) {
|
|
|
if (nativeEventName !== eventNameToWrap || !this._eventMap
|
|
|
|| !this._eventMap[eventNameToWrap]) {
|
|
|
return nativeRemoveEventListener.apply(this, arguments);
|
|
|
}
|
|
|
if (!this._eventMap[eventNameToWrap].has(cb)) {
|
|
|
return nativeRemoveEventListener.apply(this, arguments);
|
|
|
}
|
|
|
const unwrappedCb = this._eventMap[eventNameToWrap].get(cb);
|
|
|
this._eventMap[eventNameToWrap].delete(cb);
|
|
|
if (this._eventMap[eventNameToWrap].size === 0) {
|
|
|
delete this._eventMap[eventNameToWrap];
|
|
|
}
|
|
|
if (Object.keys(this._eventMap).length === 0) {
|
|
|
delete this._eventMap;
|
|
|
}
|
|
|
return nativeRemoveEventListener.apply(this, [nativeEventName,
|
|
|
unwrappedCb]);
|
|
|
};
|
|
|
|
|
|
Object.defineProperty(proto, 'on' + eventNameToWrap, {
|
|
|
get() {
|
|
|
return this['_on' + eventNameToWrap];
|
|
|
},
|
|
|
set(cb) {
|
|
|
if (this['_on' + eventNameToWrap]) {
|
|
|
this.removeEventListener(eventNameToWrap,
|
|
|
this['_on' + eventNameToWrap]);
|
|
|
delete this['_on' + eventNameToWrap];
|
|
|
}
|
|
|
if (cb) {
|
|
|
this.addEventListener(eventNameToWrap,
|
|
|
this['_on' + eventNameToWrap] = cb);
|
|
|
}
|
|
|
},
|
|
|
enumerable: true,
|
|
|
configurable: true
|
|
|
});
|
|
|
}
|
|
|
|
|
|
function disableLog(bool) {
|
|
|
if (typeof bool !== 'boolean') {
|
|
|
return new Error('Argument type: ' + typeof bool +
|
|
|
'. Please use a boolean.');
|
|
|
}
|
|
|
logDisabled_ = bool;
|
|
|
return (bool) ? 'adapter.js logging disabled' :
|
|
|
'adapter.js logging enabled';
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Disable or enable deprecation warnings
|
|
|
* @param {!boolean} bool set to true to disable warnings.
|
|
|
*/
|
|
|
function disableWarnings(bool) {
|
|
|
if (typeof bool !== 'boolean') {
|
|
|
return new Error('Argument type: ' + typeof bool +
|
|
|
'. Please use a boolean.');
|
|
|
}
|
|
|
deprecationWarnings_ = !bool;
|
|
|
return 'adapter.js deprecation warnings ' + (bool ? 'disabled' : 'enabled');
|
|
|
}
|
|
|
|
|
|
function log() {
|
|
|
if (typeof window === 'object') {
|
|
|
if (logDisabled_) {
|
|
|
return;
|
|
|
}
|
|
|
if (typeof console !== 'undefined' && typeof console.log === 'function') {
|
|
|
console.log.apply(console, arguments);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Shows a deprecation warning suggesting the modern and spec-compatible API.
|
|
|
*/
|
|
|
function deprecated(oldMethod, newMethod) {
|
|
|
if (!deprecationWarnings_) {
|
|
|
return;
|
|
|
}
|
|
|
console.warn(oldMethod + ' is deprecated, please use ' + newMethod +
|
|
|
' instead.');
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Browser detector.
|
|
|
*
|
|
|
* @return {object} result containing browser and version
|
|
|
* properties.
|
|
|
*/
|
|
|
function detectBrowser(window) {
|
|
|
// Returned result object.
|
|
|
const result = {browser: null, version: null};
|
|
|
|
|
|
// Fail early if it's not a browser
|
|
|
if (typeof window === 'undefined' || !window.navigator ||
|
|
|
!window.navigator.userAgent) {
|
|
|
result.browser = 'Not a browser.';
|
|
|
return result;
|
|
|
}
|
|
|
|
|
|
const {navigator} = window;
|
|
|
|
|
|
if (navigator.mozGetUserMedia) { // Firefox.
|
|
|
result.browser = 'firefox';
|
|
|
result.version = extractVersion(navigator.userAgent,
|
|
|
/Firefox\/(\d+)\./, 1);
|
|
|
} else if (navigator.webkitGetUserMedia ||
|
|
|
(window.isSecureContext === false && window.webkitRTCPeerConnection)) {
|
|
|
// Chrome, Chromium, Webview, Opera.
|
|
|
// Version matches Chrome/WebRTC version.
|
|
|
// Chrome 74 removed webkitGetUserMedia on http as well so we need the
|
|
|
// more complicated fallback to webkitRTCPeerConnection.
|
|
|
result.browser = 'chrome';
|
|
|
result.version = extractVersion(navigator.userAgent,
|
|
|
/Chrom(e|ium)\/(\d+)\./, 2);
|
|
|
} else if (window.RTCPeerConnection &&
|
|
|
navigator.userAgent.match(/AppleWebKit\/(\d+)\./)) { // Safari.
|
|
|
result.browser = 'safari';
|
|
|
result.version = extractVersion(navigator.userAgent,
|
|
|
/AppleWebKit\/(\d+)\./, 1);
|
|
|
result.supportsUnifiedPlan = window.RTCRtpTransceiver &&
|
|
|
'currentDirection' in window.RTCRtpTransceiver.prototype;
|
|
|
} else { // Default fallthrough: not supported.
|
|
|
result.browser = 'Not a supported browser.';
|
|
|
return result;
|
|
|
}
|
|
|
|
|
|
return result;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Checks if something is an object.
|
|
|
*
|
|
|
* @param {*} val The something you want to check.
|
|
|
* @return true if val is an object, false otherwise.
|
|
|
*/
|
|
|
function isObject(val) {
|
|
|
return Object.prototype.toString.call(val) === '[object Object]';
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Remove all empty objects and undefined values
|
|
|
* from a nested object -- an enhanced and vanilla version
|
|
|
* of Lodash's `compact`.
|
|
|
*/
|
|
|
function compactObject(data) {
|
|
|
if (!isObject(data)) {
|
|
|
return data;
|
|
|
}
|
|
|
|
|
|
return Object.keys(data).reduce(function(accumulator, key) {
|
|
|
const isObj = isObject(data[key]);
|
|
|
const value = isObj ? compactObject(data[key]) : data[key];
|
|
|
const isEmptyObject = isObj && !Object.keys(value).length;
|
|
|
if (value === undefined || isEmptyObject) {
|
|
|
return accumulator;
|
|
|
}
|
|
|
return Object.assign(accumulator, {[key]: value});
|
|
|
}, {});
|
|
|
}
|
|
|
|
|
|
/* iterates the stats graph recursively. */
|
|
|
function walkStats(stats, base, resultSet) {
|
|
|
if (!base || resultSet.has(base.id)) {
|
|
|
return;
|
|
|
}
|
|
|
resultSet.set(base.id, base);
|
|
|
Object.keys(base).forEach(name => {
|
|
|
if (name.endsWith('Id')) {
|
|
|
walkStats(stats, stats.get(base[name]), resultSet);
|
|
|
} else if (name.endsWith('Ids')) {
|
|
|
base[name].forEach(id => {
|
|
|
walkStats(stats, stats.get(id), resultSet);
|
|
|
});
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
/* filter getStats for a sender/receiver track. */
|
|
|
function filterStats(result, track, outbound) {
|
|
|
const streamStatsType = outbound ? 'outbound-rtp' : 'inbound-rtp';
|
|
|
const filteredResult = new Map();
|
|
|
if (track === null) {
|
|
|
return filteredResult;
|
|
|
}
|
|
|
const trackStats = [];
|
|
|
result.forEach(value => {
|
|
|
if (value.type === 'track' &&
|
|
|
value.trackIdentifier === track.id) {
|
|
|
trackStats.push(value);
|
|
|
}
|
|
|
});
|
|
|
trackStats.forEach(trackStat => {
|
|
|
result.forEach(stats => {
|
|
|
if (stats.type === streamStatsType && stats.trackId === trackStat.id) {
|
|
|
walkStats(result, stats, filteredResult);
|
|
|
}
|
|
|
});
|
|
|
});
|
|
|
return filteredResult;
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./src/settings/WebRtcPeer.js":
|
|
|
/*!************************************!*\
|
|
|
!*** ./src/settings/WebRtcPeer.js ***!
|
|
|
\************************************/
|
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
|
|
/*
|
|
|
* (C) Copyright 2017-2022 OpenVidu (https://openvidu.io)
|
|
|
*
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
* You may obtain a copy of the License at
|
|
|
*
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
*
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
* See the License for the specific language governing permissions and
|
|
|
* limitations under the License.
|
|
|
*
|
|
|
*/
|
|
|
|
|
|
// taken from here:
|
|
|
// https://github.com/OpenVidu/openvidu/blob/master/openvidu-browser/src/OpenViduInternal/WebRtcPeer/WebRtcPeer.ts
|
|
|
// and monkey-patched
|
|
|
const OmUtil = __webpack_require__(/*! ../main/omutils */ "../main/omutils");
|
|
|
|
|
|
const freeice = __webpack_require__(/*! freeice */ "./node_modules/freeice/index.js");
|
|
|
|
|
|
const ExceptionEventName = {
|
|
|
/**
|
|
|
* The [ICE connection state](https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection/iceConnectionState)
|
|
|
* of an [RTCPeerConnection](https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection) reached `failed` status.
|
|
|
*
|
|
|
* This is a terminal error that won't have any kind of possible recovery. If the client is still connected to OpenVidu Server,
|
|
|
* then an automatic reconnection process of the media stream is immediately performed. If the ICE connection has broken due to
|
|
|
* a total network drop, then no automatic reconnection process will be possible.
|
|
|
*
|
|
|
* {@link ExceptionEvent} objects with this {@link ExceptionEvent.name} will have as {@link ExceptionEvent.origin} property a {@link Stream} object.
|
|
|
*/
|
|
|
ICE_CONNECTION_FAILED: 'ICE_CONNECTION_FAILED',
|
|
|
|
|
|
/**
|
|
|
* The [ICE connection state](https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection/iceConnectionState)
|
|
|
* of an [RTCPeerConnection](https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection) reached `disconnected` status.
|
|
|
*
|
|
|
* This is not a terminal error, and it is possible for the ICE connection to be reconnected. If the client is still connected to
|
|
|
* OpenVidu Server and after certain timeout the ICE connection has not reached a success or terminal status, then an automatic
|
|
|
* reconnection process of the media stream is performed. If the ICE connection has broken due to a total network drop, then no
|
|
|
* automatic reconnection process will be possible.
|
|
|
*
|
|
|
* You can customize the timeout for the reconnection attempt with property {@link OpenViduAdvancedConfiguration.iceConnectionDisconnectedExceptionTimeout},
|
|
|
* which by default is 4000 milliseconds.
|
|
|
*
|
|
|
* {@link ExceptionEvent} objects with this {@link ExceptionEvent.name} will have as {@link ExceptionEvent.origin} property a {@link Stream} object.
|
|
|
*/
|
|
|
ICE_CONNECTION_DISCONNECTED: 'ICE_CONNECTION_DISCONNECTED',
|
|
|
};
|
|
|
|
|
|
class WebRtcPeer {
|
|
|
constructor(configuration) {
|
|
|
this.remoteCandidatesQueue = [];
|
|
|
this.localCandidatesQueue = [];
|
|
|
this.iceCandidateList = [];
|
|
|
this.candidategatheringdone = false;
|
|
|
|
|
|
// Same as WebRtcPeerConfiguration but without optional fields.
|
|
|
this.configuration = {
|
|
|
...configuration,
|
|
|
iceServers: !!configuration.iceServers && configuration.iceServers.length > 0 ? configuration.iceServers : freeice(),
|
|
|
mediaStream: configuration.mediaStream !== undefined ? configuration.mediaStream : null,
|
|
|
mode: !!configuration.mode ? configuration.mode : 'sendrecv',
|
|
|
id: !!configuration.id ? configuration.id : this.generateUniqueId()
|
|
|
};
|
|
|
// prettier-ignore
|
|
|
OmUtil.log(`[WebRtcPeer] configuration:\n${JSON.stringify(this.configuration, null, 2)}`);
|
|
|
|
|
|
this.pc = new RTCPeerConnection({ iceServers: this.configuration.iceServers });
|
|
|
|
|
|
this._iceCandidateListener = (event) => {
|
|
|
if (event.candidate !== null) {
|
|
|
// `RTCPeerConnectionIceEvent.candidate` is supposed to be an RTCIceCandidate:
|
|
|
// https://w3c.github.io/webrtc-pc/#dom-rtcpeerconnectioniceevent-candidate
|
|
|
//
|
|
|
// But in practice, it is actually an RTCIceCandidateInit that can be used to
|
|
|
// obtain a proper candidate, using the RTCIceCandidate constructor:
|
|
|
// https://w3c.github.io/webrtc-pc/#dom-rtcicecandidate-constructor
|
|
|
const candidateInit = event.candidate;
|
|
|
const iceCandidate = new RTCIceCandidate(candidateInit);
|
|
|
|
|
|
this.configuration.onIceCandidate(iceCandidate);
|
|
|
if (iceCandidate.candidate !== '') {
|
|
|
this.localCandidatesQueue.push(iceCandidate);
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
this.pc.addEventListener('icecandidate', this._iceCandidateListener);
|
|
|
|
|
|
this._signalingStateChangeListener = async () => {
|
|
|
if (this.pc.signalingState === 'stable') {
|
|
|
// SDP Offer/Answer finished. Add stored remote candidates.
|
|
|
while (this.iceCandidateList.length > 0) {
|
|
|
let candidate = this.iceCandidateList.shift();
|
|
|
try {
|
|
|
await this.pc.addIceCandidate(candidate);
|
|
|
} catch (error) {
|
|
|
console.error('Error when calling RTCPeerConnection#addIceCandidate for RTCPeerConnection ' + this.getId(), error);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
this.pc.addEventListener('signalingstatechange', this._signalingStateChangeListener);
|
|
|
if (this.configuration.onConnectionStateChange) {
|
|
|
this.pc.addEventListener('connectionstatechange', this.configuration.onConnectionStateChange);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
getId() {
|
|
|
return this.configuration.id;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* This method frees the resources used by WebRtcPeer
|
|
|
*/
|
|
|
dispose() {
|
|
|
OmUtil.log('Disposing WebRtcPeer');
|
|
|
if (this.pc) {
|
|
|
if (this.pc.signalingState === 'closed') {
|
|
|
return;
|
|
|
}
|
|
|
this.pc.removeEventListener('icecandidate', this._iceCandidateListener);
|
|
|
this._iceCandidateListener = undefined;
|
|
|
this.pc.removeEventListener('signalingstatechange', this._signalingStateChangeListener);
|
|
|
this._signalingStateChangeListener = undefined;
|
|
|
if (this._iceConnectionStateChangeListener) {
|
|
|
this.pc.removeEventListener('iceconnectionstatechange', this._iceConnectionStateChangeListener);
|
|
|
this._iceConnectionStateChangeListener = undefined;
|
|
|
}
|
|
|
if (this.configuration.onConnectionStateChange) {
|
|
|
this.pc.removeEventListener('connectionstatechange', this.configuration.onConnectionStateChange);
|
|
|
}
|
|
|
this.configuration = {};
|
|
|
this.pc.close();
|
|
|
this.remoteCandidatesQueue = [];
|
|
|
this.localCandidatesQueue = [];
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Creates an SDP offer from the local RTCPeerConnection to send to the other peer.
|
|
|
* Only if the negotiation was initiated by this peer.
|
|
|
*/
|
|
|
async createOffer() {
|
|
|
// TODO: Delete this conditional when all supported browsers are
|
|
|
// modern enough to implement the Transceiver methods.
|
|
|
if (!('addTransceiver' in this.pc)) {
|
|
|
OmUtil.error(
|
|
|
'[createOffer] Method RTCPeerConnection.addTransceiver() is NOT available; using LEGACY offerToReceive{Audio,Video}'
|
|
|
);
|
|
|
return this.createOfferLegacy();
|
|
|
} else {
|
|
|
OmUtil.log('[createOffer] Method RTCPeerConnection.addTransceiver() is available; using it');
|
|
|
}
|
|
|
|
|
|
// Spec doc: https://w3c.github.io/webrtc-pc/#dom-rtcpeerconnection-addtransceiver
|
|
|
|
|
|
if (this.configuration.mode !== 'recvonly') {
|
|
|
// To send media, assume that all desired media tracks have been
|
|
|
// already added by higher level code to our MediaStream.
|
|
|
|
|
|
if (!this.configuration.mediaStream) {
|
|
|
throw new Error(
|
|
|
`[WebRtcPeer.createOffer] Direction is '${this.configuration.mode}', but no stream was configured to be sent`
|
|
|
);
|
|
|
}
|
|
|
|
|
|
for (const track of this.configuration.mediaStream.getTracks()) {
|
|
|
const tcInit = {
|
|
|
direction: this.configuration.mode,
|
|
|
streams: [this.configuration.mediaStream]
|
|
|
};
|
|
|
|
|
|
if (track.kind === 'video' && this.configuration.simulcast) {
|
|
|
// Check if the requested size is enough to ask for 3 layers.
|
|
|
const trackSettings = track.getSettings();
|
|
|
const trackConsts = track.getConstraints();
|
|
|
|
|
|
const trackWidth = typeof(trackSettings.width) === 'object' ? trackConsts.width.ideal : trackConsts.width || 0;
|
|
|
const trackHeight = typeof(trackSettings.height) === 'object' ? trackConsts.height.ideal : trackConsts.height || 0;
|
|
|
OmUtil.info(`[createOffer] Video track dimensions: ${trackWidth}x${trackHeight}`);
|
|
|
|
|
|
const trackPixels = trackWidth * trackHeight;
|
|
|
let maxLayers = 0;
|
|
|
if (trackPixels >= 960 * 540) {
|
|
|
maxLayers = 3;
|
|
|
} else if (trackPixels >= 480 * 270) {
|
|
|
maxLayers = 2;
|
|
|
} else {
|
|
|
maxLayers = 1;
|
|
|
}
|
|
|
|
|
|
tcInit.sendEncodings = [];
|
|
|
for (let l = 0; l < maxLayers; l++) {
|
|
|
const layerDiv = 2 ** (maxLayers - l - 1);
|
|
|
|
|
|
const encoding = {
|
|
|
rid: 'rdiv' + layerDiv.toString(),
|
|
|
|
|
|
// @ts-ignore -- Property missing from DOM types.
|
|
|
scalabilityMode: 'L1T1'
|
|
|
};
|
|
|
|
|
|
if (['detail', 'text'].includes(track.contentHint)) {
|
|
|
// Prioritize best resolution, for maximum picture detail.
|
|
|
encoding.scaleResolutionDownBy = 1.0;
|
|
|
|
|
|
// @ts-ignore -- Property missing from DOM types.
|
|
|
encoding.maxFramerate = Math.floor(30 / layerDiv);
|
|
|
} else {
|
|
|
encoding.scaleResolutionDownBy = layerDiv;
|
|
|
}
|
|
|
|
|
|
tcInit.sendEncodings.push(encoding);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
const tc = this.pc.addTransceiver(track, tcInit);
|
|
|
|
|
|
if (track.kind === 'video') {
|
|
|
let sendParams = tc.sender.getParameters();
|
|
|
let needSetParams = false;
|
|
|
|
|
|
if (sendParams.degradationPreference && !sendParams.degradationPreference.length) {
|
|
|
// degradationPreference for video: "balanced", "maintain-framerate", "maintain-resolution".
|
|
|
// https://www.w3.org/TR/2018/CR-webrtc-20180927/#dom-rtcdegradationpreference
|
|
|
if (['detail', 'text'].includes(track.contentHint)) {
|
|
|
sendParams.degradationPreference = 'maintain-resolution';
|
|
|
} else {
|
|
|
sendParams.degradationPreference = 'balanced';
|
|
|
}
|
|
|
|
|
|
OmUtil.info(`[createOffer] Video sender Degradation Preference set: ${sendParams.degradationPreference}`);
|
|
|
|
|
|
// Firefox implements degradationPreference on each individual encoding!
|
|
|
// (set it on every element of the sendParams.encodings array)
|
|
|
|
|
|
needSetParams = true;
|
|
|
}
|
|
|
|
|
|
// Check that the simulcast encodings were applied.
|
|
|
// Firefox doesn't implement `RTCRtpTransceiverInit.sendEncodings`
|
|
|
// so the only way to enable simulcast is with `RTCRtpSender.setParameters()`.
|
|
|
//
|
|
|
// This next block can be deleted when Firefox fixes bug #1396918:
|
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=1396918
|
|
|
//
|
|
|
// NOTE: This is done in a way that is compatible with all browsers, to save on
|
|
|
// browser-conditional code. The idea comes from WebRTC Adapter.js:
|
|
|
// * https://github.com/webrtcHacks/adapter/issues/998
|
|
|
// * https://github.com/webrtcHacks/adapter/blob/v7.7.0/src/js/firefox/firefox_shim.js#L231-L255
|
|
|
if (this.configuration.simulcast) {
|
|
|
if (sendParams.encodings.length !== tcInit.sendEncodings.length) {
|
|
|
sendParams.encodings = tcInit.sendEncodings;
|
|
|
|
|
|
needSetParams = true;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
if (needSetParams) {
|
|
|
OmUtil.log(`[createOffer] Setting new RTCRtpSendParameters to video sender`);
|
|
|
try {
|
|
|
await tc.sender.setParameters(sendParams);
|
|
|
} catch (error) {
|
|
|
let message = `[WebRtcPeer.createOffer] Cannot set RTCRtpSendParameters to video sender`;
|
|
|
if (error instanceof Error) {
|
|
|
message += `: ${error.message}`;
|
|
|
}
|
|
|
throw new Error(message);
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
} else {
|
|
|
// To just receive media, create new recvonly transceivers.
|
|
|
for (const kind of ['audio', 'video']) {
|
|
|
// Check if the media kind should be used.
|
|
|
if (!this.configuration.mediaConstraints[kind]) {
|
|
|
continue;
|
|
|
}
|
|
|
|
|
|
this.configuration.mediaStream = new MediaStream();
|
|
|
this.pc.addTransceiver(kind, {
|
|
|
direction: this.configuration.mode,
|
|
|
streams: [this.configuration.mediaStream]
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
|
|
|
let sdpOffer;
|
|
|
try {
|
|
|
sdpOffer = await this.pc.createOffer();
|
|
|
} catch (error) {
|
|
|
let message = `[WebRtcPeer.createOffer] Browser failed creating an SDP Offer`;
|
|
|
if (error instanceof Error) {
|
|
|
message += `: ${error.message}`;
|
|
|
}
|
|
|
throw new Error(message);
|
|
|
}
|
|
|
|
|
|
return sdpOffer;
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Creates an SDP answer from the local RTCPeerConnection to send to the other peer
|
|
|
* Only if the negotiation was initiated by the other peer
|
|
|
*/
|
|
|
createAnswer() {
|
|
|
return new Promise((resolve, reject) => {
|
|
|
// TODO: Delete this conditional when all supported browsers are
|
|
|
// modern enough to implement the Transceiver methods.
|
|
|
if ('getTransceivers' in this.pc) {
|
|
|
OmUtil.log('[createAnswer] Method RTCPeerConnection.getTransceivers() is available; using it');
|
|
|
|
|
|
// Ensure that the PeerConnection already contains one Transceiver
|
|
|
// for each kind of media.
|
|
|
// The Transceivers should have been already created internally by
|
|
|
// the PC itself, when `pc.setRemoteDescription(sdpOffer)` was called.
|
|
|
|
|
|
for (const kind of ['audio', 'video']) {
|
|
|
// Check if the media kind should be used.
|
|
|
if (!this.configuration.mediaConstraints[kind]) {
|
|
|
continue;
|
|
|
}
|
|
|
|
|
|
let tc = this.pc.getTransceivers().find((tc) => tc.receiver.track.kind === kind);
|
|
|
|
|
|
if (tc) {
|
|
|
// Enforce our desired direction.
|
|
|
tc.direction = this.configuration.mode;
|
|
|
} else {
|
|
|
return reject(new Error(`${kind} requested, but no transceiver was created from remote description`));
|
|
|
}
|
|
|
}
|
|
|
|
|
|
this.pc
|
|
|
.createAnswer()
|
|
|
.then((sdpAnswer) => resolve(sdpAnswer))
|
|
|
.catch((error) => reject(error));
|
|
|
} else {
|
|
|
// TODO: Delete else branch when all supported browsers are
|
|
|
// modern enough to implement the Transceiver methods
|
|
|
|
|
|
let offerAudio,
|
|
|
offerVideo = true;
|
|
|
if (!!this.configuration.mediaConstraints) {
|
|
|
offerAudio =
|
|
|
typeof this.configuration.mediaConstraints.audio === 'boolean' ? this.configuration.mediaConstraints.audio : true;
|
|
|
offerVideo =
|
|
|
typeof this.configuration.mediaConstraints.video === 'boolean' ? this.configuration.mediaConstraints.video : true;
|
|
|
const constraints = {
|
|
|
offerToReceiveAudio: offerAudio,
|
|
|
offerToReceiveVideo: offerVideo
|
|
|
};
|
|
|
(this.pc).createAnswer(constraints)
|
|
|
.then((sdpAnswer) => resolve(sdpAnswer))
|
|
|
.catch((error) => reject(error));
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// else, there is nothing to do; the legacy createAnswer() options do
|
|
|
// not offer any control over which tracks are included in the answer.
|
|
|
});
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* This peer initiated negotiation. Step 1/4 of SDP offer-answer protocol
|
|
|
*/
|
|
|
processLocalOffer(offer) {
|
|
|
return new Promise((resolve, reject) => {
|
|
|
this.pc
|
|
|
.setLocalDescription(offer)
|
|
|
.then(() => {
|
|
|
const localDescription = this.pc.localDescription;
|
|
|
if (!!localDescription) {
|
|
|
OmUtil.log('Local description set', localDescription.sdp);
|
|
|
return resolve();
|
|
|
} else {
|
|
|
return reject('Local description is not defined');
|
|
|
}
|
|
|
})
|
|
|
.catch((error) => reject(error));
|
|
|
});
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Other peer initiated negotiation. Step 2/4 of SDP offer-answer protocol
|
|
|
*/
|
|
|
processRemoteOffer(sdpOffer) {
|
|
|
return new Promise((resolve, reject) => {
|
|
|
const offer = {
|
|
|
type: 'offer',
|
|
|
sdp: sdpOffer
|
|
|
};
|
|
|
OmUtil.log('SDP offer received, setting remote description', offer);
|
|
|
|
|
|
if (this.pc.signalingState === 'closed') {
|
|
|
return reject('RTCPeerConnection is closed when trying to set remote description');
|
|
|
}
|
|
|
this.setRemoteDescription(offer)
|
|
|
.then(() => resolve())
|
|
|
.catch((error) => reject(error));
|
|
|
});
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Other peer initiated negotiation. Step 3/4 of SDP offer-answer protocol
|
|
|
*/
|
|
|
processLocalAnswer(answer) {
|
|
|
return new Promise((resolve, reject) => {
|
|
|
OmUtil.log('SDP answer created, setting local description');
|
|
|
if (this.pc.signalingState === 'closed') {
|
|
|
return reject('RTCPeerConnection is closed when trying to set local description');
|
|
|
}
|
|
|
this.pc
|
|
|
.setLocalDescription(answer)
|
|
|
.then(() => resolve())
|
|
|
.catch((error) => reject(error));
|
|
|
});
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* This peer initiated negotiation. Step 4/4 of SDP offer-answer protocol
|
|
|
*/
|
|
|
processRemoteAnswer(sdpAnswer) {
|
|
|
return new Promise((resolve, reject) => {
|
|
|
const answer = {
|
|
|
type: 'answer',
|
|
|
sdp: sdpAnswer
|
|
|
};
|
|
|
OmUtil.log('SDP answer received, setting remote description');
|
|
|
|
|
|
if (this.pc.signalingState === 'closed') {
|
|
|
return reject('RTCPeerConnection is closed when trying to set remote description');
|
|
|
}
|
|
|
this.setRemoteDescription(answer)
|
|
|
.then(() => {
|
|
|
resolve();
|
|
|
})
|
|
|
.catch((error) => reject(error));
|
|
|
});
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* @hidden
|
|
|
*/
|
|
|
async setRemoteDescription(sdp) {
|
|
|
return this.pc.setRemoteDescription(sdp);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* Callback function invoked when an ICE candidate is received
|
|
|
*/
|
|
|
addIceCandidate(iceCandidate) {
|
|
|
return new Promise((resolve, reject) => {
|
|
|
OmUtil.log('Remote ICE candidate received', iceCandidate);
|
|
|
this.remoteCandidatesQueue.push(iceCandidate);
|
|
|
switch (this.pc.signalingState) {
|
|
|
case 'closed':
|
|
|
reject(new Error('PeerConnection object is closed'));
|
|
|
break;
|
|
|
case 'stable':
|
|
|
if (!!this.pc.remoteDescription) {
|
|
|
this.pc
|
|
|
.addIceCandidate(iceCandidate)
|
|
|
.then(() => resolve())
|
|
|
.catch((error) => reject(error));
|
|
|
} else {
|
|
|
this.iceCandidateList.push(iceCandidate);
|
|
|
resolve();
|
|
|
}
|
|
|
break;
|
|
|
default:
|
|
|
this.iceCandidateList.push(iceCandidate);
|
|
|
resolve();
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
addIceConnectionStateChangeListener(otherId) {
|
|
|
if (!this._iceConnectionStateChangeListener) {
|
|
|
this._iceConnectionStateChangeListener = () => {
|
|
|
const iceConnectionState = this.pc.iceConnectionState;
|
|
|
switch (iceConnectionState) {
|
|
|
case 'disconnected':
|
|
|
// Possible network disconnection
|
|
|
const msg1 =
|
|
|
'IceConnectionState of RTCPeerConnection ' +
|
|
|
this.configuration.id +
|
|
|
' (' +
|
|
|
otherId +
|
|
|
') change to "disconnected". Possible network disconnection';
|
|
|
console.warn(msg1);
|
|
|
this.configuration.onIceConnectionStateException(ExceptionEventName.ICE_CONNECTION_DISCONNECTED, msg1);
|
|
|
break;
|
|
|
case 'failed':
|
|
|
const msg2 = 'IceConnectionState of RTCPeerConnection ' + this.configuration.id + ' (' + otherId + ') to "failed"';
|
|
|
console.error(msg2);
|
|
|
this.configuration.onIceConnectionStateException(ExceptionEventName.ICE_CONNECTION_FAILED, msg2);
|
|
|
break;
|
|
|
case 'closed':
|
|
|
OmUtil.log(
|
|
|
'IceConnectionState of RTCPeerConnection ' + this.configuration.id + ' (' + otherId + ') change to "closed"'
|
|
|
);
|
|
|
break;
|
|
|
case 'new':
|
|
|
OmUtil.log('IceConnectionState of RTCPeerConnection ' + this.configuration.id + ' (' + otherId + ') change to "new"');
|
|
|
break;
|
|
|
case 'checking':
|
|
|
OmUtil.log(
|
|
|
'IceConnectionState of RTCPeerConnection ' + this.configuration.id + ' (' + otherId + ') change to "checking"'
|
|
|
);
|
|
|
break;
|
|
|
case 'connected':
|
|
|
OmUtil.log(
|
|
|
'IceConnectionState of RTCPeerConnection ' + this.configuration.id + ' (' + otherId + ') change to "connected"'
|
|
|
);
|
|
|
break;
|
|
|
case 'completed':
|
|
|
OmUtil.log(
|
|
|
'IceConnectionState of RTCPeerConnection ' + this.configuration.id + ' (' + otherId + ') change to "completed"'
|
|
|
);
|
|
|
break;
|
|
|
}
|
|
|
};
|
|
|
}
|
|
|
this.pc.addEventListener('iceconnectionstatechange', this._iceConnectionStateChangeListener);
|
|
|
}
|
|
|
|
|
|
/**
|
|
|
* @hidden
|
|
|
*/
|
|
|
generateUniqueId() {
|
|
|
return crypto.randomUUID();
|
|
|
}
|
|
|
|
|
|
get stream() {
|
|
|
return this.pc.getLocalStreams()[0] || this.pc.getRemoteStreams()[0];
|
|
|
}
|
|
|
|
|
|
// LEGACY code
|
|
|
deprecatedPeerConnectionTrackApi() {
|
|
|
for (const track of this.configuration.mediaStream.getTracks()) {
|
|
|
this.pc.addTrack(track, this.configuration.mediaStream);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// DEPRECATED LEGACY METHOD: Old WebRTC versions don't implement
|
|
|
// Transceivers, and instead depend on the deprecated
|
|
|
// "offerToReceiveAudio" and "offerToReceiveVideo".
|
|
|
createOfferLegacy() {
|
|
|
if (!!this.configuration.mediaStream) {
|
|
|
this.deprecatedPeerConnectionTrackApi();
|
|
|
}
|
|
|
|
|
|
const hasAudio = this.configuration.mediaConstraints.audio;
|
|
|
const hasVideo = this.configuration.mediaConstraints.video;
|
|
|
|
|
|
const options = {
|
|
|
offerToReceiveAudio: this.configuration.mode !== 'sendonly' && hasAudio,
|
|
|
offerToReceiveVideo: this.configuration.mode !== 'sendonly' && hasVideo
|
|
|
};
|
|
|
|
|
|
OmUtil.log('[createOfferLegacy] RTCPeerConnection.createOffer() options:', JSON.stringify(options));
|
|
|
|
|
|
return this.pc.createOffer(options);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
class WebRtcPeerRecvonly extends WebRtcPeer {
|
|
|
constructor(configuration) {
|
|
|
configuration.mode = 'recvonly';
|
|
|
super(configuration);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
class WebRtcPeerSendonly extends WebRtcPeer {
|
|
|
constructor(configuration) {
|
|
|
configuration.mode = 'sendonly';
|
|
|
super(configuration);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
class WebRtcPeerSendrecv extends WebRtcPeer {
|
|
|
constructor(configuration) {
|
|
|
configuration.mode = 'sendrecv';
|
|
|
super(configuration);
|
|
|
}
|
|
|
};
|
|
|
|
|
|
module.exports = {
|
|
|
Recvonly: WebRtcPeerRecvonly,
|
|
|
Sendonly: WebRtcPeerSendonly
|
|
|
};
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./src/settings/mic-level.js":
|
|
|
/*!***********************************!*\
|
|
|
!*** ./src/settings/mic-level.js ***!
|
|
|
\***********************************/
|
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
|
|
/* Licensed under the Apache License, Version 2.0 (the "License") http://www.apache.org/licenses/LICENSE-2.0 */
|
|
|
const VideoUtil = __webpack_require__(/*! ./video-util */ "./src/settings/video-util.js");
|
|
|
const RingBuffer = __webpack_require__(/*! ./ring-buffer */ "./src/settings/ring-buffer.js");
|
|
|
|
|
|
module.exports = class MicLevel {
|
|
|
constructor() {
|
|
|
let ctx, mic, analyser
|
|
|
, cnvs, canvasCtx, WIDTH, HEIGHT, horiz
|
|
|
, vol = .0, vals = new RingBuffer(100);
|
|
|
|
|
|
this.meterStream = (stream, _cnvs, _micActivity, _error, connectAudio) => {
|
|
|
if (!stream || stream.getAudioTracks().length < 1) {
|
|
|
return;
|
|
|
}
|
|
|
try {
|
|
|
const AudioCtx = window.AudioContext || window.webkitAudioContext;
|
|
|
if (!AudioCtx) {
|
|
|
_error("AudioContext is inaccessible");
|
|
|
return;
|
|
|
}
|
|
|
ctx = new AudioCtx();
|
|
|
analyser = ctx.createAnalyser();
|
|
|
mic = ctx.createMediaStreamSource(stream);
|
|
|
mic.connect(analyser);
|
|
|
if (connectAudio) {
|
|
|
analyser.connect(ctx.destination);
|
|
|
}
|
|
|
this.meter(analyser, _cnvs, _micActivity, _error);
|
|
|
} catch (err) {
|
|
|
_error(err);
|
|
|
}
|
|
|
};
|
|
|
this.setCanvas = (_cnvs) => {
|
|
|
cnvs = _cnvs;
|
|
|
const canvas = cnvs[0];
|
|
|
canvasCtx = canvas.getContext('2d');
|
|
|
WIDTH = canvas.width;
|
|
|
HEIGHT = canvas.height;
|
|
|
horiz = cnvs.data('orientation') === 'horizontal';
|
|
|
};
|
|
|
this.meter = (_analyser, _cnvs, _micActivity, _error) => {
|
|
|
this.setCanvas(_cnvs);
|
|
|
try {
|
|
|
analyser = _analyser;
|
|
|
analyser.minDecibels = -90;
|
|
|
analyser.maxDecibels = -10;
|
|
|
analyser.fftSize = 256;
|
|
|
const color = $('body').css('--level-color')
|
|
|
, al = analyser.frequencyBinCount
|
|
|
, arr = new Uint8Array(al);
|
|
|
function update() {
|
|
|
canvasCtx.clearRect(0, 0, WIDTH, HEIGHT);
|
|
|
if (!!analyser && cnvs.length > 0) {
|
|
|
if (cnvs.is(':visible')) {
|
|
|
analyser.getByteFrequencyData(arr);
|
|
|
let favg = 0.0;
|
|
|
for (let i = 0; i < al; ++i) {
|
|
|
favg += arr[i] * arr[i];
|
|
|
}
|
|
|
vol = Math.sqrt(favg / al);
|
|
|
vals.push(vol);
|
|
|
const min = vals.min();
|
|
|
_micActivity(vol > min + 5); // magic number
|
|
|
canvasCtx.fillStyle = color;
|
|
|
if (horiz) {
|
|
|
canvasCtx.fillRect(0, 0, WIDTH * vol / 100, HEIGHT);
|
|
|
} else {
|
|
|
const h = HEIGHT * vol / 100;
|
|
|
canvasCtx.fillRect(0, HEIGHT - h, WIDTH, h);
|
|
|
}
|
|
|
}
|
|
|
requestAnimationFrame(update);
|
|
|
}
|
|
|
}
|
|
|
update();
|
|
|
} catch (err) {
|
|
|
_error(err);
|
|
|
}
|
|
|
};
|
|
|
this.dispose = () => {
|
|
|
if (!!ctx) {
|
|
|
VideoUtil.cleanStream(mic.mediaStream);
|
|
|
VideoUtil.disconnect(mic);
|
|
|
VideoUtil.disconnect(ctx.destination);
|
|
|
ctx.close();
|
|
|
ctx = null;
|
|
|
}
|
|
|
if (!!analyser) {
|
|
|
VideoUtil.disconnect(analyser);
|
|
|
analyser = null;
|
|
|
}
|
|
|
};
|
|
|
}
|
|
|
};
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./src/settings/ring-buffer.js":
|
|
|
/*!*************************************!*\
|
|
|
!*** ./src/settings/ring-buffer.js ***!
|
|
|
\*************************************/
|
|
|
/***/ ((module) => {
|
|
|
|
|
|
/* Licensed under the Apache License, Version 2.0 (the "License") http://www.apache.org/licenses/LICENSE-2.0 */
|
|
|
module.exports = class RingBuffer {
|
|
|
constructor(length) {
|
|
|
const buffer = [];
|
|
|
let pos = 0;
|
|
|
|
|
|
this.get = (key) => {
|
|
|
return buffer[key];
|
|
|
};
|
|
|
this.push = (item) => {
|
|
|
buffer[pos] = item;
|
|
|
pos = (pos + 1) % length;
|
|
|
};
|
|
|
this.min = () => {
|
|
|
return Math.min.apply(Math, buffer);
|
|
|
}
|
|
|
}
|
|
|
};
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./src/settings/settings.js":
|
|
|
/*!**********************************!*\
|
|
|
!*** ./src/settings/settings.js ***!
|
|
|
\**********************************/
|
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
|
|
/* Licensed under the Apache License, Version 2.0 (the "License") http://www.apache.org/licenses/LICENSE-2.0 */
|
|
|
const OmUtil = __webpack_require__(/*! ../main/omutils */ "../main/omutils");
|
|
|
const Settings = __webpack_require__(/*! ../main/settings */ "../main/settings");
|
|
|
|
|
|
const MicLevel = __webpack_require__(/*! ./mic-level */ "./src/settings/mic-level.js");
|
|
|
const VideoUtil = __webpack_require__(/*! ./video-util */ "./src/settings/video-util.js");
|
|
|
const WebRtcPeer = __webpack_require__(/*! ./WebRtcPeer */ "./src/settings/WebRtcPeer.js");
|
|
|
|
|
|
const DEV_AUDIO = 'audioinput'
|
|
|
, DEV_VIDEO = 'videoinput'
|
|
|
, MsgBase = {type: 'kurento', mode: 'test'};
|
|
|
let vs, lm, s, cam, mic, res, o, rtcPeer, timer
|
|
|
, vidScroll, vid, recBtn, playBtn, recAllowed = false
|
|
|
, level;
|
|
|
|
|
|
function _load() {
|
|
|
s = Settings.load();
|
|
|
if (!s.video) {
|
|
|
const _res = $('#video-settings .cam-resolution option:selected').data();
|
|
|
s.video = {
|
|
|
cam: 0
|
|
|
, mic: 0
|
|
|
, width: _res.width
|
|
|
, height: _res.height
|
|
|
};
|
|
|
}
|
|
|
if (!s.fixed) {
|
|
|
s.fixed = {
|
|
|
enabled: false
|
|
|
, width: 120
|
|
|
, height: 90
|
|
|
};
|
|
|
}
|
|
|
return s;
|
|
|
}
|
|
|
function _save() {
|
|
|
Settings.save(s);
|
|
|
OmUtil.sendMessage({
|
|
|
type: 'av'
|
|
|
, area: 'room'
|
|
|
, settings: s
|
|
|
});
|
|
|
}
|
|
|
function _clear(_ms) {
|
|
|
const ms = _ms || (vid && vid.length === 1 ? vid[0].srcObject : null);
|
|
|
VideoUtil.cleanStream(ms);
|
|
|
if (vid && vid.length === 1) {
|
|
|
vid[0].srcObject = null;
|
|
|
}
|
|
|
VideoUtil.cleanPeer(rtcPeer);
|
|
|
if (!!lm) {
|
|
|
lm.hide();
|
|
|
}
|
|
|
if (!!level) {
|
|
|
level.dispose();
|
|
|
level = null;
|
|
|
}
|
|
|
}
|
|
|
function _close() {
|
|
|
_clear();
|
|
|
Wicket.Event.unsubscribe('/websocket/message', _onWsMessage);
|
|
|
}
|
|
|
function _onIceCandidate(candidate) {
|
|
|
OmUtil.log('Local candidate' + JSON.stringify(candidate));
|
|
|
OmUtil.sendMessage({
|
|
|
id : 'iceCandidate'
|
|
|
, candidate: candidate
|
|
|
}, MsgBase);
|
|
|
}
|
|
|
function _init(options) {
|
|
|
o = JSON.parse(JSON.stringify(options));
|
|
|
if (!!o.infoMsg) {
|
|
|
OmUtil.alert('info', o.infoMsg, 0);
|
|
|
}
|
|
|
vs = $('#video-settings');
|
|
|
lm = vs.find('.level-meter');
|
|
|
cam = vs.find('select.cam').change(function() {
|
|
|
_readValues();
|
|
|
});
|
|
|
mic = vs.find('select.mic').change(function() {
|
|
|
_readValues();
|
|
|
});
|
|
|
res = vs.find('select.cam-resolution').change(function() {
|
|
|
_readValues();
|
|
|
});
|
|
|
vidScroll = vs.find('.vid-block .video-conainer');
|
|
|
timer = vs.find('.timer');
|
|
|
vid = vidScroll.find('video');
|
|
|
recBtn = vs.find('.rec-start')
|
|
|
.click(function() {
|
|
|
recBtn.prop('disabled', true);
|
|
|
_setEnabled(true);
|
|
|
OmUtil.sendMessage({
|
|
|
id : 'wannaRecord'
|
|
|
}, MsgBase);
|
|
|
});
|
|
|
playBtn = vs.find('.play')
|
|
|
.click(function() {
|
|
|
recBtn.prop('disabled', true);
|
|
|
_setEnabled(true);
|
|
|
OmUtil.sendMessage({
|
|
|
id : 'wannaPlay'
|
|
|
}, MsgBase);
|
|
|
});
|
|
|
vs.find('.btn-save').off().click(function() {
|
|
|
_save();
|
|
|
_close();
|
|
|
vs.modal("hide");
|
|
|
});
|
|
|
vs.find('.btn-cancel').off().click(function() {
|
|
|
_close();
|
|
|
vs.modal("hide");
|
|
|
});
|
|
|
vs.off().on('hidden.bs.modal', function () {
|
|
|
_close();
|
|
|
});
|
|
|
o.width = 300;
|
|
|
o.height = 200;
|
|
|
o.mode = 'settings';
|
|
|
o.rights = (o.rights || []).join();
|
|
|
delete o.keycode;
|
|
|
vs.find('.modal-body input, .modal-body button').prop('disabled', true);
|
|
|
const rr = vs.find('.cam-resolution').parents('.sett-row');
|
|
|
if (!o.interview) {
|
|
|
rr.show();
|
|
|
} else {
|
|
|
rr.hide();
|
|
|
}
|
|
|
_load();
|
|
|
_save(); // trigger settings update
|
|
|
}
|
|
|
function _updateRec() {
|
|
|
recBtn.prop('disabled', !recAllowed || (s.video.cam < 0 && s.video.mic < 0));
|
|
|
}
|
|
|
function _setCntsDimensions(cnts) {
|
|
|
if (VideoUtil.isSafari()) {
|
|
|
let width = s.video.width;
|
|
|
//valid widths are 320, 640, 1280
|
|
|
[320, 640, 1280].some(function(w) {
|
|
|
if (width < w + 1) {
|
|
|
width = w;
|
|
|
return true;
|
|
|
}
|
|
|
return false;
|
|
|
});
|
|
|
cnts.video.width = width < 1281 ? width : 1280;
|
|
|
} else {
|
|
|
cnts.video.width = o.interview ? 320 : s.video.width;
|
|
|
cnts.video.height = o.interview ? 260 : s.video.height;
|
|
|
}
|
|
|
}
|
|
|
//each bool OR https://developer.mozilla.org/en-US/docs/Web/API/MediaTrackConstraints
|
|
|
// min/ideal/max/exact/mandatory can also be used
|
|
|
function _constraints(sd, callback) {
|
|
|
_getDevConstraints(function(devCnts) {
|
|
|
const cnts = {
|
|
|
videoEnabled: VideoUtil.hasCam(sd)
|
|
|
, audioEnabled: VideoUtil.hasMic(sd)
|
|
|
};
|
|
|
if (devCnts.video && false === o.audioOnly && s.video.cam > -1) {
|
|
|
cnts.video = {
|
|
|
frameRate: o.camera.fps
|
|
|
};
|
|
|
_setCntsDimensions(cnts)
|
|
|
if (!!s.video.camDevice) {
|
|
|
cnts.video.deviceId = {
|
|
|
ideal: s.video.camDevice
|
|
|
};
|
|
|
} else {
|
|
|
cnts.video.facingMode = {
|
|
|
ideal: 'user'
|
|
|
}
|
|
|
}
|
|
|
} else {
|
|
|
cnts.video = false;
|
|
|
}
|
|
|
if (devCnts.audio && s.video.mic > -1) {
|
|
|
cnts.audio = {
|
|
|
sampleRate: o.microphone.rate
|
|
|
, echoCancellation: o.microphone.echo
|
|
|
, noiseSuppression: o.microphone.noise
|
|
|
};
|
|
|
if (!!s.video.micDevice) {
|
|
|
cnts.audio.deviceId = {
|
|
|
ideal: s.video.micDevice
|
|
|
};
|
|
|
}
|
|
|
} else {
|
|
|
cnts.audio = false;
|
|
|
}
|
|
|
callback(cnts);
|
|
|
});
|
|
|
}
|
|
|
function _readValues(msg, func) {
|
|
|
const v = cam.find('option:selected')
|
|
|
, m = mic.find('option:selected')
|
|
|
, o = res.find('option:selected').data();
|
|
|
s.video.cam = 1 * cam.val();
|
|
|
s.video.camDevice = v.data('device-id');
|
|
|
s.video.mic = 1 * mic.val();
|
|
|
s.video.micDevice = m.data('device-id');
|
|
|
s.video.width = o.width;
|
|
|
s.video.height = o.height;
|
|
|
vid.width(o.width).height(o.height);
|
|
|
vidScroll.scrollLeft(Math.max(0, s.video.width / 2 - 150))
|
|
|
.scrollTop(Math.max(0, s.video.height / 2 - 110));
|
|
|
_clear();
|
|
|
_constraints(null, function(cnts) {
|
|
|
if (cnts.video !== false || cnts.audio !== false) {
|
|
|
const options = VideoUtil.addIceServers({
|
|
|
mediaConstraints: cnts
|
|
|
, onIceCandidate: _onIceCandidate
|
|
|
}, msg);
|
|
|
navigator.mediaDevices.getUserMedia(cnts)
|
|
|
.then(stream => {
|
|
|
VideoUtil.playSrc(vid[0], stream, true);
|
|
|
options.mediaStream = stream;
|
|
|
|
|
|
rtcPeer = new WebRtcPeer.Sendonly(options);
|
|
|
if (cnts.audio) {
|
|
|
lm.show();
|
|
|
level = new MicLevel();
|
|
|
level.meterStream(stream, lm, function(){}, OmUtil.error, false);
|
|
|
} else {
|
|
|
lm.hide();
|
|
|
}
|
|
|
return rtcPeer.createOffer();
|
|
|
})
|
|
|
.then(sdpOffer => {
|
|
|
rtcPeer.processLocalOffer(sdpOffer);
|
|
|
if (typeof(func) === 'function') {
|
|
|
func(sdpOffer.sdp, cnts);
|
|
|
} else {
|
|
|
_allowRec(true);
|
|
|
}
|
|
|
}).catch(_ => OmUtil.error('Error generating the offer'));
|
|
|
}
|
|
|
if (!msg) {
|
|
|
_updateRec();
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
function _allowRec(allow) {
|
|
|
recAllowed = allow;
|
|
|
_updateRec();
|
|
|
}
|
|
|
function _setLoading(el) {
|
|
|
el.find('option').remove();
|
|
|
el.append(OmUtil.tmpl('#settings-option-loading'));
|
|
|
}
|
|
|
function _setDisabled(els) {
|
|
|
els.forEach(function(el) {
|
|
|
el.find('option').remove();
|
|
|
el.append(OmUtil.tmpl('#settings-option-disabled'));
|
|
|
});
|
|
|
}
|
|
|
function _setSelectedDevice(dev, devIdx) {
|
|
|
let o = dev.find('option[value="' + devIdx + '"]');
|
|
|
if (o.length === 0 && devIdx !== -1) {
|
|
|
o = dev.find('option[value="0"]');
|
|
|
}
|
|
|
o.prop('selected', true);
|
|
|
}
|
|
|
function _getDevConstraints(callback) {
|
|
|
const devCnts = {audio: false, video: false, devices: []};
|
|
|
if (window.isSecureContext === false) {
|
|
|
OmUtil.error($('#settings-https-required').text());
|
|
|
return;
|
|
|
}
|
|
|
if (!navigator.mediaDevices || !navigator.mediaDevices.enumerateDevices) {
|
|
|
OmUtil.error('enumerateDevices() not supported.');
|
|
|
return;
|
|
|
}
|
|
|
navigator.mediaDevices.enumerateDevices()
|
|
|
.then(devices => devices.forEach(device => {
|
|
|
if (DEV_AUDIO === device.kind || DEV_VIDEO === device.kind) {
|
|
|
devCnts.devices.push({
|
|
|
kind: device.kind
|
|
|
, label: device.label || (device.kind + ' ' + devCnts.devices.length)
|
|
|
, deviceId: device.deviceId
|
|
|
});
|
|
|
}
|
|
|
if (DEV_AUDIO === device.kind) {
|
|
|
devCnts.audio = true;
|
|
|
} else if (DEV_VIDEO === device.kind) {
|
|
|
devCnts.video = true;
|
|
|
}
|
|
|
}))
|
|
|
.catch(() => OmUtil.error('Unable to get the list of multimedia devices'))
|
|
|
.finally(() => callback(devCnts));
|
|
|
}
|
|
|
function _initDevices() {
|
|
|
if (window.isSecureContext === false) {
|
|
|
OmUtil.error($('#settings-https-required').text());
|
|
|
return;
|
|
|
}
|
|
|
if (!navigator.mediaDevices || !navigator.mediaDevices.enumerateDevices) {
|
|
|
OmUtil.error('enumerateDevices() not supported.');
|
|
|
return;
|
|
|
}
|
|
|
_setLoading(cam);
|
|
|
_setLoading(mic);
|
|
|
_getDevConstraints(function(devCnts) {
|
|
|
if (!devCnts.audio && !devCnts.video) {
|
|
|
_setDisabled([cam, mic]);
|
|
|
return;
|
|
|
}
|
|
|
navigator.mediaDevices.getUserMedia(devCnts)
|
|
|
.then(stream => {
|
|
|
const devices = navigator.mediaDevices.enumerateDevices()
|
|
|
.catch(function(err) {
|
|
|
throw err;
|
|
|
})
|
|
|
.finally(() => _clear(stream));
|
|
|
return devices || devCnts.devices;
|
|
|
})
|
|
|
.catch(function() {
|
|
|
return devCnts.devices;
|
|
|
})
|
|
|
.then(devices => {
|
|
|
let cCount = 0, mCount = 0;
|
|
|
_load();
|
|
|
_setDisabled([cam, mic]);
|
|
|
devices.forEach(device => {
|
|
|
if (DEV_AUDIO === device.kind) {
|
|
|
const o = $('<option></option>').attr('value', mCount).text(device.label)
|
|
|
.data('device-id', device.deviceId);
|
|
|
mic.append(o);
|
|
|
mCount++;
|
|
|
} else if (DEV_VIDEO === device.kind) {
|
|
|
const o = $('<option></option>').attr('value', cCount).text(device.label)
|
|
|
.data('device-id', device.deviceId);
|
|
|
cam.append(o);
|
|
|
cCount++;
|
|
|
}
|
|
|
});
|
|
|
_setSelectedDevice(cam, s.video.cam);
|
|
|
_setSelectedDevice(mic, s.video.mic);
|
|
|
res.find('option').each(function() {
|
|
|
const o = $(this).data();
|
|
|
if (o.width === s.video.width && o.height === s.video.height) {
|
|
|
$(this).prop('selected', true);
|
|
|
return false;
|
|
|
}
|
|
|
});
|
|
|
_readValues();
|
|
|
})
|
|
|
.catch(function(err) {
|
|
|
_setDisabled([cam, mic]);
|
|
|
OmUtil.error(err);
|
|
|
});
|
|
|
});
|
|
|
}
|
|
|
function _open() {
|
|
|
Wicket.Event.subscribe('/websocket/message', _onWsMessage);
|
|
|
recAllowed = false;
|
|
|
timer.hide();
|
|
|
playBtn.prop('disabled', true);
|
|
|
vs.modal('show');
|
|
|
_load();
|
|
|
_initDevices();
|
|
|
}
|
|
|
function _setEnabled(enabled) {
|
|
|
playBtn.prop('disabled', enabled);
|
|
|
cam.prop('disabled', enabled);
|
|
|
mic.prop('disabled', enabled);
|
|
|
res.prop('disabled', enabled);
|
|
|
}
|
|
|
function _onStop() {
|
|
|
_updateRec();
|
|
|
_setEnabled(false);
|
|
|
}
|
|
|
function _onKMessage(m) {
|
|
|
OmUtil.info('Received message: ', m);
|
|
|
switch (m.id) {
|
|
|
case 'canRecord':
|
|
|
_readValues(m, function(_offerSdp, cnts) {
|
|
|
OmUtil.info('Invoking SDP offer callback function');
|
|
|
OmUtil.sendMessage({
|
|
|
id : 'record'
|
|
|
, sdpOffer: _offerSdp
|
|
|
, video: cnts.video !== false
|
|
|
, audio: cnts.audio !== false
|
|
|
}, MsgBase);
|
|
|
});
|
|
|
break;
|
|
|
case 'canPlay': {
|
|
|
const options = VideoUtil.addIceServers({
|
|
|
mediaConstraints: {audio: true, video: true}
|
|
|
, onIceCandidate: _onIceCandidate
|
|
|
}, m);
|
|
|
_clear();
|
|
|
rtcPeer = new WebRtcPeer.Recvonly(options);
|
|
|
rtcPeer.createOffer()
|
|
|
.then(sdpOffer => {
|
|
|
rtcPeer.processLocalOffer(sdpOffer);
|
|
|
OmUtil.sendMessage({
|
|
|
id : 'play'
|
|
|
, sdpOffer: sdpOffer.sdp
|
|
|
}, MsgBase);
|
|
|
})
|
|
|
.catch(_ => OmUtil.error('Error generating the offer'));
|
|
|
}
|
|
|
break;
|
|
|
case 'playResponse':
|
|
|
OmUtil.log('Play SDP answer received from server. Processing ...');
|
|
|
|
|
|
rtcPeer.processRemoteAnswer(m.sdpAnswer)
|
|
|
.then(() => {
|
|
|
const stream = rtcPeer.stream;
|
|
|
if (stream) {
|
|
|
VideoUtil.playSrc(vid[0], stream, false);
|
|
|
lm.show();
|
|
|
level = new MicLevel();
|
|
|
level.meterStream(stream, lm, function(){}, OmUtil.error, true);
|
|
|
};
|
|
|
})
|
|
|
.catch(error => OmUtil.error(error));
|
|
|
break;
|
|
|
case 'startResponse':
|
|
|
OmUtil.log('SDP answer received from server. Processing ...');
|
|
|
rtcPeer.processRemoteAnswer(m.sdpAnswer)
|
|
|
.catch(error => OmUtil.error(error));
|
|
|
break;
|
|
|
case 'iceCandidate':
|
|
|
rtcPeer.addIceCandidate(m.candidate)
|
|
|
.catch(error => OmUtil.error('Error adding candidate: ' + error));
|
|
|
break;
|
|
|
case 'recording':
|
|
|
timer.show().find('.time').text(m.time);
|
|
|
break;
|
|
|
case 'recStopped':
|
|
|
timer.hide();
|
|
|
_onStop();
|
|
|
break;
|
|
|
case 'playStopped':
|
|
|
_onStop();
|
|
|
_readValues();
|
|
|
break;
|
|
|
default:
|
|
|
// no-op
|
|
|
}
|
|
|
}
|
|
|
function _onWsMessage(jqEvent, msg) {
|
|
|
try {
|
|
|
if (msg instanceof Blob) {
|
|
|
return; //ping
|
|
|
}
|
|
|
const m = JSON.parse(msg);
|
|
|
if (m && 'kurento' === m.type) {
|
|
|
if ('test' === m.mode) {
|
|
|
_onKMessage(m);
|
|
|
}
|
|
|
switch (m.id) {
|
|
|
case 'error':
|
|
|
OmUtil.error(m.message);
|
|
|
break;
|
|
|
default:
|
|
|
//no-op
|
|
|
}
|
|
|
}
|
|
|
} catch (err) {
|
|
|
OmUtil.error(err);
|
|
|
}
|
|
|
}
|
|
|
|
|
|
module.exports = {
|
|
|
init: _init
|
|
|
, open: _open
|
|
|
, close: function() {
|
|
|
_close();
|
|
|
vs && vs.modal('hide');
|
|
|
}
|
|
|
, load: _load
|
|
|
, save: _save
|
|
|
, constraints: _constraints
|
|
|
};
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./src/settings/video-util.js":
|
|
|
/*!************************************!*\
|
|
|
!*** ./src/settings/video-util.js ***!
|
|
|
\************************************/
|
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
|
|
/* Licensed under the Apache License, Version 2.0 (the "License") http://www.apache.org/licenses/LICENSE-2.0 */
|
|
|
const OmUtil = __webpack_require__(/*! ../main/omutils */ "../main/omutils");
|
|
|
const Settings = __webpack_require__(/*! ../main/settings */ "../main/settings");
|
|
|
|
|
|
const UAParser = __webpack_require__(/*! ua-parser-js */ "./node_modules/ua-parser-js/src/ua-parser.js")
|
|
|
, ua = (typeof window !== 'undefined' && window.navigator) ? window.navigator.userAgent : ''
|
|
|
, parser = new UAParser(ua)
|
|
|
, browser = parser.getBrowser();
|
|
|
|
|
|
const WB_AREA_SEL = '.room-block .wb-block';
|
|
|
const WBA_WB_SEL = '.room-block .wb-block .wb-tab-content';
|
|
|
const VIDWIN_SEL = '.video.user-video';
|
|
|
const VID_SEL = '.video-container[id!=user-video]';
|
|
|
const CAM_ACTIVITY = 'VIDEO';
|
|
|
const MIC_ACTIVITY = 'AUDIO';
|
|
|
const SCREEN_ACTIVITY = 'SCREEN';
|
|
|
const REC_ACTIVITY = 'RECORD';
|
|
|
|
|
|
function _isSafari() {
|
|
|
return browser.name === 'Safari';
|
|
|
}
|
|
|
function _isChrome() {
|
|
|
return browser.name === 'Chrome' || browser.name === 'Chromium';
|
|
|
}
|
|
|
function _isEdge() {
|
|
|
return browser.name === 'Edge' && "MSGestureEvent" in window;
|
|
|
}
|
|
|
function _isEdgeChromium() {
|
|
|
return browser.name === 'Edge' && !("MSGestureEvent" in window);
|
|
|
}
|
|
|
|
|
|
function _getVid(uid) {
|
|
|
return 'video' + uid;
|
|
|
}
|
|
|
function _isSharing(sd) {
|
|
|
return !!sd && 'SCREEN' === sd.type && sd.activities.includes(SCREEN_ACTIVITY);
|
|
|
}
|
|
|
function _isRecording(sd) {
|
|
|
return !!sd && 'SCREEN' === sd.type && sd.activities.includes(REC_ACTIVITY);
|
|
|
}
|
|
|
function _hasActivity(sd, act) {
|
|
|
return !!sd && sd.activities.includes(act);
|
|
|
}
|
|
|
function _hasMic(sd) {
|
|
|
if (!sd) {
|
|
|
return true;
|
|
|
}
|
|
|
const enabled = sd.micEnabled !== false;
|
|
|
return sd.activities.includes(MIC_ACTIVITY) && enabled;
|
|
|
}
|
|
|
function _hasCam(sd) {
|
|
|
if (!sd) {
|
|
|
return true;
|
|
|
}
|
|
|
const enabled = sd.camEnabled !== false;
|
|
|
return sd.activities.includes(CAM_ACTIVITY) && enabled;
|
|
|
}
|
|
|
function _hasVideo(sd) {
|
|
|
return _hasCam(sd) || _isSharing(sd) || _isRecording(sd);
|
|
|
}
|
|
|
function _getRects(sel, excl) {
|
|
|
const list = [], elems = $(sel);
|
|
|
for (let i = 0; i < elems.length; ++i) {
|
|
|
if (excl !== $(elems[i]).attr('aria-describedby')) {
|
|
|
list.push(_getRect(elems[i]));
|
|
|
}
|
|
|
}
|
|
|
return list;
|
|
|
}
|
|
|
function _getRect(e) {
|
|
|
const win = $(e), winoff = win.offset();
|
|
|
return {left: winoff.left
|
|
|
, top: winoff.top
|
|
|
, right: winoff.left + win.width()
|
|
|
, bottom: winoff.top + win.height()};
|
|
|
}
|
|
|
function _container() {
|
|
|
const a = $(WB_AREA_SEL);
|
|
|
const c = a.find('.wb-area .tabs .wb-tab-content');
|
|
|
return c.length > 0 ? $(WBA_WB_SEL) : a;
|
|
|
}
|
|
|
function __processTopToBottom(area, rectNew, list) {
|
|
|
const offsetX = 20
|
|
|
, offsetY = 10;
|
|
|
|
|
|
let minY = area.bottom, posFound;
|
|
|
do {
|
|
|
posFound = true;
|
|
|
for (let i = 0; i < list.length; ++i) {
|
|
|
const rect = list[i];
|
|
|
minY = Math.min(minY, rect.bottom);
|
|
|
|
|
|
if (rectNew.left < rect.right && rectNew.right > rect.left && rectNew.top < rect.bottom && rectNew.bottom > rect.top) {
|
|
|
rectNew.left = rect.right + offsetX;
|
|
|
posFound = false;
|
|
|
}
|
|
|
if (rectNew.right >= area.right) {
|
|
|
rectNew.left = area.left;
|
|
|
rectNew.top = Math.max(minY, rectNew.top) + offsetY;
|
|
|
posFound = false;
|
|
|
}
|
|
|
if (rectNew.bottom >= area.bottom) {
|
|
|
rectNew.top = area.top;
|
|
|
posFound = true;
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
} while (!posFound);
|
|
|
return {left: rectNew.left, top: rectNew.top};
|
|
|
}
|
|
|
function __processEqualsBottomToTop(area, rectNew, list) {
|
|
|
const offsetX = 20
|
|
|
, offsetY = 10;
|
|
|
|
|
|
rectNew.bottom = area.bottom;
|
|
|
let minY = area.bottom, posFound;
|
|
|
do {
|
|
|
posFound = true;
|
|
|
for (let i = 0; i < list.length; ++i) {
|
|
|
const rect = list[i];
|
|
|
minY = Math.min(minY, rect.top);
|
|
|
|
|
|
if (rectNew.left < rect.right && rectNew.right > rect.left && rectNew.top < rect.bottom && rectNew.bottom > rect.top) {
|
|
|
rectNew.left = rect.right + offsetX;
|
|
|
posFound = false;
|
|
|
}
|
|
|
if (rectNew.right >= area.right) {
|
|
|
rectNew.left = area.left;
|
|
|
rectNew.bottom = Math.min(minY, rectNew.top) - offsetY;
|
|
|
posFound = false;
|
|
|
}
|
|
|
if (rectNew.top <= area.top) {
|
|
|
rectNew.top = area.top;
|
|
|
posFound = true;
|
|
|
break;
|
|
|
}
|
|
|
}
|
|
|
} while (!posFound);
|
|
|
return {left: rectNew.left, top: rectNew.top};
|
|
|
}
|
|
|
function _getPos(list, w, h, _processor) {
|
|
|
if (Room.getOptions().interview) {
|
|
|
return {left: 0, top: 0};
|
|
|
}
|
|
|
const wba = _container()
|
|
|
, woffset = wba.offset()
|
|
|
, area = {left: woffset.left, top: woffset.top, right: woffset.left + wba.width(), bottom: woffset.top + wba.height()}
|
|
|
, rectNew = {
|
|
|
_left: area.left
|
|
|
, _top: area.top
|
|
|
, _right: area.left + w
|
|
|
, _bottom: area.top + h
|
|
|
, get left() {
|
|
|
return this._left;
|
|
|
}
|
|
|
, set left(l) {
|
|
|
this._left = l;
|
|
|
this._right = l + w;
|
|
|
}
|
|
|
, get right() {
|
|
|
return this._right;
|
|
|
}
|
|
|
, get top() {
|
|
|
return this._top;
|
|
|
}
|
|
|
, set top(t) {
|
|
|
this._top = t;
|
|
|
this._bottom = t + h;
|
|
|
}
|
|
|
, set bottom(b) {
|
|
|
this._bottom = b;
|
|
|
this._top = b - h;
|
|
|
}
|
|
|
, get bottom() {
|
|
|
return this._bottom;
|
|
|
}
|
|
|
};
|
|
|
const processor = _processor || __processTopToBottom;
|
|
|
return processor(area, rectNew, list);
|
|
|
}
|
|
|
function _arrange() {
|
|
|
const list = [];
|
|
|
$(VIDWIN_SEL).each(function() {
|
|
|
const v = $(this);
|
|
|
v.css(_getPos(list, v.width(), v.height()));
|
|
|
list.push(_getRect(v));
|
|
|
});
|
|
|
}
|
|
|
function _arrangeResize(vSettings) {
|
|
|
const list = []
|
|
|
, size = {width: 120, height: 90};
|
|
|
if (vSettings.fixed.enabled) {
|
|
|
size.width = vSettings.fixed.width;
|
|
|
size.height = vSettings.fixed.height;
|
|
|
}
|
|
|
|
|
|
function __getDialog(_v) {
|
|
|
return $(_v).find('.video-container.ui-dialog-content');
|
|
|
}
|
|
|
$(VIDWIN_SEL).toArray().sort((v1, v2) => {
|
|
|
const c1 = __getDialog(v1).data().stream()
|
|
|
, c2 = __getDialog(v2).data().stream();
|
|
|
return c2.level - c1.level || c1.user.displayName.localeCompare(c2.user.displayName);
|
|
|
}).forEach(_v => {
|
|
|
const v = $(_v);
|
|
|
__getDialog(v)
|
|
|
.dialog('option', 'width', size.width)
|
|
|
.dialog('option', 'height', size.height);
|
|
|
v.css(_getPos(list, v.width(), v.height(), __processEqualsBottomToTop));
|
|
|
list.push(_getRect(v));
|
|
|
});
|
|
|
}
|
|
|
function _cleanStream(stream) {
|
|
|
if (!!stream) {
|
|
|
stream.getTracks().forEach(track => track.stop());
|
|
|
}
|
|
|
}
|
|
|
function _cleanPeer(rtcPeer) {
|
|
|
if (!!rtcPeer) {
|
|
|
try {
|
|
|
const pc = rtcPeer.pc;
|
|
|
if (!!pc) {
|
|
|
pc.getSenders().forEach(sender => {
|
|
|
try {
|
|
|
if (sender.track) {
|
|
|
sender.track.stop();
|
|
|
}
|
|
|
} catch(e) {
|
|
|
OmUtil.log('Failed to clean sender' + e);
|
|
|
}
|
|
|
});
|
|
|
pc.getReceivers().forEach(receiver => {
|
|
|
try {
|
|
|
if (receiver.track) {
|
|
|
receiver.track.stop();
|
|
|
}
|
|
|
} catch(e) {
|
|
|
OmUtil.log('Failed to clean receiver' + e);
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
rtcPeer.dispose();
|
|
|
} catch(e) {
|
|
|
//no-op
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
function _setPos(v, pos) {
|
|
|
if (v.dialog('instance')) {
|
|
|
v.dialog('widget').css(pos);
|
|
|
}
|
|
|
}
|
|
|
function _askPermission(callback) {
|
|
|
const perm = $('#ask-permission');
|
|
|
$('.sidebar').confirmation({
|
|
|
title: perm.attr('title')
|
|
|
, placement: Settings.isRtl ? 'right' : 'left'
|
|
|
, singleton: true
|
|
|
, rootSelector: '.sidebar'
|
|
|
, html: true
|
|
|
, content: perm.html()
|
|
|
, buttons: [{
|
|
|
class: 'btn btn-sm btn-warning'
|
|
|
, label: perm.data('btn-ok')
|
|
|
, value: perm.data('btn-ok')
|
|
|
, onClick: function() {
|
|
|
callback();
|
|
|
$('.sidebar').confirmation('dispose');
|
|
|
}
|
|
|
}]
|
|
|
});
|
|
|
$('.sidebar').confirmation('show');
|
|
|
}
|
|
|
function _disconnect(node) {
|
|
|
try {
|
|
|
node.disconnect(); //this one can throw
|
|
|
} catch (e) {
|
|
|
//no-op
|
|
|
}
|
|
|
}
|
|
|
function _sharingSupported() {
|
|
|
return (browser.name === 'Edge' && browser.major > 16)
|
|
|
|| (typeof(navigator.mediaDevices.getDisplayMedia) === 'function'
|
|
|
&& (browser.name === 'Firefox'
|
|
|
|| browser.name === 'Opera'
|
|
|
|| browser.name === 'Yandex'
|
|
|
|| _isSafari()
|
|
|
|| _isChrome()
|
|
|
|| _isEdgeChromium()
|
|
|
|| (browser.name === 'Mozilla' && browser.major > 4)
|
|
|
));
|
|
|
}
|
|
|
function _highlight(el, clazz, count) {
|
|
|
if (!el || el.length < 1 || el.hasClass('disabled') || count < 0) {
|
|
|
return;
|
|
|
}
|
|
|
el.addClass(clazz).delay(2000).queue(function(next) {
|
|
|
el.removeClass(clazz).delay(2000).queue(function(next1) {
|
|
|
_highlight(el, clazz, --count);
|
|
|
next1();
|
|
|
});
|
|
|
next();
|
|
|
});
|
|
|
}
|
|
|
function _playSrc(_video, _stream, mute) {
|
|
|
if (_stream && _video) {
|
|
|
_video.srcObject = _stream;
|
|
|
if (_video.paused) {
|
|
|
_video.play().then(() => _video.muted = mute).catch(err => {
|
|
|
if ('NotAllowedError' === err.name) {
|
|
|
_askPermission(() => _video.play().then(() => _video.muted = mute));
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
module.exports = {
|
|
|
VIDWIN_SEL: VIDWIN_SEL
|
|
|
, VID_SEL: VID_SEL
|
|
|
, CAM_ACTIVITY: CAM_ACTIVITY
|
|
|
, MIC_ACTIVITY: MIC_ACTIVITY
|
|
|
|
|
|
, getVid: _getVid
|
|
|
, isSharing: _isSharing
|
|
|
, isRecording: _isRecording
|
|
|
, hasMic: _hasMic
|
|
|
, hasCam: _hasCam
|
|
|
, hasVideo: _hasVideo
|
|
|
, hasActivity: _hasActivity
|
|
|
, getRects: _getRects
|
|
|
, getPos: _getPos
|
|
|
, container: _container
|
|
|
, arrange: _arrange
|
|
|
, arrangeResize: _arrangeResize
|
|
|
, cleanStream: _cleanStream
|
|
|
, cleanPeer: _cleanPeer
|
|
|
, addIceServers: function(opts, m) {
|
|
|
if (m && m.iceServers && m.iceServers.length > 0) {
|
|
|
opts.iceServers = m.iceServers;
|
|
|
}
|
|
|
return opts;
|
|
|
}
|
|
|
, setPos: _setPos
|
|
|
, askPermission: _askPermission
|
|
|
, disconnect: _disconnect
|
|
|
, sharingSupported: _sharingSupported
|
|
|
, highlight: _highlight
|
|
|
, playSrc: _playSrc
|
|
|
|
|
|
, browser: browser
|
|
|
, isEdge: _isEdge
|
|
|
, isEdgeChromium: _isEdgeChromium
|
|
|
, isChrome: _isChrome
|
|
|
, isSafari: _isSafari
|
|
|
};
|
|
|
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "../main/omutils":
|
|
|
/*!*************************!*\
|
|
|
!*** external "OmUtil" ***!
|
|
|
\*************************/
|
|
|
/***/ ((module) => {
|
|
|
|
|
|
"use strict";
|
|
|
module.exports = OmUtil;
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "../main/settings":
|
|
|
/*!***************************!*\
|
|
|
!*** external "Settings" ***!
|
|
|
\***************************/
|
|
|
/***/ ((module) => {
|
|
|
|
|
|
"use strict";
|
|
|
module.exports = Settings;
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/freeice/stun.json":
|
|
|
/*!****************************************!*\
|
|
|
!*** ./node_modules/freeice/stun.json ***!
|
|
|
\****************************************/
|
|
|
/***/ ((module) => {
|
|
|
|
|
|
"use strict";
|
|
|
module.exports = JSON.parse('["stun.l.google.com:19302","stun1.l.google.com:19302","stun2.l.google.com:19302","stun3.l.google.com:19302","stun4.l.google.com:19302","stun.ekiga.net","stun.ideasip.com","stun.schlund.de","stun.stunprotocol.org:3478","stun.voiparound.com","stun.voipbuster.com","stun.voipstunt.com","stun.voxgratia.org"]');
|
|
|
|
|
|
/***/ }),
|
|
|
|
|
|
/***/ "./node_modules/freeice/turn.json":
|
|
|
/*!****************************************!*\
|
|
|
!*** ./node_modules/freeice/turn.json ***!
|
|
|
\****************************************/
|
|
|
/***/ ((module) => {
|
|
|
|
|
|
"use strict";
|
|
|
module.exports = [];
|
|
|
|
|
|
/***/ })
|
|
|
|
|
|
/******/ });
|
|
|
/************************************************************************/
|
|
|
/******/ // The module cache
|
|
|
/******/ var __webpack_module_cache__ = {};
|
|
|
/******/
|
|
|
/******/ // The require function
|
|
|
/******/ function __webpack_require__(moduleId) {
|
|
|
/******/ // Check if module is in cache
|
|
|
/******/ var cachedModule = __webpack_module_cache__[moduleId];
|
|
|
/******/ if (cachedModule !== undefined) {
|
|
|
/******/ return cachedModule.exports;
|
|
|
/******/ }
|
|
|
/******/ // Create a new module (and put it into the cache)
|
|
|
/******/ var module = __webpack_module_cache__[moduleId] = {
|
|
|
/******/ // no module.id needed
|
|
|
/******/ // no module.loaded needed
|
|
|
/******/ exports: {}
|
|
|
/******/ };
|
|
|
/******/
|
|
|
/******/ // Execute the module function
|
|
|
/******/ __webpack_modules__[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
|
|
/******/
|
|
|
/******/ // Return the exports of the module
|
|
|
/******/ return module.exports;
|
|
|
/******/ }
|
|
|
/******/
|
|
|
/************************************************************************/
|
|
|
/******/ /* webpack/runtime/amd options */
|
|
|
/******/ (() => {
|
|
|
/******/ __webpack_require__.amdO = {};
|
|
|
/******/ })();
|
|
|
/******/
|
|
|
/******/ /* webpack/runtime/compat get default export */
|
|
|
/******/ (() => {
|
|
|
/******/ // getDefaultExport function for compatibility with non-harmony modules
|
|
|
/******/ __webpack_require__.n = (module) => {
|
|
|
/******/ var getter = module && module.__esModule ?
|
|
|
/******/ () => (module['default']) :
|
|
|
/******/ () => (module);
|
|
|
/******/ __webpack_require__.d(getter, { a: getter });
|
|
|
/******/ return getter;
|
|
|
/******/ };
|
|
|
/******/ })();
|
|
|
/******/
|
|
|
/******/ /* webpack/runtime/define property getters */
|
|
|
/******/ (() => {
|
|
|
/******/ // define getter functions for harmony exports
|
|
|
/******/ __webpack_require__.d = (exports, definition) => {
|
|
|
/******/ for(var key in definition) {
|
|
|
/******/ if(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {
|
|
|
/******/ Object.defineProperty(exports, key, { enumerable: true, get: definition[key] });
|
|
|
/******/ }
|
|
|
/******/ }
|
|
|
/******/ };
|
|
|
/******/ })();
|
|
|
/******/
|
|
|
/******/ /* webpack/runtime/hasOwnProperty shorthand */
|
|
|
/******/ (() => {
|
|
|
/******/ __webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))
|
|
|
/******/ })();
|
|
|
/******/
|
|
|
/******/ /* webpack/runtime/make namespace object */
|
|
|
/******/ (() => {
|
|
|
/******/ // define __esModule on exports
|
|
|
/******/ __webpack_require__.r = (exports) => {
|
|
|
/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) {
|
|
|
/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });
|
|
|
/******/ }
|
|
|
/******/ Object.defineProperty(exports, '__esModule', { value: true });
|
|
|
/******/ };
|
|
|
/******/ })();
|
|
|
/******/
|
|
|
/************************************************************************/
|
|
|
var __webpack_exports__ = {};
|
|
|
// This entry need to be wrapped in an IIFE because it need to be isolated against other modules in the chunk.
|
|
|
(() => {
|
|
|
/*!*******************************!*\
|
|
|
!*** ./src/settings/index.js ***!
|
|
|
\*******************************/
|
|
|
/* Licensed under the Apache License, Version 2.0 (the "License") http://www.apache.org/licenses/LICENSE-2.0 */
|
|
|
__webpack_require__(/*! webrtc-adapter */ "./node_modules/webrtc-adapter/src/js/adapter_core.js");
|
|
|
|
|
|
if (window.hasOwnProperty('isSecureContext') === false) {
|
|
|
window.isSecureContext = window.location.protocol == 'https:' || ["localhost", "127.0.0.1"].indexOf(window.location.hostname) !== -1;
|
|
|
}
|
|
|
|
|
|
Object.assign(window, {
|
|
|
VideoUtil: __webpack_require__(/*! ./video-util */ "./src/settings/video-util.js")
|
|
|
, MicLevel: __webpack_require__(/*! ./mic-level */ "./src/settings/mic-level.js")
|
|
|
, WebRtcPeer: __webpack_require__(/*! ./WebRtcPeer */ "./src/settings/WebRtcPeer.js")
|
|
|
, VideoSettings: __webpack_require__(/*! ./settings */ "./src/settings/settings.js")
|
|
|
});
|
|
|
|
|
|
})();
|
|
|
|
|
|
/******/ })()
|
|
|
;
|
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"settings.js","mappings":";;;;;;;;;;AAAA;AACa;;AAEb,gBAAgB,mBAAO,CAAC,oDAAW;;AAEnC;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA,qBAAqB,UAAU,mBAAO,CAAC,qDAAa;AACpD,qBAAqB,UAAU,mBAAO,CAAC,qDAAa;AACpD;;AAEA,6BAA6B;AAC7B,6BAA6B;AAC7B;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA,KAAK;AACL;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;;;;;;;;;;AC1GA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA,wBAAwB;AACxB;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,oCAAoC;AACpC;;AAEA;AACA;;AAEA;AACA;;;;;;;;;;;;AC3DA;AACa;;AAEb;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;;AAEA;AACA;AACA,gBAAgB,oBAAoB;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,kBAAkB,kBAAkB;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wCAAwC;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,IAAI;AACJ;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA,6CAA6C;AAC7C;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,oBAAoB;AACpB,2BAA2B;AAC3B;AACA;AACA;AACA,8DAA8D;AAC9D,kBAAkB,kBAAkB;AACpC;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA,KAAK;AACL,iDAAiD;AACjD;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,kBAAkB,OAAO;AAC3C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA,KAAK;AACL,GAAG;AACH;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,6CAA6C;AAC7C;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;;AAEA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,kBAAkB,kBAAkB;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,kBAAkB;AACpC;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,IAAI,IAA0B;AAC9B;AACA;;;;;;;;;;;ACjyBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,0BAA0B,cAAc;AACxC;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;;AAEA;AACA,6CAA6C;AAC7C;;AAEA;AACA;;AAEA,qCAAqC;AACrC;;AAEA;AACA,oCAAoC,kBAAkB;AACtD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sCAAsC;AACtC;AACA;AACA;AACA,kCAAkC;AAClC;AACA;AACA;AACA;AACA,sCAAsC;AACtC;AACA;AACA;AACA,kCAAkC;AAClC;AACA;AACA,8BAA8B;AAC9B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;;AAET;;AAEA;AACA;AACA;AACA,oCAAoC,mBAAmB;AACvD;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,gCAAgC,IAAI;AACpC;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,8CAA8C,IAAI;AAClD;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oCAAoC,IAAI;AACxC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA,sDAAsD,gBAAgB;AACtE;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA,8CAA8C,GAAG;AACjD;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,wBAAwB;AACxB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,uBAAuB;AACvB;AACA;AACA;AACA;AACA;;AAEA;;AAEA,sDAAsD;AACtD;;AAEA,sBAAsB;AACtB;;AAEA,+BAA+B;AAC/B;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA,kCAAkC,IAAI;AACtC;;AAEA,8CAA8C;AAC9C;;AAEA,uBAAuB;AACvB;;AAEA,+BAA+B,0CAA0C;AACzE;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA,kDAAkD,IAAI,WAAW,IAAI;AACrE;AACA;AACA;AACA;AACA;;AAEA;AACA,mDAAmD;AACnD;AACA,sBAAsB,SAAS;AAC/B;AACA,kCAAkC;AAClC;AACA,yBAAyB;AACzB;;AAEA;AACA;AACA;;AAEA;AACA,wDAAwD,EAAE;AAC1D;AACA,wCAAwC;AACxC,4BAA4B,IAAI;AAChC;;AAEA;AACA,gCAAgC,EAAE,WAAW,EAAE;AAC/C,iBAAiB;AACjB;AACA;AACA,wBAAwB,qBAAqB,IAAI,cAAc;AAC/D;AACA;AACA,wBAAwB,KAAK,EAAE;AAC/B;AACA;;AAEA;AACA,eAAe;AACf,0BAA0B,EAAE;AAC5B;;AAEA;AACA;AACA,wBAAwB,EAAE,iBAAiB;AAC3C;;AAEA;AACA,2BAA2B,EAAE,UAAU;AACvC;;AAEA;AACA;AACA;AACA,qCAAqC,IAAI;AACzC;AACA,gCAAgC,IAAI;AACpC;;AAEA;AACA,gCAAgC,EAAE,gBAAgB,EAAE,GAAG,aAAa,IAAI;AACxE;AACA;AACA,qBAAqB;AACrB;AACA;;AAEA;AACA;AACA,6DAA6D,EAAE,WAAW,EAAE;AAC5E;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,sBAAsB,eAAe,IAAI;AACzC;;AAEA;AACA,gCAAgC,EAAE,WAAW,EAAE,yDAAyD,IAAI;AAC5G;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,sBAAsB,EAAE,YAAY,EAAE;AACtC;AACA;AACA;AACA;;AAEA;AACA,wBAAwB,SAAS;AACjC;AACA;AACA,qBAAqB;AACrB;;AAEA;AACA,0CAA0C,MAAM;AAChD;AACA;AACA;;AAEA;AACA;AACA;AACA,qBAAqB,IAAI,IAAI;;AAE7B;AACA;AACA,wDAAwD;AACxD;;AAEA;AACA,sBAAsB;AACtB;;AAEA;AACA,sBAAsB;AACtB,yBAAyB,GAAG;AAC5B;AACA;AACA;AACA,eAAe,0BAA0B,IAAI;AAC7C;;AAEA;AACA;AACA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,kCAAkC,IAAI;AACtC,gCAAgC,EAAE;AAClC,gCAAgC,IAAI;AACpC;AACA;;AAEA;AACA;AACA,4BAA4B;AAC5B;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,IAAI;AAC/B;AACA;AACA;AACA;AACA;AACA,qBAAqB,EAAE;AACvB;AACA;AACA;AACA,qBAAqB,EAAE;AACvB;AACA,sBAAsB,EAAE;AACxB;AACA,sBAAsB,EAAE;AACxB;AACA;AACA;AACA,uBAAuB,EAAE;AACzB,yCAAyC,EAAE;AAC3C;AACA,uBAAuB,IAAI;AAC3B;AACA,+BAA+B,IAAI;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2BAA2B,EAAE;AAC7B;AACA,sBAAsB;AACtB;AACA,sBAAsB;AACtB;;AAEA;AACA;AACA;;AAEA;AACA;AACA,0BAA0B;AAC1B;AACA,mBAAmB;AACnB;AACA;AACA;AACA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA;AACA,sBAAsB,EAAE;AACxB;AACA,qCAAqC;AACrC;AACA;AACA,+CAA+C,WAAW,IAAI,IAAI;AAClE;AACA,qDAAqD;AACrD;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,sBAAsB;AACtB;AACA;AACA;AACA,kCAAkC,WAAW;AAC7C;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,sBAAsB;AACtB;AACA,sBAAsB,QAAQ,IAAI;AAClC;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA,wBAAwB,IAAI,cAAc;AAC1C;AACA,wBAAwB,IAAI;AAC5B;AACA,8BAA8B;AAC9B;AACA,+BAA+B;AAC/B;AACA,8BAA8B,IAAI,EAAE;AACpC;AACA;;AAEA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,yBAAyB,IAAI;AAC7B;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA,8BAA8B;AAC9B;AACA;AACA;;AAEA;AACA,wBAAwB,IAAI,6BAA6B;AACzD,oBAAoB;AACpB;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA,sBAAsB;AACtB;AACA,0CAA0C;AAC1C;AACA,4DAA4D,SAAS;AACrE;AACA,mBAAmB;AACnB;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA,uBAAuB;AACvB;AACA;;AAEA;AACA;AACA,qBAAqB,YAAY;;AAEjC;AACA;AACA;AACA,oCAAoC;AACpC;AACA;AACA;AACA;AACA,+BAA+B,IAAI,mCAAmC,IAAI;AAC1E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0EAA0E;AAC1E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA,YAAY,QAAa;AACzB;AACA;AACA,QAAQ,gBAAgB;AACxB,MAAM;AACN;AACA,YAAY,UAAc,kBAAkB,wBAAU;AACtD,YAAY,mCAAO;AACnB;AACA,aAAa;AAAA,kGAAC;AACd,UAAU;AACV;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,CAAC;;;;;;;;;;;;;;;;;ACj7BD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEa;;AAEuC;;AAEpD;AACA,EAAE,mEAAc,EAAE,2DAA2D;AAC7E,iEAAe,OAAO,EAAC;;;;;;;;;;;;;;;;;;;;;;;ACfvB;AACA;AACA;AACA;AACA;AACA;AACA;AACiC;;AAEjC;AACmD;AACG;AACH;AACP;AACjB;;AAE3B;AACO,yBAAyB,QAAQ,IAAI;AAC5C;AACA;AACA;AACA,CAAC;AACD;AACA,kBAAkB,uCAAS;AAC3B,yBAAyB,iDAAmB;;AAE5C;AACA;AACA,cAAc;AACd,oBAAoB,kDAAoB;AACxC,gBAAgB,8CAAgB;AAChC,qBAAqB,mDAAqB;AAC1C;AACA,OAAO;AACP;;AAEA;AACA;AACA;AACA,WAAW,gDAAU,KAAK,mEAA6B;AACvD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,gDAAU;;AAEtC;AACA,MAAM,wEAAyC;AAC/C,MAAM,8EAA+C;;AAErD,MAAM,iEAA2B;AACjC,MAAM,gEAA0B;AAChC,MAAM,mEAA6B;AACnC,MAAM,4DAAsB;AAC5B,MAAM,wEAAkC;AACxC,MAAM,uEAAiC;AACvC,MAAM,6DAAuB;AAC7B,MAAM,2EAAqC;AAC3C,MAAM,qEAA+B;;AAErC,MAAM,6DAA8B;AACpC,MAAM,0EAA2C;AACjD,MAAM,6DAA8B;AACpC,MAAM,4DAA6B;AACnC,MAAM,gEAAiC;AACvC,MAAM,gEAAiC;AACvC;AACA;AACA,WAAW,kDAAW,KAAK,qEAA8B;AACzD;AACA;AACA;AACA;AACA;AACA;AACA,4BAA4B,kDAAW;;AAEvC;AACA,MAAM,wEAAyC;AAC/C,MAAM,8EAA+C;;AAErD,MAAM,mEAA4B;AAClC,MAAM,qEAA8B;AACpC,MAAM,8DAAuB;AAC7B,MAAM,mEAA4B;AAClC,MAAM,qEAA8B;AACpC,MAAM,uEAAgC;AACtC,MAAM,qEAA8B;AACpC,MAAM,qEAA8B;AACpC,MAAM,oEAA6B;AACnC,MAAM,kEAA2B;AACjC,MAAM,mEAA4B;;AAElC,MAAM,6DAA8B;AACpC,MAAM,6DAA8B;AACpC,MAAM,4DAA6B;AACnC,MAAM,gEAAiC;AACvC;AACA;AACA,WAAW,gDAAU;AACrB;AACA;AACA;AACA;AACA;AACA,4BAA4B,gDAAU;;AAEtC;AACA,MAAM,wEAAyC;AAC/C,MAAM,8EAA+C;;AAErD,MAAM,qEAA+B;AACrC,MAAM,sEAAgC;AACtC,MAAM,iEAA2B;AACjC,MAAM,oEAA8B;AACpC,MAAM,qEAA+B;AACrC,MAAM,0EAAoC;AAC1C,MAAM,iEAA2B;AACjC,MAAM,iEAA2B;;AAEjC,MAAM,6DAA8B;AACpC,MAAM,0EAA2C;AACjD,MAAM,4DAA6B;AACnC,MAAM,gEAAiC;AACvC,MAAM,gEAAiC;AACvC;AACA;AACA;AACA;AACA;;AAEA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AC1IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACa;AACwB;;AAEW;AACM;;AAE/C;AACP;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB;AAChB,4BAA4B;AAC5B;;AAEA;AACA;AACA;AACA,mCAAmC;AACnC;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA,gBAAgB;AAChB,4BAA4B;AAC5B;AACA;AACA;AACA;AACA,mCAAmC;AACnC;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA,IAAI,8DAA6B;AACjC;AACA;AACA,WAAW,QAAQ,sBAAsB;AACzC;AACA;AACA,KAAK;AACL;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,sCAAsC;AACtC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,wBAAwB;AACxB;AACA;AACA,SAAS;AACT;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;;AAEO;AACP;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA,SAAS;AACT;AACA,OAAO;;AAEP;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT,KAAK;AACL;AACA;;AAEO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ,kDAAiB;AACzB;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI,8DAA6B;AACjC;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,QAAQ,kDAAiB;AACzB;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA,QAAQ;AACR;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,SAAS;AACT,QAAQ;AACR;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,uBAAuB;AACvB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;;AAEH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;;AAEL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;;AAEP;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,2BAA2B;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;;AAEA;AACO;AACP,EAAE,8DAA6B;AAC/B;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;;;;;;;;;;;;;;;AC7rBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACa;AACN;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;;;;;;;;;;;;;;;;;ACjDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACa;AACwB;AACrC,gBAAgB,0CAAS;;AAElB;AACP;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,yDAAyD;AACzD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2DAA2D,YAAY;AACvE;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mDAAmD;AACnD;AACA;AACA;AACA,qBAAqB;AACrB,qBAAqB;AACrB;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP,KAAK;AACL;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;AC5LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACa;;AAEc;AACM;;AAE1B;AACP;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,8BAA8B,yDAAuB;AACrD;AACA;AACA;AACA,aAAa,4BAA4B;AACzC;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,EAAE,2DAA6B;AAC/B;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA,GAAG;AACH;;AAEO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA,EAAE,2DAA6B;AAC/B;AACA,8BAA8B,yDAAuB;AACrD;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA,GAAG;AACH;;AAEO;AACP;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA,qBAAqB,wDAAsB;AAC3C;AACA;AACA,oBAAoB,qDAAmB;AACvC;AACA;AACA,KAAK;AACL;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,kBAAkB,sDAAoB;AACtC;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe,cAAc;AAC7B;AACA;AACA;AACA;AACA,aAAa;AACb;AACA;AACA,WAAW;AACX;AACA;;AAEA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA,UAAU;AACV;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;;AAEA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE,2DAA6B;AAC/B;AACA;AACA,GAAG;AACH;;;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP,KAAK;AACL;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA,GAAG;;AAEH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT,QAAQ;AACR;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA,iCAAiC,mBAAmB;AACpD;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0CAA0C,mBAAmB;AAC7D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AC7cA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACa;;AAEqB;AACc;AACM;;AAE/C;AACP;AACA;AACA;AACA;AACA;AACA,gBAAgB;AAChB;AACA,KAAK;AACL;AACA;;AAEO;AACP;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,2BAA2B;AAC3B;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,2CAA2C;AAC3C;AACA;AACA;AACA,aAAa;AACb,YAAY;AACZ;AACA;AACA;AACA;AACA;AACA,2CAA2C;AAC3C;AACA,eAAe;AACf,aAAa;AACb;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE,2DAA6B;AAC/B;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,MAAM,8CAAgB;AACtB;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0CAA0C,KAAK;AAC/C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,QAAQ;AACzB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;AACf;AACA,eAAe;AACf;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gEAAgE;AAChE;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;;;;;;;;;;;;;;;;AC3SA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACa;;AAEN;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6BAA6B;AAC7B,QAAQ;AACR;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACa;;AAEqB;;AAE3B;AACP;AACA;;AAEA;AACA;AACA,IAAI,8CAAgB;AACpB;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACa;AACqB;;AAE3B;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX,SAAS;AACT;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,aAAa;AACb,WAAW;AACX;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;;AAEO;AACP;AACA,2BAA2B;AAC3B;AACA,OAAO,OAAO,iDAAmB;AACjC;AACA;;AAEA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wBAAwB,gCAAgC;AACxD;AACA;AACA,YAAY,8CAAgB;AAC5B;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA,gBAAgB;AAChB;AACA,KAAK;AACL;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA,YAAY;AACZ;AACA;AACA,cAAc;AACd;AACA;AACA;AACA,UAAU;AACV;AACA,wCAAwC,sBAAsB;AAC9D;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,cAAc;AACd;AACA;AACA,YAAY;AACZ;AACA;AACA,cAAc;AACd;AACA;AACA;AACA,UAAU;AACV;AACA,wCAAwC,sBAAsB;AAC9D;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;;;;;;;;;;;;;;;;;;;;;;;;;;AC9VA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACa;;AAEb;AACA;;AAEA;AACA;AACA;AACA,WAAW,SAAS;AACpB,WAAW,SAAS;AACpB,WAAW,SAAS;AACpB,YAAY,SAAS;AACrB;AACO;AACP;AACA;AACA;;AAEA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA,GAAG;AACH;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA,WAAW,UAAU;AACrB;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;;AAEO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,YAAY,QAAQ;AACpB;AACA;AACO;AACP;AACA,kBAAkB;;AAElB;AACA;AACA;AACA;AACA;AACA;;AAEA,SAAS,WAAW;;AAEpB,mCAAmC;AACnC;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ,2DAA2D;AAC3D;AACA;AACA;AACA;AACA;AACA,IAAI,OAAO;AACX;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA,WAAW,GAAG;AACd;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACO;AACP;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uCAAuC,aAAa;AACpD,GAAG,IAAI;AACP;;AAEA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA,OAAO;AACP;AACA,GAAG;AACH;;AAEA;AACO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA,KAAK;AACL,GAAG;AACH;AACA;;;;;;;;;;;;ACrQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,eAAe,mBAAO,CAAC,wCAAiB;;AAExC,gBAAgB,mBAAO,CAAC,gDAAS;;AAEjC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK,sBAAsB,mBAAmB,2BAA2B,cAAc,6BAA6B,YAAY,cAAc;AAC9I;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,8EAA8E,8EAA8E;AAC5J;AACA;AACA,KAAK,sBAAsB,mBAAmB,2BAA2B,cAAc,6BAA6B,YAAY,cAAc;AAC9I;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,6CAA6C,4CAA4C;;AAEzF,oCAAoC,2CAA2C;;AAE/E;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,+EAA+E,4BAA4B,YAAY;AACvH;AACA;AACA,IAAI;AACJ,qFAAqF;AACrF;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,+CAA+C,wBAAwB;AACvE;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;AACA;AACA,0DAA0D,WAAW,GAAG,YAAY;;AAEpF;AACA;AACA;AACA;AACA,OAAO;AACP;AACA,OAAO;AACP;AACA;;AAEA;AACA,qBAAqB,eAAe;AACpC;;AAEA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA,QAAQ;AACR;AACA;;AAEA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;;AAEA,4EAA4E,iCAAiC;;AAE7G;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA,wBAAwB,cAAc;AACtC;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;;AAEA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA,oBAAoB,cAAc;AAClC;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,wFAAwF;;AAExF;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;AACA,OAAO;AACP,iCAAiC,MAAM;AACvC;AACA;;AAEA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA,oCAAoC;AACpC;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA,KAAK;AACL;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;;AAEA;AACA;AACA;AACA;;AAEA;;AAEA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;;;;;;;;;;;ACrlBA;AACA,kBAAkB,mBAAO,CAAC,kDAAc;AACxC,mBAAmB,mBAAO,CAAC,oDAAe;;AAE1C;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,uBAAuB,QAAQ;AAC/B;AACA;AACA;AACA;AACA;AACA,oCAAoC;AACpC;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;AC7FA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;ACjBA;AACA,eAAe,mBAAO,CAAC,wCAAiB;AACxC,iBAAiB,mBAAO,CAAC,0CAAkB;;AAE3C,iBAAiB,mBAAO,CAAC,gDAAa;AACtC,kBAAkB,mBAAO,CAAC,kDAAc;AACxC,mBAAmB,mBAAO,CAAC,kDAAc;;AAEzC;AACA;AACA,cAAc;AACd;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA,EAAE;AACF;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ,GAAG;AACH;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA,EAAE;AACF;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,gDAAgD;AAChD,OAAO;AACP;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA,EAAE;AACF;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB;AAClB;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA;AACA,MAAM;AACN;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN;AACA;AACA,IAAI;AACJ;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA,IAAI;AACJ;AACA;AACA;AACA,IAAI;AACJ,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL,IAAI;AACJ;AACA;AACA;AACA,uBAAuB;AACvB;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,MAAM;AACN,KAAK;AACL;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,gDAAgD;AAChD;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;AC7dA;AACA,eAAe,mBAAO,CAAC,wCAAiB;AACxC,iBAAiB,mBAAO,CAAC,0CAAkB;;AAE3C,iBAAiB,mBAAO,CAAC,kEAAc;AACvC;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,kBAAkB;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,SAAS;AACT;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA,kBAAkB,iBAAiB;AACnC;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH,SAAS;AACT;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,kBAAkB,iBAAiB;AACnC;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH,SAAS;AACT;AACA;AACA;AACA,UAAU;AACV;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA;AACA,KAAK;AACL;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH,EAAE;AACF;AACA;AACA;AACA;AACA,qBAAqB;AACrB,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;;ACnWA;;;;;;;;;;;ACAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;UCAA;UACA;;UAEA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;;UAEA;UACA;;UAEA;UACA;UACA;;;;;WCtBA;;;;;WCAA;WACA;WACA;WACA;WACA;WACA,iCAAiC,WAAW;WAC5C;WACA;;;;;WCPA;WACA;WACA;WACA;WACA,yCAAyC,wCAAwC;WACjF;WACA;WACA;;;;;WCPA;;;;;WCAA;WACA;WACA;WACA,uDAAuD,iBAAiB;WACxE;WACA,gDAAgD,aAAa;WAC7D;;;;;;;;;;ACNA;AACA,mBAAO,CAAC,4EAAgB;;AAExB;AACA;AACA;;AAEA;AACA,YAAY,mBAAO,CAAC,kDAAc;AAClC,aAAa,mBAAO,CAAC,gDAAa;AAClC,eAAe,mBAAO,CAAC,kDAAc;AACrC,kBAAkB,mBAAO,CAAC,8CAAY;AACtC,CAAC","sources":["webpack://om-frontend/./node_modules/freeice/index.js","webpack://om-frontend/./node_modules/normalice/index.js","webpack://om-frontend/./node_modules/sdp/sdp.js","webpack://om-frontend/./node_modules/ua-parser-js/src/ua-parser.js","webpack://om-frontend/./node_modules/webrtc-adapter/src/js/adapter_core.js","webpack://om-frontend/./node_modules/webrtc-adapter/src/js/adapter_factory.js","webpack://om-frontend/./node_modules/webrtc-adapter/src/js/chrome/chrome_shim.js","webpack://om-frontend/./node_modules/webrtc-adapter/src/js/chrome/getdisplaymedia.js","webpack://om-frontend/./node_modules/webrtc-adapter/src/js/chrome/getusermedia.js","webpack://om-frontend/./node_modules/webrtc-adapter/src/js/common_shim.js","webpack://om-frontend/./node_modules/webrtc-adapter/src/js/firefox/firefox_shim.js","webpack://om-frontend/./node_modules/webrtc-adapter/src/js/firefox/getdisplaymedia.js","webpack://om-frontend/./node_modules/webrtc-adapter/src/js/firefox/getusermedia.js","webpack://om-frontend/./node_modules/webrtc-adapter/src/js/safari/safari_shim.js","webpack://om-frontend/./node_modules/webrtc-adapter/src/js/utils.js","webpack://om-frontend/./src/settings/WebRtcPeer.js","webpack://om-frontend/./src/settings/mic-level.js","webpack://om-frontend/./src/settings/ring-buffer.js","webpack://om-frontend/./src/settings/settings.js","webpack://om-frontend/./src/settings/video-util.js","webpack://om-frontend/external var \"OmUtil\"","webpack://om-frontend/external var \"Settings\"","webpack://om-frontend/webpack/bootstrap","webpack://om-frontend/webpack/runtime/amd options","webpack://om-frontend/webpack/runtime/compat get default export","webpack://om-frontend/webpack/runtime/define property getters","webpack://om-frontend/webpack/runtime/hasOwnProperty shorthand","webpack://om-frontend/webpack/runtime/make namespace object","webpack://om-frontend/./src/settings/index.js"],"sourcesContent":["/* jshint node: true */\n'use strict';\n\nvar normalice = require('normalice');\n\n/**\n  # freeice\n\n  The `freeice` module is a simple way of getting random STUN or TURN server\n  for your WebRTC application.  The list of servers (just STUN at this stage)\n  were sourced from this [gist](https://gist.github.com/zziuni/3741933).\n\n  ## Example Use\n\n  The following demonstrates how you can use `freeice` with\n  [rtc-quickconnect](https://github.com/rtc-io/rtc-quickconnect):\n\n  <<< examples/quickconnect.js\n\n  As the `freeice` module generates ice servers in a list compliant with the\n  WebRTC spec you will be able to use it with raw `RTCPeerConnection`\n  constructors and other WebRTC libraries.\n\n  ## Hey, don't use my STUN/TURN server!\n\n  If for some reason your free STUN or TURN server ends up in the\n  list of servers ([stun](https://github.com/DamonOehlman/freeice/blob/master/stun.json) or\n  [turn](https://github.com/DamonOehlman/freeice/blob/master/turn.json))\n  that is used in this module, you can feel\n  free to open an issue on this repository and those servers will be removed\n  within 24 hours (or sooner).  This is the quickest and probably the most\n  polite way to have something removed (and provides us some visibility\n  if someone opens a pull request requesting that a server is added).\n\n  ## Please add my server!\n\n  If you have a server that you wish to add to the list, that's awesome! I'm\n  sure I speak on behalf of a whole pile of WebRTC developers who say thanks.\n  To get it into the list, feel free to either open a pull request or if you\n  find that process a bit daunting then just create an issue requesting\n  the addition of the server (make sure you provide all the details, and if\n  you have a Terms of Service then including that in the PR/issue would be\n  awesome).\n\n  ## I know of a free server, can I add it?\n\n  Sure, if you do your homework and make sure it is ok to use (I'm currently\n  in the process of reviewing the terms of those STUN servers included from\n  the original list).  If it's ok to go, then please see the previous entry\n  for how to add it.\n\n  ## Current List of Servers\n\n  * current as at the time of last `README.md` file generation\n\n  ### STUN\n\n  <<< stun.json\n\n  ### TURN\n\n  <<< turn.json\n\n**/\n\nvar freeice = function(opts) {\n  // if a list of servers has been provided, then use it instead of defaults\n  var servers = {\n    stun: (opts || {}).stun || require('./stun.json'),\n    turn: (opts || {}).turn || require('./turn.json')\n  };\n\n  var stunCount = (opts || {}).stunCount || 2;\n  var turnCount = (opts || {}).turnCount || 0;\n  var selected;\n\n  function getServers(type, count) {\n    var out = [];\n    var input = [].concat(servers[type]);\n    var idx;\n\n    while (input.length && out.length < count) {\n      idx = (Math.random() * input.length) | 0;\n      out = out.concat(input.splice(idx, 1));\n    }\n\n    return out.map(function(url) {\n        //If it's a not a string, don't try to \"normalice\" it otherwise using type:url will screw it up\n        if ((typeof url !== 'string') && (! (url instanceof String))) {\n            return url;\n        } else {\n            return normalice(type + ':' + url);\n        }\n    });\n  }\n\n  // add stun servers\n  selected = [].concat(getServers('stun', stunCount));\n\n  if (turnCount) {\n    selected = selected.concat(getServers('turn', turnCount));\n  }\n\n  return selected;\n};\n\nmodule.exports = freeice;","/**\n  # normalice\n\n  Normalize an ice server configuration object (or plain old string) into a format\n  that is usable in all browsers supporting WebRTC.  Primarily this module is designed\n  to help with the transition of the `url` attribute of the configuration object to\n  the `urls` attribute.\n\n  ## Example Usage\n\n  <<< examples/simple.js\n\n**/\n\nvar protocols = [\n  'stun:',\n  'turn:'\n];\n\nmodule.exports = function(input) {\n  var url = (input || {}).url || input;\n  var protocol;\n  var parts;\n  var output = {};\n\n  // if we don't have a string url, then allow the input to passthrough\n  if (typeof url != 'string' && (! (url instanceof String))) {\n    return input;\n  }\n\n  // trim the url string, and convert to an array\n  url = url.trim();\n\n  // if the protocol is not known, then passthrough\n  protocol = protocols[protocols.indexOf(url.slice(0, 5))];\n  if (! protocol) {\n    return input;\n  }\n\n  // now let's attack the remaining url parts\n  url = url.slice(5);\n  parts = url.split('@');\n\n  output.username = input.username;\n  output.credential = input.credential;\n  // if we have an authentication part, then set the credentials\n  if (parts.length > 1) {\n    url = parts[1];\n    parts = parts[0].split(':');\n\n    // add the output credential and username\n    output.username = parts[0];\n    output.credential = (input || {}).credential || parts[1] || '';\n  }\n\n  output.url = protocol + url;\n  output.urls = [ output.url ];\n\n  return output;\n};\n","/* eslint-env node */\n'use strict';\n\n// SDP helpers.\nconst SDPUtils = {};\n\n// Generate an alphanumeric identifier for cname or mids.\n// TODO: use UUIDs instead? https://gist.github.com/jed/982883\nSDPUtils.generateIdentifier = function() {\n  return Math.random().toString(36).substring(2, 12);\n};\n\n// The RTCP CNAME used by all peerconnections from the same JS.\nSDPUtils.localCName = SDPUtils.generateIdentifier();\n\n// Splits SDP into lines, dealing with both CRLF and LF.\nSDPUtils.splitLines = function(blob) {\n  return blob.trim().split('\\n').map(line => line.trim());\n};\n// Splits SDP into sessionpart and mediasections. Ensures CRLF.\nSDPUtils.splitSections = function(blob) {\n  const parts = blob.split('\\nm=');\n  return parts.map((part, index) => (index > 0 ?\n    'm=' + part : part).trim() + '\\r\\n');\n};\n\n// Returns the session description.\nSDPUtils.getDescription = function(blob) {\n  const sections = SDPUtils.splitSections(blob);\n  return sections && sections[0];\n};\n\n// Returns the individual media sections.\nSDPUtils.getMediaSections = function(blob) {\n  const sections = SDPUtils.splitSections(blob);\n  sections.shift();\n  return sections;\n};\n\n// Returns lines that start with a certain prefix.\nSDPUtils.matchPrefix = function(blob, prefix) {\n  return SDPUtils.splitLines(blob).filter(line => line.indexOf(prefix) === 0);\n};\n\n// Parses an ICE candidate line. Sample input:\n// candidate:702786350 2 udp 41819902 8.8.8.8 60769 typ relay raddr 8.8.8.8\n// rport 55996\"\n// Input can be prefixed with a=.\nSDPUtils.parseCandidate = function(line) {\n  let parts;\n  // Parse both variants.\n  if (line.indexOf('a=candidate:') === 0) {\n    parts = line.substring(12).split(' ');\n  } else {\n    parts = line.substring(10).split(' ');\n  }\n\n  const candidate = {\n    foundation: parts[0],\n    component: {1: 'rtp', 2: 'rtcp'}[parts[1]] || parts[1],\n    protocol: parts[2].toLowerCase(),\n    priority: parseInt(parts[3], 10),\n    ip: parts[4],\n    address: parts[4], // address is an alias for ip.\n    port: parseInt(parts[5], 10),\n    // skip parts[6] == 'typ'\n    type: parts[7],\n  };\n\n  for (let i = 8; i < parts.length; i += 2) {\n    switch (parts[i]) {\n      case 'raddr':\n        candidate.relatedAddress = parts[i + 1];\n        break;\n      case 'rport':\n        candidate.relatedPort = parseInt(parts[i + 1], 10);\n        break;\n      case 'tcptype':\n        candidate.tcpType = parts[i + 1];\n        break;\n      case 'ufrag':\n        candidate.ufrag = parts[i + 1]; // for backward compatibility.\n        candidate.usernameFragment = parts[i + 1];\n        break;\n      default: // extension handling, in particular ufrag. Don't overwrite.\n        if (candidate[parts[i]] === undefined) {\n          candidate[parts[i]] = parts[i + 1];\n        }\n        break;\n    }\n  }\n  return candidate;\n};\n\n// Translates a candidate object into SDP candidate attribute.\n// This does not include the a= prefix!\nSDPUtils.writeCandidate = function(candidate) {\n  const sdp = [];\n  sdp.push(candidate.foundation);\n\n  const component = candidate.component;\n  if (component === 'rtp') {\n    sdp.push(1);\n  } else if (component === 'rtcp') {\n    sdp.push(2);\n  } else {\n    sdp.push(component);\n  }\n  sdp.push(candidate.protocol.toUpperCase());\n  sdp.push(candidate.priority);\n  sdp.push(candidate.address || candidate.ip);\n  sdp.push(candidate.port);\n\n  const type = candidate.type;\n  sdp.push('typ');\n  sdp.push(type);\n  if (type !== 'host' && candidate.relatedAddress &&\n      candidate.relatedPort) {\n    sdp.push('raddr');\n    sdp.push(candidate.relatedAddress);\n    sdp.push('rport');\n    sdp.push(candidate.relatedPort);\n  }\n  if (candidate.tcpType && candidate.protocol.toLowerCase() === 'tcp') {\n    sdp.push('tcptype');\n    sdp.push(candidate.tcpType);\n  }\n  if (candidate.usernameFragment || candidate.ufrag) {\n    sdp.push('ufrag');\n    sdp.push(candidate.usernameFragment || candidate.ufrag);\n  }\n  return 'candidate:' + sdp.join(' ');\n};\n\n// Parses an ice-options line, returns an array of option tags.\n// Sample input:\n// a=ice-options:foo bar\nSDPUtils.parseIceOptions = function(line) {\n  return line.substring(14).split(' ');\n};\n\n// Parses a rtpmap line, returns RTCRtpCoddecParameters. Sample input:\n// a=rtpmap:111 opus/48000/2\nSDPUtils.parseRtpMap = function(line) {\n  let parts = line.substring(9).split(' ');\n  const parsed = {\n    payloadType: parseInt(parts.shift(), 10), // was: id\n  };\n\n  parts = parts[0].split('/');\n\n  parsed.name = parts[0];\n  parsed.clockRate = parseInt(parts[1], 10); // was: clockrate\n  parsed.channels = parts.length === 3 ? parseInt(parts[2], 10) : 1;\n  // legacy alias, got renamed back to channels in ORTC.\n  parsed.numChannels = parsed.channels;\n  return parsed;\n};\n\n// Generates a rtpmap line from RTCRtpCodecCapability or\n// RTCRtpCodecParameters.\nSDPUtils.writeRtpMap = function(codec) {\n  let pt = codec.payloadType;\n  if (codec.preferredPayloadType !== undefined) {\n    pt = codec.preferredPayloadType;\n  }\n  const channels = codec.channels || codec.numChannels || 1;\n  return 'a=rtpmap:' + pt + ' ' + codec.name + '/' + codec.clockRate +\n      (channels !== 1 ? '/' + channels : '') + '\\r\\n';\n};\n\n// Parses a extmap line (headerextension from RFC 5285). Sample input:\n// a=extmap:2 urn:ietf:params:rtp-hdrext:toffset\n// a=extmap:2/sendonly urn:ietf:params:rtp-hdrext:toffset\nSDPUtils.parseExtmap = function(line) {\n  const parts = line.substring(9).split(' ');\n  return {\n    id: parseInt(parts[0], 10),\n    direction: parts[0].indexOf('/') > 0 ? parts[0].split('/')[1] : 'sendrecv',\n    uri: parts[1],\n    attributes: parts.slice(2).join(' '),\n  };\n};\n\n// Generates an extmap line from RTCRtpHeaderExtensionParameters or\n// RTCRtpHeaderExtension.\nSDPUtils.writeExtmap = function(headerExtension) {\n  return 'a=extmap:' + (headerExtension.id || headerExtension.preferredId) +\n      (headerExtension.direction && headerExtension.direction !== 'sendrecv'\n        ? '/' + headerExtension.direction\n        : '') +\n      ' ' + headerExtension.uri +\n      (headerExtension.attributes ? ' ' + headerExtension.attributes : '') +\n      '\\r\\n';\n};\n\n// Parses a fmtp line, returns dictionary. Sample input:\n// a=fmtp:96 vbr=on;cng=on\n// Also deals with vbr=on; cng=on\nSDPUtils.parseFmtp = function(line) {\n  const parsed = {};\n  let kv;\n  const parts = line.substring(line.indexOf(' ') + 1).split(';');\n  for (let j = 0; j < parts.length; j++) {\n    kv = parts[j].trim().split('=');\n    parsed[kv[0].trim()] = kv[1];\n  }\n  return parsed;\n};\n\n// Generates a fmtp line from RTCRtpCodecCapability or RTCRtpCodecParameters.\nSDPUtils.writeFmtp = function(codec) {\n  let line = '';\n  let pt = codec.payloadType;\n  if (codec.preferredPayloadType !== undefined) {\n    pt = codec.preferredPayloadType;\n  }\n  if (codec.parameters && Object.keys(codec.parameters).length) {\n    const params = [];\n    Object.keys(codec.parameters).forEach(param => {\n      if (codec.parameters[param] !== undefined) {\n        params.push(param + '=' + codec.parameters[param]);\n      } else {\n        params.push(param);\n      }\n    });\n    line += 'a=fmtp:' + pt + ' ' + params.join(';') + '\\r\\n';\n  }\n  return line;\n};\n\n// Parses a rtcp-fb line, returns RTCPRtcpFeedback object. Sample input:\n// a=rtcp-fb:98 nack rpsi\nSDPUtils.parseRtcpFb = function(line) {\n  const parts = line.substring(line.indexOf(' ') + 1).split(' ');\n  return {\n    type: parts.shift(),\n    parameter: parts.join(' '),\n  };\n};\n\n// Generate a=rtcp-fb lines from RTCRtpCodecCapability or RTCRtpCodecParameters.\nSDPUtils.writeRtcpFb = function(codec) {\n  let lines = '';\n  let pt = codec.payloadType;\n  if (codec.preferredPayloadType !== undefined) {\n    pt = codec.preferredPayloadType;\n  }\n  if (codec.rtcpFeedback && codec.rtcpFeedback.length) {\n    // FIXME: special handling for trr-int?\n    codec.rtcpFeedback.forEach(fb => {\n      lines += 'a=rtcp-fb:' + pt + ' ' + fb.type +\n      (fb.parameter && fb.parameter.length ? ' ' + fb.parameter : '') +\n          '\\r\\n';\n    });\n  }\n  return lines;\n};\n\n// Parses a RFC 5576 ssrc media attribute. Sample input:\n// a=ssrc:3735928559 cname:something\nSDPUtils.parseSsrcMedia = function(line) {\n  const sp = line.indexOf(' ');\n  const parts = {\n    ssrc: parseInt(line.substring(7, sp), 10),\n  };\n  const colon = line.indexOf(':', sp);\n  if (colon > -1) {\n    parts.attribute = line.substring(sp + 1, colon);\n    parts.value = line.substring(colon + 1);\n  } else {\n    parts.attribute = line.substring(sp + 1);\n  }\n  return parts;\n};\n\n// Parse a ssrc-group line (see RFC 5576). Sample input:\n// a=ssrc-group:semantics 12 34\nSDPUtils.parseSsrcGroup = function(line) {\n  const parts = line.substring(13).split(' ');\n  return {\n    semantics: parts.shift(),\n    ssrcs: parts.map(ssrc => parseInt(ssrc, 10)),\n  };\n};\n\n// Extracts the MID (RFC 5888) from a media section.\n// Returns the MID or undefined if no mid line was found.\nSDPUtils.getMid = function(mediaSection) {\n  const mid = SDPUtils.matchPrefix(mediaSection, 'a=mid:')[0];\n  if (mid) {\n    return mid.substring(6);\n  }\n};\n\n// Parses a fingerprint line for DTLS-SRTP.\nSDPUtils.parseFingerprint = function(line) {\n  const parts = line.substring(14).split(' ');\n  return {\n    algorithm: parts[0].toLowerCase(), // algorithm is case-sensitive in Edge.\n    value: parts[1].toUpperCase(), // the definition is upper-case in RFC 4572.\n  };\n};\n\n// Extracts DTLS parameters from SDP media section or sessionpart.\n// FIXME: for consistency with other functions this should only\n//   get the fingerprint line as input. See also getIceParameters.\nSDPUtils.getDtlsParameters = function(mediaSection, sessionpart) {\n  const lines = SDPUtils.matchPrefix(mediaSection + sessionpart,\n    'a=fingerprint:');\n  // Note: a=setup line is ignored since we use the 'auto' role in Edge.\n  return {\n    role: 'auto',\n    fingerprints: lines.map(SDPUtils.parseFingerprint),\n  };\n};\n\n// Serializes DTLS parameters to SDP.\nSDPUtils.writeDtlsParameters = function(params, setupType) {\n  let sdp = 'a=setup:' + setupType + '\\r\\n';\n  params.fingerprints.forEach(fp => {\n    sdp += 'a=fingerprint:' + fp.algorithm + ' ' + fp.value + '\\r\\n';\n  });\n  return sdp;\n};\n\n// Parses a=crypto lines into\n//   https://rawgit.com/aboba/edgertc/master/msortc-rs4.html#dictionary-rtcsrtpsdesparameters-members\nSDPUtils.parseCryptoLine = function(line) {\n  const parts = line.substring(9).split(' ');\n  return {\n    tag: parseInt(parts[0], 10),\n    cryptoSuite: parts[1],\n    keyParams: parts[2],\n    sessionParams: parts.slice(3),\n  };\n};\n\nSDPUtils.writeCryptoLine = function(parameters) {\n  return 'a=crypto:' + parameters.tag + ' ' +\n    parameters.cryptoSuite + ' ' +\n    (typeof parameters.keyParams === 'object'\n      ? SDPUtils.writeCryptoKeyParams(parameters.keyParams)\n      : parameters.keyParams) +\n    (parameters.sessionParams ? ' ' + parameters.sessionParams.join(' ') : '') +\n    '\\r\\n';\n};\n\n// Parses the crypto key parameters into\n//   https://rawgit.com/aboba/edgertc/master/msortc-rs4.html#rtcsrtpkeyparam*\nSDPUtils.parseCryptoKeyParams = function(keyParams) {\n  if (keyParams.indexOf('inline:') !== 0) {\n    return null;\n  }\n  const parts = keyParams.substring(7).split('|');\n  return {\n    keyMethod: 'inline',\n    keySalt: parts[0],\n    lifeTime: parts[1],\n    mkiValue: parts[2] ? parts[2].split(':')[0] : undefined,\n    mkiLength: parts[2] ? parts[2].split(':')[1] : undefined,\n  };\n};\n\nSDPUtils.writeCryptoKeyParams = function(keyParams) {\n  return keyParams.keyMethod + ':'\n    + keyParams.keySalt +\n    (keyParams.lifeTime ? '|' + keyParams.lifeTime : '') +\n    (keyParams.mkiValue && keyParams.mkiLength\n      ? '|' + keyParams.mkiValue + ':' + keyParams.mkiLength\n      : '');\n};\n\n// Extracts all SDES parameters.\nSDPUtils.getCryptoParameters = function(mediaSection, sessionpart) {\n  const lines = SDPUtils.matchPrefix(mediaSection + sessionpart,\n    'a=crypto:');\n  return lines.map(SDPUtils.parseCryptoLine);\n};\n\n// Parses ICE information from SDP media section or sessionpart.\n// FIXME: for consistency with other functions this should only\n//   get the ice-ufrag and ice-pwd lines as input.\nSDPUtils.getIceParameters = function(mediaSection, sessionpart) {\n  const ufrag = SDPUtils.matchPrefix(mediaSection + sessionpart,\n    'a=ice-ufrag:')[0];\n  const pwd = SDPUtils.matchPrefix(mediaSection + sessionpart,\n    'a=ice-pwd:')[0];\n  if (!(ufrag && pwd)) {\n    return null;\n  }\n  return {\n    usernameFragment: ufrag.substring(12),\n    password: pwd.substring(10),\n  };\n};\n\n// Serializes ICE parameters to SDP.\nSDPUtils.writeIceParameters = function(params) {\n  let sdp = 'a=ice-ufrag:' + params.usernameFragment + '\\r\\n' +\n      'a=ice-pwd:' + params.password + '\\r\\n';\n  if (params.iceLite) {\n    sdp += 'a=ice-lite\\r\\n';\n  }\n  return sdp;\n};\n\n// Parses the SDP media section and returns RTCRtpParameters.\nSDPUtils.parseRtpParameters = function(mediaSection) {\n  const description = {\n    codecs: [],\n    headerExtensions: [],\n    fecMechanisms: [],\n    rtcp: [],\n  };\n  const lines = SDPUtils.splitLines(mediaSection);\n  const mline = lines[0].split(' ');\n  description.profile = mline[2];\n  for (let i = 3; i < mline.length; i++) { // find all codecs from mline[3..]\n    const pt = mline[i];\n    const rtpmapline = SDPUtils.matchPrefix(\n      mediaSection, 'a=rtpmap:' + pt + ' ')[0];\n    if (rtpmapline) {\n      const codec = SDPUtils.parseRtpMap(rtpmapline);\n      const fmtps = SDPUtils.matchPrefix(\n        mediaSection, 'a=fmtp:' + pt + ' ');\n      // Only the first a=fmtp:<pt> is considered.\n      codec.parameters = fmtps.length ? SDPUtils.parseFmtp(fmtps[0]) : {};\n      codec.rtcpFeedback = SDPUtils.matchPrefix(\n        mediaSection, 'a=rtcp-fb:' + pt + ' ')\n        .map(SDPUtils.parseRtcpFb);\n      description.codecs.push(codec);\n      // parse FEC mechanisms from rtpmap lines.\n      switch (codec.name.toUpperCase()) {\n        case 'RED':\n        case 'ULPFEC':\n          description.fecMechanisms.push(codec.name.toUpperCase());\n          break;\n        default: // only RED and ULPFEC are recognized as FEC mechanisms.\n          break;\n      }\n    }\n  }\n  SDPUtils.matchPrefix(mediaSection, 'a=extmap:').forEach(line => {\n    description.headerExtensions.push(SDPUtils.parseExtmap(line));\n  });\n  const wildcardRtcpFb = SDPUtils.matchPrefix(mediaSection, 'a=rtcp-fb:* ')\n    .map(SDPUtils.parseRtcpFb);\n  description.codecs.forEach(codec => {\n    wildcardRtcpFb.forEach(fb=> {\n      const duplicate = codec.rtcpFeedback.find(existingFeedback => {\n        return existingFeedback.type === fb.type &&\n          existingFeedback.parameter === fb.parameter;\n      });\n      if (!duplicate) {\n        codec.rtcpFeedback.push(fb);\n      }\n    });\n  });\n  // FIXME: parse rtcp.\n  return description;\n};\n\n// Generates parts of the SDP media section describing the capabilities /\n// parameters.\nSDPUtils.writeRtpDescription = function(kind, caps) {\n  let sdp = '';\n\n  // Build the mline.\n  sdp += 'm=' + kind + ' ';\n  sdp += caps.codecs.length > 0 ? '9' : '0'; // reject if no codecs.\n  sdp += ' ' + (caps.profile || 'UDP/TLS/RTP/SAVPF') + ' ';\n  sdp += caps.codecs.map(codec => {\n    if (codec.preferredPayloadType !== undefined) {\n      return codec.preferredPayloadType;\n    }\n    return codec.payloadType;\n  }).join(' ') + '\\r\\n';\n\n  sdp += 'c=IN IP4 0.0.0.0\\r\\n';\n  sdp += 'a=rtcp:9 IN IP4 0.0.0.0\\r\\n';\n\n  // Add a=rtpmap lines for each codec. Also fmtp and rtcp-fb.\n  caps.codecs.forEach(codec => {\n    sdp += SDPUtils.writeRtpMap(codec);\n    sdp += SDPUtils.writeFmtp(codec);\n    sdp += SDPUtils.writeRtcpFb(codec);\n  });\n  let maxptime = 0;\n  caps.codecs.forEach(codec => {\n    if (codec.maxptime > maxptime) {\n      maxptime = codec.maxptime;\n    }\n  });\n  if (maxptime > 0) {\n    sdp += 'a=maxptime:' + maxptime + '\\r\\n';\n  }\n\n  if (caps.headerExtensions) {\n    caps.headerExtensions.forEach(extension => {\n      sdp += SDPUtils.writeExtmap(extension);\n    });\n  }\n  // FIXME: write fecMechanisms.\n  return sdp;\n};\n\n// Parses the SDP media section and returns an array of\n// RTCRtpEncodingParameters.\nSDPUtils.parseRtpEncodingParameters = function(mediaSection) {\n  const encodingParameters = [];\n  const description = SDPUtils.parseRtpParameters(mediaSection);\n  const hasRed = description.fecMechanisms.indexOf('RED') !== -1;\n  const hasUlpfec = description.fecMechanisms.indexOf('ULPFEC') !== -1;\n\n  // filter a=ssrc:... cname:, ignore PlanB-msid\n  const ssrcs = SDPUtils.matchPrefix(mediaSection, 'a=ssrc:')\n    .map(line => SDPUtils.parseSsrcMedia(line))\n    .filter(parts => parts.attribute === 'cname');\n  const primarySsrc = ssrcs.length > 0 && ssrcs[0].ssrc;\n  let secondarySsrc;\n\n  const flows = SDPUtils.matchPrefix(mediaSection, 'a=ssrc-group:FID')\n    .map(line => {\n      const parts = line.substring(17).split(' ');\n      return parts.map(part => parseInt(part, 10));\n    });\n  if (flows.length > 0 && flows[0].length > 1 && flows[0][0] === primarySsrc) {\n    secondarySsrc = flows[0][1];\n  }\n\n  description.codecs.forEach(codec => {\n    if (codec.name.toUpperCase() === 'RTX' && codec.parameters.apt) {\n      let encParam = {\n        ssrc: primarySsrc,\n        codecPayloadType: parseInt(codec.parameters.apt, 10),\n      };\n      if (primarySsrc && secondarySsrc) {\n        encParam.rtx = {ssrc: secondarySsrc};\n      }\n      encodingParameters.push(encParam);\n      if (hasRed) {\n        encParam = JSON.parse(JSON.stringify(encParam));\n        encParam.fec = {\n          ssrc: primarySsrc,\n          mechanism: hasUlpfec ? 'red+ulpfec' : 'red',\n        };\n        encodingParameters.push(encParam);\n      }\n    }\n  });\n  if (encodingParameters.length === 0 && primarySsrc) {\n    encodingParameters.push({\n      ssrc: primarySsrc,\n    });\n  }\n\n  // we support both b=AS and b=TIAS but interpret AS as TIAS.\n  let bandwidth = SDPUtils.matchPrefix(mediaSection, 'b=');\n  if (bandwidth.length) {\n    if (bandwidth[0].indexOf('b=TIAS:') === 0) {\n      bandwidth = parseInt(bandwidth[0].substring(7), 10);\n    } else if (bandwidth[0].indexOf('b=AS:') === 0) {\n      // use formula from JSEP to convert b=AS to TIAS value.\n      bandwidth = parseInt(bandwidth[0].substring(5), 10) * 1000 * 0.95\n          - (50 * 40 * 8);\n    } else {\n      bandwidth = undefined;\n    }\n    encodingParameters.forEach(params => {\n      params.maxBitrate = bandwidth;\n    });\n  }\n  return encodingParameters;\n};\n\n// parses http://draft.ortc.org/#rtcrtcpparameters*\nSDPUtils.parseRtcpParameters = function(mediaSection) {\n  const rtcpParameters = {};\n\n  // Gets the first SSRC. Note that with RTX there might be multiple\n  // SSRCs.\n  const remoteSsrc = SDPUtils.matchPrefix(mediaSection, 'a=ssrc:')\n    .map(line => SDPUtils.parseSsrcMedia(line))\n    .filter(obj => obj.attribute === 'cname')[0];\n  if (remoteSsrc) {\n    rtcpParameters.cname = remoteSsrc.value;\n    rtcpParameters.ssrc = remoteSsrc.ssrc;\n  }\n\n  // Edge uses the compound attribute instead of reducedSize\n  // compound is !reducedSize\n  const rsize = SDPUtils.matchPrefix(mediaSection, 'a=rtcp-rsize');\n  rtcpParameters.reducedSize = rsize.length > 0;\n  rtcpParameters.compound = rsize.length === 0;\n\n  // parses the rtcp-mux attrіbute.\n  // Note that Edge does not support unmuxed RTCP.\n  const mux = SDPUtils.matchPrefix(mediaSection, 'a=rtcp-mux');\n  rtcpParameters.mux = mux.length > 0;\n\n  return rtcpParameters;\n};\n\nSDPUtils.writeRtcpParameters = function(rtcpParameters) {\n  let sdp = '';\n  if (rtcpParameters.reducedSize) {\n    sdp += 'a=rtcp-rsize\\r\\n';\n  }\n  if (rtcpParameters.mux) {\n    sdp += 'a=rtcp-mux\\r\\n';\n  }\n  if (rtcpParameters.ssrc !== undefined && rtcpParameters.cname) {\n    sdp += 'a=ssrc:' + rtcpParameters.ssrc +\n      ' cname:' + rtcpParameters.cname + '\\r\\n';\n  }\n  return sdp;\n};\n\n\n// parses either a=msid: or a=ssrc:... msid lines and returns\n// the id of the MediaStream and MediaStreamTrack.\nSDPUtils.parseMsid = function(mediaSection) {\n  let parts;\n  const spec = SDPUtils.matchPrefix(mediaSection, 'a=msid:');\n  if (spec.length === 1) {\n    parts = spec[0].substring(7).split(' ');\n    return {stream: parts[0], track: parts[1]};\n  }\n  const planB = SDPUtils.matchPrefix(mediaSection, 'a=ssrc:')\n    .map(line => SDPUtils.parseSsrcMedia(line))\n    .filter(msidParts => msidParts.attribute === 'msid');\n  if (planB.length > 0) {\n    parts = planB[0].value.split(' ');\n    return {stream: parts[0], track: parts[1]};\n  }\n};\n\n// SCTP\n// parses draft-ietf-mmusic-sctp-sdp-26 first and falls back\n// to draft-ietf-mmusic-sctp-sdp-05\nSDPUtils.parseSctpDescription = function(mediaSection) {\n  const mline = SDPUtils.parseMLine(mediaSection);\n  const maxSizeLine = SDPUtils.matchPrefix(mediaSection, 'a=max-message-size:');\n  let maxMessageSize;\n  if (maxSizeLine.length > 0) {\n    maxMessageSize = parseInt(maxSizeLine[0].substring(19), 10);\n  }\n  if (isNaN(maxMessageSize)) {\n    maxMessageSize = 65536;\n  }\n  const sctpPort = SDPUtils.matchPrefix(mediaSection, 'a=sctp-port:');\n  if (sctpPort.length > 0) {\n    return {\n      port: parseInt(sctpPort[0].substring(12), 10),\n      protocol: mline.fmt,\n      maxMessageSize,\n    };\n  }\n  const sctpMapLines = SDPUtils.matchPrefix(mediaSection, 'a=sctpmap:');\n  if (sctpMapLines.length > 0) {\n    const parts = sctpMapLines[0]\n      .substring(10)\n      .split(' ');\n    return {\n      port: parseInt(parts[0], 10),\n      protocol: parts[1],\n      maxMessageSize,\n    };\n  }\n};\n\n// SCTP\n// outputs the draft-ietf-mmusic-sctp-sdp-26 version that all browsers\n// support by now receiving in this format, unless we originally parsed\n// as the draft-ietf-mmusic-sctp-sdp-05 format (indicated by the m-line\n// protocol of DTLS/SCTP -- without UDP/ or TCP/)\nSDPUtils.writeSctpDescription = function(media, sctp) {\n  let output = [];\n  if (media.protocol !== 'DTLS/SCTP') {\n    output = [\n      'm=' + media.kind + ' 9 ' + media.protocol + ' ' + sctp.protocol + '\\r\\n',\n      'c=IN IP4 0.0.0.0\\r\\n',\n      'a=sctp-port:' + sctp.port + '\\r\\n',\n    ];\n  } else {\n    output = [\n      'm=' + media.kind + ' 9 ' + media.protocol + ' ' + sctp.port + '\\r\\n',\n      'c=IN IP4 0.0.0.0\\r\\n',\n      'a=sctpmap:' + sctp.port + ' ' + sctp.protocol + ' 65535\\r\\n',\n    ];\n  }\n  if (sctp.maxMessageSize !== undefined) {\n    output.push('a=max-message-size:' + sctp.maxMessageSize + '\\r\\n');\n  }\n  return output.join('');\n};\n\n// Generate a session ID for SDP.\n// https://tools.ietf.org/html/draft-ietf-rtcweb-jsep-20#section-5.2.1\n// recommends using a cryptographically random +ve 64-bit value\n// but right now this should be acceptable and within the right range\nSDPUtils.generateSessionId = function() {\n  return Math.random().toString().substr(2, 22);\n};\n\n// Write boiler plate for start of SDP\n// sessId argument is optional - if not supplied it will\n// be generated randomly\n// sessVersion is optional and defaults to 2\n// sessUser is optional and defaults to 'thisisadapterortc'\nSDPUtils.writeSessionBoilerplate = function(sessId, sessVer, sessUser) {\n  let sessionId;\n  const version = sessVer !== undefined ? sessVer : 2;\n  if (sessId) {\n    sessionId = sessId;\n  } else {\n    sessionId = SDPUtils.generateSessionId();\n  }\n  const user = sessUser || 'thisisadapterortc';\n  // FIXME: sess-id should be an NTP timestamp.\n  return 'v=0\\r\\n' +\n      'o=' + user + ' ' + sessionId + ' ' + version +\n        ' IN IP4 127.0.0.1\\r\\n' +\n      's=-\\r\\n' +\n      't=0 0\\r\\n';\n};\n\n// Gets the direction from the mediaSection or the sessionpart.\nSDPUtils.getDirection = function(mediaSection, sessionpart) {\n  // Look for sendrecv, sendonly, recvonly, inactive, default to sendrecv.\n  const lines = SDPUtils.splitLines(mediaSection);\n  for (let i = 0; i < lines.length; i++) {\n    switch (lines[i]) {\n      case 'a=sendrecv':\n      case 'a=sendonly':\n      case 'a=recvonly':\n      case 'a=inactive':\n        return lines[i].substring(2);\n      default:\n        // FIXME: What should happen here?\n    }\n  }\n  if (sessionpart) {\n    return SDPUtils.getDirection(sessionpart);\n  }\n  return 'sendrecv';\n};\n\nSDPUtils.getKind = function(mediaSection) {\n  const lines = SDPUtils.splitLines(mediaSection);\n  const mline = lines[0].split(' ');\n  return mline[0].substring(2);\n};\n\nSDPUtils.isRejected = function(mediaSection) {\n  return mediaSection.split(' ', 2)[1] === '0';\n};\n\nSDPUtils.parseMLine = function(mediaSection) {\n  const lines = SDPUtils.splitLines(mediaSection);\n  const parts = lines[0].substring(2).split(' ');\n  return {\n    kind: parts[0],\n    port: parseInt(parts[1], 10),\n    protocol: parts[2],\n    fmt: parts.slice(3).join(' '),\n  };\n};\n\nSDPUtils.parseOLine = function(mediaSection) {\n  const line = SDPUtils.matchPrefix(mediaSection, 'o=')[0];\n  const parts = line.substring(2).split(' ');\n  return {\n    username: parts[0],\n    sessionId: parts[1],\n    sessionVersion: parseInt(parts[2], 10),\n    netType: parts[3],\n    addressType: parts[4],\n    address: parts[5],\n  };\n};\n\n// a very naive interpretation of a valid SDP.\nSDPUtils.isValidSDP = function(blob) {\n  if (typeof blob !== 'string' || blob.length === 0) {\n    return false;\n  }\n  const lines = SDPUtils.splitLines(blob);\n  for (let i = 0; i < lines.length; i++) {\n    if (lines[i].length < 2 || lines[i].charAt(1) !== '=') {\n      return false;\n    }\n    // TODO: check the modifier a bit more.\n  }\n  return true;\n};\n\n// Expose public methods.\nif (typeof module === 'object') {\n  module.exports = SDPUtils;\n}\n","/////////////////////////////////////////////////////////////////////////////////\n/* UAParser.js v1.0.37\n   Copyright © 2012-2021 Faisal Salman <f@faisalman.com>\n   MIT License *//*\n   Detect Browser, Engine, OS, CPU, and Device type/model from User-Agent data.\n   Supports browser & node.js environment. \n   Demo   : https://faisalman.github.io/ua-parser-js\n   Source : https://github.com/faisalman/ua-parser-js */\n/////////////////////////////////////////////////////////////////////////////////\n\n(function (window, undefined) {\n\n    'use strict';\n\n    //////////////\n    // Constants\n    /////////////\n\n\n    var LIBVERSION  = '1.0.37',\n        EMPTY       = '',\n        UNKNOWN     = '?',\n        FUNC_TYPE   = 'function',\n        UNDEF_TYPE  = 'undefined',\n        OBJ_TYPE    = 'object',\n        STR_TYPE    = 'string',\n        MAJOR       = 'major',\n        MODEL       = 'model',\n        NAME        = 'name',\n        TYPE        = 'type',\n        VENDOR      = 'vendor',\n        VERSION     = 'version',\n        ARCHITECTURE= 'architecture',\n        CONSOLE     = 'console',\n        MOBILE      = 'mobile',\n        TABLET      = 'tablet',\n        SMARTTV     = 'smarttv',\n        WEARABLE    = 'wearable',\n        EMBEDDED    = 'embedded',\n        UA_MAX_LENGTH = 500;\n\n    var AMAZON  = 'Amazon',\n        APPLE   = 'Apple',\n        ASUS    = 'ASUS',\n        BLACKBERRY = 'BlackBerry',\n        BROWSER = 'Browser',\n        CHROME  = 'Chrome',\n        EDGE    = 'Edge',\n        FIREFOX = 'Firefox',\n        GOOGLE  = 'Google',\n        HUAWEI  = 'Huawei',\n        LG      = 'LG',\n        MICROSOFT = 'Microsoft',\n        MOTOROLA  = 'Motorola',\n        OPERA   = 'Opera',\n        SAMSUNG = 'Samsung',\n        SHARP   = 'Sharp',\n        SONY    = 'Sony',\n        XIAOMI  = 'Xiaomi',\n        ZEBRA   = 'Zebra',\n        FACEBOOK    = 'Facebook',\n        CHROMIUM_OS = 'Chromium OS',\n        MAC_OS  = 'Mac OS';\n\n    ///////////\n    // Helper\n    //////////\n\n    var extend = function (regexes, extensions) {\n            var mergedRegexes = {};\n            for (var i in regexes) {\n                if (extensions[i] && extensions[i].length % 2 === 0) {\n                    mergedRegexes[i] = extensions[i].concat(regexes[i]);\n                } else {\n                    mergedRegexes[i] = regexes[i];\n                }\n            }\n            return mergedRegexes;\n        },\n        enumerize = function (arr) {\n            var enums = {};\n            for (var i=0; i<arr.length; i++) {\n                enums[arr[i].toUpperCase()] = arr[i];\n            }\n            return enums;\n        },\n        has = function (str1, str2) {\n            return typeof str1 === STR_TYPE ? lowerize(str2).indexOf(lowerize(str1)) !== -1 : false;\n        },\n        lowerize = function (str) {\n            return str.toLowerCase();\n        },\n        majorize = function (version) {\n            return typeof(version) === STR_TYPE ? version.replace(/[^\\d\\.]/g, EMPTY).split('.')[0] : undefined;\n        },\n        trim = function (str, len) {\n            if (typeof(str) === STR_TYPE) {\n                str = str.replace(/^\\s\\s*/, EMPTY);\n                return typeof(len) === UNDEF_TYPE ? str : str.substring(0, UA_MAX_LENGTH);\n            }\n    };\n\n    ///////////////\n    // Map helper\n    //////////////\n\n    var rgxMapper = function (ua, arrays) {\n\n            var i = 0, j, k, p, q, matches, match;\n\n            // loop through all regexes maps\n            while (i < arrays.length && !matches) {\n\n                var regex = arrays[i],       // even sequence (0,2,4,..)\n                    props = arrays[i + 1];   // odd sequence (1,3,5,..)\n                j = k = 0;\n\n                // try matching uastring with regexes\n                while (j < regex.length && !matches) {\n\n                    if (!regex[j]) { break; }\n                    matches = regex[j++].exec(ua);\n\n                    if (!!matches) {\n                        for (p = 0; p < props.length; p++) {\n                            match = matches[++k];\n                            q = props[p];\n                            // check if given property is actually array\n                            if (typeof q === OBJ_TYPE && q.length > 0) {\n                                if (q.length === 2) {\n                                    if (typeof q[1] == FUNC_TYPE) {\n                                        // assign modified match\n                                        this[q[0]] = q[1].call(this, match);\n                                    } else {\n                                        // assign given value, ignore regex match\n                                        this[q[0]] = q[1];\n                                    }\n                                } else if (q.length === 3) {\n                                    // check whether function or regex\n                                    if (typeof q[1] === FUNC_TYPE && !(q[1].exec && q[1].test)) {\n                                        // call function (usually string mapper)\n                                        this[q[0]] = match ? q[1].call(this, match, q[2]) : undefined;\n                                    } else {\n                                        // sanitize match using given regex\n                                        this[q[0]] = match ? match.replace(q[1], q[2]) : undefined;\n                                    }\n                                } else if (q.length === 4) {\n                                        this[q[0]] = match ? q[3].call(this, match.replace(q[1], q[2])) : undefined;\n                                }\n                            } else {\n                                this[q] = match ? match : undefined;\n                            }\n                        }\n                    }\n                }\n                i += 2;\n            }\n        },\n\n        strMapper = function (str, map) {\n\n            for (var i in map) {\n                // check if current value is array\n                if (typeof map[i] === OBJ_TYPE && map[i].length > 0) {\n                    for (var j = 0; j < map[i].length; j++) {\n                        if (has(map[i][j], str)) {\n                            return (i === UNKNOWN) ? undefined : i;\n                        }\n                    }\n                } else if (has(map[i], str)) {\n                    return (i === UNKNOWN) ? undefined : i;\n                }\n            }\n            return str;\n    };\n\n    ///////////////\n    // String map\n    //////////////\n\n    // Safari < 3.0\n    var oldSafariMap = {\n            '1.0'   : '/8',\n            '1.2'   : '/1',\n            '1.3'   : '/3',\n            '2.0'   : '/412',\n            '2.0.2' : '/416',\n            '2.0.3' : '/417',\n            '2.0.4' : '/419',\n            '?'     : '/'\n        },\n        windowsVersionMap = {\n            'ME'        : '4.90',\n            'NT 3.11'   : 'NT3.51',\n            'NT 4.0'    : 'NT4.0',\n            '2000'      : 'NT 5.0',\n            'XP'        : ['NT 5.1', 'NT 5.2'],\n            'Vista'     : 'NT 6.0',\n            '7'         : 'NT 6.1',\n            '8'         : 'NT 6.2',\n            '8.1'       : 'NT 6.3',\n            '10'        : ['NT 6.4', 'NT 10.0'],\n            'RT'        : 'ARM'\n    };\n\n    //////////////\n    // Regex map\n    /////////////\n\n    var regexes = {\n\n        browser : [[\n\n            /\\b(?:crmo|crios)\\/([\\w\\.]+)/i                                      // Chrome for Android/iOS\n            ], [VERSION, [NAME, 'Chrome']], [\n            /edg(?:e|ios|a)?\\/([\\w\\.]+)/i                                       // Microsoft Edge\n            ], [VERSION, [NAME, 'Edge']], [\n\n            // Presto based\n            /(opera mini)\\/([-\\w\\.]+)/i,                                        // Opera Mini\n            /(opera [mobiletab]{3,6})\\b.+version\\/([-\\w\\.]+)/i,                 // Opera Mobi/Tablet\n            /(opera)(?:.+version\\/|[\\/ ]+)([\\w\\.]+)/i                           // Opera\n            ], [NAME, VERSION], [\n            /opios[\\/ ]+([\\w\\.]+)/i                                             // Opera mini on iphone >= 8.0\n            ], [VERSION, [NAME, OPERA+' Mini']], [\n            /\\bopr\\/([\\w\\.]+)/i                                                 // Opera Webkit\n            ], [VERSION, [NAME, OPERA]], [\n\n            // Mixed\n            /\\bb[ai]*d(?:uhd|[ub]*[aekoprswx]{5,6})[\\/ ]?([\\w\\.]+)/i            // Baidu\n            ], [VERSION, [NAME, 'Baidu']], [\n            /(kindle)\\/([\\w\\.]+)/i,                                             // Kindle\n            /(lunascape|maxthon|netfront|jasmine|blazer)[\\/ ]?([\\w\\.]*)/i,      // Lunascape/Maxthon/Netfront/Jasmine/Blazer\n            // Trident based\n            /(avant|iemobile|slim)\\s?(?:browser)?[\\/ ]?([\\w\\.]*)/i,             // Avant/IEMobile/SlimBrowser\n            /(?:ms|\\()(ie) ([\\w\\.]+)/i,                                         // Internet Explorer\n\n            // Webkit/KHTML based                                               // Flock/RockMelt/Midori/Epiphany/Silk/Skyfire/Bolt/Iron/Iridium/PhantomJS/Bowser/QupZilla/Falkon\n            /(flock|rockmelt|midori|epiphany|silk|skyfire|bolt|iron|vivaldi|iridium|phantomjs|bowser|quark|qupzilla|falkon|rekonq|puffin|brave|whale(?!.+naver)|qqbrowserlite|qq|duckduckgo)\\/([-\\w\\.]+)/i,\n                                                                                // Rekonq/Puffin/Brave/Whale/QQBrowserLite/QQ, aka ShouQ\n            /(heytap|ovi)browser\\/([\\d\\.]+)/i,                                  // Heytap/Ovi\n            /(weibo)__([\\d\\.]+)/i                                               // Weibo\n            ], [NAME, VERSION], [\n            /(?:\\buc? ?browser|(?:juc.+)ucweb)[\\/ ]?([\\w\\.]+)/i                 // UCBrowser\n            ], [VERSION, [NAME, 'UC'+BROWSER]], [\n            /microm.+\\bqbcore\\/([\\w\\.]+)/i,                                     // WeChat Desktop for Windows Built-in Browser\n            /\\bqbcore\\/([\\w\\.]+).+microm/i,\n            /micromessenger\\/([\\w\\.]+)/i                                        // WeChat\n            ], [VERSION, [NAME, 'WeChat']], [\n            /konqueror\\/([\\w\\.]+)/i                                             // Konqueror\n            ], [VERSION, [NAME, 'Konqueror']], [\n            /trident.+rv[: ]([\\w\\.]{1,9})\\b.+like gecko/i                       // IE11\n            ], [VERSION, [NAME, 'IE']], [\n            /ya(?:search)?browser\\/([\\w\\.]+)/i                                  // Yandex\n            ], [VERSION, [NAME, 'Yandex']], [\n            /slbrowser\\/([\\w\\.]+)/i                                             // Smart Lenovo Browser\n            ], [VERSION, [NAME, 'Smart Lenovo '+BROWSER]], [\n            /(avast|avg)\\/([\\w\\.]+)/i                                           // Avast/AVG Secure Browser\n            ], [[NAME, /(.+)/, '$1 Secure '+BROWSER], VERSION], [\n            /\\bfocus\\/([\\w\\.]+)/i                                               // Firefox Focus\n            ], [VERSION, [NAME, FIREFOX+' Focus']], [\n            /\\bopt\\/([\\w\\.]+)/i                                                 // Opera Touch\n            ], [VERSION, [NAME, OPERA+' Touch']], [\n            /coc_coc\\w+\\/([\\w\\.]+)/i                                            // Coc Coc Browser\n            ], [VERSION, [NAME, 'Coc Coc']], [\n            /dolfin\\/([\\w\\.]+)/i                                                // Dolphin\n            ], [VERSION, [NAME, 'Dolphin']], [\n            /coast\\/([\\w\\.]+)/i                                                 // Opera Coast\n            ], [VERSION, [NAME, OPERA+' Coast']], [\n            /miuibrowser\\/([\\w\\.]+)/i                                           // MIUI Browser\n            ], [VERSION, [NAME, 'MIUI '+BROWSER]], [\n            /fxios\\/([-\\w\\.]+)/i                                                // Firefox for iOS\n            ], [VERSION, [NAME, FIREFOX]], [\n            /\\bqihu|(qi?ho?o?|360)browser/i                                     // 360\n            ], [[NAME, '360 ' + BROWSER]], [\n            /(oculus|sailfish|huawei|vivo)browser\\/([\\w\\.]+)/i\n            ], [[NAME, /(.+)/, '$1 ' + BROWSER], VERSION], [                    // Oculus/Sailfish/HuaweiBrowser/VivoBrowser\n            /samsungbrowser\\/([\\w\\.]+)/i                                        // Samsung Internet\n            ], [VERSION, [NAME, SAMSUNG + ' Internet']], [\n            /(comodo_dragon)\\/([\\w\\.]+)/i                                       // Comodo Dragon\n            ], [[NAME, /_/g, ' '], VERSION], [\n            /metasr[\\/ ]?([\\d\\.]+)/i                                            // Sogou Explorer\n            ], [VERSION, [NAME, 'Sogou Explorer']], [\n            /(sogou)mo\\w+\\/([\\d\\.]+)/i                                          // Sogou Mobile\n            ], [[NAME, 'Sogou Mobile'], VERSION], [\n            /(electron)\\/([\\w\\.]+) safari/i,                                    // Electron-based App\n            /(tesla)(?: qtcarbrowser|\\/(20\\d\\d\\.[-\\w\\.]+))/i,                   // Tesla\n            /m?(qqbrowser|2345Explorer)[\\/ ]?([\\w\\.]+)/i                        // QQBrowser/2345 Browser\n            ], [NAME, VERSION], [\n            /(lbbrowser)/i,                                                     // LieBao Browser\n            /\\[(linkedin)app\\]/i                                                // LinkedIn App for iOS & Android\n            ], [NAME], [\n\n            // WebView\n            /((?:fban\\/fbios|fb_iab\\/fb4a)(?!.+fbav)|;fbav\\/([\\w\\.]+);)/i       // Facebook App for iOS & Android\n            ], [[NAME, FACEBOOK], VERSION], [\n            /(Klarna)\\/([\\w\\.]+)/i,                                             // Klarna Shopping Browser for iOS & Android\n            /(kakao(?:talk|story))[\\/ ]([\\w\\.]+)/i,                             // Kakao App\n            /(naver)\\(.*?(\\d+\\.[\\w\\.]+).*\\)/i,                                  // Naver InApp\n            /safari (line)\\/([\\w\\.]+)/i,                                        // Line App for iOS\n            /\\b(line)\\/([\\w\\.]+)\\/iab/i,                                        // Line App for Android\n            /(alipay)client\\/([\\w\\.]+)/i,                                       // Alipay\n            /(chromium|instagram|snapchat)[\\/ ]([-\\w\\.]+)/i                     // Chromium/Instagram/Snapchat\n            ], [NAME, VERSION], [\n            /\\bgsa\\/([\\w\\.]+) .*safari\\//i                                      // Google Search Appliance on iOS\n            ], [VERSION, [NAME, 'GSA']], [\n            /musical_ly(?:.+app_?version\\/|_)([\\w\\.]+)/i                        // TikTok\n            ], [VERSION, [NAME, 'TikTok']], [\n\n            /headlesschrome(?:\\/([\\w\\.]+)| )/i                                  // Chrome Headless\n            ], [VERSION, [NAME, CHROME+' Headless']], [\n\n            / wv\\).+(chrome)\\/([\\w\\.]+)/i                                       // Chrome WebView\n            ], [[NAME, CHROME+' WebView'], VERSION], [\n\n            /droid.+ version\\/([\\w\\.]+)\\b.+(?:mobile safari|safari)/i           // Android Browser\n            ], [VERSION, [NAME, 'Android '+BROWSER]], [\n\n            /(chrome|omniweb|arora|[tizenoka]{5} ?browser)\\/v?([\\w\\.]+)/i       // Chrome/OmniWeb/Arora/Tizen/Nokia\n            ], [NAME, VERSION], [\n\n            /version\\/([\\w\\.\\,]+) .*mobile\\/\\w+ (safari)/i                      // Mobile Safari\n            ], [VERSION, [NAME, 'Mobile Safari']], [\n            /version\\/([\\w(\\.|\\,)]+) .*(mobile ?safari|safari)/i                // Safari & Safari Mobile\n            ], [VERSION, NAME], [\n            /webkit.+?(mobile ?safari|safari)(\\/[\\w\\.]+)/i                      // Safari < 3.0\n            ], [NAME, [VERSION, strMapper, oldSafariMap]], [\n\n            /(webkit|khtml)\\/([\\w\\.]+)/i\n            ], [NAME, VERSION], [\n\n            // Gecko based\n            /(navigator|netscape\\d?)\\/([-\\w\\.]+)/i                              // Netscape\n            ], [[NAME, 'Netscape'], VERSION], [\n            /mobile vr; rv:([\\w\\.]+)\\).+firefox/i                               // Firefox Reality\n            ], [VERSION, [NAME, FIREFOX+' Reality']], [\n            /ekiohf.+(flow)\\/([\\w\\.]+)/i,                                       // Flow\n            /(swiftfox)/i,                                                      // Swiftfox\n            /(icedragon|iceweasel|camino|chimera|fennec|maemo browser|minimo|conkeror|klar)[\\/ ]?([\\w\\.\\+]+)/i,\n                                                                                // IceDragon/Iceweasel/Camino/Chimera/Fennec/Maemo/Minimo/Conkeror/Klar\n            /(seamonkey|k-meleon|icecat|iceape|firebird|phoenix|palemoon|basilisk|waterfox)\\/([-\\w\\.]+)$/i,\n                                                                                // Firefox/SeaMonkey/K-Meleon/IceCat/IceApe/Firebird/Phoenix\n            /(firefox)\\/([\\w\\.]+)/i,                                            // Other Firefox-based\n            /(mozilla)\\/([\\w\\.]+) .+rv\\:.+gecko\\/\\d+/i,                         // Mozilla\n\n            // Other\n            /(polaris|lynx|dillo|icab|doris|amaya|w3m|netsurf|sleipnir|obigo|mosaic|(?:go|ice|up)[\\. ]?browser)[-\\/ ]?v?([\\w\\.]+)/i,\n                                                                                // Polaris/Lynx/Dillo/iCab/Doris/Amaya/w3m/NetSurf/Sleipnir/Obigo/Mosaic/Go/ICE/UP.Browser\n            /(links) \\(([\\w\\.]+)/i,                                             // Links\n            /panasonic;(viera)/i                                                // Panasonic Viera\n            ], [NAME, VERSION], [\n            \n            /(cobalt)\\/([\\w\\.]+)/i                                              // Cobalt\n            ], [NAME, [VERSION, /master.|lts./, \"\"]]\n        ],\n\n        cpu : [[\n\n            /(?:(amd|x(?:(?:86|64)[-_])?|wow|win)64)[;\\)]/i                     // AMD64 (x64)\n            ], [[ARCHITECTURE, 'amd64']], [\n\n            /(ia32(?=;))/i                                                      // IA32 (quicktime)\n            ], [[ARCHITECTURE, lowerize]], [\n\n            /((?:i[346]|x)86)[;\\)]/i                                            // IA32 (x86)\n            ], [[ARCHITECTURE, 'ia32']], [\n\n            /\\b(aarch64|arm(v?8e?l?|_?64))\\b/i                                 // ARM64\n            ], [[ARCHITECTURE, 'arm64']], [\n\n            /\\b(arm(?:v[67])?ht?n?[fl]p?)\\b/i                                   // ARMHF\n            ], [[ARCHITECTURE, 'armhf']], [\n\n            // PocketPC mistakenly identified as PowerPC\n            /windows (ce|mobile); ppc;/i\n            ], [[ARCHITECTURE, 'arm']], [\n\n            /((?:ppc|powerpc)(?:64)?)(?: mac|;|\\))/i                            // PowerPC\n            ], [[ARCHITECTURE, /ower/, EMPTY, lowerize]], [\n\n            /(sun4\\w)[;\\)]/i                                                    // SPARC\n            ], [[ARCHITECTURE, 'sparc']], [\n\n            /((?:avr32|ia64(?=;))|68k(?=\\))|\\barm(?=v(?:[1-7]|[5-7]1)l?|;|eabi)|(?=atmel )avr|(?:irix|mips|sparc)(?:64)?\\b|pa-risc)/i\n                                                                                // IA64, 68K, ARM/64, AVR/32, IRIX/64, MIPS/64, SPARC/64, PA-RISC\n            ], [[ARCHITECTURE, lowerize]]\n        ],\n\n        device : [[\n\n            //////////////////////////\n            // MOBILES & TABLETS\n            /////////////////////////\n\n            // Samsung\n            /\\b(sch-i[89]0\\d|shw-m380s|sm-[ptx]\\w{2,4}|gt-[pn]\\d{2,4}|sgh-t8[56]9|nexus 10)/i\n            ], [MODEL, [VENDOR, SAMSUNG], [TYPE, TABLET]], [\n            /\\b((?:s[cgp]h|gt|sm)-\\w+|sc[g-]?[\\d]+a?|galaxy nexus)/i,\n            /samsung[- ]([-\\w]+)/i,\n            /sec-(sgh\\w+)/i\n            ], [MODEL, [VENDOR, SAMSUNG], [TYPE, MOBILE]], [\n\n            // Apple\n            /(?:\\/|\\()(ip(?:hone|od)[\\w, ]*)(?:\\/|;)/i                          // iPod/iPhone\n            ], [MODEL, [VENDOR, APPLE], [TYPE, MOBILE]], [\n            /\\((ipad);[-\\w\\),; ]+apple/i,                                       // iPad\n            /applecoremedia\\/[\\w\\.]+ \\((ipad)/i,\n            /\\b(ipad)\\d\\d?,\\d\\d?[;\\]].+ios/i\n            ], [MODEL, [VENDOR, APPLE], [TYPE, TABLET]], [\n            /(macintosh);/i\n            ], [MODEL, [VENDOR, APPLE]], [\n\n            // Sharp\n            /\\b(sh-?[altvz]?\\d\\d[a-ekm]?)/i\n            ], [MODEL, [VENDOR, SHARP], [TYPE, MOBILE]], [\n\n            // Huawei\n            /\\b((?:ag[rs][23]?|bah2?|sht?|btv)-a?[lw]\\d{2})\\b(?!.+d\\/s)/i\n            ], [MODEL, [VENDOR, HUAWEI], [TYPE, TABLET]], [\n            /(?:huawei|honor)([-\\w ]+)[;\\)]/i,\n            /\\b(nexus 6p|\\w{2,4}e?-[atu]?[ln][\\dx][012359c][adn]?)\\b(?!.+d\\/s)/i\n            ], [MODEL, [VENDOR, HUAWEI], [TYPE, MOBILE]], [\n\n            // Xiaomi\n            /\\b(poco[\\w ]+|m2\\d{3}j\\d\\d[a-z]{2})(?: bui|\\))/i,                  // Xiaomi POCO\n            /\\b; (\\w+) build\\/hm\\1/i,                                           // Xiaomi Hongmi 'numeric' models\n            /\\b(hm[-_ ]?note?[_ ]?(?:\\d\\w)?) bui/i,                             // Xiaomi Hongmi\n            /\\b(redmi[\\-_ ]?(?:note|k)?[\\w_ ]+)(?: bui|\\))/i,                   // Xiaomi Redmi\n            /oid[^\\)]+; (m?[12][0-389][01]\\w{3,6}[c-y])( bui|; wv|\\))/i,        // Xiaomi Redmi 'numeric' models\n            /\\b(mi[-_ ]?(?:a\\d|one|one[_ ]plus|note lte|max|cc)?[_ ]?(?:\\d?\\w?)[_ ]?(?:plus|se|lite)?)(?: bui|\\))/i // Xiaomi Mi\n            ], [[MODEL, /_/g, ' '], [VENDOR, XIAOMI], [TYPE, MOBILE]], [\n            /oid[^\\)]+; (2\\d{4}(283|rpbf)[cgl])( bui|\\))/i,                     // Redmi Pad\n            /\\b(mi[-_ ]?(?:pad)(?:[\\w_ ]+))(?: bui|\\))/i                        // Mi Pad tablets\n            ],[[MODEL, /_/g, ' '], [VENDOR, XIAOMI], [TYPE, TABLET]], [\n\n            // OPPO\n            /; (\\w+) bui.+ oppo/i,\n            /\\b(cph[12]\\d{3}|p(?:af|c[al]|d\\w|e[ar])[mt]\\d0|x9007|a101op)\\b/i\n            ], [MODEL, [VENDOR, 'OPPO'], [TYPE, MOBILE]], [\n\n            // Vivo\n            /vivo (\\w+)(?: bui|\\))/i,\n            /\\b(v[12]\\d{3}\\w?[at])(?: bui|;)/i\n            ], [MODEL, [VENDOR, 'Vivo'], [TYPE, MOBILE]], [\n\n            // Realme\n            /\\b(rmx[1-3]\\d{3})(?: bui|;|\\))/i\n            ], [MODEL, [VENDOR, 'Realme'], [TYPE, MOBILE]], [\n\n            // Motorola\n            /\\b(milestone|droid(?:[2-4x]| (?:bionic|x2|pro|razr))?:?( 4g)?)\\b[\\w ]+build\\//i,\n            /\\bmot(?:orola)?[- ](\\w*)/i,\n            /((?:moto[\\w\\(\\) ]+|xt\\d{3,4}|nexus 6)(?= bui|\\)))/i\n            ], [MODEL, [VENDOR, MOTOROLA], [TYPE, MOBILE]], [\n            /\\b(mz60\\d|xoom[2 ]{0,2}) build\\//i\n            ], [MODEL, [VENDOR, MOTOROLA], [TYPE, TABLET]], [\n\n            // LG\n            /((?=lg)?[vl]k\\-?\\d{3}) bui| 3\\.[-\\w; ]{10}lg?-([06cv9]{3,4})/i\n            ], [MODEL, [VENDOR, LG], [TYPE, TABLET]], [\n            /(lm(?:-?f100[nv]?|-[\\w\\.]+)(?= bui|\\))|nexus [45])/i,\n            /\\blg[-e;\\/ ]+((?!browser|netcast|android tv)\\w+)/i,\n            /\\blg-?([\\d\\w]+) bui/i\n            ], [MODEL, [VENDOR, LG], [TYPE, MOBILE]], [\n\n            // Lenovo\n            /(ideatab[-\\w ]+)/i,\n            /lenovo ?(s[56]000[-\\w]+|tab(?:[\\w ]+)|yt[-\\d\\w]{6}|tb[-\\d\\w]{6})/i\n            ], [MODEL, [VENDOR, 'Lenovo'], [TYPE, TABLET]], [\n\n            // Nokia\n            /(?:maemo|nokia).*(n900|lumia \\d+)/i,\n            /nokia[-_ ]?([-\\w\\.]*)/i\n            ], [[MODEL, /_/g, ' '], [VENDOR, 'Nokia'], [TYPE, MOBILE]], [\n\n            // Google\n            /(pixel c)\\b/i                                                      // Google Pixel C\n            ], [MODEL, [VENDOR, GOOGLE], [TYPE, TABLET]], [\n            /droid.+; (pixel[\\daxl ]{0,6})(?: bui|\\))/i                         // Google Pixel\n            ], [MODEL, [VENDOR, GOOGLE], [TYPE, MOBILE]], [\n\n            // Sony\n            /droid.+ (a?\\d[0-2]{2}so|[c-g]\\d{4}|so[-gl]\\w+|xq-a\\w[4-7][12])(?= bui|\\).+chrome\\/(?![1-6]{0,1}\\d\\.))/i\n            ], [MODEL, [VENDOR, SONY], [TYPE, MOBILE]], [\n            /sony tablet [ps]/i,\n            /\\b(?:sony)?sgp\\w+(?: bui|\\))/i\n            ], [[MODEL, 'Xperia Tablet'], [VENDOR, SONY], [TYPE, TABLET]], [\n\n            // OnePlus\n            / (kb2005|in20[12]5|be20[12][59])\\b/i,\n            /(?:one)?(?:plus)? (a\\d0\\d\\d)(?: b|\\))/i\n            ], [MODEL, [VENDOR, 'OnePlus'], [TYPE, MOBILE]], [\n\n            // Amazon\n            /(alexa)webm/i,\n            /(kf[a-z]{2}wi|aeo[c-r]{2})( bui|\\))/i,                             // Kindle Fire without Silk / Echo Show\n            /(kf[a-z]+)( bui|\\)).+silk\\//i                                      // Kindle Fire HD\n            ], [MODEL, [VENDOR, AMAZON], [TYPE, TABLET]], [\n            /((?:sd|kf)[0349hijorstuw]+)( bui|\\)).+silk\\//i                     // Fire Phone\n            ], [[MODEL, /(.+)/g, 'Fire Phone $1'], [VENDOR, AMAZON], [TYPE, MOBILE]], [\n\n            // BlackBerry\n            /(playbook);[-\\w\\),; ]+(rim)/i                                      // BlackBerry PlayBook\n            ], [MODEL, VENDOR, [TYPE, TABLET]], [\n            /\\b((?:bb[a-f]|st[hv])100-\\d)/i,\n            /\\(bb10; (\\w+)/i                                                    // BlackBerry 10\n            ], [MODEL, [VENDOR, BLACKBERRY], [TYPE, MOBILE]], [\n\n            // Asus\n            /(?:\\b|asus_)(transfo[prime ]{4,10} \\w+|eeepc|slider \\w+|nexus 7|padfone|p00[cj])/i\n            ], [MODEL, [VENDOR, ASUS], [TYPE, TABLET]], [\n            / (z[bes]6[027][012][km][ls]|zenfone \\d\\w?)\\b/i\n            ], [MODEL, [VENDOR, ASUS], [TYPE, MOBILE]], [\n\n            // HTC\n            /(nexus 9)/i                                                        // HTC Nexus 9\n            ], [MODEL, [VENDOR, 'HTC'], [TYPE, TABLET]], [\n            /(htc)[-;_ ]{1,2}([\\w ]+(?=\\)| bui)|\\w+)/i,                         // HTC\n\n            // ZTE\n            /(zte)[- ]([\\w ]+?)(?: bui|\\/|\\))/i,\n            /(alcatel|geeksphone|nexian|panasonic(?!(?:;|\\.))|sony(?!-bra))[-_ ]?([-\\w]*)/i         // Alcatel/GeeksPhone/Nexian/Panasonic/Sony\n            ], [VENDOR, [MODEL, /_/g, ' '], [TYPE, MOBILE]], [\n\n            // Acer\n            /droid.+; ([ab][1-7]-?[0178a]\\d\\d?)/i\n            ], [MODEL, [VENDOR, 'Acer'], [TYPE, TABLET]], [\n\n            // Meizu\n            /droid.+; (m[1-5] note) bui/i,\n            /\\bmz-([-\\w]{2,})/i\n            ], [MODEL, [VENDOR, 'Meizu'], [TYPE, MOBILE]], [\n                \n            // Ulefone\n            /; ((?:power )?armor(?:[\\w ]{0,8}))(?: bui|\\))/i\n            ], [MODEL, [VENDOR, 'Ulefone'], [TYPE, MOBILE]], [\n\n            // MIXED\n            /(blackberry|benq|palm(?=\\-)|sonyericsson|acer|asus|dell|meizu|motorola|polytron|infinix|tecno)[-_ ]?([-\\w]*)/i,\n                                                                                // BlackBerry/BenQ/Palm/Sony-Ericsson/Acer/Asus/Dell/Meizu/Motorola/Polytron\n            /(hp) ([\\w ]+\\w)/i,                                                 // HP iPAQ\n            /(asus)-?(\\w+)/i,                                                   // Asus\n            /(microsoft); (lumia[\\w ]+)/i,                                      // Microsoft Lumia\n            /(lenovo)[-_ ]?([-\\w]+)/i,                                          // Lenovo\n            /(jolla)/i,                                                         // Jolla\n            /(oppo) ?([\\w ]+) bui/i                                             // OPPO\n            ], [VENDOR, MODEL, [TYPE, MOBILE]], [\n\n            /(kobo)\\s(ereader|touch)/i,                                         // Kobo\n            /(archos) (gamepad2?)/i,                                            // Archos\n            /(hp).+(touchpad(?!.+tablet)|tablet)/i,                             // HP TouchPad\n            /(kindle)\\/([\\w\\.]+)/i,                                             // Kindle\n            /(nook)[\\w ]+build\\/(\\w+)/i,                                        // Nook\n            /(dell) (strea[kpr\\d ]*[\\dko])/i,                                   // Dell Streak\n            /(le[- ]+pan)[- ]+(\\w{1,9}) bui/i,                                  // Le Pan Tablets\n            /(trinity)[- ]*(t\\d{3}) bui/i,                                      // Trinity Tablets\n            /(gigaset)[- ]+(q\\w{1,9}) bui/i,                                    // Gigaset Tablets\n            /(vodafone) ([\\w ]+)(?:\\)| bui)/i                                   // Vodafone\n            ], [VENDOR, MODEL, [TYPE, TABLET]], [\n\n            /(surface duo)/i                                                    // Surface Duo\n            ], [MODEL, [VENDOR, MICROSOFT], [TYPE, TABLET]], [\n            /droid [\\d\\.]+; (fp\\du?)(?: b|\\))/i                                 // Fairphone\n            ], [MODEL, [VENDOR, 'Fairphone'], [TYPE, MOBILE]], [\n            /(u304aa)/i                                                         // AT&T\n            ], [MODEL, [VENDOR, 'AT&T'], [TYPE, MOBILE]], [\n            /\\bsie-(\\w*)/i                                                      // Siemens\n            ], [MODEL, [VENDOR, 'Siemens'], [TYPE, MOBILE]], [\n            /\\b(rct\\w+) b/i                                                     // RCA Tablets\n            ], [MODEL, [VENDOR, 'RCA'], [TYPE, TABLET]], [\n            /\\b(venue[\\d ]{2,7}) b/i                                            // Dell Venue Tablets\n            ], [MODEL, [VENDOR, 'Dell'], [TYPE, TABLET]], [\n            /\\b(q(?:mv|ta)\\w+) b/i                                              // Verizon Tablet\n            ], [MODEL, [VENDOR, 'Verizon'], [TYPE, TABLET]], [\n            /\\b(?:barnes[& ]+noble |bn[rt])([\\w\\+ ]*) b/i                       // Barnes & Noble Tablet\n            ], [MODEL, [VENDOR, 'Barnes & Noble'], [TYPE, TABLET]], [\n            /\\b(tm\\d{3}\\w+) b/i\n            ], [MODEL, [VENDOR, 'NuVision'], [TYPE, TABLET]], [\n            /\\b(k88) b/i                                                        // ZTE K Series Tablet\n            ], [MODEL, [VENDOR, 'ZTE'], [TYPE, TABLET]], [\n            /\\b(nx\\d{3}j) b/i                                                   // ZTE Nubia\n            ], [MODEL, [VENDOR, 'ZTE'], [TYPE, MOBILE]], [\n            /\\b(gen\\d{3}) b.+49h/i                                              // Swiss GEN Mobile\n            ], [MODEL, [VENDOR, 'Swiss'], [TYPE, MOBILE]], [\n            /\\b(zur\\d{3}) b/i                                                   // Swiss ZUR Tablet\n            ], [MODEL, [VENDOR, 'Swiss'], [TYPE, TABLET]], [\n            /\\b((zeki)?tb.*\\b) b/i                                              // Zeki Tablets\n            ], [MODEL, [VENDOR, 'Zeki'], [TYPE, TABLET]], [\n            /\\b([yr]\\d{2}) b/i,\n            /\\b(dragon[- ]+touch |dt)(\\w{5}) b/i                                // Dragon Touch Tablet\n            ], [[VENDOR, 'Dragon Touch'], MODEL, [TYPE, TABLET]], [\n            /\\b(ns-?\\w{0,9}) b/i                                                // Insignia Tablets\n            ], [MODEL, [VENDOR, 'Insignia'], [TYPE, TABLET]], [\n            /\\b((nxa|next)-?\\w{0,9}) b/i                                        // NextBook Tablets\n            ], [MODEL, [VENDOR, 'NextBook'], [TYPE, TABLET]], [\n            /\\b(xtreme\\_)?(v(1[045]|2[015]|[3469]0|7[05])) b/i                  // Voice Xtreme Phones\n            ], [[VENDOR, 'Voice'], MODEL, [TYPE, MOBILE]], [\n            /\\b(lvtel\\-)?(v1[12]) b/i                                           // LvTel Phones\n            ], [[VENDOR, 'LvTel'], MODEL, [TYPE, MOBILE]], [\n            /\\b(ph-1) /i                                                        // Essential PH-1\n            ], [MODEL, [VENDOR, 'Essential'], [TYPE, MOBILE]], [\n            /\\b(v(100md|700na|7011|917g).*\\b) b/i                               // Envizen Tablets\n            ], [MODEL, [VENDOR, 'Envizen'], [TYPE, TABLET]], [\n            /\\b(trio[-\\w\\. ]+) b/i                                              // MachSpeed Tablets\n            ], [MODEL, [VENDOR, 'MachSpeed'], [TYPE, TABLET]], [\n            /\\btu_(1491) b/i                                                    // Rotor Tablets\n            ], [MODEL, [VENDOR, 'Rotor'], [TYPE, TABLET]], [\n            /(shield[\\w ]+) b/i                                                 // Nvidia Shield Tablets\n            ], [MODEL, [VENDOR, 'Nvidia'], [TYPE, TABLET]], [\n            /(sprint) (\\w+)/i                                                   // Sprint Phones\n            ], [VENDOR, MODEL, [TYPE, MOBILE]], [\n            /(kin\\.[onetw]{3})/i                                                // Microsoft Kin\n            ], [[MODEL, /\\./g, ' '], [VENDOR, MICROSOFT], [TYPE, MOBILE]], [\n            /droid.+; (cc6666?|et5[16]|mc[239][23]x?|vc8[03]x?)\\)/i             // Zebra\n            ], [MODEL, [VENDOR, ZEBRA], [TYPE, TABLET]], [\n            /droid.+; (ec30|ps20|tc[2-8]\\d[kx])\\)/i\n            ], [MODEL, [VENDOR, ZEBRA], [TYPE, MOBILE]], [\n\n            ///////////////////\n            // SMARTTVS\n            ///////////////////\n\n            /smart-tv.+(samsung)/i                                              // Samsung\n            ], [VENDOR, [TYPE, SMARTTV]], [\n            /hbbtv.+maple;(\\d+)/i\n            ], [[MODEL, /^/, 'SmartTV'], [VENDOR, SAMSUNG], [TYPE, SMARTTV]], [\n            /(nux; netcast.+smarttv|lg (netcast\\.tv-201\\d|android tv))/i        // LG SmartTV\n            ], [[VENDOR, LG], [TYPE, SMARTTV]], [\n            /(apple) ?tv/i                                                      // Apple TV\n            ], [VENDOR, [MODEL, APPLE+' TV'], [TYPE, SMARTTV]], [\n            /crkey/i                                                            // Google Chromecast\n            ], [[MODEL, CHROME+'cast'], [VENDOR, GOOGLE], [TYPE, SMARTTV]], [\n            /droid.+aft(\\w+)( bui|\\))/i                                         // Fire TV\n            ], [MODEL, [VENDOR, AMAZON], [TYPE, SMARTTV]], [\n            /\\(dtv[\\);].+(aquos)/i,\n            /(aquos-tv[\\w ]+)\\)/i                                               // Sharp\n            ], [MODEL, [VENDOR, SHARP], [TYPE, SMARTTV]],[\n            /(bravia[\\w ]+)( bui|\\))/i                                              // Sony\n            ], [MODEL, [VENDOR, SONY], [TYPE, SMARTTV]], [\n            /(mitv-\\w{5}) bui/i                                                 // Xiaomi\n            ], [MODEL, [VENDOR, XIAOMI], [TYPE, SMARTTV]], [\n            /Hbbtv.*(technisat) (.*);/i                                         // TechniSAT\n            ], [VENDOR, MODEL, [TYPE, SMARTTV]], [\n            /\\b(roku)[\\dx]*[\\)\\/]((?:dvp-)?[\\d\\.]*)/i,                          // Roku\n            /hbbtv\\/\\d+\\.\\d+\\.\\d+ +\\([\\w\\+ ]*; *([\\w\\d][^;]*);([^;]*)/i         // HbbTV devices\n            ], [[VENDOR, trim], [MODEL, trim], [TYPE, SMARTTV]], [\n            /\\b(android tv|smart[- ]?tv|opera tv|tv; rv:)\\b/i                   // SmartTV from Unidentified Vendors\n            ], [[TYPE, SMARTTV]], [\n\n            ///////////////////\n            // CONSOLES\n            ///////////////////\n\n            /(ouya)/i,                                                          // Ouya\n            /(nintendo) ([wids3utch]+)/i                                        // Nintendo\n            ], [VENDOR, MODEL, [TYPE, CONSOLE]], [\n            /droid.+; (shield) bui/i                                            // Nvidia\n            ], [MODEL, [VENDOR, 'Nvidia'], [TYPE, CONSOLE]], [\n            /(playstation [345portablevi]+)/i                                   // Playstation\n            ], [MODEL, [VENDOR, SONY], [TYPE, CONSOLE]], [\n            /\\b(xbox(?: one)?(?!; xbox))[\\); ]/i                                // Microsoft Xbox\n            ], [MODEL, [VENDOR, MICROSOFT], [TYPE, CONSOLE]], [\n\n            ///////////////////\n            // WEARABLES\n            ///////////////////\n\n            /((pebble))app/i                                                    // Pebble\n            ], [VENDOR, MODEL, [TYPE, WEARABLE]], [\n            /(watch)(?: ?os[,\\/]|\\d,\\d\\/)[\\d\\.]+/i                              // Apple Watch\n            ], [MODEL, [VENDOR, APPLE], [TYPE, WEARABLE]], [\n            /droid.+; (glass) \\d/i                                              // Google Glass\n            ], [MODEL, [VENDOR, GOOGLE], [TYPE, WEARABLE]], [\n            /droid.+; (wt63?0{2,3})\\)/i\n            ], [MODEL, [VENDOR, ZEBRA], [TYPE, WEARABLE]], [\n            /(quest( 2| pro)?)/i                                                // Oculus Quest\n            ], [MODEL, [VENDOR, FACEBOOK], [TYPE, WEARABLE]], [\n\n            ///////////////////\n            // EMBEDDED\n            ///////////////////\n\n            /(tesla)(?: qtcarbrowser|\\/[-\\w\\.]+)/i                              // Tesla\n            ], [VENDOR, [TYPE, EMBEDDED]], [\n            /(aeobc)\\b/i                                                        // Echo Dot\n            ], [MODEL, [VENDOR, AMAZON], [TYPE, EMBEDDED]], [\n\n            ////////////////////\n            // MIXED (GENERIC)\n            ///////////////////\n\n            /droid .+?; ([^;]+?)(?: bui|; wv\\)|\\) applew).+? mobile safari/i    // Android Phones from Unidentified Vendors\n            ], [MODEL, [TYPE, MOBILE]], [\n            /droid .+?; ([^;]+?)(?: bui|\\) applew).+?(?! mobile) safari/i       // Android Tablets from Unidentified Vendors\n            ], [MODEL, [TYPE, TABLET]], [\n            /\\b((tablet|tab)[;\\/]|focus\\/\\d(?!.+mobile))/i                      // Unidentifiable Tablet\n            ], [[TYPE, TABLET]], [\n            /(phone|mobile(?:[;\\/]| [ \\w\\/\\.]*safari)|pda(?=.+windows ce))/i    // Unidentifiable Mobile\n            ], [[TYPE, MOBILE]], [\n            /(android[-\\w\\. ]{0,9});.+buil/i                                    // Generic Android Device\n            ], [MODEL, [VENDOR, 'Generic']]\n        ],\n\n        engine : [[\n\n            /windows.+ edge\\/([\\w\\.]+)/i                                       // EdgeHTML\n            ], [VERSION, [NAME, EDGE+'HTML']], [\n\n            /webkit\\/537\\.36.+chrome\\/(?!27)([\\w\\.]+)/i                         // Blink\n            ], [VERSION, [NAME, 'Blink']], [\n\n            /(presto)\\/([\\w\\.]+)/i,                                             // Presto\n            /(webkit|trident|netfront|netsurf|amaya|lynx|w3m|goanna)\\/([\\w\\.]+)/i, // WebKit/Trident/NetFront/NetSurf/Amaya/Lynx/w3m/Goanna\n            /ekioh(flow)\\/([\\w\\.]+)/i,                                          // Flow\n            /(khtml|tasman|links)[\\/ ]\\(?([\\w\\.]+)/i,                           // KHTML/Tasman/Links\n            /(icab)[\\/ ]([23]\\.[\\d\\.]+)/i,                                      // iCab\n            /\\b(libweb)/i\n            ], [NAME, VERSION], [\n\n            /rv\\:([\\w\\.]{1,9})\\b.+(gecko)/i                                     // Gecko\n            ], [VERSION, NAME]\n        ],\n\n        os : [[\n\n            // Windows\n            /microsoft (windows) (vista|xp)/i                                   // Windows (iTunes)\n            ], [NAME, VERSION], [\n            /(windows (?:phone(?: os)?|mobile))[\\/ ]?([\\d\\.\\w ]*)/i             // Windows Phone\n            ], [NAME, [VERSION, strMapper, windowsVersionMap]], [\n            /windows nt 6\\.2; (arm)/i,                                        // Windows RT\n            /windows[\\/ ]?([ntce\\d\\. ]+\\w)(?!.+xbox)/i,\n            /(?:win(?=3|9|n)|win 9x )([nt\\d\\.]+)/i\n            ], [[VERSION, strMapper, windowsVersionMap], [NAME, 'Windows']], [\n\n            // iOS/macOS\n            /ip[honead]{2,4}\\b(?:.*os ([\\w]+) like mac|; opera)/i,              // iOS\n            /(?:ios;fbsv\\/|iphone.+ios[\\/ ])([\\d\\.]+)/i,\n            /cfnetwork\\/.+darwin/i\n            ], [[VERSION, /_/g, '.'], [NAME, 'iOS']], [\n            /(mac os x) ?([\\w\\. ]*)/i,\n            /(macintosh|mac_powerpc\\b)(?!.+haiku)/i                             // Mac OS\n            ], [[NAME, MAC_OS], [VERSION, /_/g, '.']], [\n\n            // Mobile OSes\n            /droid ([\\w\\.]+)\\b.+(android[- ]x86|harmonyos)/i                    // Android-x86/HarmonyOS\n            ], [VERSION, NAME], [                                               // Android/WebOS/QNX/Bada/RIM/Maemo/MeeGo/Sailfish OS\n            /(android|webos|qnx|bada|rim tablet os|maemo|meego|sailfish)[-\\/ ]?([\\w\\.]*)/i,\n            /(blackberry)\\w*\\/([\\w\\.]*)/i,                                      // Blackberry\n            /(tizen|kaios)[\\/ ]([\\w\\.]+)/i,                                     // Tizen/KaiOS\n            /\\((series40);/i                                                    // Series 40\n            ], [NAME, VERSION], [\n            /\\(bb(10);/i                                                        // BlackBerry 10\n            ], [VERSION, [NAME, BLACKBERRY]], [\n            /(?:symbian ?os|symbos|s60(?=;)|series60)[-\\/ ]?([\\w\\.]*)/i         // Symbian\n            ], [VERSION, [NAME, 'Symbian']], [\n            /mozilla\\/[\\d\\.]+ \\((?:mobile|tablet|tv|mobile; [\\w ]+); rv:.+ gecko\\/([\\w\\.]+)/i // Firefox OS\n            ], [VERSION, [NAME, FIREFOX+' OS']], [\n            /web0s;.+rt(tv)/i,\n            /\\b(?:hp)?wos(?:browser)?\\/([\\w\\.]+)/i                              // WebOS\n            ], [VERSION, [NAME, 'webOS']], [\n            /watch(?: ?os[,\\/]|\\d,\\d\\/)([\\d\\.]+)/i                              // watchOS\n            ], [VERSION, [NAME, 'watchOS']], [\n\n            // Google Chromecast\n            /crkey\\/([\\d\\.]+)/i                                                 // Google Chromecast\n            ], [VERSION, [NAME, CHROME+'cast']], [\n            /(cros) [\\w]+(?:\\)| ([\\w\\.]+)\\b)/i                                  // Chromium OS\n            ], [[NAME, CHROMIUM_OS], VERSION],[\n\n            // Smart TVs\n            /panasonic;(viera)/i,                                               // Panasonic Viera\n            /(netrange)mmh/i,                                                   // Netrange\n            /(nettv)\\/(\\d+\\.[\\w\\.]+)/i,                                         // NetTV\n\n            // Console\n            /(nintendo|playstation) ([wids345portablevuch]+)/i,                 // Nintendo/Playstation\n            /(xbox); +xbox ([^\\);]+)/i,                                         // Microsoft Xbox (360, One, X, S, Series X, Series S)\n\n            // Other\n            /\\b(joli|palm)\\b ?(?:os)?\\/?([\\w\\.]*)/i,                            // Joli/Palm\n            /(mint)[\\/\\(\\) ]?(\\w*)/i,                                           // Mint\n            /(mageia|vectorlinux)[; ]/i,                                        // Mageia/VectorLinux\n            /([kxln]?ubuntu|debian|suse|opensuse|gentoo|arch(?= linux)|slackware|fedora|mandriva|centos|pclinuxos|red ?hat|zenwalk|linpus|raspbian|plan 9|minix|risc os|contiki|deepin|manjaro|elementary os|sabayon|linspire)(?: gnu\\/linux)?(?: enterprise)?(?:[- ]linux)?(?:-gnu)?[-\\/ ]?(?!chrom|package)([-\\w\\.]*)/i,\n                                                                                // Ubuntu/Debian/SUSE/Gentoo/Arch/Slackware/Fedora/Mandriva/CentOS/PCLinuxOS/RedHat/Zenwalk/Linpus/Raspbian/Plan9/Minix/RISCOS/Contiki/Deepin/Manjaro/elementary/Sabayon/Linspire\n            /(hurd|linux) ?([\\w\\.]*)/i,                                         // Hurd/Linux\n            /(gnu) ?([\\w\\.]*)/i,                                                // GNU\n            /\\b([-frentopcghs]{0,5}bsd|dragonfly)[\\/ ]?(?!amd|[ix346]{1,2}86)([\\w\\.]*)/i, // FreeBSD/NetBSD/OpenBSD/PC-BSD/GhostBSD/DragonFly\n            /(haiku) (\\w+)/i                                                    // Haiku\n            ], [NAME, VERSION], [\n            /(sunos) ?([\\w\\.\\d]*)/i                                             // Solaris\n            ], [[NAME, 'Solaris'], VERSION], [\n            /((?:open)?solaris)[-\\/ ]?([\\w\\.]*)/i,                              // Solaris\n            /(aix) ((\\d)(?=\\.|\\)| )[\\w\\.])*/i,                                  // AIX\n            /\\b(beos|os\\/2|amigaos|morphos|openvms|fuchsia|hp-ux|serenityos)/i, // BeOS/OS2/AmigaOS/MorphOS/OpenVMS/Fuchsia/HP-UX/SerenityOS\n            /(unix) ?([\\w\\.]*)/i                                                // UNIX\n            ], [NAME, VERSION]\n        ]\n    };\n\n    /////////////////\n    // Constructor\n    ////////////////\n\n    var UAParser = function (ua, extensions) {\n\n        if (typeof ua === OBJ_TYPE) {\n            extensions = ua;\n            ua = undefined;\n        }\n\n        if (!(this instanceof UAParser)) {\n            return new UAParser(ua, extensions).getResult();\n        }\n\n        var _navigator = (typeof window !== UNDEF_TYPE && window.navigator) ? window.navigator : undefined;\n        var _ua = ua || ((_navigator && _navigator.userAgent) ? _navigator.userAgent : EMPTY);\n        var _uach = (_navigator && _navigator.userAgentData) ? _navigator.userAgentData : undefined;\n        var _rgxmap = extensions ? extend(regexes, extensions) : regexes;\n        var _isSelfNav = _navigator && _navigator.userAgent == _ua;\n\n        this.getBrowser = function () {\n            var _browser = {};\n            _browser[NAME] = undefined;\n            _browser[VERSION] = undefined;\n            rgxMapper.call(_browser, _ua, _rgxmap.browser);\n            _browser[MAJOR] = majorize(_browser[VERSION]);\n            // Brave-specific detection\n            if (_isSelfNav && _navigator && _navigator.brave && typeof _navigator.brave.isBrave == FUNC_TYPE) {\n                _browser[NAME] = 'Brave';\n            }\n            return _browser;\n        };\n        this.getCPU = function () {\n            var _cpu = {};\n            _cpu[ARCHITECTURE] = undefined;\n            rgxMapper.call(_cpu, _ua, _rgxmap.cpu);\n            return _cpu;\n        };\n        this.getDevice = function () {\n            var _device = {};\n            _device[VENDOR] = undefined;\n            _device[MODEL] = undefined;\n            _device[TYPE] = undefined;\n            rgxMapper.call(_device, _ua, _rgxmap.device);\n            if (_isSelfNav && !_device[TYPE] && _uach && _uach.mobile) {\n                _device[TYPE] = MOBILE;\n            }\n            // iPadOS-specific detection: identified as Mac, but has some iOS-only properties\n            if (_isSelfNav && _device[MODEL] == 'Macintosh' && _navigator && typeof _navigator.standalone !== UNDEF_TYPE && _navigator.maxTouchPoints && _navigator.maxTouchPoints > 2) {\n                _device[MODEL] = 'iPad';\n                _device[TYPE] = TABLET;\n            }\n            return _device;\n        };\n        this.getEngine = function () {\n            var _engine = {};\n            _engine[NAME] = undefined;\n            _engine[VERSION] = undefined;\n            rgxMapper.call(_engine, _ua, _rgxmap.engine);\n            return _engine;\n        };\n        this.getOS = function () {\n            var _os = {};\n            _os[NAME] = undefined;\n            _os[VERSION] = undefined;\n            rgxMapper.call(_os, _ua, _rgxmap.os);\n            if (_isSelfNav && !_os[NAME] && _uach && _uach.platform != 'Unknown') {\n                _os[NAME] = _uach.platform  \n                                    .replace(/chrome os/i, CHROMIUM_OS)\n                                    .replace(/macos/i, MAC_OS);           // backward compatibility\n            }\n            return _os;\n        };\n        this.getResult = function () {\n            return {\n                ua      : this.getUA(),\n                browser : this.getBrowser(),\n                engine  : this.getEngine(),\n                os      : this.getOS(),\n                device  : this.getDevice(),\n                cpu     : this.getCPU()\n            };\n        };\n        this.getUA = function () {\n            return _ua;\n        };\n        this.setUA = function (ua) {\n            _ua = (typeof ua === STR_TYPE && ua.length > UA_MAX_LENGTH) ? trim(ua, UA_MAX_LENGTH) : ua;\n            return this;\n        };\n        this.setUA(_ua);\n        return this;\n    };\n\n    UAParser.VERSION = LIBVERSION;\n    UAParser.BROWSER =  enumerize([NAME, VERSION, MAJOR]);\n    UAParser.CPU = enumerize([ARCHITECTURE]);\n    UAParser.DEVICE = enumerize([MODEL, VENDOR, TYPE, CONSOLE, MOBILE, SMARTTV, TABLET, WEARABLE, EMBEDDED]);\n    UAParser.ENGINE = UAParser.OS = enumerize([NAME, VERSION]);\n\n    ///////////\n    // Export\n    //////////\n\n    // check js environment\n    if (typeof(exports) !== UNDEF_TYPE) {\n        // nodejs env\n        if (typeof module !== UNDEF_TYPE && module.exports) {\n            exports = module.exports = UAParser;\n        }\n        exports.UAParser = UAParser;\n    } else {\n        // requirejs env (optional)\n        if (typeof(define) === FUNC_TYPE && define.amd) {\n            define(function () {\n                return UAParser;\n            });\n        } else if (typeof window !== UNDEF_TYPE) {\n            // browser env\n            window.UAParser = UAParser;\n        }\n    }\n\n    // jQuery/Zepto specific (optional)\n    // Note:\n    //   In AMD env the global scope should be kept clean, but jQuery is an exception.\n    //   jQuery always exports to global scope, unless jQuery.noConflict(true) is used,\n    //   and we should catch that.\n    var $ = typeof window !== UNDEF_TYPE && (window.jQuery || window.Zepto);\n    if ($ && !$.ua) {\n        var parser = new UAParser();\n        $.ua = parser.getResult();\n        $.ua.get = function () {\n            return parser.getUA();\n        };\n        $.ua.set = function (ua) {\n            parser.setUA(ua);\n            var result = parser.getResult();\n            for (var prop in result) {\n                $.ua[prop] = result[prop];\n            }\n        };\n    }\n\n})(typeof window === 'object' ? window : this);\n","/*\n *  Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.\n *\n *  Use of this source code is governed by a BSD-style license\n *  that can be found in the LICENSE file in the root of the source\n *  tree.\n */\n/* eslint-env node */\n\n'use strict';\n\nimport {adapterFactory} from './adapter_factory.js';\n\nconst adapter =\n  adapterFactory({window: typeof window === 'undefined' ? undefined : window});\nexport default adapter;\n","/*\n *  Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.\n *\n *  Use of this source code is governed by a BSD-style license\n *  that can be found in the LICENSE file in the root of the source\n *  tree.\n */\nimport * as utils from './utils';\n\n// Browser shims.\nimport * as chromeShim from './chrome/chrome_shim';\nimport * as firefoxShim from './firefox/firefox_shim';\nimport * as safariShim from './safari/safari_shim';\nimport * as commonShim from './common_shim';\nimport * as sdp from 'sdp';\n\n// Shimming starts here.\nexport function adapterFactory({window} = {}, options = {\n  shimChrome: true,\n  shimFirefox: true,\n  shimSafari: true,\n}) {\n  // Utils.\n  const logging = utils.log;\n  const browserDetails = utils.detectBrowser(window);\n\n  const adapter = {\n    browserDetails,\n    commonShim,\n    extractVersion: utils.extractVersion,\n    disableLog: utils.disableLog,\n    disableWarnings: utils.disableWarnings,\n    // Expose sdp as a convenience. For production apps include directly.\n    sdp,\n  };\n\n  // Shim browser if found.\n  switch (browserDetails.browser) {\n    case 'chrome':\n      if (!chromeShim || !chromeShim.shimPeerConnection ||\n          !options.shimChrome) {\n        logging('Chrome shim is not included in this adapter release.');\n        return adapter;\n      }\n      if (browserDetails.version === null) {\n        logging('Chrome shim can not determine version, not shimming.');\n        return adapter;\n      }\n      logging('adapter.js shimming chrome.');\n      // Export to the adapter global object visible in the browser.\n      adapter.browserShim = chromeShim;\n\n      // Must be called before shimPeerConnection.\n      commonShim.shimAddIceCandidateNullOrEmpty(window, browserDetails);\n      commonShim.shimParameterlessSetLocalDescription(window, browserDetails);\n\n      chromeShim.shimGetUserMedia(window, browserDetails);\n      chromeShim.shimMediaStream(window, browserDetails);\n      chromeShim.shimPeerConnection(window, browserDetails);\n      chromeShim.shimOnTrack(window, browserDetails);\n      chromeShim.shimAddTrackRemoveTrack(window, browserDetails);\n      chromeShim.shimGetSendersWithDtmf(window, browserDetails);\n      chromeShim.shimGetStats(window, browserDetails);\n      chromeShim.shimSenderReceiverGetStats(window, browserDetails);\n      chromeShim.fixNegotiationNeeded(window, browserDetails);\n\n      commonShim.shimRTCIceCandidate(window, browserDetails);\n      commonShim.shimRTCIceCandidateRelayProtocol(window, browserDetails);\n      commonShim.shimConnectionState(window, browserDetails);\n      commonShim.shimMaxMessageSize(window, browserDetails);\n      commonShim.shimSendThrowTypeError(window, browserDetails);\n      commonShim.removeExtmapAllowMixed(window, browserDetails);\n      break;\n    case 'firefox':\n      if (!firefoxShim || !firefoxShim.shimPeerConnection ||\n          !options.shimFirefox) {\n        logging('Firefox shim is not included in this adapter release.');\n        return adapter;\n      }\n      logging('adapter.js shimming firefox.');\n      // Export to the adapter global object visible in the browser.\n      adapter.browserShim = firefoxShim;\n\n      // Must be called before shimPeerConnection.\n      commonShim.shimAddIceCandidateNullOrEmpty(window, browserDetails);\n      commonShim.shimParameterlessSetLocalDescription(window, browserDetails);\n\n      firefoxShim.shimGetUserMedia(window, browserDetails);\n      firefoxShim.shimPeerConnection(window, browserDetails);\n      firefoxShim.shimOnTrack(window, browserDetails);\n      firefoxShim.shimRemoveStream(window, browserDetails);\n      firefoxShim.shimSenderGetStats(window, browserDetails);\n      firefoxShim.shimReceiverGetStats(window, browserDetails);\n      firefoxShim.shimRTCDataChannel(window, browserDetails);\n      firefoxShim.shimAddTransceiver(window, browserDetails);\n      firefoxShim.shimGetParameters(window, browserDetails);\n      firefoxShim.shimCreateOffer(window, browserDetails);\n      firefoxShim.shimCreateAnswer(window, browserDetails);\n\n      commonShim.shimRTCIceCandidate(window, browserDetails);\n      commonShim.shimConnectionState(window, browserDetails);\n      commonShim.shimMaxMessageSize(window, browserDetails);\n      commonShim.shimSendThrowTypeError(window, browserDetails);\n      break;\n    case 'safari':\n      if (!safariShim || !options.shimSafari) {\n        logging('Safari shim is not included in this adapter release.');\n        return adapter;\n      }\n      logging('adapter.js shimming safari.');\n      // Export to the adapter global object visible in the browser.\n      adapter.browserShim = safariShim;\n\n      // Must be called before shimCallbackAPI.\n      commonShim.shimAddIceCandidateNullOrEmpty(window, browserDetails);\n      commonShim.shimParameterlessSetLocalDescription(window, browserDetails);\n\n      safariShim.shimRTCIceServerUrls(window, browserDetails);\n      safariShim.shimCreateOfferLegacy(window, browserDetails);\n      safariShim.shimCallbacksAPI(window, browserDetails);\n      safariShim.shimLocalStreamsAPI(window, browserDetails);\n      safariShim.shimRemoteStreamsAPI(window, browserDetails);\n      safariShim.shimTrackEventTransceiver(window, browserDetails);\n      safariShim.shimGetUserMedia(window, browserDetails);\n      safariShim.shimAudioContext(window, browserDetails);\n\n      commonShim.shimRTCIceCandidate(window, browserDetails);\n      commonShim.shimRTCIceCandidateRelayProtocol(window, browserDetails);\n      commonShim.shimMaxMessageSize(window, browserDetails);\n      commonShim.shimSendThrowTypeError(window, browserDetails);\n      commonShim.removeExtmapAllowMixed(window, browserDetails);\n      break;\n    default:\n      logging('Unsupported browser!');\n      break;\n  }\n\n  return adapter;\n}\n","/*\n *  Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.\n *\n *  Use of this source code is governed by a BSD-style license\n *  that can be found in the LICENSE file in the root of the source\n *  tree.\n */\n/* eslint-env node */\n'use strict';\nimport * as utils from '../utils.js';\n\nexport {shimGetUserMedia} from './getusermedia';\nexport {shimGetDisplayMedia} from './getdisplaymedia';\n\nexport function shimMediaStream(window) {\n  window.MediaStream = window.MediaStream || window.webkitMediaStream;\n}\n\nexport function shimOnTrack(window) {\n  if (typeof window === 'object' && window.RTCPeerConnection && !('ontrack' in\n      window.RTCPeerConnection.prototype)) {\n    Object.defineProperty(window.RTCPeerConnection.prototype, 'ontrack', {\n      get() {\n        return this._ontrack;\n      },\n      set(f) {\n        if (this._ontrack) {\n          this.removeEventListener('track', this._ontrack);\n        }\n        this.addEventListener('track', this._ontrack = f);\n      },\n      enumerable: true,\n      configurable: true\n    });\n    const origSetRemoteDescription =\n        window.RTCPeerConnection.prototype.setRemoteDescription;\n    window.RTCPeerConnection.prototype.setRemoteDescription =\n      function setRemoteDescription() {\n        if (!this._ontrackpoly) {\n          this._ontrackpoly = (e) => {\n            // onaddstream does not fire when a track is added to an existing\n            // stream. But stream.onaddtrack is implemented so we use that.\n            e.stream.addEventListener('addtrack', te => {\n              let receiver;\n              if (window.RTCPeerConnection.prototype.getReceivers) {\n                receiver = this.getReceivers()\n                  .find(r => r.track && r.track.id === te.track.id);\n              } else {\n                receiver = {track: te.track};\n              }\n\n              const event = new Event('track');\n              event.track = te.track;\n              event.receiver = receiver;\n              event.transceiver = {receiver};\n              event.streams = [e.stream];\n              this.dispatchEvent(event);\n            });\n            e.stream.getTracks().forEach(track => {\n              let receiver;\n              if (window.RTCPeerConnection.prototype.getReceivers) {\n                receiver = this.getReceivers()\n                  .find(r => r.track && r.track.id === track.id);\n              } else {\n                receiver = {track};\n              }\n              const event = new Event('track');\n              event.track = track;\n              event.receiver = receiver;\n              event.transceiver = {receiver};\n              event.streams = [e.stream];\n              this.dispatchEvent(event);\n            });\n          };\n          this.addEventListener('addstream', this._ontrackpoly);\n        }\n        return origSetRemoteDescription.apply(this, arguments);\n      };\n  } else {\n    // even if RTCRtpTransceiver is in window, it is only used and\n    // emitted in unified-plan. Unfortunately this means we need\n    // to unconditionally wrap the event.\n    utils.wrapPeerConnectionEvent(window, 'track', e => {\n      if (!e.transceiver) {\n        Object.defineProperty(e, 'transceiver',\n          {value: {receiver: e.receiver}});\n      }\n      return e;\n    });\n  }\n}\n\nexport function shimGetSendersWithDtmf(window) {\n  // Overrides addTrack/removeTrack, depends on shimAddTrackRemoveTrack.\n  if (typeof window === 'object' && window.RTCPeerConnection &&\n      !('getSenders' in window.RTCPeerConnection.prototype) &&\n      'createDTMFSender' in window.RTCPeerConnection.prototype) {\n    const shimSenderWithDtmf = function(pc, track) {\n      return {\n        track,\n        get dtmf() {\n          if (this._dtmf === undefined) {\n            if (track.kind === 'audio') {\n              this._dtmf = pc.createDTMFSender(track);\n            } else {\n              this._dtmf = null;\n            }\n          }\n          return this._dtmf;\n        },\n        _pc: pc\n      };\n    };\n\n    // augment addTrack when getSenders is not available.\n    if (!window.RTCPeerConnection.prototype.getSenders) {\n      window.RTCPeerConnection.prototype.getSenders = function getSenders() {\n        this._senders = this._senders || [];\n        return this._senders.slice(); // return a copy of the internal state.\n      };\n      const origAddTrack = window.RTCPeerConnection.prototype.addTrack;\n      window.RTCPeerConnection.prototype.addTrack =\n        function addTrack(track, stream) {\n          let sender = origAddTrack.apply(this, arguments);\n          if (!sender) {\n            sender = shimSenderWithDtmf(this, track);\n            this._senders.push(sender);\n          }\n          return sender;\n        };\n\n      const origRemoveTrack = window.RTCPeerConnection.prototype.removeTrack;\n      window.RTCPeerConnection.prototype.removeTrack =\n        function removeTrack(sender) {\n          origRemoveTrack.apply(this, arguments);\n          const idx = this._senders.indexOf(sender);\n          if (idx !== -1) {\n            this._senders.splice(idx, 1);\n          }\n        };\n    }\n    const origAddStream = window.RTCPeerConnection.prototype.addStream;\n    window.RTCPeerConnection.prototype.addStream = function addStream(stream) {\n      this._senders = this._senders || [];\n      origAddStream.apply(this, [stream]);\n      stream.getTracks().forEach(track => {\n        this._senders.push(shimSenderWithDtmf(this, track));\n      });\n    };\n\n    const origRemoveStream = window.RTCPeerConnection.prototype.removeStream;\n    window.RTCPeerConnection.prototype.removeStream =\n      function removeStream(stream) {\n        this._senders = this._senders || [];\n        origRemoveStream.apply(this, [stream]);\n\n        stream.getTracks().forEach(track => {\n          const sender = this._senders.find(s => s.track === track);\n          if (sender) { // remove sender\n            this._senders.splice(this._senders.indexOf(sender), 1);\n          }\n        });\n      };\n  } else if (typeof window === 'object' && window.RTCPeerConnection &&\n             'getSenders' in window.RTCPeerConnection.prototype &&\n             'createDTMFSender' in window.RTCPeerConnection.prototype &&\n             window.RTCRtpSender &&\n             !('dtmf' in window.RTCRtpSender.prototype)) {\n    const origGetSenders = window.RTCPeerConnection.prototype.getSenders;\n    window.RTCPeerConnection.prototype.getSenders = function getSenders() {\n      const senders = origGetSenders.apply(this, []);\n      senders.forEach(sender => sender._pc = this);\n      return senders;\n    };\n\n    Object.defineProperty(window.RTCRtpSender.prototype, 'dtmf', {\n      get() {\n        if (this._dtmf === undefined) {\n          if (this.track.kind === 'audio') {\n            this._dtmf = this._pc.createDTMFSender(this.track);\n          } else {\n            this._dtmf = null;\n          }\n        }\n        return this._dtmf;\n      }\n    });\n  }\n}\n\nexport function shimGetStats(window) {\n  if (!window.RTCPeerConnection) {\n    return;\n  }\n\n  const origGetStats = window.RTCPeerConnection.prototype.getStats;\n  window.RTCPeerConnection.prototype.getStats = function getStats() {\n    const [selector, onSucc, onErr] = arguments;\n\n    // If selector is a function then we are in the old style stats so just\n    // pass back the original getStats format to avoid breaking old users.\n    if (arguments.length > 0 && typeof selector === 'function') {\n      return origGetStats.apply(this, arguments);\n    }\n\n    // When spec-style getStats is supported, return those when called with\n    // either no arguments or the selector argument is null.\n    if (origGetStats.length === 0 && (arguments.length === 0 ||\n        typeof selector !== 'function')) {\n      return origGetStats.apply(this, []);\n    }\n\n    const fixChromeStats_ = function(response) {\n      const standardReport = {};\n      const reports = response.result();\n      reports.forEach(report => {\n        const standardStats = {\n          id: report.id,\n          timestamp: report.timestamp,\n          type: {\n            localcandidate: 'local-candidate',\n            remotecandidate: 'remote-candidate'\n          }[report.type] || report.type\n        };\n        report.names().forEach(name => {\n          standardStats[name] = report.stat(name);\n        });\n        standardReport[standardStats.id] = standardStats;\n      });\n\n      return standardReport;\n    };\n\n    // shim getStats with maplike support\n    const makeMapStats = function(stats) {\n      return new Map(Object.keys(stats).map(key => [key, stats[key]]));\n    };\n\n    if (arguments.length >= 2) {\n      const successCallbackWrapper_ = function(response) {\n        onSucc(makeMapStats(fixChromeStats_(response)));\n      };\n\n      return origGetStats.apply(this, [successCallbackWrapper_,\n        selector]);\n    }\n\n    // promise-support\n    return new Promise((resolve, reject) => {\n      origGetStats.apply(this, [\n        function(response) {\n          resolve(makeMapStats(fixChromeStats_(response)));\n        }, reject]);\n    }).then(onSucc, onErr);\n  };\n}\n\nexport function shimSenderReceiverGetStats(window) {\n  if (!(typeof window === 'object' && window.RTCPeerConnection &&\n      window.RTCRtpSender && window.RTCRtpReceiver)) {\n    return;\n  }\n\n  // shim sender stats.\n  if (!('getStats' in window.RTCRtpSender.prototype)) {\n    const origGetSenders = window.RTCPeerConnection.prototype.getSenders;\n    if (origGetSenders) {\n      window.RTCPeerConnection.prototype.getSenders = function getSenders() {\n        const senders = origGetSenders.apply(this, []);\n        senders.forEach(sender => sender._pc = this);\n        return senders;\n      };\n    }\n\n    const origAddTrack = window.RTCPeerConnection.prototype.addTrack;\n    if (origAddTrack) {\n      window.RTCPeerConnection.prototype.addTrack = function addTrack() {\n        const sender = origAddTrack.apply(this, arguments);\n        sender._pc = this;\n        return sender;\n      };\n    }\n    window.RTCRtpSender.prototype.getStats = function getStats() {\n      const sender = this;\n      return this._pc.getStats().then(result =>\n        /* Note: this will include stats of all senders that\n         *   send a track with the same id as sender.track as\n         *   it is not possible to identify the RTCRtpSender.\n         */\n        utils.filterStats(result, sender.track, true));\n    };\n  }\n\n  // shim receiver stats.\n  if (!('getStats' in window.RTCRtpReceiver.prototype)) {\n    const origGetReceivers = window.RTCPeerConnection.prototype.getReceivers;\n    if (origGetReceivers) {\n      window.RTCPeerConnection.prototype.getReceivers =\n        function getReceivers() {\n          const receivers = origGetReceivers.apply(this, []);\n          receivers.forEach(receiver => receiver._pc = this);\n          return receivers;\n        };\n    }\n    utils.wrapPeerConnectionEvent(window, 'track', e => {\n      e.receiver._pc = e.srcElement;\n      return e;\n    });\n    window.RTCRtpReceiver.prototype.getStats = function getStats() {\n      const receiver = this;\n      return this._pc.getStats().then(result =>\n        utils.filterStats(result, receiver.track, false));\n    };\n  }\n\n  if (!('getStats' in window.RTCRtpSender.prototype &&\n      'getStats' in window.RTCRtpReceiver.prototype)) {\n    return;\n  }\n\n  // shim RTCPeerConnection.getStats(track).\n  const origGetStats = window.RTCPeerConnection.prototype.getStats;\n  window.RTCPeerConnection.prototype.getStats = function getStats() {\n    if (arguments.length > 0 &&\n        arguments[0] instanceof window.MediaStreamTrack) {\n      const track = arguments[0];\n      let sender;\n      let receiver;\n      let err;\n      this.getSenders().forEach(s => {\n        if (s.track === track) {\n          if (sender) {\n            err = true;\n          } else {\n            sender = s;\n          }\n        }\n      });\n      this.getReceivers().forEach(r => {\n        if (r.track === track) {\n          if (receiver) {\n            err = true;\n          } else {\n            receiver = r;\n          }\n        }\n        return r.track === track;\n      });\n      if (err || (sender && receiver)) {\n        return Promise.reject(new DOMException(\n          'There are more than one sender or receiver for the track.',\n          'InvalidAccessError'));\n      } else if (sender) {\n        return sender.getStats();\n      } else if (receiver) {\n        return receiver.getStats();\n      }\n      return Promise.reject(new DOMException(\n        'There is no sender or receiver for the track.',\n        'InvalidAccessError'));\n    }\n    return origGetStats.apply(this, arguments);\n  };\n}\n\nexport function shimAddTrackRemoveTrackWithNative(window) {\n  // shim addTrack/removeTrack with native variants in order to make\n  // the interactions with legacy getLocalStreams behave as in other browsers.\n  // Keeps a mapping stream.id => [stream, rtpsenders...]\n  window.RTCPeerConnection.prototype.getLocalStreams =\n    function getLocalStreams() {\n      this._shimmedLocalStreams = this._shimmedLocalStreams || {};\n      return Object.keys(this._shimmedLocalStreams)\n        .map(streamId => this._shimmedLocalStreams[streamId][0]);\n    };\n\n  const origAddTrack = window.RTCPeerConnection.prototype.addTrack;\n  window.RTCPeerConnection.prototype.addTrack =\n    function addTrack(track, stream) {\n      if (!stream) {\n        return origAddTrack.apply(this, arguments);\n      }\n      this._shimmedLocalStreams = this._shimmedLocalStreams || {};\n\n      const sender = origAddTrack.apply(this, arguments);\n      if (!this._shimmedLocalStreams[stream.id]) {\n        this._shimmedLocalStreams[stream.id] = [stream, sender];\n      } else if (this._shimmedLocalStreams[stream.id].indexOf(sender) === -1) {\n        this._shimmedLocalStreams[stream.id].push(sender);\n      }\n      return sender;\n    };\n\n  const origAddStream = window.RTCPeerConnection.prototype.addStream;\n  window.RTCPeerConnection.prototype.addStream = function addStream(stream) {\n    this._shimmedLocalStreams = this._shimmedLocalStreams || {};\n\n    stream.getTracks().forEach(track => {\n      const alreadyExists = this.getSenders().find(s => s.track === track);\n      if (alreadyExists) {\n        throw new DOMException('Track already exists.',\n          'InvalidAccessError');\n      }\n    });\n    const existingSenders = this.getSenders();\n    origAddStream.apply(this, arguments);\n    const newSenders = this.getSenders()\n      .filter(newSender => existingSenders.indexOf(newSender) === -1);\n    this._shimmedLocalStreams[stream.id] = [stream].concat(newSenders);\n  };\n\n  const origRemoveStream = window.RTCPeerConnection.prototype.removeStream;\n  window.RTCPeerConnection.prototype.removeStream =\n    function removeStream(stream) {\n      this._shimmedLocalStreams = this._shimmedLocalStreams || {};\n      delete this._shimmedLocalStreams[stream.id];\n      return origRemoveStream.apply(this, arguments);\n    };\n\n  const origRemoveTrack = window.RTCPeerConnection.prototype.removeTrack;\n  window.RTCPeerConnection.prototype.removeTrack =\n    function removeTrack(sender) {\n      this._shimmedLocalStreams = this._shimmedLocalStreams || {};\n      if (sender) {\n        Object.keys(this._shimmedLocalStreams).forEach(streamId => {\n          const idx = this._shimmedLocalStreams[streamId].indexOf(sender);\n          if (idx !== -1) {\n            this._shimmedLocalStreams[streamId].splice(idx, 1);\n          }\n          if (this._shimmedLocalStreams[streamId].length === 1) {\n            delete this._shimmedLocalStreams[streamId];\n          }\n        });\n      }\n      return origRemoveTrack.apply(this, arguments);\n    };\n}\n\nexport function shimAddTrackRemoveTrack(window, browserDetails) {\n  if (!window.RTCPeerConnection) {\n    return;\n  }\n  // shim addTrack and removeTrack.\n  if (window.RTCPeerConnection.prototype.addTrack &&\n      browserDetails.version >= 65) {\n    return shimAddTrackRemoveTrackWithNative(window);\n  }\n\n  // also shim pc.getLocalStreams when addTrack is shimmed\n  // to return the original streams.\n  const origGetLocalStreams = window.RTCPeerConnection.prototype\n    .getLocalStreams;\n  window.RTCPeerConnection.prototype.getLocalStreams =\n    function getLocalStreams() {\n      const nativeStreams = origGetLocalStreams.apply(this);\n      this._reverseStreams = this._reverseStreams || {};\n      return nativeStreams.map(stream => this._reverseStreams[stream.id]);\n    };\n\n  const origAddStream = window.RTCPeerConnection.prototype.addStream;\n  window.RTCPeerConnection.prototype.addStream = function addStream(stream) {\n    this._streams = this._streams || {};\n    this._reverseStreams = this._reverseStreams || {};\n\n    stream.getTracks().forEach(track => {\n      const alreadyExists = this.getSenders().find(s => s.track === track);\n      if (alreadyExists) {\n        throw new DOMException('Track already exists.',\n          'InvalidAccessError');\n      }\n    });\n    // Add identity mapping for consistency with addTrack.\n    // Unless this is being used with a stream from addTrack.\n    if (!this._reverseStreams[stream.id]) {\n      const newStream = new window.MediaStream(stream.getTracks());\n      this._streams[stream.id] = newStream;\n      this._reverseStreams[newStream.id] = stream;\n      stream = newStream;\n    }\n    origAddStream.apply(this, [stream]);\n  };\n\n  const origRemoveStream = window.RTCPeerConnection.prototype.removeStream;\n  window.RTCPeerConnection.prototype.removeStream =\n    function removeStream(stream) {\n      this._streams = this._streams || {};\n      this._reverseStreams = this._reverseStreams || {};\n\n      origRemoveStream.apply(this, [(this._streams[stream.id] || stream)]);\n      delete this._reverseStreams[(this._streams[stream.id] ?\n        this._streams[stream.id].id : stream.id)];\n      delete this._streams[stream.id];\n    };\n\n  window.RTCPeerConnection.prototype.addTrack =\n    function addTrack(track, stream) {\n      if (this.signalingState === 'closed') {\n        throw new DOMException(\n          'The RTCPeerConnection\\'s signalingState is \\'closed\\'.',\n          'InvalidStateError');\n      }\n      const streams = [].slice.call(arguments, 1);\n      if (streams.length !== 1 ||\n          !streams[0].getTracks().find(t => t === track)) {\n        // this is not fully correct but all we can manage without\n        // [[associated MediaStreams]] internal slot.\n        throw new DOMException(\n          'The adapter.js addTrack polyfill only supports a single ' +\n          ' stream which is associated with the specified track.',\n          'NotSupportedError');\n      }\n\n      const alreadyExists = this.getSenders().find(s => s.track === track);\n      if (alreadyExists) {\n        throw new DOMException('Track already exists.',\n          'InvalidAccessError');\n      }\n\n      this._streams = this._streams || {};\n      this._reverseStreams = this._reverseStreams || {};\n      const oldStream = this._streams[stream.id];\n      if (oldStream) {\n        // this is using odd Chrome behaviour, use with caution:\n        // https://bugs.chromium.org/p/webrtc/issues/detail?id=7815\n        // Note: we rely on the high-level addTrack/dtmf shim to\n        // create the sender with a dtmf sender.\n        oldStream.addTrack(track);\n\n        // Trigger ONN async.\n        Promise.resolve().then(() => {\n          this.dispatchEvent(new Event('negotiationneeded'));\n        });\n      } else {\n        const newStream = new window.MediaStream([track]);\n        this._streams[stream.id] = newStream;\n        this._reverseStreams[newStream.id] = stream;\n        this.addStream(newStream);\n      }\n      return this.getSenders().find(s => s.track === track);\n    };\n\n  // replace the internal stream id with the external one and\n  // vice versa.\n  function replaceInternalStreamId(pc, description) {\n    let sdp = description.sdp;\n    Object.keys(pc._reverseStreams || []).forEach(internalId => {\n      const externalStream = pc._reverseStreams[internalId];\n      const internalStream = pc._streams[externalStream.id];\n      sdp = sdp.replace(new RegExp(internalStream.id, 'g'),\n        externalStream.id);\n    });\n    return new RTCSessionDescription({\n      type: description.type,\n      sdp\n    });\n  }\n  function replaceExternalStreamId(pc, description) {\n    let sdp = description.sdp;\n    Object.keys(pc._reverseStreams || []).forEach(internalId => {\n      const externalStream = pc._reverseStreams[internalId];\n      const internalStream = pc._streams[externalStream.id];\n      sdp = sdp.replace(new RegExp(externalStream.id, 'g'),\n        internalStream.id);\n    });\n    return new RTCSessionDescription({\n      type: description.type,\n      sdp\n    });\n  }\n  ['createOffer', 'createAnswer'].forEach(function(method) {\n    const nativeMethod = window.RTCPeerConnection.prototype[method];\n    const methodObj = {[method]() {\n      const args = arguments;\n      const isLegacyCall = arguments.length &&\n          typeof arguments[0] === 'function';\n      if (isLegacyCall) {\n        return nativeMethod.apply(this, [\n          (description) => {\n            const desc = replaceInternalStreamId(this, description);\n            args[0].apply(null, [desc]);\n          },\n          (err) => {\n            if (args[1]) {\n              args[1].apply(null, err);\n            }\n          }, arguments[2]\n        ]);\n      }\n      return nativeMethod.apply(this, arguments)\n        .then(description => replaceInternalStreamId(this, description));\n    }};\n    window.RTCPeerConnection.prototype[method] = methodObj[method];\n  });\n\n  const origSetLocalDescription =\n      window.RTCPeerConnection.prototype.setLocalDescription;\n  window.RTCPeerConnection.prototype.setLocalDescription =\n    function setLocalDescription() {\n      if (!arguments.length || !arguments[0].type) {\n        return origSetLocalDescription.apply(this, arguments);\n      }\n      arguments[0] = replaceExternalStreamId(this, arguments[0]);\n      return origSetLocalDescription.apply(this, arguments);\n    };\n\n  // TODO: mangle getStats: https://w3c.github.io/webrtc-stats/#dom-rtcmediastreamstats-streamidentifier\n\n  const origLocalDescription = Object.getOwnPropertyDescriptor(\n    window.RTCPeerConnection.prototype, 'localDescription');\n  Object.defineProperty(window.RTCPeerConnection.prototype,\n    'localDescription', {\n      get() {\n        const description = origLocalDescription.get.apply(this);\n        if (description.type === '') {\n          return description;\n        }\n        return replaceInternalStreamId(this, description);\n      }\n    });\n\n  window.RTCPeerConnection.prototype.removeTrack =\n    function removeTrack(sender) {\n      if (this.signalingState === 'closed') {\n        throw new DOMException(\n          'The RTCPeerConnection\\'s signalingState is \\'closed\\'.',\n          'InvalidStateError');\n      }\n      // We can not yet check for sender instanceof RTCRtpSender\n      // since we shim RTPSender. So we check if sender._pc is set.\n      if (!sender._pc) {\n        throw new DOMException('Argument 1 of RTCPeerConnection.removeTrack ' +\n            'does not implement interface RTCRtpSender.', 'TypeError');\n      }\n      const isLocal = sender._pc === this;\n      if (!isLocal) {\n        throw new DOMException('Sender was not created by this connection.',\n          'InvalidAccessError');\n      }\n\n      // Search for the native stream the senders track belongs to.\n      this._streams = this._streams || {};\n      let stream;\n      Object.keys(this._streams).forEach(streamid => {\n        const hasTrack = this._streams[streamid].getTracks()\n          .find(track => sender.track === track);\n        if (hasTrack) {\n          stream = this._streams[streamid];\n        }\n      });\n\n      if (stream) {\n        if (stream.getTracks().length === 1) {\n          // if this is the last track of the stream, remove the stream. This\n          // takes care of any shimmed _senders.\n          this.removeStream(this._reverseStreams[stream.id]);\n        } else {\n          // relying on the same odd chrome behaviour as above.\n          stream.removeTrack(sender.track);\n        }\n        this.dispatchEvent(new Event('negotiationneeded'));\n      }\n    };\n}\n\nexport function shimPeerConnection(window, browserDetails) {\n  if (!window.RTCPeerConnection && window.webkitRTCPeerConnection) {\n    // very basic support for old versions.\n    window.RTCPeerConnection = window.webkitRTCPeerConnection;\n  }\n  if (!window.RTCPeerConnection) {\n    return;\n  }\n\n  // shim implicit creation of RTCSessionDescription/RTCIceCandidate\n  if (browserDetails.version < 53) {\n    ['setLocalDescription', 'setRemoteDescription', 'addIceCandidate']\n      .forEach(function(method) {\n        const nativeMethod = window.RTCPeerConnection.prototype[method];\n        const methodObj = {[method]() {\n          arguments[0] = new ((method === 'addIceCandidate') ?\n            window.RTCIceCandidate :\n            window.RTCSessionDescription)(arguments[0]);\n          return nativeMethod.apply(this, arguments);\n        }};\n        window.RTCPeerConnection.prototype[method] = methodObj[method];\n      });\n  }\n}\n\n// Attempt to fix ONN in plan-b mode.\nexport function fixNegotiationNeeded(window, browserDetails) {\n  utils.wrapPeerConnectionEvent(window, 'negotiationneeded', e => {\n    const pc = e.target;\n    if (browserDetails.version < 72 || (pc.getConfiguration &&\n        pc.getConfiguration().sdpSemantics === 'plan-b')) {\n      if (pc.signalingState !== 'stable') {\n        return;\n      }\n    }\n    return e;\n  });\n}\n","/*\n *  Copyright (c) 2018 The adapter.js project authors. All Rights Reserved.\n *\n *  Use of this source code is governed by a BSD-style license\n *  that can be found in the LICENSE file in the root of the source\n *  tree.\n */\n/* eslint-env node */\n'use strict';\nexport function shimGetDisplayMedia(window, getSourceId) {\n  if (window.navigator.mediaDevices &&\n    'getDisplayMedia' in window.navigator.mediaDevices) {\n    return;\n  }\n  if (!(window.navigator.mediaDevices)) {\n    return;\n  }\n  // getSourceId is a function that returns a promise resolving with\n  // the sourceId of the screen/window/tab to be shared.\n  if (typeof getSourceId !== 'function') {\n    console.error('shimGetDisplayMedia: getSourceId argument is not ' +\n        'a function');\n    return;\n  }\n  window.navigator.mediaDevices.getDisplayMedia =\n    function getDisplayMedia(constraints) {\n      return getSourceId(constraints)\n        .then(sourceId => {\n          const widthSpecified = constraints.video && constraints.video.width;\n          const heightSpecified = constraints.video &&\n            constraints.video.height;\n          const frameRateSpecified = constraints.video &&\n            constraints.video.frameRate;\n          constraints.video = {\n            mandatory: {\n              chromeMediaSource: 'desktop',\n              chromeMediaSourceId: sourceId,\n              maxFrameRate: frameRateSpecified || 3\n            }\n          };\n          if (widthSpecified) {\n            constraints.video.mandatory.maxWidth = widthSpecified;\n          }\n          if (heightSpecified) {\n            constraints.video.mandatory.maxHeight = heightSpecified;\n          }\n          return window.navigator.mediaDevices.getUserMedia(constraints);\n        });\n    };\n}\n","/*\n *  Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.\n *\n *  Use of this source code is governed by a BSD-style license\n *  that can be found in the LICENSE file in the root of the source\n *  tree.\n */\n/* eslint-env node */\n'use strict';\nimport * as utils from '../utils.js';\nconst logging = utils.log;\n\nexport function shimGetUserMedia(window, browserDetails) {\n  const navigator = window && window.navigator;\n\n  if (!navigator.mediaDevices) {\n    return;\n  }\n\n  const constraintsToChrome_ = function(c) {\n    if (typeof c !== 'object' || c.mandatory || c.optional) {\n      return c;\n    }\n    const cc = {};\n    Object.keys(c).forEach(key => {\n      if (key === 'require' || key === 'advanced' || key === 'mediaSource') {\n        return;\n      }\n      const r = (typeof c[key] === 'object') ? c[key] : {ideal: c[key]};\n      if (r.exact !== undefined && typeof r.exact === 'number') {\n        r.min = r.max = r.exact;\n      }\n      const oldname_ = function(prefix, name) {\n        if (prefix) {\n          return prefix + name.charAt(0).toUpperCase() + name.slice(1);\n        }\n        return (name === 'deviceId') ? 'sourceId' : name;\n      };\n      if (r.ideal !== undefined) {\n        cc.optional = cc.optional || [];\n        let oc = {};\n        if (typeof r.ideal === 'number') {\n          oc[oldname_('min', key)] = r.ideal;\n          cc.optional.push(oc);\n          oc = {};\n          oc[oldname_('max', key)] = r.ideal;\n          cc.optional.push(oc);\n        } else {\n          oc[oldname_('', key)] = r.ideal;\n          cc.optional.push(oc);\n        }\n      }\n      if (r.exact !== undefined && typeof r.exact !== 'number') {\n        cc.mandatory = cc.mandatory || {};\n        cc.mandatory[oldname_('', key)] = r.exact;\n      } else {\n        ['min', 'max'].forEach(mix => {\n          if (r[mix] !== undefined) {\n            cc.mandatory = cc.mandatory || {};\n            cc.mandatory[oldname_(mix, key)] = r[mix];\n          }\n        });\n      }\n    });\n    if (c.advanced) {\n      cc.optional = (cc.optional || []).concat(c.advanced);\n    }\n    return cc;\n  };\n\n  const shimConstraints_ = function(constraints, func) {\n    if (browserDetails.version >= 61) {\n      return func(constraints);\n    }\n    constraints = JSON.parse(JSON.stringify(constraints));\n    if (constraints && typeof constraints.audio === 'object') {\n      const remap = function(obj, a, b) {\n        if (a in obj && !(b in obj)) {\n          obj[b] = obj[a];\n          delete obj[a];\n        }\n      };\n      constraints = JSON.parse(JSON.stringify(constraints));\n      remap(constraints.audio, 'autoGainControl', 'googAutoGainControl');\n      remap(constraints.audio, 'noiseSuppression', 'googNoiseSuppression');\n      constraints.audio = constraintsToChrome_(constraints.audio);\n    }\n    if (constraints && typeof constraints.video === 'object') {\n      // Shim facingMode for mobile & surface pro.\n      let face = constraints.video.facingMode;\n      face = face && ((typeof face === 'object') ? face : {ideal: face});\n      const getSupportedFacingModeLies = browserDetails.version < 66;\n\n      if ((face && (face.exact === 'user' || face.exact === 'environment' ||\n                    face.ideal === 'user' || face.ideal === 'environment')) &&\n          !(navigator.mediaDevices.getSupportedConstraints &&\n            navigator.mediaDevices.getSupportedConstraints().facingMode &&\n            !getSupportedFacingModeLies)) {\n        delete constraints.video.facingMode;\n        let matches;\n        if (face.exact === 'environment' || face.ideal === 'environment') {\n          matches = ['back', 'rear'];\n        } else if (face.exact === 'user' || face.ideal === 'user') {\n          matches = ['front'];\n        }\n        if (matches) {\n          // Look for matches in label, or use last cam for back (typical).\n          return navigator.mediaDevices.enumerateDevices()\n            .then(devices => {\n              devices = devices.filter(d => d.kind === 'videoinput');\n              let dev = devices.find(d => matches.some(match =>\n                d.label.toLowerCase().includes(match)));\n              if (!dev && devices.length && matches.includes('back')) {\n                dev = devices[devices.length - 1]; // more likely the back cam\n              }\n              if (dev) {\n                constraints.video.deviceId = face.exact\n                  ? {exact: dev.deviceId}\n                  : {ideal: dev.deviceId};\n              }\n              constraints.video = constraintsToChrome_(constraints.video);\n              logging('chrome: ' + JSON.stringify(constraints));\n              return func(constraints);\n            });\n        }\n      }\n      constraints.video = constraintsToChrome_(constraints.video);\n    }\n    logging('chrome: ' + JSON.stringify(constraints));\n    return func(constraints);\n  };\n\n  const shimError_ = function(e) {\n    if (browserDetails.version >= 64) {\n      return e;\n    }\n    return {\n      name: {\n        PermissionDeniedError: 'NotAllowedError',\n        PermissionDismissedError: 'NotAllowedError',\n        InvalidStateError: 'NotAllowedError',\n        DevicesNotFoundError: 'NotFoundError',\n        ConstraintNotSatisfiedError: 'OverconstrainedError',\n        TrackStartError: 'NotReadableError',\n        MediaDeviceFailedDueToShutdown: 'NotAllowedError',\n        MediaDeviceKillSwitchOn: 'NotAllowedError',\n        TabCaptureError: 'AbortError',\n        ScreenCaptureError: 'AbortError',\n        DeviceCaptureError: 'AbortError'\n      }[e.name] || e.name,\n      message: e.message,\n      constraint: e.constraint || e.constraintName,\n      toString() {\n        return this.name + (this.message && ': ') + this.message;\n      }\n    };\n  };\n\n  const getUserMedia_ = function(constraints, onSuccess, onError) {\n    shimConstraints_(constraints, c => {\n      navigator.webkitGetUserMedia(c, onSuccess, e => {\n        if (onError) {\n          onError(shimError_(e));\n        }\n      });\n    });\n  };\n  navigator.getUserMedia = getUserMedia_.bind(navigator);\n\n  // Even though Chrome 45 has navigator.mediaDevices and a getUserMedia\n  // function which returns a Promise, it does not accept spec-style\n  // constraints.\n  if (navigator.mediaDevices.getUserMedia) {\n    const origGetUserMedia = navigator.mediaDevices.getUserMedia.\n      bind(navigator.mediaDevices);\n    navigator.mediaDevices.getUserMedia = function(cs) {\n      return shimConstraints_(cs, c => origGetUserMedia(c).then(stream => {\n        if (c.audio && !stream.getAudioTracks().length ||\n            c.video && !stream.getVideoTracks().length) {\n          stream.getTracks().forEach(track => {\n            track.stop();\n          });\n          throw new DOMException('', 'NotFoundError');\n        }\n        return stream;\n      }, e => Promise.reject(shimError_(e))));\n    };\n  }\n}\n","/*\n *  Copyright (c) 2017 The WebRTC project authors. All Rights Reserved.\n *\n *  Use of this source code is governed by a BSD-style license\n *  that can be found in the LICENSE file in the root of the source\n *  tree.\n */\n/* eslint-env node */\n'use strict';\n\nimport SDPUtils from 'sdp';\nimport * as utils from './utils';\n\nexport function shimRTCIceCandidate(window) {\n  // foundation is arbitrarily chosen as an indicator for full support for\n  // https://w3c.github.io/webrtc-pc/#rtcicecandidate-interface\n  if (!window.RTCIceCandidate || (window.RTCIceCandidate && 'foundation' in\n      window.RTCIceCandidate.prototype)) {\n    return;\n  }\n\n  const NativeRTCIceCandidate = window.RTCIceCandidate;\n  window.RTCIceCandidate = function RTCIceCandidate(args) {\n    // Remove the a= which shouldn't be part of the candidate string.\n    if (typeof args === 'object' && args.candidate &&\n        args.candidate.indexOf('a=') === 0) {\n      args = JSON.parse(JSON.stringify(args));\n      args.candidate = args.candidate.substring(2);\n    }\n\n    if (args.candidate && args.candidate.length) {\n      // Augment the native candidate with the parsed fields.\n      const nativeCandidate = new NativeRTCIceCandidate(args);\n      const parsedCandidate = SDPUtils.parseCandidate(args.candidate);\n      for (const key in parsedCandidate) {\n        if (!(key in nativeCandidate)) {\n          Object.defineProperty(nativeCandidate, key,\n            {value: parsedCandidate[key]});\n        }\n      }\n\n      // Override serializer to not serialize the extra attributes.\n      nativeCandidate.toJSON = function toJSON() {\n        return {\n          candidate: nativeCandidate.candidate,\n          sdpMid: nativeCandidate.sdpMid,\n          sdpMLineIndex: nativeCandidate.sdpMLineIndex,\n          usernameFragment: nativeCandidate.usernameFragment,\n        };\n      };\n      return nativeCandidate;\n    }\n    return new NativeRTCIceCandidate(args);\n  };\n  window.RTCIceCandidate.prototype = NativeRTCIceCandidate.prototype;\n\n  // Hook up the augmented candidate in onicecandidate and\n  // addEventListener('icecandidate', ...)\n  utils.wrapPeerConnectionEvent(window, 'icecandidate', e => {\n    if (e.candidate) {\n      Object.defineProperty(e, 'candidate', {\n        value: new window.RTCIceCandidate(e.candidate),\n        writable: 'false'\n      });\n    }\n    return e;\n  });\n}\n\nexport function shimRTCIceCandidateRelayProtocol(window) {\n  if (!window.RTCIceCandidate || (window.RTCIceCandidate && 'relayProtocol' in\n      window.RTCIceCandidate.prototype)) {\n    return;\n  }\n\n  // Hook up the augmented candidate in onicecandidate and\n  // addEventListener('icecandidate', ...)\n  utils.wrapPeerConnectionEvent(window, 'icecandidate', e => {\n    if (e.candidate) {\n      const parsedCandidate = SDPUtils.parseCandidate(e.candidate.candidate);\n      if (parsedCandidate.type === 'relay') {\n        // This is a libwebrtc-specific mapping of local type preference\n        // to relayProtocol.\n        e.candidate.relayProtocol = {\n          0: 'tls',\n          1: 'tcp',\n          2: 'udp',\n        }[parsedCandidate.priority >> 24];\n      }\n    }\n    return e;\n  });\n}\n\nexport function shimMaxMessageSize(window, browserDetails) {\n  if (!window.RTCPeerConnection) {\n    return;\n  }\n\n  if (!('sctp' in window.RTCPeerConnection.prototype)) {\n    Object.defineProperty(window.RTCPeerConnection.prototype, 'sctp', {\n      get() {\n        return typeof this._sctp === 'undefined' ? null : this._sctp;\n      }\n    });\n  }\n\n  const sctpInDescription = function(description) {\n    if (!description || !description.sdp) {\n      return false;\n    }\n    const sections = SDPUtils.splitSections(description.sdp);\n    sections.shift();\n    return sections.some(mediaSection => {\n      const mLine = SDPUtils.parseMLine(mediaSection);\n      return mLine && mLine.kind === 'application'\n          && mLine.protocol.indexOf('SCTP') !== -1;\n    });\n  };\n\n  const getRemoteFirefoxVersion = function(description) {\n    // TODO: Is there a better solution for detecting Firefox?\n    const match = description.sdp.match(/mozilla...THIS_IS_SDPARTA-(\\d+)/);\n    if (match === null || match.length < 2) {\n      return -1;\n    }\n    const version = parseInt(match[1], 10);\n    // Test for NaN (yes, this is ugly)\n    return version !== version ? -1 : version;\n  };\n\n  const getCanSendMaxMessageSize = function(remoteIsFirefox) {\n    // Every implementation we know can send at least 64 KiB.\n    // Note: Although Chrome is technically able to send up to 256 KiB, the\n    //       data does not reach the other peer reliably.\n    //       See: https://bugs.chromium.org/p/webrtc/issues/detail?id=8419\n    let canSendMaxMessageSize = 65536;\n    if (browserDetails.browser === 'firefox') {\n      if (browserDetails.version < 57) {\n        if (remoteIsFirefox === -1) {\n          // FF < 57 will send in 16 KiB chunks using the deprecated PPID\n          // fragmentation.\n          canSendMaxMessageSize = 16384;\n        } else {\n          // However, other FF (and RAWRTC) can reassemble PPID-fragmented\n          // messages. Thus, supporting ~2 GiB when sending.\n          canSendMaxMessageSize = 2147483637;\n        }\n      } else if (browserDetails.version < 60) {\n        // Currently, all FF >= 57 will reset the remote maximum message size\n        // to the default value when a data channel is created at a later\n        // stage. :(\n        // See: https://bugzilla.mozilla.org/show_bug.cgi?id=1426831\n        canSendMaxMessageSize =\n          browserDetails.version === 57 ? 65535 : 65536;\n      } else {\n        // FF >= 60 supports sending ~2 GiB\n        canSendMaxMessageSize = 2147483637;\n      }\n    }\n    return canSendMaxMessageSize;\n  };\n\n  const getMaxMessageSize = function(description, remoteIsFirefox) {\n    // Note: 65536 bytes is the default value from the SDP spec. Also,\n    //       every implementation we know supports receiving 65536 bytes.\n    let maxMessageSize = 65536;\n\n    // FF 57 has a slightly incorrect default remote max message size, so\n    // we need to adjust it here to avoid a failure when sending.\n    // See: https://bugzilla.mozilla.org/show_bug.cgi?id=1425697\n    if (browserDetails.browser === 'firefox'\n         && browserDetails.version === 57) {\n      maxMessageSize = 65535;\n    }\n\n    const match = SDPUtils.matchPrefix(description.sdp,\n      'a=max-message-size:');\n    if (match.length > 0) {\n      maxMessageSize = parseInt(match[0].substring(19), 10);\n    } else if (browserDetails.browser === 'firefox' &&\n                remoteIsFirefox !== -1) {\n      // If the maximum message size is not present in the remote SDP and\n      // both local and remote are Firefox, the remote peer can receive\n      // ~2 GiB.\n      maxMessageSize = 2147483637;\n    }\n    return maxMessageSize;\n  };\n\n  const origSetRemoteDescription =\n      window.RTCPeerConnection.prototype.setRemoteDescription;\n  window.RTCPeerConnection.prototype.setRemoteDescription =\n    function setRemoteDescription() {\n      this._sctp = null;\n      // Chrome decided to not expose .sctp in plan-b mode.\n      // As usual, adapter.js has to do an 'ugly worakaround'\n      // to cover up the mess.\n      if (browserDetails.browser === 'chrome' && browserDetails.version >= 76) {\n        const {sdpSemantics} = this.getConfiguration();\n        if (sdpSemantics === 'plan-b') {\n          Object.defineProperty(this, 'sctp', {\n            get() {\n              return typeof this._sctp === 'undefined' ? null : this._sctp;\n            },\n            enumerable: true,\n            configurable: true,\n          });\n        }\n      }\n\n      if (sctpInDescription(arguments[0])) {\n        // Check if the remote is FF.\n        const isFirefox = getRemoteFirefoxVersion(arguments[0]);\n\n        // Get the maximum message size the local peer is capable of sending\n        const canSendMMS = getCanSendMaxMessageSize(isFirefox);\n\n        // Get the maximum message size of the remote peer.\n        const remoteMMS = getMaxMessageSize(arguments[0], isFirefox);\n\n        // Determine final maximum message size\n        let maxMessageSize;\n        if (canSendMMS === 0 && remoteMMS === 0) {\n          maxMessageSize = Number.POSITIVE_INFINITY;\n        } else if (canSendMMS === 0 || remoteMMS === 0) {\n          maxMessageSize = Math.max(canSendMMS, remoteMMS);\n        } else {\n          maxMessageSize = Math.min(canSendMMS, remoteMMS);\n        }\n\n        // Create a dummy RTCSctpTransport object and the 'maxMessageSize'\n        // attribute.\n        const sctp = {};\n        Object.defineProperty(sctp, 'maxMessageSize', {\n          get() {\n            return maxMessageSize;\n          }\n        });\n        this._sctp = sctp;\n      }\n\n      return origSetRemoteDescription.apply(this, arguments);\n    };\n}\n\nexport function shimSendThrowTypeError(window) {\n  if (!(window.RTCPeerConnection &&\n      'createDataChannel' in window.RTCPeerConnection.prototype)) {\n    return;\n  }\n\n  // Note: Although Firefox >= 57 has a native implementation, the maximum\n  //       message size can be reset for all data channels at a later stage.\n  //       See: https://bugzilla.mozilla.org/show_bug.cgi?id=1426831\n\n  function wrapDcSend(dc, pc) {\n    const origDataChannelSend = dc.send;\n    dc.send = function send() {\n      const data = arguments[0];\n      const length = data.length || data.size || data.byteLength;\n      if (dc.readyState === 'open' &&\n          pc.sctp && length > pc.sctp.maxMessageSize) {\n        throw new TypeError('Message too large (can send a maximum of ' +\n          pc.sctp.maxMessageSize + ' bytes)');\n      }\n      return origDataChannelSend.apply(dc, arguments);\n    };\n  }\n  const origCreateDataChannel =\n    window.RTCPeerConnection.prototype.createDataChannel;\n  window.RTCPeerConnection.prototype.createDataChannel =\n    function createDataChannel() {\n      const dataChannel = origCreateDataChannel.apply(this, arguments);\n      wrapDcSend(dataChannel, this);\n      return dataChannel;\n    };\n  utils.wrapPeerConnectionEvent(window, 'datachannel', e => {\n    wrapDcSend(e.channel, e.target);\n    return e;\n  });\n}\n\n\n/* shims RTCConnectionState by pretending it is the same as iceConnectionState.\n * See https://bugs.chromium.org/p/webrtc/issues/detail?id=6145#c12\n * for why this is a valid hack in Chrome. In Firefox it is slightly incorrect\n * since DTLS failures would be hidden. See\n * https://bugzilla.mozilla.org/show_bug.cgi?id=1265827\n * for the Firefox tracking bug.\n */\nexport function shimConnectionState(window) {\n  if (!window.RTCPeerConnection ||\n      'connectionState' in window.RTCPeerConnection.prototype) {\n    return;\n  }\n  const proto = window.RTCPeerConnection.prototype;\n  Object.defineProperty(proto, 'connectionState', {\n    get() {\n      return {\n        completed: 'connected',\n        checking: 'connecting'\n      }[this.iceConnectionState] || this.iceConnectionState;\n    },\n    enumerable: true,\n    configurable: true\n  });\n  Object.defineProperty(proto, 'onconnectionstatechange', {\n    get() {\n      return this._onconnectionstatechange || null;\n    },\n    set(cb) {\n      if (this._onconnectionstatechange) {\n        this.removeEventListener('connectionstatechange',\n          this._onconnectionstatechange);\n        delete this._onconnectionstatechange;\n      }\n      if (cb) {\n        this.addEventListener('connectionstatechange',\n          this._onconnectionstatechange = cb);\n      }\n    },\n    enumerable: true,\n    configurable: true\n  });\n\n  ['setLocalDescription', 'setRemoteDescription'].forEach((method) => {\n    const origMethod = proto[method];\n    proto[method] = function() {\n      if (!this._connectionstatechangepoly) {\n        this._connectionstatechangepoly = e => {\n          const pc = e.target;\n          if (pc._lastConnectionState !== pc.connectionState) {\n            pc._lastConnectionState = pc.connectionState;\n            const newEvent = new Event('connectionstatechange', e);\n            pc.dispatchEvent(newEvent);\n          }\n          return e;\n        };\n        this.addEventListener('iceconnectionstatechange',\n          this._connectionstatechangepoly);\n      }\n      return origMethod.apply(this, arguments);\n    };\n  });\n}\n\nexport function removeExtmapAllowMixed(window, browserDetails) {\n  /* remove a=extmap-allow-mixed for webrtc.org < M71 */\n  if (!window.RTCPeerConnection) {\n    return;\n  }\n  if (browserDetails.browser === 'chrome' && browserDetails.version >= 71) {\n    return;\n  }\n  if (browserDetails.browser === 'safari' && browserDetails.version >= 605) {\n    return;\n  }\n  const nativeSRD = window.RTCPeerConnection.prototype.setRemoteDescription;\n  window.RTCPeerConnection.prototype.setRemoteDescription =\n  function setRemoteDescription(desc) {\n    if (desc && desc.sdp && desc.sdp.indexOf('\\na=extmap-allow-mixed') !== -1) {\n      const sdp = desc.sdp.split('\\n').filter((line) => {\n        return line.trim() !== 'a=extmap-allow-mixed';\n      }).join('\\n');\n      // Safari enforces read-only-ness of RTCSessionDescription fields.\n      if (window.RTCSessionDescription &&\n          desc instanceof window.RTCSessionDescription) {\n        arguments[0] = new window.RTCSessionDescription({\n          type: desc.type,\n          sdp,\n        });\n      } else {\n        desc.sdp = sdp;\n      }\n    }\n    return nativeSRD.apply(this, arguments);\n  };\n}\n\nexport function shimAddIceCandidateNullOrEmpty(window, browserDetails) {\n  // Support for addIceCandidate(null or undefined)\n  // as well as addIceCandidate({candidate: \"\", ...})\n  // https://bugs.chromium.org/p/chromium/issues/detail?id=978582\n  // Note: must be called before other polyfills which change the signature.\n  if (!(window.RTCPeerConnection && window.RTCPeerConnection.prototype)) {\n    return;\n  }\n  const nativeAddIceCandidate =\n      window.RTCPeerConnection.prototype.addIceCandidate;\n  if (!nativeAddIceCandidate || nativeAddIceCandidate.length === 0) {\n    return;\n  }\n  window.RTCPeerConnection.prototype.addIceCandidate =\n    function addIceCandidate() {\n      if (!arguments[0]) {\n        if (arguments[1]) {\n          arguments[1].apply(null);\n        }\n        return Promise.resolve();\n      }\n      // Firefox 68+ emits and processes {candidate: \"\", ...}, ignore\n      // in older versions.\n      // Native support for ignoring exists for Chrome M77+.\n      // Safari ignores as well, exact version unknown but works in the same\n      // version that also ignores addIceCandidate(null).\n      if (((browserDetails.browser === 'chrome' && browserDetails.version < 78)\n           || (browserDetails.browser === 'firefox'\n               && browserDetails.version < 68)\n           || (browserDetails.browser === 'safari'))\n          && arguments[0] && arguments[0].candidate === '') {\n        return Promise.resolve();\n      }\n      return nativeAddIceCandidate.apply(this, arguments);\n    };\n}\n\n// Note: Make sure to call this ahead of APIs that modify\n// setLocalDescription.length\nexport function shimParameterlessSetLocalDescription(window, browserDetails) {\n  if (!(window.RTCPeerConnection && window.RTCPeerConnection.prototype)) {\n    return;\n  }\n  const nativeSetLocalDescription =\n      window.RTCPeerConnection.prototype.setLocalDescription;\n  if (!nativeSetLocalDescription || nativeSetLocalDescription.length === 0) {\n    return;\n  }\n  window.RTCPeerConnection.prototype.setLocalDescription =\n    function setLocalDescription() {\n      let desc = arguments[0] || {};\n      if (typeof desc !== 'object' || (desc.type && desc.sdp)) {\n        return nativeSetLocalDescription.apply(this, arguments);\n      }\n      // The remaining steps should technically happen when SLD comes off the\n      // RTCPeerConnection's operations chain (not ahead of going on it), but\n      // this is too difficult to shim. Instead, this shim only covers the\n      // common case where the operations chain is empty. This is imperfect, but\n      // should cover many cases. Rationale: Even if we can't reduce the glare\n      // window to zero on imperfect implementations, there's value in tapping\n      // into the perfect negotiation pattern that several browsers support.\n      desc = {type: desc.type, sdp: desc.sdp};\n      if (!desc.type) {\n        switch (this.signalingState) {\n          case 'stable':\n          case 'have-local-offer':\n          case 'have-remote-pranswer':\n            desc.type = 'offer';\n            break;\n          default:\n            desc.type = 'answer';\n            break;\n        }\n      }\n      if (desc.sdp || (desc.type !== 'offer' && desc.type !== 'answer')) {\n        return nativeSetLocalDescription.apply(this, [desc]);\n      }\n      const func = desc.type === 'offer' ? this.createOffer : this.createAnswer;\n      return func.apply(this)\n        .then(d => nativeSetLocalDescription.apply(this, [d]));\n    };\n}\n","/*\n *  Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.\n *\n *  Use of this source code is governed by a BSD-style license\n *  that can be found in the LICENSE file in the root of the source\n *  tree.\n */\n/* eslint-env node */\n'use strict';\n\nimport * as utils from '../utils';\nexport {shimGetUserMedia} from './getusermedia';\nexport {shimGetDisplayMedia} from './getdisplaymedia';\n\nexport function shimOnTrack(window) {\n  if (typeof window === 'object' && window.RTCTrackEvent &&\n      ('receiver' in window.RTCTrackEvent.prototype) &&\n      !('transceiver' in window.RTCTrackEvent.prototype)) {\n    Object.defineProperty(window.RTCTrackEvent.prototype, 'transceiver', {\n      get() {\n        return {receiver: this.receiver};\n      }\n    });\n  }\n}\n\nexport function shimPeerConnection(window, browserDetails) {\n  if (typeof window !== 'object' ||\n      !(window.RTCPeerConnection || window.mozRTCPeerConnection)) {\n    return; // probably media.peerconnection.enabled=false in about:config\n  }\n  if (!window.RTCPeerConnection && window.mozRTCPeerConnection) {\n    // very basic support for old versions.\n    window.RTCPeerConnection = window.mozRTCPeerConnection;\n  }\n\n  if (browserDetails.version < 53) {\n    // shim away need for obsolete RTCIceCandidate/RTCSessionDescription.\n    ['setLocalDescription', 'setRemoteDescription', 'addIceCandidate']\n      .forEach(function(method) {\n        const nativeMethod = window.RTCPeerConnection.prototype[method];\n        const methodObj = {[method]() {\n          arguments[0] = new ((method === 'addIceCandidate') ?\n            window.RTCIceCandidate :\n            window.RTCSessionDescription)(arguments[0]);\n          return nativeMethod.apply(this, arguments);\n        }};\n        window.RTCPeerConnection.prototype[method] = methodObj[method];\n      });\n  }\n\n  const modernStatsTypes = {\n    inboundrtp: 'inbound-rtp',\n    outboundrtp: 'outbound-rtp',\n    candidatepair: 'candidate-pair',\n    localcandidate: 'local-candidate',\n    remotecandidate: 'remote-candidate'\n  };\n\n  const nativeGetStats = window.RTCPeerConnection.prototype.getStats;\n  window.RTCPeerConnection.prototype.getStats = function getStats() {\n    const [selector, onSucc, onErr] = arguments;\n    return nativeGetStats.apply(this, [selector || null])\n      .then(stats => {\n        if (browserDetails.version < 53 && !onSucc) {\n          // Shim only promise getStats with spec-hyphens in type names\n          // Leave callback version alone; misc old uses of forEach before Map\n          try {\n            stats.forEach(stat => {\n              stat.type = modernStatsTypes[stat.type] || stat.type;\n            });\n          } catch (e) {\n            if (e.name !== 'TypeError') {\n              throw e;\n            }\n            // Avoid TypeError: \"type\" is read-only, in old versions. 34-43ish\n            stats.forEach((stat, i) => {\n              stats.set(i, Object.assign({}, stat, {\n                type: modernStatsTypes[stat.type] || stat.type\n              }));\n            });\n          }\n        }\n        return stats;\n      })\n      .then(onSucc, onErr);\n  };\n}\n\nexport function shimSenderGetStats(window) {\n  if (!(typeof window === 'object' && window.RTCPeerConnection &&\n      window.RTCRtpSender)) {\n    return;\n  }\n  if (window.RTCRtpSender && 'getStats' in window.RTCRtpSender.prototype) {\n    return;\n  }\n  const origGetSenders = window.RTCPeerConnection.prototype.getSenders;\n  if (origGetSenders) {\n    window.RTCPeerConnection.prototype.getSenders = function getSenders() {\n      const senders = origGetSenders.apply(this, []);\n      senders.forEach(sender => sender._pc = this);\n      return senders;\n    };\n  }\n\n  const origAddTrack = window.RTCPeerConnection.prototype.addTrack;\n  if (origAddTrack) {\n    window.RTCPeerConnection.prototype.addTrack = function addTrack() {\n      const sender = origAddTrack.apply(this, arguments);\n      sender._pc = this;\n      return sender;\n    };\n  }\n  window.RTCRtpSender.prototype.getStats = function getStats() {\n    return this.track ? this._pc.getStats(this.track) :\n      Promise.resolve(new Map());\n  };\n}\n\nexport function shimReceiverGetStats(window) {\n  if (!(typeof window === 'object' && window.RTCPeerConnection &&\n      window.RTCRtpSender)) {\n    return;\n  }\n  if (window.RTCRtpSender && 'getStats' in window.RTCRtpReceiver.prototype) {\n    return;\n  }\n  const origGetReceivers = window.RTCPeerConnection.prototype.getReceivers;\n  if (origGetReceivers) {\n    window.RTCPeerConnection.prototype.getReceivers = function getReceivers() {\n      const receivers = origGetReceivers.apply(this, []);\n      receivers.forEach(receiver => receiver._pc = this);\n      return receivers;\n    };\n  }\n  utils.wrapPeerConnectionEvent(window, 'track', e => {\n    e.receiver._pc = e.srcElement;\n    return e;\n  });\n  window.RTCRtpReceiver.prototype.getStats = function getStats() {\n    return this._pc.getStats(this.track);\n  };\n}\n\nexport function shimRemoveStream(window) {\n  if (!window.RTCPeerConnection ||\n      'removeStream' in window.RTCPeerConnection.prototype) {\n    return;\n  }\n  window.RTCPeerConnection.prototype.removeStream =\n    function removeStream(stream) {\n      utils.deprecated('removeStream', 'removeTrack');\n      this.getSenders().forEach(sender => {\n        if (sender.track && stream.getTracks().includes(sender.track)) {\n          this.removeTrack(sender);\n        }\n      });\n    };\n}\n\nexport function shimRTCDataChannel(window) {\n  // rename DataChannel to RTCDataChannel (native fix in FF60):\n  // https://bugzilla.mozilla.org/show_bug.cgi?id=1173851\n  if (window.DataChannel && !window.RTCDataChannel) {\n    window.RTCDataChannel = window.DataChannel;\n  }\n}\n\nexport function shimAddTransceiver(window) {\n  // https://github.com/webrtcHacks/adapter/issues/998#issuecomment-516921647\n  // Firefox ignores the init sendEncodings options passed to addTransceiver\n  // https://bugzilla.mozilla.org/show_bug.cgi?id=1396918\n  if (!(typeof window === 'object' && window.RTCPeerConnection)) {\n    return;\n  }\n  const origAddTransceiver = window.RTCPeerConnection.prototype.addTransceiver;\n  if (origAddTransceiver) {\n    window.RTCPeerConnection.prototype.addTransceiver =\n      function addTransceiver() {\n        this.setParametersPromises = [];\n        // WebIDL input coercion and validation\n        let sendEncodings = arguments[1] && arguments[1].sendEncodings;\n        if (sendEncodings === undefined) {\n          sendEncodings = [];\n        }\n        sendEncodings = [...sendEncodings];\n        const shouldPerformCheck = sendEncodings.length > 0;\n        if (shouldPerformCheck) {\n          // If sendEncodings params are provided, validate grammar\n          sendEncodings.forEach((encodingParam) => {\n            if ('rid' in encodingParam) {\n              const ridRegex = /^[a-z0-9]{0,16}$/i;\n              if (!ridRegex.test(encodingParam.rid)) {\n                throw new TypeError('Invalid RID value provided.');\n              }\n            }\n            if ('scaleResolutionDownBy' in encodingParam) {\n              if (!(parseFloat(encodingParam.scaleResolutionDownBy) >= 1.0)) {\n                throw new RangeError('scale_resolution_down_by must be >= 1.0');\n              }\n            }\n            if ('maxFramerate' in encodingParam) {\n              if (!(parseFloat(encodingParam.maxFramerate) >= 0)) {\n                throw new RangeError('max_framerate must be >= 0.0');\n              }\n            }\n          });\n        }\n        const transceiver = origAddTransceiver.apply(this, arguments);\n        if (shouldPerformCheck) {\n          // Check if the init options were applied. If not we do this in an\n          // asynchronous way and save the promise reference in a global object.\n          // This is an ugly hack, but at the same time is way more robust than\n          // checking the sender parameters before and after the createOffer\n          // Also note that after the createoffer we are not 100% sure that\n          // the params were asynchronously applied so we might miss the\n          // opportunity to recreate offer.\n          const {sender} = transceiver;\n          const params = sender.getParameters();\n          if (!('encodings' in params) ||\n              // Avoid being fooled by patched getParameters() below.\n              (params.encodings.length === 1 &&\n               Object.keys(params.encodings[0]).length === 0)) {\n            params.encodings = sendEncodings;\n            sender.sendEncodings = sendEncodings;\n            this.setParametersPromises.push(sender.setParameters(params)\n              .then(() => {\n                delete sender.sendEncodings;\n              }).catch(() => {\n                delete sender.sendEncodings;\n              })\n            );\n          }\n        }\n        return transceiver;\n      };\n  }\n}\n\nexport function shimGetParameters(window) {\n  if (!(typeof window === 'object' && window.RTCRtpSender)) {\n    return;\n  }\n  const origGetParameters = window.RTCRtpSender.prototype.getParameters;\n  if (origGetParameters) {\n    window.RTCRtpSender.prototype.getParameters =\n      function getParameters() {\n        const params = origGetParameters.apply(this, arguments);\n        if (!('encodings' in params)) {\n          params.encodings = [].concat(this.sendEncodings || [{}]);\n        }\n        return params;\n      };\n  }\n}\n\nexport function shimCreateOffer(window) {\n  // https://github.com/webrtcHacks/adapter/issues/998#issuecomment-516921647\n  // Firefox ignores the init sendEncodings options passed to addTransceiver\n  // https://bugzilla.mozilla.org/show_bug.cgi?id=1396918\n  if (!(typeof window === 'object' && window.RTCPeerConnection)) {\n    return;\n  }\n  const origCreateOffer = window.RTCPeerConnection.prototype.createOffer;\n  window.RTCPeerConnection.prototype.createOffer = function createOffer() {\n    if (this.setParametersPromises && this.setParametersPromises.length) {\n      return Promise.all(this.setParametersPromises)\n        .then(() => {\n          return origCreateOffer.apply(this, arguments);\n        })\n        .finally(() => {\n          this.setParametersPromises = [];\n        });\n    }\n    return origCreateOffer.apply(this, arguments);\n  };\n}\n\nexport function shimCreateAnswer(window) {\n  // https://github.com/webrtcHacks/adapter/issues/998#issuecomment-516921647\n  // Firefox ignores the init sendEncodings options passed to addTransceiver\n  // https://bugzilla.mozilla.org/show_bug.cgi?id=1396918\n  if (!(typeof window === 'object' && window.RTCPeerConnection)) {\n    return;\n  }\n  const origCreateAnswer = window.RTCPeerConnection.prototype.createAnswer;\n  window.RTCPeerConnection.prototype.createAnswer = function createAnswer() {\n    if (this.setParametersPromises && this.setParametersPromises.length) {\n      return Promise.all(this.setParametersPromises)\n        .then(() => {\n          return origCreateAnswer.apply(this, arguments);\n        })\n        .finally(() => {\n          this.setParametersPromises = [];\n        });\n    }\n    return origCreateAnswer.apply(this, arguments);\n  };\n}\n","/*\n *  Copyright (c) 2018 The adapter.js project authors. All Rights Reserved.\n *\n *  Use of this source code is governed by a BSD-style license\n *  that can be found in the LICENSE file in the root of the source\n *  tree.\n */\n/* eslint-env node */\n'use strict';\n\nexport function shimGetDisplayMedia(window, preferredMediaSource) {\n  if (window.navigator.mediaDevices &&\n    'getDisplayMedia' in window.navigator.mediaDevices) {\n    return;\n  }\n  if (!(window.navigator.mediaDevices)) {\n    return;\n  }\n  window.navigator.mediaDevices.getDisplayMedia =\n    function getDisplayMedia(constraints) {\n      if (!(constraints && constraints.video)) {\n        const err = new DOMException('getDisplayMedia without video ' +\n            'constraints is undefined');\n        err.name = 'NotFoundError';\n        // from https://heycam.github.io/webidl/#idl-DOMException-error-names\n        err.code = 8;\n        return Promise.reject(err);\n      }\n      if (constraints.video === true) {\n        constraints.video = {mediaSource: preferredMediaSource};\n      } else {\n        constraints.video.mediaSource = preferredMediaSource;\n      }\n      return window.navigator.mediaDevices.getUserMedia(constraints);\n    };\n}\n","/*\n *  Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.\n *\n *  Use of this source code is governed by a BSD-style license\n *  that can be found in the LICENSE file in the root of the source\n *  tree.\n */\n/* eslint-env node */\n'use strict';\n\nimport * as utils from '../utils';\n\nexport function shimGetUserMedia(window, browserDetails) {\n  const navigator = window && window.navigator;\n  const MediaStreamTrack = window && window.MediaStreamTrack;\n\n  navigator.getUserMedia = function(constraints, onSuccess, onError) {\n    // Replace Firefox 44+'s deprecation warning with unprefixed version.\n    utils.deprecated('navigator.getUserMedia',\n      'navigator.mediaDevices.getUserMedia');\n    navigator.mediaDevices.getUserMedia(constraints).then(onSuccess, onError);\n  };\n\n  if (!(browserDetails.version > 55 &&\n      'autoGainControl' in navigator.mediaDevices.getSupportedConstraints())) {\n    const remap = function(obj, a, b) {\n      if (a in obj && !(b in obj)) {\n        obj[b] = obj[a];\n        delete obj[a];\n      }\n    };\n\n    const nativeGetUserMedia = navigator.mediaDevices.getUserMedia.\n      bind(navigator.mediaDevices);\n    navigator.mediaDevices.getUserMedia = function(c) {\n      if (typeof c === 'object' && typeof c.audio === 'object') {\n        c = JSON.parse(JSON.stringify(c));\n        remap(c.audio, 'autoGainControl', 'mozAutoGainControl');\n        remap(c.audio, 'noiseSuppression', 'mozNoiseSuppression');\n      }\n      return nativeGetUserMedia(c);\n    };\n\n    if (MediaStreamTrack && MediaStreamTrack.prototype.getSettings) {\n      const nativeGetSettings = MediaStreamTrack.prototype.getSettings;\n      MediaStreamTrack.prototype.getSettings = function() {\n        const obj = nativeGetSettings.apply(this, arguments);\n        remap(obj, 'mozAutoGainControl', 'autoGainControl');\n        remap(obj, 'mozNoiseSuppression', 'noiseSuppression');\n        return obj;\n      };\n    }\n\n    if (MediaStreamTrack && MediaStreamTrack.prototype.applyConstraints) {\n      const nativeApplyConstraints =\n        MediaStreamTrack.prototype.applyConstraints;\n      MediaStreamTrack.prototype.applyConstraints = function(c) {\n        if (this.kind === 'audio' && typeof c === 'object') {\n          c = JSON.parse(JSON.stringify(c));\n          remap(c, 'autoGainControl', 'mozAutoGainControl');\n          remap(c, 'noiseSuppression', 'mozNoiseSuppression');\n        }\n        return nativeApplyConstraints.apply(this, [c]);\n      };\n    }\n  }\n}\n","/*\n *  Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.\n *\n *  Use of this source code is governed by a BSD-style license\n *  that can be found in the LICENSE file in the root of the source\n *  tree.\n */\n'use strict';\nimport * as utils from '../utils';\n\nexport function shimLocalStreamsAPI(window) {\n  if (typeof window !== 'object' || !window.RTCPeerConnection) {\n    return;\n  }\n  if (!('getLocalStreams' in window.RTCPeerConnection.prototype)) {\n    window.RTCPeerConnection.prototype.getLocalStreams =\n      function getLocalStreams() {\n        if (!this._localStreams) {\n          this._localStreams = [];\n        }\n        return this._localStreams;\n      };\n  }\n  if (!('addStream' in window.RTCPeerConnection.prototype)) {\n    const _addTrack = window.RTCPeerConnection.prototype.addTrack;\n    window.RTCPeerConnection.prototype.addStream = function addStream(stream) {\n      if (!this._localStreams) {\n        this._localStreams = [];\n      }\n      if (!this._localStreams.includes(stream)) {\n        this._localStreams.push(stream);\n      }\n      // Try to emulate Chrome's behaviour of adding in audio-video order.\n      // Safari orders by track id.\n      stream.getAudioTracks().forEach(track => _addTrack.call(this, track,\n        stream));\n      stream.getVideoTracks().forEach(track => _addTrack.call(this, track,\n        stream));\n    };\n\n    window.RTCPeerConnection.prototype.addTrack =\n      function addTrack(track, ...streams) {\n        if (streams) {\n          streams.forEach((stream) => {\n            if (!this._localStreams) {\n              this._localStreams = [stream];\n            } else if (!this._localStreams.includes(stream)) {\n              this._localStreams.push(stream);\n            }\n          });\n        }\n        return _addTrack.apply(this, arguments);\n      };\n  }\n  if (!('removeStream' in window.RTCPeerConnection.prototype)) {\n    window.RTCPeerConnection.prototype.removeStream =\n      function removeStream(stream) {\n        if (!this._localStreams) {\n          this._localStreams = [];\n        }\n        const index = this._localStreams.indexOf(stream);\n        if (index === -1) {\n          return;\n        }\n        this._localStreams.splice(index, 1);\n        const tracks = stream.getTracks();\n        this.getSenders().forEach(sender => {\n          if (tracks.includes(sender.track)) {\n            this.removeTrack(sender);\n          }\n        });\n      };\n  }\n}\n\nexport function shimRemoteStreamsAPI(window) {\n  if (typeof window !== 'object' || !window.RTCPeerConnection) {\n    return;\n  }\n  if (!('getRemoteStreams' in window.RTCPeerConnection.prototype)) {\n    window.RTCPeerConnection.prototype.getRemoteStreams =\n      function getRemoteStreams() {\n        return this._remoteStreams ? this._remoteStreams : [];\n      };\n  }\n  if (!('onaddstream' in window.RTCPeerConnection.prototype)) {\n    Object.defineProperty(window.RTCPeerConnection.prototype, 'onaddstream', {\n      get() {\n        return this._onaddstream;\n      },\n      set(f) {\n        if (this._onaddstream) {\n          this.removeEventListener('addstream', this._onaddstream);\n          this.removeEventListener('track', this._onaddstreampoly);\n        }\n        this.addEventListener('addstream', this._onaddstream = f);\n        this.addEventListener('track', this._onaddstreampoly = (e) => {\n          e.streams.forEach(stream => {\n            if (!this._remoteStreams) {\n              this._remoteStreams = [];\n            }\n            if (this._remoteStreams.includes(stream)) {\n              return;\n            }\n            this._remoteStreams.push(stream);\n            const event = new Event('addstream');\n            event.stream = stream;\n            this.dispatchEvent(event);\n          });\n        });\n      }\n    });\n    const origSetRemoteDescription =\n      window.RTCPeerConnection.prototype.setRemoteDescription;\n    window.RTCPeerConnection.prototype.setRemoteDescription =\n      function setRemoteDescription() {\n        const pc = this;\n        if (!this._onaddstreampoly) {\n          this.addEventListener('track', this._onaddstreampoly = function(e) {\n            e.streams.forEach(stream => {\n              if (!pc._remoteStreams) {\n                pc._remoteStreams = [];\n              }\n              if (pc._remoteStreams.indexOf(stream) >= 0) {\n                return;\n              }\n              pc._remoteStreams.push(stream);\n              const event = new Event('addstream');\n              event.stream = stream;\n              pc.dispatchEvent(event);\n            });\n          });\n        }\n        return origSetRemoteDescription.apply(pc, arguments);\n      };\n  }\n}\n\nexport function shimCallbacksAPI(window) {\n  if (typeof window !== 'object' || !window.RTCPeerConnection) {\n    return;\n  }\n  const prototype = window.RTCPeerConnection.prototype;\n  const origCreateOffer = prototype.createOffer;\n  const origCreateAnswer = prototype.createAnswer;\n  const setLocalDescription = prototype.setLocalDescription;\n  const setRemoteDescription = prototype.setRemoteDescription;\n  const addIceCandidate = prototype.addIceCandidate;\n\n  prototype.createOffer =\n    function createOffer(successCallback, failureCallback) {\n      const options = (arguments.length >= 2) ? arguments[2] : arguments[0];\n      const promise = origCreateOffer.apply(this, [options]);\n      if (!failureCallback) {\n        return promise;\n      }\n      promise.then(successCallback, failureCallback);\n      return Promise.resolve();\n    };\n\n  prototype.createAnswer =\n    function createAnswer(successCallback, failureCallback) {\n      const options = (arguments.length >= 2) ? arguments[2] : arguments[0];\n      const promise = origCreateAnswer.apply(this, [options]);\n      if (!failureCallback) {\n        return promise;\n      }\n      promise.then(successCallback, failureCallback);\n      return Promise.resolve();\n    };\n\n  let withCallback = function(description, successCallback, failureCallback) {\n    const promise = setLocalDescription.apply(this, [description]);\n    if (!failureCallback) {\n      return promise;\n    }\n    promise.then(successCallback, failureCallback);\n    return Promise.resolve();\n  };\n  prototype.setLocalDescription = withCallback;\n\n  withCallback = function(description, successCallback, failureCallback) {\n    const promise = setRemoteDescription.apply(this, [description]);\n    if (!failureCallback) {\n      return promise;\n    }\n    promise.then(successCallback, failureCallback);\n    return Promise.resolve();\n  };\n  prototype.setRemoteDescription = withCallback;\n\n  withCallback = function(candidate, successCallback, failureCallback) {\n    const promise = addIceCandidate.apply(this, [candidate]);\n    if (!failureCallback) {\n      return promise;\n    }\n    promise.then(successCallback, failureCallback);\n    return Promise.resolve();\n  };\n  prototype.addIceCandidate = withCallback;\n}\n\nexport function shimGetUserMedia(window) {\n  const navigator = window && window.navigator;\n\n  if (navigator.mediaDevices && navigator.mediaDevices.getUserMedia) {\n    // shim not needed in Safari 12.1\n    const mediaDevices = navigator.mediaDevices;\n    const _getUserMedia = mediaDevices.getUserMedia.bind(mediaDevices);\n    navigator.mediaDevices.getUserMedia = (constraints) => {\n      return _getUserMedia(shimConstraints(constraints));\n    };\n  }\n\n  if (!navigator.getUserMedia && navigator.mediaDevices &&\n    navigator.mediaDevices.getUserMedia) {\n    navigator.getUserMedia = function getUserMedia(constraints, cb, errcb) {\n      navigator.mediaDevices.getUserMedia(constraints)\n        .then(cb, errcb);\n    }.bind(navigator);\n  }\n}\n\nexport function shimConstraints(constraints) {\n  if (constraints && constraints.video !== undefined) {\n    return Object.assign({},\n      constraints,\n      {video: utils.compactObject(constraints.video)}\n    );\n  }\n\n  return constraints;\n}\n\nexport function shimRTCIceServerUrls(window) {\n  if (!window.RTCPeerConnection) {\n    return;\n  }\n  // migrate from non-spec RTCIceServer.url to RTCIceServer.urls\n  const OrigPeerConnection = window.RTCPeerConnection;\n  window.RTCPeerConnection =\n    function RTCPeerConnection(pcConfig, pcConstraints) {\n      if (pcConfig && pcConfig.iceServers) {\n        const newIceServers = [];\n        for (let i = 0; i < pcConfig.iceServers.length; i++) {\n          let server = pcConfig.iceServers[i];\n          if (server.urls === undefined && server.url) {\n            utils.deprecated('RTCIceServer.url', 'RTCIceServer.urls');\n            server = JSON.parse(JSON.stringify(server));\n            server.urls = server.url;\n            delete server.url;\n            newIceServers.push(server);\n          } else {\n            newIceServers.push(pcConfig.iceServers[i]);\n          }\n        }\n        pcConfig.iceServers = newIceServers;\n      }\n      return new OrigPeerConnection(pcConfig, pcConstraints);\n    };\n  window.RTCPeerConnection.prototype = OrigPeerConnection.prototype;\n  // wrap static methods. Currently just generateCertificate.\n  if ('generateCertificate' in OrigPeerConnection) {\n    Object.defineProperty(window.RTCPeerConnection, 'generateCertificate', {\n      get() {\n        return OrigPeerConnection.generateCertificate;\n      }\n    });\n  }\n}\n\nexport function shimTrackEventTransceiver(window) {\n  // Add event.transceiver member over deprecated event.receiver\n  if (typeof window === 'object' && window.RTCTrackEvent &&\n      'receiver' in window.RTCTrackEvent.prototype &&\n      !('transceiver' in window.RTCTrackEvent.prototype)) {\n    Object.defineProperty(window.RTCTrackEvent.prototype, 'transceiver', {\n      get() {\n        return {receiver: this.receiver};\n      }\n    });\n  }\n}\n\nexport function shimCreateOfferLegacy(window) {\n  const origCreateOffer = window.RTCPeerConnection.prototype.createOffer;\n  window.RTCPeerConnection.prototype.createOffer =\n    function createOffer(offerOptions) {\n      if (offerOptions) {\n        if (typeof offerOptions.offerToReceiveAudio !== 'undefined') {\n          // support bit values\n          offerOptions.offerToReceiveAudio =\n            !!offerOptions.offerToReceiveAudio;\n        }\n        const audioTransceiver = this.getTransceivers().find(transceiver =>\n          transceiver.receiver.track.kind === 'audio');\n        if (offerOptions.offerToReceiveAudio === false && audioTransceiver) {\n          if (audioTransceiver.direction === 'sendrecv') {\n            if (audioTransceiver.setDirection) {\n              audioTransceiver.setDirection('sendonly');\n            } else {\n              audioTransceiver.direction = 'sendonly';\n            }\n          } else if (audioTransceiver.direction === 'recvonly') {\n            if (audioTransceiver.setDirection) {\n              audioTransceiver.setDirection('inactive');\n            } else {\n              audioTransceiver.direction = 'inactive';\n            }\n          }\n        } else if (offerOptions.offerToReceiveAudio === true &&\n            !audioTransceiver) {\n          this.addTransceiver('audio', {direction: 'recvonly'});\n        }\n\n        if (typeof offerOptions.offerToReceiveVideo !== 'undefined') {\n          // support bit values\n          offerOptions.offerToReceiveVideo =\n            !!offerOptions.offerToReceiveVideo;\n        }\n        const videoTransceiver = this.getTransceivers().find(transceiver =>\n          transceiver.receiver.track.kind === 'video');\n        if (offerOptions.offerToReceiveVideo === false && videoTransceiver) {\n          if (videoTransceiver.direction === 'sendrecv') {\n            if (videoTransceiver.setDirection) {\n              videoTransceiver.setDirection('sendonly');\n            } else {\n              videoTransceiver.direction = 'sendonly';\n            }\n          } else if (videoTransceiver.direction === 'recvonly') {\n            if (videoTransceiver.setDirection) {\n              videoTransceiver.setDirection('inactive');\n            } else {\n              videoTransceiver.direction = 'inactive';\n            }\n          }\n        } else if (offerOptions.offerToReceiveVideo === true &&\n            !videoTransceiver) {\n          this.addTransceiver('video', {direction: 'recvonly'});\n        }\n      }\n      return origCreateOffer.apply(this, arguments);\n    };\n}\n\nexport function shimAudioContext(window) {\n  if (typeof window !== 'object' || window.AudioContext) {\n    return;\n  }\n  window.AudioContext = window.webkitAudioContext;\n}\n\n","/*\n *  Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.\n *\n *  Use of this source code is governed by a BSD-style license\n *  that can be found in the LICENSE file in the root of the source\n *  tree.\n */\n/* eslint-env node */\n'use strict';\n\nlet logDisabled_ = true;\nlet deprecationWarnings_ = true;\n\n/**\n * Extract browser version out of the provided user agent string.\n *\n * @param {!string} uastring userAgent string.\n * @param {!string} expr Regular expression used as match criteria.\n * @param {!number} pos position in the version string to be returned.\n * @return {!number} browser version.\n */\nexport function extractVersion(uastring, expr, pos) {\n  const match = uastring.match(expr);\n  return match && match.length >= pos && parseInt(match[pos], 10);\n}\n\n// Wraps the peerconnection event eventNameToWrap in a function\n// which returns the modified event object (or false to prevent\n// the event).\nexport function wrapPeerConnectionEvent(window, eventNameToWrap, wrapper) {\n  if (!window.RTCPeerConnection) {\n    return;\n  }\n  const proto = window.RTCPeerConnection.prototype;\n  const nativeAddEventListener = proto.addEventListener;\n  proto.addEventListener = function(nativeEventName, cb) {\n    if (nativeEventName !== eventNameToWrap) {\n      return nativeAddEventListener.apply(this, arguments);\n    }\n    const wrappedCallback = (e) => {\n      const modifiedEvent = wrapper(e);\n      if (modifiedEvent) {\n        if (cb.handleEvent) {\n          cb.handleEvent(modifiedEvent);\n        } else {\n          cb(modifiedEvent);\n        }\n      }\n    };\n    this._eventMap = this._eventMap || {};\n    if (!this._eventMap[eventNameToWrap]) {\n      this._eventMap[eventNameToWrap] = new Map();\n    }\n    this._eventMap[eventNameToWrap].set(cb, wrappedCallback);\n    return nativeAddEventListener.apply(this, [nativeEventName,\n      wrappedCallback]);\n  };\n\n  const nativeRemoveEventListener = proto.removeEventListener;\n  proto.removeEventListener = function(nativeEventName, cb) {\n    if (nativeEventName !== eventNameToWrap || !this._eventMap\n        || !this._eventMap[eventNameToWrap]) {\n      return nativeRemoveEventListener.apply(this, arguments);\n    }\n    if (!this._eventMap[eventNameToWrap].has(cb)) {\n      return nativeRemoveEventListener.apply(this, arguments);\n    }\n    const unwrappedCb = this._eventMap[eventNameToWrap].get(cb);\n    this._eventMap[eventNameToWrap].delete(cb);\n    if (this._eventMap[eventNameToWrap].size === 0) {\n      delete this._eventMap[eventNameToWrap];\n    }\n    if (Object.keys(this._eventMap).length === 0) {\n      delete this._eventMap;\n    }\n    return nativeRemoveEventListener.apply(this, [nativeEventName,\n      unwrappedCb]);\n  };\n\n  Object.defineProperty(proto, 'on' + eventNameToWrap, {\n    get() {\n      return this['_on' + eventNameToWrap];\n    },\n    set(cb) {\n      if (this['_on' + eventNameToWrap]) {\n        this.removeEventListener(eventNameToWrap,\n          this['_on' + eventNameToWrap]);\n        delete this['_on' + eventNameToWrap];\n      }\n      if (cb) {\n        this.addEventListener(eventNameToWrap,\n          this['_on' + eventNameToWrap] = cb);\n      }\n    },\n    enumerable: true,\n    configurable: true\n  });\n}\n\nexport function disableLog(bool) {\n  if (typeof bool !== 'boolean') {\n    return new Error('Argument type: ' + typeof bool +\n        '. Please use a boolean.');\n  }\n  logDisabled_ = bool;\n  return (bool) ? 'adapter.js logging disabled' :\n    'adapter.js logging enabled';\n}\n\n/**\n * Disable or enable deprecation warnings\n * @param {!boolean} bool set to true to disable warnings.\n */\nexport function disableWarnings(bool) {\n  if (typeof bool !== 'boolean') {\n    return new Error('Argument type: ' + typeof bool +\n        '. Please use a boolean.');\n  }\n  deprecationWarnings_ = !bool;\n  return 'adapter.js deprecation warnings ' + (bool ? 'disabled' : 'enabled');\n}\n\nexport function log() {\n  if (typeof window === 'object') {\n    if (logDisabled_) {\n      return;\n    }\n    if (typeof console !== 'undefined' && typeof console.log === 'function') {\n      console.log.apply(console, arguments);\n    }\n  }\n}\n\n/**\n * Shows a deprecation warning suggesting the modern and spec-compatible API.\n */\nexport function deprecated(oldMethod, newMethod) {\n  if (!deprecationWarnings_) {\n    return;\n  }\n  console.warn(oldMethod + ' is deprecated, please use ' + newMethod +\n      ' instead.');\n}\n\n/**\n * Browser detector.\n *\n * @return {object} result containing browser and version\n *     properties.\n */\nexport function detectBrowser(window) {\n  // Returned result object.\n  const result = {browser: null, version: null};\n\n  // Fail early if it's not a browser\n  if (typeof window === 'undefined' || !window.navigator ||\n      !window.navigator.userAgent) {\n    result.browser = 'Not a browser.';\n    return result;\n  }\n\n  const {navigator} = window;\n\n  if (navigator.mozGetUserMedia) { // Firefox.\n    result.browser = 'firefox';\n    result.version = extractVersion(navigator.userAgent,\n      /Firefox\\/(\\d+)\\./, 1);\n  } else if (navigator.webkitGetUserMedia ||\n      (window.isSecureContext === false && window.webkitRTCPeerConnection)) {\n    // Chrome, Chromium, Webview, Opera.\n    // Version matches Chrome/WebRTC version.\n    // Chrome 74 removed webkitGetUserMedia on http as well so we need the\n    // more complicated fallback to webkitRTCPeerConnection.\n    result.browser = 'chrome';\n    result.version = extractVersion(navigator.userAgent,\n      /Chrom(e|ium)\\/(\\d+)\\./, 2);\n  } else if (window.RTCPeerConnection &&\n      navigator.userAgent.match(/AppleWebKit\\/(\\d+)\\./)) { // Safari.\n    result.browser = 'safari';\n    result.version = extractVersion(navigator.userAgent,\n      /AppleWebKit\\/(\\d+)\\./, 1);\n    result.supportsUnifiedPlan = window.RTCRtpTransceiver &&\n        'currentDirection' in window.RTCRtpTransceiver.prototype;\n  } else { // Default fallthrough: not supported.\n    result.browser = 'Not a supported browser.';\n    return result;\n  }\n\n  return result;\n}\n\n/**\n * Checks if something is an object.\n *\n * @param {*} val The something you want to check.\n * @return true if val is an object, false otherwise.\n */\nfunction isObject(val) {\n  return Object.prototype.toString.call(val) === '[object Object]';\n}\n\n/**\n * Remove all empty objects and undefined values\n * from a nested object -- an enhanced and vanilla version\n * of Lodash's `compact`.\n */\nexport function compactObject(data) {\n  if (!isObject(data)) {\n    return data;\n  }\n\n  return Object.keys(data).reduce(function(accumulator, key) {\n    const isObj = isObject(data[key]);\n    const value = isObj ? compactObject(data[key]) : data[key];\n    const isEmptyObject = isObj && !Object.keys(value).length;\n    if (value === undefined || isEmptyObject) {\n      return accumulator;\n    }\n    return Object.assign(accumulator, {[key]: value});\n  }, {});\n}\n\n/* iterates the stats graph recursively. */\nexport function walkStats(stats, base, resultSet) {\n  if (!base || resultSet.has(base.id)) {\n    return;\n  }\n  resultSet.set(base.id, base);\n  Object.keys(base).forEach(name => {\n    if (name.endsWith('Id')) {\n      walkStats(stats, stats.get(base[name]), resultSet);\n    } else if (name.endsWith('Ids')) {\n      base[name].forEach(id => {\n        walkStats(stats, stats.get(id), resultSet);\n      });\n    }\n  });\n}\n\n/* filter getStats for a sender/receiver track. */\nexport function filterStats(result, track, outbound) {\n  const streamStatsType = outbound ? 'outbound-rtp' : 'inbound-rtp';\n  const filteredResult = new Map();\n  if (track === null) {\n    return filteredResult;\n  }\n  const trackStats = [];\n  result.forEach(value => {\n    if (value.type === 'track' &&\n        value.trackIdentifier === track.id) {\n      trackStats.push(value);\n    }\n  });\n  trackStats.forEach(trackStat => {\n    result.forEach(stats => {\n      if (stats.type === streamStatsType && stats.trackId === trackStat.id) {\n        walkStats(result, stats, filteredResult);\n      }\n    });\n  });\n  return filteredResult;\n}\n\n","/*\n * (C) Copyright 2017-2022 OpenVidu (https://openvidu.io)\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n *   http://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n *\n */\n\n// taken from here:\n// https://github.com/OpenVidu/openvidu/blob/master/openvidu-browser/src/OpenViduInternal/WebRtcPeer/WebRtcPeer.ts\n// and monkey-patched\nconst OmUtil = require('../main/omutils');\n\nconst freeice = require('freeice');\n\nconst ExceptionEventName = {\n\t/**\n\t * The [ICE connection state](https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection/iceConnectionState)\n\t * of an [RTCPeerConnection](https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection) reached `failed` status.\n\t *\n\t * This is a terminal error that won't have any kind of possible recovery. If the client is still connected to OpenVidu Server,\n\t * then an automatic reconnection process of the media stream is immediately performed. If the ICE connection has broken due to\n\t * a total network drop, then no automatic reconnection process will be possible.\n\t *\n\t * {@link ExceptionEvent} objects with this {@link ExceptionEvent.name} will have as {@link ExceptionEvent.origin} property a {@link Stream} object.\n\t */\n\t ICE_CONNECTION_FAILED: 'ICE_CONNECTION_FAILED',\n\n\t/**\n\t * The [ICE connection state](https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection/iceConnectionState)\n\t * of an [RTCPeerConnection](https://developer.mozilla.org/en-US/docs/Web/API/RTCPeerConnection) reached `disconnected` status.\n\t *\n\t * This is not a terminal error, and it is possible for the ICE connection to be reconnected. If the client is still connected to\n\t * OpenVidu Server and after certain timeout the ICE connection has not reached a success or terminal status, then an automatic\n\t * reconnection process of the media stream is performed. If the ICE connection has broken due to a total network drop, then no\n\t * automatic reconnection process will be possible.\n\t *\n\t * You can customize the timeout for the reconnection attempt with property {@link OpenViduAdvancedConfiguration.iceConnectionDisconnectedExceptionTimeout},\n\t * which by default is 4000 milliseconds.\n\t *\n\t * {@link ExceptionEvent} objects with this {@link ExceptionEvent.name} will have as {@link ExceptionEvent.origin} property a {@link Stream} object.\n\t */\n\t ICE_CONNECTION_DISCONNECTED: 'ICE_CONNECTION_DISCONNECTED',\n};\n\nclass WebRtcPeer {\n\tconstructor(configuration) {\n\t\tthis.remoteCandidatesQueue = [];\n\t\tthis.localCandidatesQueue = [];\n\t\tthis.iceCandidateList = [];\n\t\tthis.candidategatheringdone = false;\n\n\t\t// Same as WebRtcPeerConfiguration but without optional fields.\n\t\tthis.configuration = {\n\t\t\t...configuration,\n\t\t\ticeServers: !!configuration.iceServers && configuration.iceServers.length > 0 ? configuration.iceServers : freeice(),\n\t\t\tmediaStream: configuration.mediaStream !== undefined ? configuration.mediaStream : null,\n\t\t\tmode: !!configuration.mode ? configuration.mode : 'sendrecv',\n\t\t\tid: !!configuration.id ? configuration.id : this.generateUniqueId()\n\t\t};\n\t\t// prettier-ignore\n\t\tOmUtil.log(`[WebRtcPeer] configuration:\\n${JSON.stringify(this.configuration, null, 2)}`);\n\n\t\tthis.pc = new RTCPeerConnection({ iceServers: this.configuration.iceServers });\n\n\t\tthis._iceCandidateListener = (event) => {\n\t\t\tif (event.candidate !== null) {\n\t\t\t\t// `RTCPeerConnectionIceEvent.candidate` is supposed to be an RTCIceCandidate:\n\t\t\t\t// https://w3c.github.io/webrtc-pc/#dom-rtcpeerconnectioniceevent-candidate\n\t\t\t\t//\n\t\t\t\t// But in practice, it is actually an RTCIceCandidateInit that can be used to\n\t\t\t\t// obtain a proper candidate, using the RTCIceCandidate constructor:\n\t\t\t\t// https://w3c.github.io/webrtc-pc/#dom-rtcicecandidate-constructor\n\t\t\t\tconst candidateInit = event.candidate;\n\t\t\t\tconst iceCandidate = new RTCIceCandidate(candidateInit);\n\n\t\t\t\tthis.configuration.onIceCandidate(iceCandidate);\n\t\t\t\tif (iceCandidate.candidate !== '') {\n\t\t\t\t\tthis.localCandidatesQueue.push(iceCandidate);\n\t\t\t\t}\n\t\t\t}\n\t\t};\n\t\tthis.pc.addEventListener('icecandidate', this._iceCandidateListener);\n\n\t\tthis._signalingStateChangeListener = async () => {\n\t\t\tif (this.pc.signalingState === 'stable') {\n\t\t\t\t// SDP Offer/Answer finished. Add stored remote candidates.\n\t\t\t\twhile (this.iceCandidateList.length > 0) {\n\t\t\t\t\tlet candidate = this.iceCandidateList.shift();\n\t\t\t\t\ttry {\n\t\t\t\t\t\tawait this.pc.addIceCandidate(candidate);\n\t\t\t\t\t} catch (error) {\n\t\t\t\t\t\tconsole.error('Error when calling RTCPeerConnection#addIceCandidate for RTCPeerConnection ' + this.getId(), error);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t};\n\t\tthis.pc.addEventListener('signalingstatechange', this._signalingStateChangeListener);\n\t\tif (this.configuration.onConnectionStateChange) {\n\t\t\tthis.pc.addEventListener('connectionstatechange', this.configuration.onConnectionStateChange);\n\t\t}\n\t}\n\n\tgetId() {\n\t\treturn this.configuration.id;\n\t}\n\n\t/**\n\t * This method frees the resources used by WebRtcPeer\n\t */\n\tdispose() {\n\t\tOmUtil.log('Disposing WebRtcPeer');\n\t\tif (this.pc) {\n\t\t\tif (this.pc.signalingState === 'closed') {\n\t\t\t\treturn;\n\t\t\t}\n\t\t\tthis.pc.removeEventListener('icecandidate', this._iceCandidateListener);\n\t\t\tthis._iceCandidateListener = undefined;\n\t\t\tthis.pc.removeEventListener('signalingstatechange', this._signalingStateChangeListener);\n\t\t\tthis._signalingStateChangeListener = undefined;\n\t\t\tif (this._iceConnectionStateChangeListener) {\n\t\t\t\tthis.pc.removeEventListener('iceconnectionstatechange', this._iceConnectionStateChangeListener);\n\t\t\t\tthis._iceConnectionStateChangeListener = undefined;\n\t\t\t}\n\t\t\tif (this.configuration.onConnectionStateChange) {\n\t\t\t\tthis.pc.removeEventListener('connectionstatechange', this.configuration.onConnectionStateChange);\n\t\t\t}\n\t\t\tthis.configuration = {};\n\t\t\tthis.pc.close();\n\t\t\tthis.remoteCandidatesQueue = [];\n\t\t\tthis.localCandidatesQueue = [];\n\t\t}\n\t}\n\n\t/**\n\t * Creates an SDP offer from the local RTCPeerConnection to send to the other peer.\n\t * Only if the negotiation was initiated by this peer.\n\t */\n\tasync createOffer() {\n\t\t// TODO: Delete this conditional when all supported browsers are\n\t\t// modern enough to implement the Transceiver methods.\n\t\tif (!('addTransceiver' in this.pc)) {\n\t\t\tOmUtil.error(\n\t\t\t\t'[createOffer] Method RTCPeerConnection.addTransceiver() is NOT available; using LEGACY offerToReceive{Audio,Video}'\n\t\t\t);\n\t\t\treturn this.createOfferLegacy();\n\t\t} else {\n\t\t\tOmUtil.log('[createOffer] Method RTCPeerConnection.addTransceiver() is available; using it');\n\t\t}\n\n\t\t// Spec doc: https://w3c.github.io/webrtc-pc/#dom-rtcpeerconnection-addtransceiver\n\n\t\tif (this.configuration.mode !== 'recvonly') {\n\t\t\t// To send media, assume that all desired media tracks have been\n\t\t\t// already added by higher level code to our MediaStream.\n\n\t\t\tif (!this.configuration.mediaStream) {\n\t\t\t\tthrow new Error(\n\t\t\t\t\t`[WebRtcPeer.createOffer] Direction is '${this.configuration.mode}', but no stream was configured to be sent`\n\t\t\t\t);\n\t\t\t}\n\n\t\t\tfor (const track of this.configuration.mediaStream.getTracks()) {\n\t\t\t\tconst tcInit = {\n\t\t\t\t\tdirection: this.configuration.mode,\n\t\t\t\t\tstreams: [this.configuration.mediaStream]\n\t\t\t\t};\n\n\t\t\t\tif (track.kind === 'video' && this.configuration.simulcast) {\n\t\t\t\t\t// Check if the requested size is enough to ask for 3 layers.\n\t\t\t\t\tconst trackSettings = track.getSettings();\n\t\t\t\t\tconst trackConsts = track.getConstraints();\n\n\t\t\t\t\tconst trackWidth = typeof(trackSettings.width) === 'object' ? trackConsts.width.ideal : trackConsts.width || 0;\n\t\t\t\t\tconst trackHeight = typeof(trackSettings.height) === 'object' ? trackConsts.height.ideal : trackConsts.height || 0;\n\t\t\t\t\tOmUtil.info(`[createOffer] Video track dimensions: ${trackWidth}x${trackHeight}`);\n\n\t\t\t\t\tconst trackPixels = trackWidth * trackHeight;\n\t\t\t\t\tlet maxLayers = 0;\n\t\t\t\t\tif (trackPixels >= 960 * 540) {\n\t\t\t\t\t\tmaxLayers = 3;\n\t\t\t\t\t} else if (trackPixels >= 480 * 270) {\n\t\t\t\t\t\tmaxLayers = 2;\n\t\t\t\t\t} else {\n\t\t\t\t\t\tmaxLayers = 1;\n\t\t\t\t\t}\n\n\t\t\t\t\ttcInit.sendEncodings = [];\n\t\t\t\t\tfor (let l = 0; l < maxLayers; l++) {\n\t\t\t\t\t\tconst layerDiv = 2 ** (maxLayers - l - 1);\n\n\t\t\t\t\t\tconst encoding = {\n\t\t\t\t\t\t\trid: 'rdiv' + layerDiv.toString(),\n\n\t\t\t\t\t\t\t// @ts-ignore -- Property missing from DOM types.\n\t\t\t\t\t\t\tscalabilityMode: 'L1T1'\n\t\t\t\t\t\t};\n\n\t\t\t\t\t\tif (['detail', 'text'].includes(track.contentHint)) {\n\t\t\t\t\t\t\t// Prioritize best resolution, for maximum picture detail.\n\t\t\t\t\t\t\tencoding.scaleResolutionDownBy = 1.0;\n\n\t\t\t\t\t\t\t// @ts-ignore -- Property missing from DOM types.\n\t\t\t\t\t\t\tencoding.maxFramerate = Math.floor(30 / layerDiv);\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tencoding.scaleResolutionDownBy = layerDiv;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\ttcInit.sendEncodings.push(encoding);\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tconst tc = this.pc.addTransceiver(track, tcInit);\n\n\t\t\t\tif (track.kind === 'video') {\n\t\t\t\t\tlet sendParams = tc.sender.getParameters();\n\t\t\t\t\tlet needSetParams = false;\n\n\t\t\t\t\tif (sendParams.degradationPreference && !sendParams.degradationPreference.length) {\n\t\t\t\t\t\t// degradationPreference for video: \"balanced\", \"maintain-framerate\", \"maintain-resolution\".\n\t\t\t\t\t\t// https://www.w3.org/TR/2018/CR-webrtc-20180927/#dom-rtcdegradationpreference\n\t\t\t\t\t\tif (['detail', 'text'].includes(track.contentHint)) {\n\t\t\t\t\t\t\tsendParams.degradationPreference = 'maintain-resolution';\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tsendParams.degradationPreference = 'balanced';\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tOmUtil.info(`[createOffer] Video sender Degradation Preference set: ${sendParams.degradationPreference}`);\n\n\t\t\t\t\t\t// Firefox implements degradationPreference on each individual encoding!\n\t\t\t\t\t\t// (set it on every element of the sendParams.encodings array)\n\n\t\t\t\t\t\tneedSetParams = true;\n\t\t\t\t\t}\n\n\t\t\t\t\t// Check that the simulcast encodings were applied.\n\t\t\t\t\t// Firefox doesn't implement `RTCRtpTransceiverInit.sendEncodings`\n\t\t\t\t\t// so the only way to enable simulcast is with `RTCRtpSender.setParameters()`.\n\t\t\t\t\t//\n\t\t\t\t\t// This next block can be deleted when Firefox fixes bug #1396918:\n\t\t\t\t\t// https://bugzilla.mozilla.org/show_bug.cgi?id=1396918\n\t\t\t\t\t//\n\t\t\t\t\t// NOTE: This is done in a way that is compatible with all browsers, to save on\n\t\t\t\t\t// browser-conditional code. The idea comes from WebRTC Adapter.js:\n\t\t\t\t\t// * https://github.com/webrtcHacks/adapter/issues/998\n\t\t\t\t\t// * https://github.com/webrtcHacks/adapter/blob/v7.7.0/src/js/firefox/firefox_shim.js#L231-L255\n\t\t\t\t\tif (this.configuration.simulcast) {\n\t\t\t\t\t\tif (sendParams.encodings.length !== tcInit.sendEncodings.length) {\n\t\t\t\t\t\t\tsendParams.encodings = tcInit.sendEncodings;\n\n\t\t\t\t\t\t\tneedSetParams = true;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tif (needSetParams) {\n\t\t\t\t\t\tOmUtil.log(`[createOffer] Setting new RTCRtpSendParameters to video sender`);\n\t\t\t\t\t\ttry {\n\t\t\t\t\t\t\tawait tc.sender.setParameters(sendParams);\n\t\t\t\t\t\t} catch (error) {\n\t\t\t\t\t\t\tlet message = `[WebRtcPeer.createOffer] Cannot set RTCRtpSendParameters to video sender`;\n\t\t\t\t\t\t\tif (error instanceof Error) {\n\t\t\t\t\t\t\t\tmessage += `: ${error.message}`;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tthrow new Error(message);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t} else {\n\t\t\t// To just receive media, create new recvonly transceivers.\n\t\t\tfor (const kind of ['audio', 'video']) {\n\t\t\t\t// Check if the media kind should be used.\n\t\t\t\tif (!this.configuration.mediaConstraints[kind]) {\n\t\t\t\t\tcontinue;\n\t\t\t\t}\n\n\t\t\t\tthis.configuration.mediaStream = new MediaStream();\n\t\t\t\tthis.pc.addTransceiver(kind, {\n\t\t\t\t\tdirection: this.configuration.mode,\n\t\t\t\t\tstreams: [this.configuration.mediaStream]\n\t\t\t\t});\n\t\t\t}\n\t\t}\n\n\t\tlet sdpOffer;\n\t\ttry {\n\t\t\tsdpOffer = await this.pc.createOffer();\n\t\t} catch (error) {\n\t\t\tlet message = `[WebRtcPeer.createOffer] Browser failed creating an SDP Offer`;\n\t\t\tif (error instanceof Error) {\n\t\t\t\tmessage += `: ${error.message}`;\n\t\t\t}\n\t\t\tthrow new Error(message);\n\t\t}\n\n\t\treturn sdpOffer;\n\t}\n\n\t/**\n\t * Creates an SDP answer from the local RTCPeerConnection to send to the other peer\n\t * Only if the negotiation was initiated by the other peer\n\t */\n\tcreateAnswer() {\n\t\treturn new Promise((resolve, reject) => {\n\t\t\t// TODO: Delete this conditional when all supported browsers are\n\t\t\t// modern enough to implement the Transceiver methods.\n\t\t\tif ('getTransceivers' in this.pc) {\n\t\t\t\tOmUtil.log('[createAnswer] Method RTCPeerConnection.getTransceivers() is available; using it');\n\n\t\t\t\t// Ensure that the PeerConnection already contains one Transceiver\n\t\t\t\t// for each kind of media.\n\t\t\t\t// The Transceivers should have been already created internally by\n\t\t\t\t// the PC itself, when `pc.setRemoteDescription(sdpOffer)` was called.\n\n\t\t\t\tfor (const kind of ['audio', 'video']) {\n\t\t\t\t\t// Check if the media kind should be used.\n\t\t\t\t\tif (!this.configuration.mediaConstraints[kind]) {\n\t\t\t\t\t\tcontinue;\n\t\t\t\t\t}\n\n\t\t\t\t\tlet tc = this.pc.getTransceivers().find((tc) => tc.receiver.track.kind === kind);\n\n\t\t\t\t\tif (tc) {\n\t\t\t\t\t\t// Enforce our desired direction.\n\t\t\t\t\t\ttc.direction = this.configuration.mode;\n\t\t\t\t\t} else {\n\t\t\t\t\t\treturn reject(new Error(`${kind} requested, but no transceiver was created from remote description`));\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tthis.pc\n\t\t\t\t\t.createAnswer()\n\t\t\t\t\t.then((sdpAnswer) => resolve(sdpAnswer))\n\t\t\t\t\t.catch((error) => reject(error));\n\t\t\t} else {\n\t\t\t\t// TODO: Delete else branch when all supported browsers are\n\t\t\t\t// modern enough to implement the Transceiver methods\n\n\t\t\t\tlet offerAudio,\n\t\t\t\t\tofferVideo = true;\n\t\t\t\tif (!!this.configuration.mediaConstraints) {\n\t\t\t\t\tofferAudio =\n\t\t\t\t\t\ttypeof this.configuration.mediaConstraints.audio === 'boolean' ? this.configuration.mediaConstraints.audio : true;\n\t\t\t\t\tofferVideo =\n\t\t\t\t\t\ttypeof this.configuration.mediaConstraints.video === 'boolean' ? this.configuration.mediaConstraints.video : true;\n\t\t\t\t\tconst constraints = {\n\t\t\t\t\t\tofferToReceiveAudio: offerAudio,\n\t\t\t\t\t\tofferToReceiveVideo: offerVideo\n\t\t\t\t\t};\n\t\t\t\t\t(this.pc).createAnswer(constraints)\n\t\t\t\t\t\t.then((sdpAnswer) => resolve(sdpAnswer))\n\t\t\t\t\t\t.catch((error) => reject(error));\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// else, there is nothing to do; the legacy createAnswer() options do\n\t\t\t// not offer any control over which tracks are included in the answer.\n\t\t});\n\t}\n\n\t/**\n\t * This peer initiated negotiation. Step 1/4 of SDP offer-answer protocol\n\t */\n\tprocessLocalOffer(offer) {\n\t\treturn new Promise((resolve, reject) => {\n\t\t\tthis.pc\n\t\t\t\t.setLocalDescription(offer)\n\t\t\t\t.then(() => {\n\t\t\t\t\tconst localDescription = this.pc.localDescription;\n\t\t\t\t\tif (!!localDescription) {\n\t\t\t\t\t\tOmUtil.log('Local description set', localDescription.sdp);\n\t\t\t\t\t\treturn resolve();\n\t\t\t\t\t} else {\n\t\t\t\t\t\treturn reject('Local description is not defined');\n\t\t\t\t\t}\n\t\t\t\t})\n\t\t\t\t.catch((error) => reject(error));\n\t\t});\n\t}\n\n\t/**\n\t * Other peer initiated negotiation. Step 2/4 of SDP offer-answer protocol\n\t */\n\tprocessRemoteOffer(sdpOffer) {\n\t\treturn new Promise((resolve, reject) => {\n\t\t\tconst offer = {\n\t\t\t\ttype: 'offer',\n\t\t\t\tsdp: sdpOffer\n\t\t\t};\n\t\t\tOmUtil.log('SDP offer received, setting remote description', offer);\n\n\t\t\tif (this.pc.signalingState === 'closed') {\n\t\t\t\treturn reject('RTCPeerConnection is closed when trying to set remote description');\n\t\t\t}\n\t\t\tthis.setRemoteDescription(offer)\n\t\t\t\t.then(() => resolve())\n\t\t\t\t.catch((error) => reject(error));\n\t\t});\n\t}\n\n\t/**\n\t * Other peer initiated negotiation. Step 3/4 of SDP offer-answer protocol\n\t */\n\tprocessLocalAnswer(answer) {\n\t\treturn new Promise((resolve, reject) => {\n\t\t\tOmUtil.log('SDP answer created, setting local description');\n\t\t\tif (this.pc.signalingState === 'closed') {\n\t\t\t\treturn reject('RTCPeerConnection is closed when trying to set local description');\n\t\t\t}\n\t\t\tthis.pc\n\t\t\t\t.setLocalDescription(answer)\n\t\t\t\t.then(() => resolve())\n\t\t\t\t.catch((error) => reject(error));\n\t\t});\n\t}\n\n\t/**\n\t * This peer initiated negotiation. Step 4/4 of SDP offer-answer protocol\n\t */\n\tprocessRemoteAnswer(sdpAnswer) {\n\t\treturn new Promise((resolve, reject) => {\n\t\t\tconst answer = {\n\t\t\t\ttype: 'answer',\n\t\t\t\tsdp: sdpAnswer\n\t\t\t};\n\t\t\tOmUtil.log('SDP answer received, setting remote description');\n\n\t\t\tif (this.pc.signalingState === 'closed') {\n\t\t\t\treturn reject('RTCPeerConnection is closed when trying to set remote description');\n\t\t\t}\n\t\t\tthis.setRemoteDescription(answer)\n\t\t\t\t.then(() => {\n\t\t\t\t\tresolve();\n\t\t\t\t})\n\t\t\t\t.catch((error) => reject(error));\n\t\t});\n\t}\n\n\t/**\n\t * @hidden\n\t */\n\tasync setRemoteDescription(sdp) {\n\t\treturn this.pc.setRemoteDescription(sdp);\n\t}\n\n\t/**\n\t * Callback function invoked when an ICE candidate is received\n\t */\n\taddIceCandidate(iceCandidate) {\n\t\treturn new Promise((resolve, reject) => {\n\t\t\tOmUtil.log('Remote ICE candidate received', iceCandidate);\n\t\t\tthis.remoteCandidatesQueue.push(iceCandidate);\n\t\t\tswitch (this.pc.signalingState) {\n\t\t\t\tcase 'closed':\n\t\t\t\t\treject(new Error('PeerConnection object is closed'));\n\t\t\t\t\tbreak;\n\t\t\t\tcase 'stable':\n\t\t\t\t\tif (!!this.pc.remoteDescription) {\n\t\t\t\t\t\tthis.pc\n\t\t\t\t\t\t\t.addIceCandidate(iceCandidate)\n\t\t\t\t\t\t\t.then(() => resolve())\n\t\t\t\t\t\t\t.catch((error) => reject(error));\n\t\t\t\t\t} else {\n\t\t\t\t\t\tthis.iceCandidateList.push(iceCandidate);\n\t\t\t\t\t\tresolve();\n\t\t\t\t\t}\n\t\t\t\t\tbreak;\n\t\t\t\tdefault:\n\t\t\t\t\tthis.iceCandidateList.push(iceCandidate);\n\t\t\t\t\tresolve();\n\t\t\t}\n\t\t});\n\t}\n\n\taddIceConnectionStateChangeListener(otherId) {\n\t\tif (!this._iceConnectionStateChangeListener) {\n\t\t\tthis._iceConnectionStateChangeListener = () => {\n\t\t\t\tconst iceConnectionState = this.pc.iceConnectionState;\n\t\t\t\tswitch (iceConnectionState) {\n\t\t\t\t\tcase 'disconnected':\n\t\t\t\t\t\t// Possible network disconnection\n\t\t\t\t\t\tconst msg1 =\n\t\t\t\t\t\t\t'IceConnectionState of RTCPeerConnection ' +\n\t\t\t\t\t\t\tthis.configuration.id +\n\t\t\t\t\t\t\t' (' +\n\t\t\t\t\t\t\totherId +\n\t\t\t\t\t\t\t') change to \"disconnected\". Possible network disconnection';\n\t\t\t\t\t\tconsole.warn(msg1);\n\t\t\t\t\t\tthis.configuration.onIceConnectionStateException(ExceptionEventName.ICE_CONNECTION_DISCONNECTED, msg1);\n\t\t\t\t\t\tbreak;\n\t\t\t\t\tcase 'failed':\n\t\t\t\t\t\tconst msg2 = 'IceConnectionState of RTCPeerConnection ' + this.configuration.id + ' (' + otherId + ') to \"failed\"';\n\t\t\t\t\t\tconsole.error(msg2);\n\t\t\t\t\t\tthis.configuration.onIceConnectionStateException(ExceptionEventName.ICE_CONNECTION_FAILED, msg2);\n\t\t\t\t\t\tbreak;\n\t\t\t\t\tcase 'closed':\n\t\t\t\t\t\tOmUtil.log(\n\t\t\t\t\t\t\t'IceConnectionState of RTCPeerConnection ' + this.configuration.id + ' (' + otherId + ') change to \"closed\"'\n\t\t\t\t\t\t);\n\t\t\t\t\t\tbreak;\n\t\t\t\t\tcase 'new':\n\t\t\t\t\t\tOmUtil.log('IceConnectionState of RTCPeerConnection ' + this.configuration.id + ' (' + otherId + ') change to \"new\"');\n\t\t\t\t\t\tbreak;\n\t\t\t\t\tcase 'checking':\n\t\t\t\t\t\tOmUtil.log(\n\t\t\t\t\t\t\t'IceConnectionState of RTCPeerConnection ' + this.configuration.id + ' (' + otherId + ') change to \"checking\"'\n\t\t\t\t\t\t);\n\t\t\t\t\t\tbreak;\n\t\t\t\t\tcase 'connected':\n\t\t\t\t\t\tOmUtil.log(\n\t\t\t\t\t\t\t'IceConnectionState of RTCPeerConnection ' + this.configuration.id + ' (' + otherId + ') change to \"connected\"'\n\t\t\t\t\t\t);\n\t\t\t\t\t\tbreak;\n\t\t\t\t\tcase 'completed':\n\t\t\t\t\t\tOmUtil.log(\n\t\t\t\t\t\t\t'IceConnectionState of RTCPeerConnection ' + this.configuration.id + ' (' + otherId + ') change to \"completed\"'\n\t\t\t\t\t\t);\n\t\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t};\n\t\t}\n\t\tthis.pc.addEventListener('iceconnectionstatechange', this._iceConnectionStateChangeListener);\n\t}\n\n\t/**\n\t * @hidden\n\t */\n\tgenerateUniqueId() {\n\t\treturn crypto.randomUUID();\n\t}\n\n\tget stream() {\n\t\treturn this.pc.getLocalStreams()[0] || this.pc.getRemoteStreams()[0];\n\t}\n\n\t// LEGACY code\n\tdeprecatedPeerConnectionTrackApi() {\n\t\tfor (const track of this.configuration.mediaStream.getTracks()) {\n\t\t\tthis.pc.addTrack(track, this.configuration.mediaStream);\n\t\t}\n\t}\n\n\t// DEPRECATED LEGACY METHOD: Old WebRTC versions don't implement\n\t// Transceivers, and instead depend on the deprecated\n\t// \"offerToReceiveAudio\" and \"offerToReceiveVideo\".\n\tcreateOfferLegacy() {\n\t\tif (!!this.configuration.mediaStream) {\n\t\t\tthis.deprecatedPeerConnectionTrackApi();\n\t\t}\n\n\t\tconst hasAudio = this.configuration.mediaConstraints.audio;\n\t\tconst hasVideo = this.configuration.mediaConstraints.video;\n\n\t\tconst options = {\n\t\t\tofferToReceiveAudio: this.configuration.mode !== 'sendonly' && hasAudio,\n\t\t\tofferToReceiveVideo: this.configuration.mode !== 'sendonly' && hasVideo\n\t\t};\n\n\t\tOmUtil.log('[createOfferLegacy] RTCPeerConnection.createOffer() options:', JSON.stringify(options));\n\n\t\treturn this.pc.createOffer(options);\n\t}\n}\n\nclass WebRtcPeerRecvonly extends WebRtcPeer {\n\tconstructor(configuration) {\n\t\tconfiguration.mode = 'recvonly';\n\t\tsuper(configuration);\n\t}\n};\n\nclass WebRtcPeerSendonly extends WebRtcPeer {\n\tconstructor(configuration) {\n\t\tconfiguration.mode = 'sendonly';\n\t\tsuper(configuration);\n\t}\n};\n\nclass WebRtcPeerSendrecv extends WebRtcPeer {\n\tconstructor(configuration) {\n\t\tconfiguration.mode = 'sendrecv';\n\t\tsuper(configuration);\n\t}\n};\n\nmodule.exports = {\n\tRecvonly: WebRtcPeerRecvonly,\n\tSendonly: WebRtcPeerSendonly\n};\n","/* Licensed under the Apache License, Version 2.0 (the \"License\") http://www.apache.org/licenses/LICENSE-2.0 */\nconst VideoUtil = require('./video-util');\nconst RingBuffer = require('./ring-buffer');\n\nmodule.exports = class MicLevel {\n\tconstructor() {\n\t\tlet ctx, mic, analyser\n\t\t\t, cnvs, canvasCtx, WIDTH, HEIGHT, horiz\n\t\t\t, vol = .0, vals = new RingBuffer(100);\n\n\t\tthis.meterStream = (stream, _cnvs, _micActivity, _error, connectAudio) => {\n\t\t\tif (!stream || stream.getAudioTracks().length < 1) {\n\t\t\t\treturn;\n\t\t\t}\n\t\t\ttry {\n\t\t\t\tconst AudioCtx = window.AudioContext || window.webkitAudioContext;\n\t\t\t\tif (!AudioCtx) {\n\t\t\t\t\t_error(\"AudioContext is inaccessible\");\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t\tctx = new AudioCtx();\n\t\t\t\tanalyser = ctx.createAnalyser();\n\t\t\t\tmic = ctx.createMediaStreamSource(stream);\n\t\t\t\tmic.connect(analyser);\n\t\t\t\tif (connectAudio) {\n\t\t\t\t\tanalyser.connect(ctx.destination);\n\t\t\t\t}\n\t\t\t\tthis.meter(analyser, _cnvs, _micActivity, _error);\n\t\t\t} catch (err) {\n\t\t\t\t_error(err);\n\t\t\t}\n\t\t};\n\t\tthis.setCanvas = (_cnvs) => {\n\t\t\tcnvs = _cnvs;\n\t\t\tconst canvas = cnvs[0];\n\t\t\tcanvasCtx = canvas.getContext('2d');\n\t\t\tWIDTH = canvas.width;\n\t\t\tHEIGHT = canvas.height;\n\t\t\thoriz = cnvs.data('orientation') === 'horizontal';\n\t\t};\n\t\tthis.meter = (_analyser, _cnvs, _micActivity, _error) => {\n\t\t\tthis.setCanvas(_cnvs);\n\t\t\ttry {\n\t\t\t\tanalyser = _analyser;\n\t\t\t\tanalyser.minDecibels = -90;\n\t\t\t\tanalyser.maxDecibels = -10;\n\t\t\t\tanalyser.fftSize = 256;\n\t\t\t\tconst color = $('body').css('--level-color')\n\t\t\t\t\t, al = analyser.frequencyBinCount\n\t\t\t\t\t, arr = new Uint8Array(al);\n\t\t\t\tfunction update() {\n\t\t\t\t\tcanvasCtx.clearRect(0, 0, WIDTH, HEIGHT);\n\t\t\t\t\tif (!!analyser && cnvs.length > 0) {\n\t\t\t\t\t\tif (cnvs.is(':visible')) {\n\t\t\t\t\t\t\tanalyser.getByteFrequencyData(arr);\n\t\t\t\t\t\t\tlet favg = 0.0;\n\t\t\t\t\t\t\tfor (let i = 0; i < al; ++i) {\n\t\t\t\t\t\t\t\tfavg += arr[i] * arr[i];\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tvol = Math.sqrt(favg / al);\n\t\t\t\t\t\t\tvals.push(vol);\n\t\t\t\t\t\t\tconst min = vals.min();\n\t\t\t\t\t\t\t_micActivity(vol > min + 5); // magic number\n\t\t\t\t\t\t\tcanvasCtx.fillStyle = color;\n\t\t\t\t\t\t\tif (horiz) {\n\t\t\t\t\t\t\t\tcanvasCtx.fillRect(0, 0, WIDTH * vol / 100, HEIGHT);\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tconst h = HEIGHT * vol / 100;\n\t\t\t\t\t\t\t\tcanvasCtx.fillRect(0, HEIGHT - h, WIDTH, h);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t\trequestAnimationFrame(update);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\tupdate();\n\t\t\t} catch (err) {\n\t\t\t\t_error(err);\n\t\t\t}\n\t\t};\n\t\tthis.dispose = () => {\n\t\t\tif (!!ctx) {\n\t\t\t\tVideoUtil.cleanStream(mic.mediaStream);\n\t\t\t\tVideoUtil.disconnect(mic);\n\t\t\t\tVideoUtil.disconnect(ctx.destination);\n\t\t\t\tctx.close();\n\t\t\t\tctx = null;\n\t\t\t}\n\t\t\tif (!!analyser) {\n\t\t\t\tVideoUtil.disconnect(analyser);\n\t\t\t\tanalyser = null;\n\t\t\t}\n\t\t};\n\t}\n};\n","/* Licensed under the Apache License, Version 2.0 (the \"License\") http://www.apache.org/licenses/LICENSE-2.0 */\nmodule.exports = class RingBuffer {\n\tconstructor(length) {\n\t\tconst buffer = [];\n\t\tlet pos = 0;\n\n\t\tthis.get = (key) => {\n\t\t\treturn buffer[key];\n\t\t};\n\t\tthis.push = (item) => {\n\t\t\tbuffer[pos] = item;\n\t\t\tpos = (pos + 1) % length;\n\t\t};\n\t\tthis.min = () => {\n\t\t\treturn Math.min.apply(Math, buffer);\n\t\t}\n\t}\n};\n","/* Licensed under the Apache License, Version 2.0 (the \"License\") http://www.apache.org/licenses/LICENSE-2.0 */\nconst OmUtil = require('../main/omutils');\nconst Settings = require('../main/settings');\n\nconst MicLevel = require('./mic-level');\nconst VideoUtil = require('./video-util');\nconst WebRtcPeer = require('./WebRtcPeer');\n\nconst DEV_AUDIO = 'audioinput'\n\t, DEV_VIDEO = 'videoinput'\n\t, MsgBase = {type: 'kurento', mode: 'test'};\nlet vs, lm, s, cam, mic, res, o, rtcPeer, timer\n\t, vidScroll, vid, recBtn, playBtn, recAllowed = false\n\t, level;\n\nfunction _load() {\n\ts = Settings.load();\n\tif (!s.video) {\n\t\tconst _res = $('#video-settings .cam-resolution option:selected').data();\n\t\ts.video = {\n\t\t\tcam: 0\n\t\t\t, mic: 0\n\t\t\t, width: _res.width\n\t\t\t, height: _res.height\n\t\t};\n\t}\n\tif (!s.fixed) {\n\t\ts.fixed = {\n\t\t\tenabled: false\n\t\t\t, width: 120\n\t\t\t, height: 90\n\t\t};\n\t}\n\treturn s;\n}\nfunction _save() {\n\tSettings.save(s);\n\tOmUtil.sendMessage({\n\t\ttype: 'av'\n\t\t, area: 'room'\n\t\t, settings: s\n\t});\n}\nfunction _clear(_ms) {\n\tconst ms = _ms || (vid && vid.length === 1 ? vid[0].srcObject : null);\n\tVideoUtil.cleanStream(ms);\n\tif (vid && vid.length === 1) {\n\t\tvid[0].srcObject = null;\n\t}\n\tVideoUtil.cleanPeer(rtcPeer);\n\tif (!!lm) {\n\t\tlm.hide();\n\t}\n\tif (!!level) {\n\t\tlevel.dispose();\n\t\tlevel = null;\n\t}\n}\nfunction _close() {\n\t_clear();\n\tWicket.Event.unsubscribe('/websocket/message', _onWsMessage);\n}\nfunction _onIceCandidate(candidate) {\n\tOmUtil.log('Local candidate' + JSON.stringify(candidate));\n\tOmUtil.sendMessage({\n\t\tid : 'iceCandidate'\n\t\t, candidate: candidate\n\t}, MsgBase);\n}\nfunction _init(options) {\n\to = JSON.parse(JSON.stringify(options));\n\tif (!!o.infoMsg) {\n\t\tOmUtil.alert('info', o.infoMsg, 0);\n\t}\n\tvs = $('#video-settings');\n\tlm = vs.find('.level-meter');\n\tcam = vs.find('select.cam').change(function() {\n\t\t_readValues();\n\t});\n\tmic = vs.find('select.mic').change(function() {\n\t\t_readValues();\n\t});\n\tres = vs.find('select.cam-resolution').change(function() {\n\t\t_readValues();\n\t});\n\tvidScroll = vs.find('.vid-block .video-conainer');\n\ttimer = vs.find('.timer');\n\tvid = vidScroll.find('video');\n\trecBtn = vs.find('.rec-start')\n\t\t.click(function() {\n\t\t\trecBtn.prop('disabled', true);\n\t\t\t_setEnabled(true);\n\t\t\tOmUtil.sendMessage({\n\t\t\t\tid : 'wannaRecord'\n\t\t\t}, MsgBase);\n\t\t});\n\tplayBtn = vs.find('.play')\n\t\t.click(function() {\n\t\t\trecBtn.prop('disabled', true);\n\t\t\t_setEnabled(true);\n\t\t\tOmUtil.sendMessage({\n\t\t\t\tid : 'wannaPlay'\n\t\t\t}, MsgBase);\n\t\t});\n\tvs.find('.btn-save').off().click(function() {\n\t\t_save();\n\t\t_close();\n\t\tvs.modal(\"hide\");\n\t});\n\tvs.find('.btn-cancel').off().click(function() {\n\t\t_close();\n\t\tvs.modal(\"hide\");\n\t});\n\tvs.off().on('hidden.bs.modal', function () {\n\t\t_close();\n\t});\n\to.width = 300;\n\to.height = 200;\n\to.mode = 'settings';\n\to.rights = (o.rights || []).join();\n\tdelete o.keycode;\n\tvs.find('.modal-body input, .modal-body button').prop('disabled', true);\n\tconst rr = vs.find('.cam-resolution').parents('.sett-row');\n\tif (!o.interview) {\n\t\trr.show();\n\t} else {\n\t\trr.hide();\n\t}\n\t_load();\n\t_save(); // trigger settings update\n}\nfunction _updateRec() {\n\trecBtn.prop('disabled', !recAllowed || (s.video.cam < 0 && s.video.mic < 0));\n}\nfunction _setCntsDimensions(cnts) {\n\tif (VideoUtil.isSafari()) {\n\t\tlet width = s.video.width;\n\t\t//valid widths are 320, 640, 1280\n\t\t[320, 640, 1280].some(function(w) {\n\t\t\tif (width < w + 1) {\n\t\t\t\twidth = w;\n\t\t\t\treturn true;\n\t\t\t}\n\t\t\treturn false;\n\t\t});\n\t\tcnts.video.width = width < 1281 ? width : 1280;\n\t} else {\n\t\tcnts.video.width = o.interview ? 320 : s.video.width;\n\t\tcnts.video.height = o.interview ? 260 : s.video.height;\n\t}\n}\n//each bool OR https://developer.mozilla.org/en-US/docs/Web/API/MediaTrackConstraints\n// min/ideal/max/exact/mandatory can also be used\nfunction _constraints(sd, callback) {\n\t_getDevConstraints(function(devCnts) {\n\t\tconst cnts = {\n\t\t\tvideoEnabled: VideoUtil.hasCam(sd)\n\t\t\t, audioEnabled: VideoUtil.hasMic(sd)\n\t\t};\n\t\tif (devCnts.video && false === o.audioOnly && s.video.cam > -1) {\n\t\t\tcnts.video = {\n\t\t\t\tframeRate: o.camera.fps\n\t\t\t};\n\t\t\t_setCntsDimensions(cnts)\n\t\t\tif (!!s.video.camDevice) {\n\t\t\t\tcnts.video.deviceId = {\n\t\t\t\t\tideal: s.video.camDevice\n\t\t\t\t};\n\t\t\t} else {\n\t\t\t\tcnts.video.facingMode = {\n\t\t\t\t\tideal: 'user'\n\t\t\t\t}\n\t\t\t}\n\t\t} else {\n\t\t\tcnts.video = false;\n\t\t}\n\t\tif (devCnts.audio && s.video.mic > -1) {\n\t\t\tcnts.audio = {\n\t\t\t\tsampleRate: o.microphone.rate\n\t\t\t\t, echoCancellation: o.microphone.echo\n\t\t\t\t, noiseSuppression: o.microphone.noise\n\t\t\t};\n\t\t\tif (!!s.video.micDevice) {\n\t\t\t\tcnts.audio.deviceId = {\n\t\t\t\t\tideal: s.video.micDevice\n\t\t\t\t};\n\t\t\t}\n\t\t} else {\n\t\t\tcnts.audio = false;\n\t\t}\n\t\tcallback(cnts);\n\t});\n}\nfunction _readValues(msg, func) {\n\tconst v = cam.find('option:selected')\n\t\t, m = mic.find('option:selected')\n\t\t, o = res.find('option:selected').data();\n\ts.video.cam = 1 * cam.val();\n\ts.video.camDevice = v.data('device-id');\n\ts.video.mic = 1 * mic.val();\n\ts.video.micDevice = m.data('device-id');\n\ts.video.width = o.width;\n\ts.video.height = o.height;\n\tvid.width(o.width).height(o.height);\n\tvidScroll.scrollLeft(Math.max(0, s.video.width / 2 - 150))\n\t\t.scrollTop(Math.max(0, s.video.height / 2 - 110));\n\t_clear();\n\t_constraints(null, function(cnts) {\n\t\tif (cnts.video !== false || cnts.audio !== false) {\n\t\t\tconst options = VideoUtil.addIceServers({\n\t\t\t\tmediaConstraints: cnts\n\t\t\t\t, onIceCandidate: _onIceCandidate\n\t\t\t}, msg);\n\t\t\tnavigator.mediaDevices.getUserMedia(cnts)\n\t\t\t\t.then(stream => {\n\t\t\t\t\tVideoUtil.playSrc(vid[0], stream, true);\n\t\t\t\t\toptions.mediaStream = stream;\n\n\t\t\t\t\trtcPeer = new WebRtcPeer.Sendonly(options);\n\t\t\t\t\tif (cnts.audio) {\n\t\t\t\t\t\tlm.show();\n\t\t\t\t\t\tlevel = new MicLevel();\n\t\t\t\t\t\tlevel.meterStream(stream, lm, function(){}, OmUtil.error, false);\n\t\t\t\t\t} else {\n\t\t\t\t\t\tlm.hide();\n\t\t\t\t\t}\n\t\t\t\t\treturn rtcPeer.createOffer();\n\t\t\t\t})\n\t\t\t\t.then(sdpOffer => {\n\t\t\t\t\trtcPeer.processLocalOffer(sdpOffer);\n\t\t\t\t\tif (typeof(func) === 'function') {\n\t\t\t\t\t\tfunc(sdpOffer.sdp, cnts);\n\t\t\t\t\t} else {\n\t\t\t\t\t\t_allowRec(true);\n\t\t\t\t\t}\n\t\t\t\t}).catch(_ => OmUtil.error('Error generating the offer'));\n\t\t}\n\t\tif (!msg) {\n\t\t\t_updateRec();\n\t\t}\n\t});\n}\n\nfunction _allowRec(allow) {\n\trecAllowed = allow;\n\t_updateRec();\n}\nfunction _setLoading(el) {\n\tel.find('option').remove();\n\tel.append(OmUtil.tmpl('#settings-option-loading'));\n}\nfunction _setDisabled(els) {\n\tels.forEach(function(el) {\n\t\tel.find('option').remove();\n\t\tel.append(OmUtil.tmpl('#settings-option-disabled'));\n\t});\n}\nfunction _setSelectedDevice(dev, devIdx) {\n\tlet o = dev.find('option[value=\"' + devIdx + '\"]');\n\tif (o.length === 0 && devIdx !== -1) {\n\t\to = dev.find('option[value=\"0\"]');\n\t}\n\to.prop('selected', true);\n}\nfunction _getDevConstraints(callback) {\n\tconst devCnts = {audio: false, video: false, devices: []};\n\tif (window.isSecureContext === false) {\n\t\tOmUtil.error($('#settings-https-required').text());\n\t\treturn;\n\t}\n\tif (!navigator.mediaDevices || !navigator.mediaDevices.enumerateDevices) {\n\t\tOmUtil.error('enumerateDevices() not supported.');\n\t\treturn;\n\t}\n\tnavigator.mediaDevices.enumerateDevices()\n\t\t.then(devices => devices.forEach(device => {\n\t\t\t\tif (DEV_AUDIO === device.kind || DEV_VIDEO === device.kind) {\n\t\t\t\t\tdevCnts.devices.push({\n\t\t\t\t\t\tkind: device.kind\n\t\t\t\t\t\t, label: device.label || (device.kind + ' ' + devCnts.devices.length)\n\t\t\t\t\t\t, deviceId: device.deviceId\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t\tif (DEV_AUDIO === device.kind) {\n\t\t\t\t\tdevCnts.audio = true;\n\t\t\t\t} else if (DEV_VIDEO === device.kind) {\n\t\t\t\t\tdevCnts.video = true;\n\t\t\t\t}\n\t\t\t}))\n\t\t.catch(() => OmUtil.error('Unable to get the list of multimedia devices'))\n\t\t.finally(() => callback(devCnts));\n}\nfunction _initDevices() {\n\tif (window.isSecureContext === false) {\n\t\tOmUtil.error($('#settings-https-required').text());\n\t\treturn;\n\t}\n\tif (!navigator.mediaDevices || !navigator.mediaDevices.enumerateDevices) {\n\t\tOmUtil.error('enumerateDevices() not supported.');\n\t\treturn;\n\t}\n\t_setLoading(cam);\n\t_setLoading(mic);\n\t_getDevConstraints(function(devCnts) {\n\t\tif (!devCnts.audio && !devCnts.video) {\n\t\t\t_setDisabled([cam, mic]);\n\t\t\treturn;\n\t\t}\n\t\tnavigator.mediaDevices.getUserMedia(devCnts)\n\t\t\t.then(stream => {\n\t\t\t\tconst devices = navigator.mediaDevices.enumerateDevices()\n\t\t\t\t\t.catch(function(err) {\n\t\t\t\t\t\tthrow err;\n\t\t\t\t\t})\n\t\t\t\t\t.finally(() => _clear(stream));\n\t\t\t\treturn devices || devCnts.devices;\n\t\t\t})\n\t\t\t.catch(function() {\n\t\t\t\treturn devCnts.devices;\n\t\t\t})\n\t\t\t.then(devices => {\n\t\t\t\tlet cCount = 0, mCount = 0;\n\t\t\t\t_load();\n\t\t\t\t_setDisabled([cam, mic]);\n\t\t\t\tdevices.forEach(device => {\n\t\t\t\t\tif (DEV_AUDIO === device.kind) {\n\t\t\t\t\t\tconst o = $('<option></option>').attr('value', mCount).text(device.label)\n\t\t\t\t\t\t\t.data('device-id', device.deviceId);\n\t\t\t\t\t\tmic.append(o);\n\t\t\t\t\t\tmCount++;\n\t\t\t\t\t} else if (DEV_VIDEO === device.kind) {\n\t\t\t\t\t\tconst o = $('<option></option>').attr('value', cCount).text(device.label)\n\t\t\t\t\t\t\t.data('device-id', device.deviceId);\n\t\t\t\t\t\tcam.append(o);\n\t\t\t\t\t\tcCount++;\n\t\t\t\t\t}\n\t\t\t\t});\n\t\t\t\t_setSelectedDevice(cam, s.video.cam);\n\t\t\t\t_setSelectedDevice(mic, s.video.mic);\n\t\t\t\tres.find('option').each(function() {\n\t\t\t\t\tconst o = $(this).data();\n\t\t\t\t\tif (o.width === s.video.width && o.height === s.video.height) {\n\t\t\t\t\t\t$(this).prop('selected', true);\n\t\t\t\t\t\treturn false;\n\t\t\t\t\t}\n\t\t\t\t});\n\t\t\t\t_readValues();\n\t\t\t})\n\t\t\t.catch(function(err) {\n\t\t\t\t_setDisabled([cam, mic]);\n\t\t\t\tOmUtil.error(err);\n\t\t\t});\n\t});\n}\nfunction _open() {\n\tWicket.Event.subscribe('/websocket/message', _onWsMessage);\n\trecAllowed = false;\n\ttimer.hide();\n\tplayBtn.prop('disabled', true);\n\tvs.modal('show');\n\t_load();\n\t_initDevices();\n}\nfunction _setEnabled(enabled) {\n\tplayBtn.prop('disabled', enabled);\n\tcam.prop('disabled', enabled);\n\tmic.prop('disabled', enabled);\n\tres.prop('disabled', enabled);\n}\nfunction _onStop() {\n\t_updateRec();\n\t_setEnabled(false);\n}\nfunction _onKMessage(m) {\n\tOmUtil.info('Received message: ', m);\n\tswitch (m.id) {\n\t\tcase 'canRecord':\n\t\t\t_readValues(m, function(_offerSdp, cnts) {\n\t\t\t\tOmUtil.info('Invoking SDP offer callback function');\n\t\t\t\tOmUtil.sendMessage({\n\t\t\t\t\tid : 'record'\n\t\t\t\t\t, sdpOffer: _offerSdp\n\t\t\t\t\t, video: cnts.video !== false\n\t\t\t\t\t, audio: cnts.audio !== false\n\t\t\t\t}, MsgBase);\n\t\t\t});\n\t\t\tbreak;\n\t\tcase 'canPlay': {\n\t\t\tconst options = VideoUtil.addIceServers({\n\t\t\t\tmediaConstraints: {audio: true, video: true}\n\t\t\t\t, onIceCandidate: _onIceCandidate\n\t\t\t}, m);\n\t\t\t_clear();\n\t\t\trtcPeer = new WebRtcPeer.Recvonly(options);\n\t\t\trtcPeer.createOffer()\n\t\t\t\t.then(sdpOffer => {\n\t\t\t\t\trtcPeer.processLocalOffer(sdpOffer);\n\t\t\t\t\tOmUtil.sendMessage({\n\t\t\t\t\t\tid : 'play'\n\t\t\t\t\t\t, sdpOffer: sdpOffer.sdp\n\t\t\t\t\t}, MsgBase);\n\t\t\t\t})\n\t\t\t\t.catch(_ => OmUtil.error('Error generating the offer'));\n\t\t\t}\n\t\t\tbreak;\n\t\tcase 'playResponse':\n\t\t\tOmUtil.log('Play SDP answer received from server. Processing ...');\n\n\t\t\trtcPeer.processRemoteAnswer(m.sdpAnswer)\n\t\t\t\t.then(() => {\n\t\t\t\t\tconst stream = rtcPeer.stream;\n\t\t\t\t\tif (stream) {\n\t\t\t\t\t\tVideoUtil.playSrc(vid[0], stream, false);\n\t\t\t\t\t\tlm.show();\n\t\t\t\t\t\tlevel = new MicLevel();\n\t\t\t\t\t\tlevel.meterStream(stream, lm, function(){}, OmUtil.error, true);\n\t\t\t\t\t};\n\t\t\t\t})\n\t\t\t\t.catch(error => OmUtil.error(error));\n\t\t\tbreak;\n\t\tcase 'startResponse':\n\t\t\tOmUtil.log('SDP answer received from server. Processing ...');\n\t\t\trtcPeer.processRemoteAnswer(m.sdpAnswer)\n\t\t\t\t.catch(error => OmUtil.error(error));\n\t\t\tbreak;\n\t\tcase 'iceCandidate':\n\t\t\trtcPeer.addIceCandidate(m.candidate)\n\t\t\t\t.catch(error => OmUtil.error('Error adding candidate: ' + error));\n\t\t\tbreak;\n\t\tcase 'recording':\n\t\t\ttimer.show().find('.time').text(m.time);\n\t\t\tbreak;\n\t\tcase 'recStopped':\n\t\t\ttimer.hide();\n\t\t\t_onStop();\n\t\t\tbreak;\n\t\tcase 'playStopped':\n\t\t\t_onStop();\n\t\t\t_readValues();\n\t\t\tbreak;\n\t\tdefault:\n\t\t\t// no-op\n\t}\n}\nfunction _onWsMessage(jqEvent, msg) {\n\ttry {\n\t\tif (msg instanceof Blob) {\n\t\t\treturn; //ping\n\t\t}\n\t\tconst m = JSON.parse(msg);\n\t\tif (m && 'kurento' === m.type) {\n\t\t\tif ('test' === m.mode) {\n\t\t\t\t_onKMessage(m);\n\t\t\t}\n\t\t\tswitch (m.id) {\n\t\t\t\tcase 'error':\n\t\t\t\t\tOmUtil.error(m.message);\n\t\t\t\t\tbreak;\n\t\t\t\tdefault:\n\t\t\t\t\t//no-op\n\t\t\t}\n\t\t}\n\t} catch (err) {\n\t\tOmUtil.error(err);\n\t}\n}\n\nmodule.exports = {\n\tinit: _init\n\t, open: _open\n\t, close: function() {\n\t\t_close();\n\t\tvs && vs.modal('hide');\n\t}\n\t, load: _load\n\t, save: _save\n\t, constraints: _constraints\n};\n","/* Licensed under the Apache License, Version 2.0 (the \"License\") http://www.apache.org/licenses/LICENSE-2.0 */\nconst OmUtil = require('../main/omutils');\nconst Settings = require('../main/settings');\n\nconst UAParser = require('ua-parser-js')\n\t, ua = (typeof window !== 'undefined' && window.navigator) ? window.navigator.userAgent : ''\n\t, parser = new UAParser(ua)\n\t, browser = parser.getBrowser();\n\nconst WB_AREA_SEL = '.room-block .wb-block';\nconst WBA_WB_SEL = '.room-block .wb-block .wb-tab-content';\nconst VIDWIN_SEL = '.video.user-video';\nconst VID_SEL = '.video-container[id!=user-video]';\nconst CAM_ACTIVITY = 'VIDEO';\nconst MIC_ACTIVITY = 'AUDIO';\nconst SCREEN_ACTIVITY = 'SCREEN';\nconst REC_ACTIVITY = 'RECORD';\n\nfunction _isSafari() {\n\treturn browser.name === 'Safari';\n}\nfunction _isChrome() {\n\treturn browser.name === 'Chrome' || browser.name === 'Chromium';\n}\nfunction _isEdge() {\n\treturn browser.name === 'Edge' && \"MSGestureEvent\" in window;\n}\nfunction _isEdgeChromium() {\n\treturn browser.name === 'Edge' && !(\"MSGestureEvent\" in window);\n}\n\nfunction _getVid(uid) {\n\treturn 'video' + uid;\n}\nfunction _isSharing(sd) {\n\treturn !!sd && 'SCREEN' === sd.type && sd.activities.includes(SCREEN_ACTIVITY);\n}\nfunction _isRecording(sd) {\n\treturn !!sd && 'SCREEN' === sd.type && sd.activities.includes(REC_ACTIVITY);\n}\nfunction _hasActivity(sd, act) {\n\treturn !!sd && sd.activities.includes(act);\n}\nfunction _hasMic(sd) {\n\tif (!sd) {\n\t\treturn true;\n\t}\n\tconst enabled = sd.micEnabled !== false;\n\treturn sd.activities.includes(MIC_ACTIVITY) && enabled;\n}\nfunction _hasCam(sd) {\n\tif (!sd) {\n\t\treturn true;\n\t}\n\tconst enabled = sd.camEnabled !== false;\n\treturn sd.activities.includes(CAM_ACTIVITY) && enabled;\n}\nfunction _hasVideo(sd) {\n\treturn _hasCam(sd) || _isSharing(sd) || _isRecording(sd);\n}\nfunction _getRects(sel, excl) {\n\tconst list = [], elems = $(sel);\n\tfor (let i = 0; i < elems.length; ++i) {\n\t\tif (excl !== $(elems[i]).attr('aria-describedby')) {\n\t\t\tlist.push(_getRect(elems[i]));\n\t\t}\n\t}\n\treturn list;\n}\nfunction _getRect(e) {\n\tconst win = $(e), winoff = win.offset();\n\treturn {left: winoff.left\n\t\t, top: winoff.top\n\t\t, right: winoff.left + win.width()\n\t\t, bottom: winoff.top + win.height()};\n}\nfunction _container() {\n\tconst a = $(WB_AREA_SEL);\n\tconst c = a.find('.wb-area .tabs .wb-tab-content');\n\treturn c.length > 0 ? $(WBA_WB_SEL) : a;\n}\nfunction __processTopToBottom(area, rectNew, list) {\n\tconst offsetX = 20\n\t\t, offsetY = 10;\n\n\tlet minY = area.bottom, posFound;\n\tdo {\n\t\tposFound = true;\n\t\tfor (let i = 0; i < list.length; ++i) {\n\t\t\tconst rect = list[i];\n\t\t\tminY = Math.min(minY, rect.bottom);\n\n\t\t\tif (rectNew.left < rect.right && rectNew.right > rect.left && rectNew.top < rect.bottom && rectNew.bottom > rect.top) {\n\t\t\t\trectNew.left = rect.right + offsetX;\n\t\t\t\tposFound = false;\n\t\t\t}\n\t\t\tif (rectNew.right >= area.right) {\n\t\t\t\trectNew.left = area.left;\n\t\t\t\trectNew.top = Math.max(minY, rectNew.top) + offsetY;\n\t\t\t\tposFound = false;\n\t\t\t}\n\t\t\tif (rectNew.bottom >= area.bottom) {\n\t\t\t\trectNew.top = area.top;\n\t\t\t\tposFound = true;\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\t} while (!posFound);\n\treturn {left: rectNew.left, top: rectNew.top};\n}\nfunction __processEqualsBottomToTop(area, rectNew, list) {\n\tconst offsetX = 20\n\t\t, offsetY = 10;\n\n\trectNew.bottom = area.bottom;\n\tlet minY = area.bottom, posFound;\n\tdo {\n\t\tposFound = true;\n\t\tfor (let i = 0; i < list.length; ++i) {\n\t\t\tconst rect = list[i];\n\t\t\tminY = Math.min(minY, rect.top);\n\n\t\t\tif (rectNew.left < rect.right && rectNew.right > rect.left && rectNew.top < rect.bottom && rectNew.bottom > rect.top) {\n\t\t\t\trectNew.left = rect.right + offsetX;\n\t\t\t\tposFound = false;\n\t\t\t}\n\t\t\tif (rectNew.right >= area.right) {\n\t\t\t\trectNew.left = area.left;\n\t\t\t\trectNew.bottom = Math.min(minY, rectNew.top) - offsetY;\n\t\t\t\tposFound = false;\n\t\t\t}\n\t\t\tif (rectNew.top <= area.top) {\n\t\t\t\trectNew.top = area.top;\n\t\t\t\tposFound = true;\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\t} while (!posFound);\n\treturn {left: rectNew.left, top: rectNew.top};\n}\nfunction _getPos(list, w, h, _processor) {\n\tif (Room.getOptions().interview) {\n\t\treturn {left: 0, top: 0};\n\t}\n\tconst wba = _container()\n\t\t, woffset = wba.offset()\n\t\t, area = {left: woffset.left, top: woffset.top, right: woffset.left + wba.width(), bottom: woffset.top + wba.height()}\n\t\t, rectNew = {\n\t\t\t_left: area.left\n\t\t\t, _top: area.top\n\t\t\t, _right: area.left + w\n\t\t\t, _bottom: area.top + h\n\t\t\t, get left() {\n\t\t\t\treturn this._left;\n\t\t\t}\n\t\t\t, set left(l) {\n\t\t\t\tthis._left = l;\n\t\t\t\tthis._right = l + w;\n\t\t\t}\n\t\t\t, get right() {\n\t\t\t\treturn this._right;\n\t\t\t}\n\t\t\t, get top() {\n\t\t\t\treturn this._top;\n\t\t\t}\n\t\t\t, set top(t) {\n\t\t\t\tthis._top = t;\n\t\t\t\tthis._bottom = t + h;\n\t\t\t}\n\t\t\t, set bottom(b) {\n\t\t\t\tthis._bottom = b;\n\t\t\t\tthis._top = b - h;\n\t\t\t}\n\t\t\t, get bottom() {\n\t\t\t\treturn this._bottom;\n\t\t\t}\n\t\t};\n\tconst processor = _processor || __processTopToBottom;\n\treturn processor(area, rectNew, list);\n}\nfunction _arrange() {\n\tconst list = [];\n\t$(VIDWIN_SEL).each(function() {\n\t\tconst v = $(this);\n\t\tv.css(_getPos(list, v.width(), v.height()));\n\t\tlist.push(_getRect(v));\n\t});\n}\nfunction _arrangeResize(vSettings) {\n\tconst list = []\n\t\t, size = {width: 120, height: 90};\n\tif (vSettings.fixed.enabled) {\n\t\tsize.width = vSettings.fixed.width;\n\t\tsize.height = vSettings.fixed.height;\n\t}\n\n\tfunction __getDialog(_v) {\n\t\treturn $(_v).find('.video-container.ui-dialog-content');\n\t}\n\t$(VIDWIN_SEL).toArray().sort((v1, v2) => {\n\t\tconst c1 = __getDialog(v1).data().stream()\n\t\t\t, c2 = __getDialog(v2).data().stream();\n\t\treturn c2.level - c1.level || c1.user.displayName.localeCompare(c2.user.displayName);\n\t}).forEach(_v => {\n\t\tconst v = $(_v);\n\t\t__getDialog(v)\n\t\t\t.dialog('option', 'width', size.width)\n\t\t\t.dialog('option', 'height', size.height);\n\t\tv.css(_getPos(list, v.width(), v.height(), __processEqualsBottomToTop));\n\t\tlist.push(_getRect(v));\n\t});\n}\nfunction _cleanStream(stream) {\n\tif (!!stream) {\n\t\tstream.getTracks().forEach(track => track.stop());\n\t}\n}\nfunction _cleanPeer(rtcPeer) {\n\tif (!!rtcPeer) {\n\t\ttry {\n\t\t\tconst pc = rtcPeer.pc;\n\t\t\tif (!!pc) {\n\t\t\t\tpc.getSenders().forEach(sender => {\n\t\t\t\t\ttry {\n\t\t\t\t\t\tif (sender.track) {\n\t\t\t\t\t\t\tsender.track.stop();\n\t\t\t\t\t\t}\n\t\t\t\t\t} catch(e) {\n\t\t\t\t\t\tOmUtil.log('Failed to clean sender' + e);\n\t\t\t\t\t}\n\t\t\t\t});\n\t\t\t\tpc.getReceivers().forEach(receiver => {\n\t\t\t\t\ttry {\n\t\t\t\t\t\tif (receiver.track) {\n\t\t\t\t\t\t\treceiver.track.stop();\n\t\t\t\t\t\t}\n\t\t\t\t\t} catch(e) {\n\t\t\t\t\t\tOmUtil.log('Failed to clean receiver' + e);\n\t\t\t\t\t}\n\t\t\t\t});\n\t\t\t}\n\t\t\trtcPeer.dispose();\n\t\t} catch(e) {\n\t\t\t//no-op\n\t\t}\n\t}\n}\nfunction _setPos(v, pos) {\n\tif (v.dialog('instance')) {\n\t\tv.dialog('widget').css(pos);\n\t}\n}\nfunction _askPermission(callback) {\n\tconst perm = $('#ask-permission');\n\t$('.sidebar').confirmation({\n\t\ttitle: perm.attr('title')\n\t\t, placement: Settings.isRtl ? 'right' : 'left'\n\t\t, singleton: true\n\t\t, rootSelector: '.sidebar'\n\t\t, html: true\n\t\t, content: perm.html()\n\t\t, buttons: [{\n\t\t\tclass: 'btn btn-sm btn-warning'\n\t\t\t, label: perm.data('btn-ok')\n\t\t\t, value: perm.data('btn-ok')\n\t\t\t, onClick: function() {\n\t\t\t\tcallback();\n\t\t\t\t$('.sidebar').confirmation('dispose');\n\t\t\t}\n\t\t}]\n\t});\n\t$('.sidebar').confirmation('show');\n}\nfunction _disconnect(node) {\n\ttry {\n\t\tnode.disconnect(); //this one can throw\n\t} catch (e) {\n\t\t//no-op\n\t}\n}\nfunction _sharingSupported() {\n\treturn (browser.name === 'Edge' && browser.major > 16)\n\t\t|| (typeof(navigator.mediaDevices.getDisplayMedia) === 'function'\n\t\t\t&& (browser.name === 'Firefox'\n\t\t\t\t|| browser.name === 'Opera'\n\t\t\t\t|| browser.name === 'Yandex'\n\t\t\t\t|| _isSafari()\n\t\t\t\t|| _isChrome()\n\t\t\t\t|| _isEdgeChromium()\n\t\t\t\t|| (browser.name === 'Mozilla' && browser.major > 4)\n\t\t\t));\n}\nfunction _highlight(el, clazz, count) {\n\tif (!el || el.length < 1 || el.hasClass('disabled') || count < 0) {\n\t\treturn;\n\t}\n\tel.addClass(clazz).delay(2000).queue(function(next) {\n\t\tel.removeClass(clazz).delay(2000).queue(function(next1) {\n\t\t\t_highlight(el, clazz, --count);\n\t\t\tnext1();\n\t\t});\n\t\tnext();\n\t});\n}\nfunction _playSrc(_video, _stream, mute) {\n\tif (_stream && _video) {\n\t\t_video.srcObject = _stream;\n\t\tif (_video.paused) {\n\t\t\t_video.play().then(() => _video.muted = mute).catch(err => {\n\t\t\t\tif ('NotAllowedError' === err.name) {\n\t\t\t\t\t_askPermission(() => _video.play().then(() => _video.muted = mute));\n\t\t\t\t}\n\t\t\t});\n\t\t}\n\t}\n}\n\nmodule.exports = {\n\tVIDWIN_SEL: VIDWIN_SEL\n\t, VID_SEL: VID_SEL\n\t, CAM_ACTIVITY: CAM_ACTIVITY\n\t, MIC_ACTIVITY: MIC_ACTIVITY\n\n\t, getVid: _getVid\n\t, isSharing: _isSharing\n\t, isRecording: _isRecording\n\t, hasMic: _hasMic\n\t, hasCam: _hasCam\n\t, hasVideo: _hasVideo\n\t, hasActivity: _hasActivity\n\t, getRects: _getRects\n\t, getPos: _getPos\n\t, container: _container\n\t, arrange: _arrange\n\t, arrangeResize: _arrangeResize\n\t, cleanStream: _cleanStream\n\t, cleanPeer: _cleanPeer\n\t, addIceServers: function(opts, m) {\n\t\tif (m && m.iceServers && m.iceServers.length > 0) {\n\t\t\topts.iceServers = m.iceServers;\n\t\t}\n\t\treturn opts;\n\t}\n\t, setPos: _setPos\n\t, askPermission: _askPermission\n\t, disconnect: _disconnect\n\t, sharingSupported: _sharingSupported\n\t, highlight: _highlight\n\t, playSrc: _playSrc\n\n\t, browser: browser\n\t, isEdge: _isEdge\n\t, isEdgeChromium: _isEdgeChromium\n\t, isChrome: _isChrome\n\t, isSafari: _isSafari\n};\n","module.exports = OmUtil;","module.exports = Settings;","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","__webpack_require__.amdO = {};","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","/* Licensed under the Apache License, Version 2.0 (the \"License\") http://www.apache.org/licenses/LICENSE-2.0 */\nrequire('webrtc-adapter');\n\nif (window.hasOwnProperty('isSecureContext') === false) {\n\twindow.isSecureContext = window.location.protocol == 'https:' || [\"localhost\", \"127.0.0.1\"].indexOf(window.location.hostname) !== -1;\n}\n\nObject.assign(window, {\n\tVideoUtil: require('./video-util')\n\t, MicLevel: require('./mic-level')\n\t, WebRtcPeer: require('./WebRtcPeer')\n\t, VideoSettings: require('./settings')\n});\n"],"names":[],"sourceRoot":""}
|