!function(e, t) { "object" == typeof exports && "object" == typeof module ? module.exports = t() : "function" == typeof define && define.amd ? define([], t) : "object" == typeof exports ? exports.spstypescriptexample = t() : e.spstypescriptexample = t() }(this, (()=>(()=>{ "use strict"; var e = { d: (t,s)=>{ for (var n in s) e.o(s, n) && !e.o(t, n) && Object.defineProperty(t, n, { enumerable: !0, get: s[n] }) } , o: (e,t)=>Object.prototype.hasOwnProperty.call(e, t), r: e=>{ "undefined" != typeof Symbol && Symbol.toStringTag && Object.defineProperty(e, Symbol.toStringTag, { value: "Module" }), Object.defineProperty(e, "__esModule", { value: !0 }) } } , t = {}; e.r(t), e.d(t, { PixelStreamingApplicationStyles: ()=>d }); var s = { 539: e=>{ const t = { generateIdentifier: function() { return Math.random().toString(36).substring(2, 12) } }; t.localCName = t.generateIdentifier(), t.splitLines = function(e) { return e.trim().split("\n").map((e=>e.trim())) } , t.splitSections = function(e) { return e.split("\nm=").map(((e,t)=>(t > 0 ? "m=" + e : e).trim() + "\r\n")) } , t.getDescription = function(e) { const s = t.splitSections(e); return s && s[0] } , t.getMediaSections = function(e) { const s = t.splitSections(e); return s.shift(), s } , t.matchPrefix = function(e, s) { return t.splitLines(e).filter((e=>0 === e.indexOf(s))) } , t.parseCandidate = function(e) { let t; t = 0 === e.indexOf("a=candidate:") ? e.substring(12).split(" ") : e.substring(10).split(" "); const s = { foundation: t[0], component: { 1: "rtp", 2: "rtcp" }[t[1]] || t[1], protocol: t[2].toLowerCase(), priority: parseInt(t[3], 10), ip: t[4], address: t[4], port: parseInt(t[5], 10), type: t[7] }; for (let e = 8; e < t.length; e += 2) switch (t[e]) { case "raddr": s.relatedAddress = t[e + 1]; break; case "rport": s.relatedPort = parseInt(t[e + 1], 10); break; case "tcptype": s.tcpType = t[e + 1]; break; case "ufrag": s.ufrag = t[e + 1], s.usernameFragment = t[e + 1]; break; default: void 0 === s[t[e]] && (s[t[e]] = t[e + 1]) } return s } , t.writeCandidate = function(e) { const t = []; t.push(e.foundation); const s = e.component; "rtp" === s ? t.push(1) : "rtcp" === s ? t.push(2) : t.push(s), t.push(e.protocol.toUpperCase()), t.push(e.priority), t.push(e.address || e.ip), t.push(e.port); const n = e.type; return t.push("typ"), t.push(n), "host" !== n && e.relatedAddress && e.relatedPort && (t.push("raddr"), t.push(e.relatedAddress), t.push("rport"), t.push(e.relatedPort)), e.tcpType && "tcp" === e.protocol.toLowerCase() && (t.push("tcptype"), t.push(e.tcpType)), (e.usernameFragment || e.ufrag) && (t.push("ufrag"), t.push(e.usernameFragment || e.ufrag)), "candidate:" + t.join(" ") } , t.parseIceOptions = function(e) { return e.substring(14).split(" ") } , t.parseRtpMap = function(e) { let t = e.substring(9).split(" "); const s = { payloadType: parseInt(t.shift(), 10) }; return t = t[0].split("/"), s.name = t[0], s.clockRate = parseInt(t[1], 10), s.channels = 3 === t.length ? parseInt(t[2], 10) : 1, s.numChannels = s.channels, s } , t.writeRtpMap = function(e) { let t = e.payloadType; void 0 !== e.preferredPayloadType && (t = e.preferredPayloadType); const s = e.channels || e.numChannels || 1; return "a=rtpmap:" + t + " " + e.name + "/" + e.clockRate + (1 !== s ? "/" + s : "") + "\r\n" } , t.parseExtmap = function(e) { const t = e.substring(9).split(" "); return { id: parseInt(t[0], 10), direction: t[0].indexOf("/") > 0 ? t[0].split("/")[1] : "sendrecv", uri: t[1], attributes: t.slice(2).join(" ") } } , t.writeExtmap = function(e) { return "a=extmap:" + (e.id || e.preferredId) + (e.direction && "sendrecv" !== e.direction ? "/" + e.direction : "") + " " + e.uri + (e.attributes ? " " + e.attributes : "") + "\r\n" } , t.parseFmtp = function(e) { const t = {}; let s; const n = e.substring(e.indexOf(" ") + 1).split(";"); for (let e = 0; e < n.length; e++) s = n[e].trim().split("="), t[s[0].trim()] = s[1]; return t } , t.writeFmtp = function(e) { let t = "" , s = e.payloadType; if (void 0 !== e.preferredPayloadType && (s = e.preferredPayloadType), e.parameters && Object.keys(e.parameters).length) { const n = []; Object.keys(e.parameters).forEach((t=>{ void 0 !== e.parameters[t] ? n.push(t + "=" + e.parameters[t]) : n.push(t) } )), t += "a=fmtp:" + s + " " + n.join(";") + "\r\n" } return t } , t.parseRtcpFb = function(e) { const t = e.substring(e.indexOf(" ") + 1).split(" "); return { type: t.shift(), parameter: t.join(" ") } } , t.writeRtcpFb = function(e) { let t = "" , s = e.payloadType; return void 0 !== e.preferredPayloadType && (s = e.preferredPayloadType), e.rtcpFeedback && e.rtcpFeedback.length && e.rtcpFeedback.forEach((e=>{ t += "a=rtcp-fb:" + s + " " + e.type + (e.parameter && e.parameter.length ? " " + e.parameter : "") + "\r\n" } )), t } , t.parseSsrcMedia = function(e) { const t = e.indexOf(" ") , s = { ssrc: parseInt(e.substring(7, t), 10) } , n = e.indexOf(":", t); return n > -1 ? (s.attribute = e.substring(t + 1, n), s.value = e.substring(n + 1)) : s.attribute = e.substring(t + 1), s } , t.parseSsrcGroup = function(e) { const t = e.substring(13).split(" "); return { semantics: t.shift(), ssrcs: t.map((e=>parseInt(e, 10))) } } , t.getMid = function(e) { const s = t.matchPrefix(e, "a=mid:")[0]; if (s) return s.substring(6) } , t.parseFingerprint = function(e) { const t = e.substring(14).split(" "); return { algorithm: t[0].toLowerCase(), value: t[1].toUpperCase() } } , t.getDtlsParameters = function(e, s) { return { role: "auto", fingerprints: t.matchPrefix(e + s, "a=fingerprint:").map(t.parseFingerprint) } } , t.writeDtlsParameters = function(e, t) { let s = "a=setup:" + t + "\r\n"; return e.fingerprints.forEach((e=>{ s += "a=fingerprint:" + e.algorithm + " " + e.value + "\r\n" } )), s } , t.parseCryptoLine = function(e) { const t = e.substring(9).split(" "); return { tag: parseInt(t[0], 10), cryptoSuite: t[1], keyParams: t[2], sessionParams: t.slice(3) } } , t.writeCryptoLine = function(e) { return "a=crypto:" + e.tag + " " + e.cryptoSuite + " " + ("object" == typeof e.keyParams ? t.writeCryptoKeyParams(e.keyParams) : e.keyParams) + (e.sessionParams ? " " + e.sessionParams.join(" ") : "") + "\r\n" } , t.parseCryptoKeyParams = function(e) { if (0 !== e.indexOf("inline:")) return null; const t = e.substring(7).split("|"); return { keyMethod: "inline", keySalt: t[0], lifeTime: t[1], mkiValue: t[2] ? t[2].split(":")[0] : void 0, mkiLength: t[2] ? t[2].split(":")[1] : void 0 } } , t.writeCryptoKeyParams = function(e) { return e.keyMethod + ":" + e.keySalt + (e.lifeTime ? "|" + e.lifeTime : "") + (e.mkiValue && e.mkiLength ? "|" + e.mkiValue + ":" + e.mkiLength : "") } , t.getCryptoParameters = function(e, s) { return t.matchPrefix(e + s, "a=crypto:").map(t.parseCryptoLine) } , t.getIceParameters = function(e, s) { const n = t.matchPrefix(e + s, "a=ice-ufrag:")[0] , i = t.matchPrefix(e + s, "a=ice-pwd:")[0]; return n && i ? { usernameFragment: n.substring(12), password: i.substring(10) } : null } , t.writeIceParameters = function(e) { let t = "a=ice-ufrag:" + e.usernameFragment + "\r\na=ice-pwd:" + e.password + "\r\n"; return e.iceLite && (t += "a=ice-lite\r\n"), t } , t.parseRtpParameters = function(e) { const s = { codecs: [], headerExtensions: [], fecMechanisms: [], rtcp: [] } , n = t.splitLines(e)[0].split(" "); s.profile = n[2]; for (let i = 3; i < n.length; i++) { const r = n[i] , o = t.matchPrefix(e, "a=rtpmap:" + r + " ")[0]; if (o) { const n = t.parseRtpMap(o) , i = t.matchPrefix(e, "a=fmtp:" + r + " "); switch (n.parameters = i.length ? t.parseFmtp(i[0]) : {}, n.rtcpFeedback = t.matchPrefix(e, "a=rtcp-fb:" + r + " ").map(t.parseRtcpFb), s.codecs.push(n), n.name.toUpperCase()) { case "RED": case "ULPFEC": s.fecMechanisms.push(n.name.toUpperCase()) } } } t.matchPrefix(e, "a=extmap:").forEach((e=>{ s.headerExtensions.push(t.parseExtmap(e)) } )); const i = t.matchPrefix(e, "a=rtcp-fb:* ").map(t.parseRtcpFb); return s.codecs.forEach((e=>{ i.forEach((t=>{ e.rtcpFeedback.find((e=>e.type === t.type && e.parameter === t.parameter)) || e.rtcpFeedback.push(t) } )) } )), s } , t.writeRtpDescription = function(e, s) { let n = ""; n += "m=" + e + " ", n += s.codecs.length > 0 ? "9" : "0", n += " " + (s.profile || "UDP/TLS/RTP/SAVPF") + " ", n += s.codecs.map((e=>void 0 !== e.preferredPayloadType ? e.preferredPayloadType : e.payloadType)).join(" ") + "\r\n", n += "c=IN IP4 0.0.0.0\r\n", n += "a=rtcp:9 IN IP4 0.0.0.0\r\n", s.codecs.forEach((e=>{ n += t.writeRtpMap(e), n += t.writeFmtp(e), n += t.writeRtcpFb(e) } )); let i = 0; return s.codecs.forEach((e=>{ e.maxptime > i && (i = e.maxptime) } )), i > 0 && (n += "a=maxptime:" + i + "\r\n"), s.headerExtensions && s.headerExtensions.forEach((e=>{ n += t.writeExtmap(e) } )), n } , t.parseRtpEncodingParameters = function(e) { const s = [] , n = t.parseRtpParameters(e) , i = -1 !== n.fecMechanisms.indexOf("RED") , r = -1 !== n.fecMechanisms.indexOf("ULPFEC") , o = t.matchPrefix(e, "a=ssrc:").map((e=>t.parseSsrcMedia(e))).filter((e=>"cname" === e.attribute)) , a = o.length > 0 && o[0].ssrc; let l; const c = t.matchPrefix(e, "a=ssrc-group:FID").map((e=>e.substring(17).split(" ").map((e=>parseInt(e, 10))))); c.length > 0 && c[0].length > 1 && c[0][0] === a && (l = c[0][1]), n.codecs.forEach((e=>{ if ("RTX" === e.name.toUpperCase() && e.parameters.apt) { let t = { ssrc: a, codecPayloadType: parseInt(e.parameters.apt, 10) }; a && l && (t.rtx = { ssrc: l }), s.push(t), i && (t = JSON.parse(JSON.stringify(t)), t.fec = { ssrc: a, mechanism: r ? "red+ulpfec" : "red" }, s.push(t)) } } )), 0 === s.length && a && s.push({ ssrc: a }); let d = t.matchPrefix(e, "b="); return d.length && (d = 0 === d[0].indexOf("b=TIAS:") ? parseInt(d[0].substring(7), 10) : 0 === d[0].indexOf("b=AS:") ? 1e3 * parseInt(d[0].substring(5), 10) * .95 - 16e3 : void 0, s.forEach((e=>{ e.maxBitrate = d } ))), s } , t.parseRtcpParameters = function(e) { const s = {} , n = t.matchPrefix(e, "a=ssrc:").map((e=>t.parseSsrcMedia(e))).filter((e=>"cname" === e.attribute))[0]; n && (s.cname = n.value, s.ssrc = n.ssrc); const i = t.matchPrefix(e, "a=rtcp-rsize"); s.reducedSize = i.length > 0, s.compound = 0 === i.length; const r = t.matchPrefix(e, "a=rtcp-mux"); return s.mux = r.length > 0, s } , t.writeRtcpParameters = function(e) { let t = ""; return e.reducedSize && (t += "a=rtcp-rsize\r\n"), e.mux && (t += "a=rtcp-mux\r\n"), void 0 !== e.ssrc && e.cname && (t += "a=ssrc:" + e.ssrc + " cname:" + e.cname + "\r\n"), t } , t.parseMsid = function(e) { let s; const n = t.matchPrefix(e, "a=msid:"); if (1 === n.length) return s = n[0].substring(7).split(" "), { stream: s[0], track: s[1] }; const i = t.matchPrefix(e, "a=ssrc:").map((e=>t.parseSsrcMedia(e))).filter((e=>"msid" === e.attribute)); return i.length > 0 ? (s = i[0].value.split(" "), { stream: s[0], track: s[1] }) : void 0 } , t.parseSctpDescription = function(e) { const s = t.parseMLine(e) , n = t.matchPrefix(e, "a=max-message-size:"); let i; n.length > 0 && (i = parseInt(n[0].substring(19), 10)), isNaN(i) && (i = 65536); const r = t.matchPrefix(e, "a=sctp-port:"); if (r.length > 0) return { port: parseInt(r[0].substring(12), 10), protocol: s.fmt, maxMessageSize: i }; const o = t.matchPrefix(e, "a=sctpmap:"); if (o.length > 0) { const e = o[0].substring(10).split(" "); return { port: parseInt(e[0], 10), protocol: e[1], maxMessageSize: i } } } , t.writeSctpDescription = function(e, t) { let s = []; return s = "DTLS/SCTP" !== e.protocol ? ["m=" + e.kind + " 9 " + e.protocol + " " + t.protocol + "\r\n", "c=IN IP4 0.0.0.0\r\n", "a=sctp-port:" + t.port + "\r\n"] : ["m=" + e.kind + " 9 " + e.protocol + " " + t.port + "\r\n", "c=IN IP4 0.0.0.0\r\n", "a=sctpmap:" + t.port + " " + t.protocol + " 65535\r\n"], void 0 !== t.maxMessageSize && s.push("a=max-message-size:" + t.maxMessageSize + "\r\n"), s.join("") } , t.generateSessionId = function() { return Math.random().toString().substr(2, 22) } , t.writeSessionBoilerplate = function(e, s, n) { let i; const r = void 0 !== s ? s : 2; return i = e || t.generateSessionId(), "v=0\r\no=" + (n || "thisisadapterortc") + " " + i + " " + r + " IN IP4 127.0.0.1\r\ns=-\r\nt=0 0\r\n" } , t.getDirection = function(e, s) { const n = t.splitLines(e); for (let e = 0; e < n.length; e++) switch (n[e]) { case "a=sendrecv": case "a=sendonly": case "a=recvonly": case "a=inactive": return n[e].substring(2) } return s ? t.getDirection(s) : "sendrecv" } , t.getKind = function(e) { return t.splitLines(e)[0].split(" ")[0].substring(2) } , t.isRejected = function(e) { return "0" === e.split(" ", 2)[1] } , t.parseMLine = function(e) { const s = t.splitLines(e)[0].substring(2).split(" "); return { kind: s[0], port: parseInt(s[1], 10), protocol: s[2], fmt: s.slice(3).join(" ") } } , t.parseOLine = function(e) { const s = t.matchPrefix(e, "o=")[0].substring(2).split(" "); return { username: s[0], sessionId: s[1], sessionVersion: parseInt(s[2], 10), netType: s[3], addressType: s[4], address: s[5] } } , t.isValidSDP = function(e) { if ("string" != typeof e || 0 === e.length) return !1; const s = t.splitLines(e); for (let e = 0; e < s.length; e++) if (s[e].length < 2 || "=" !== s[e].charAt(1)) return !1; return !0 } , e.exports = t } } , n = {}; function i(e) { var t = n[e]; if (void 0 !== t) return t.exports; var r = n[e] = { exports: {} }; return s[e](r, r.exports, i), r.exports } i.d = (e,t)=>{ for (var s in t) i.o(t, s) && !i.o(e, s) && Object.defineProperty(e, s, { enumerable: !0, get: t[s] }) } , i.o = (e,t)=>Object.prototype.hasOwnProperty.call(e, t); var r = {}; (()=>{ i.d(r, { ym: ()=>xi, EW: ()=>Mi, Dz: ()=>bt, hV: ()=>Et, Mx: ()=>Pi, z$: ()=>wt, J0: ()=>Tt, De: ()=>xt, L3: ()=>ki, rl: ()=>Ri, $C: ()=>Mt, Nu: ()=>Pt, NR: ()=>Li, zg: ()=>kt, f1: ()=>Ai, vU: ()=>Rt, rv: ()=>Lt, Nh: ()=>At, IY: ()=>Fi, ss: ()=>Ft, dP: ()=>bi, cf: ()=>Ot, fz: ()=>Hi, Yd: ()=>It, U2: ()=>Vi, jf: ()=>qi, D0: ()=>Ei, lO: ()=>Qi, iM: ()=>Dt, Je: ()=>Ki, qy: ()=>_t, YK: ()=>wi, Eu: ()=>Xi, ce: ()=>Ut, sK: ()=>zt, Ok: ()=>Bt, q5: ()=>Nt, RZ: ()=>Oi, g: ()=>Gt, XY: ()=>Ii, Fk: ()=>Di, YU: ()=>Ji, FU: ()=>ji, Ib: ()=>Ht, Az: ()=>Wt, Iw: ()=>Vt, qY: ()=>Qt, db: ()=>qt, Zh: ()=>_i, ek: ()=>Ui, MG: ()=>zi, gE: ()=>Bi, tW: ()=>Ni, Tn: ()=>Kt, wr: ()=>Gi, x_: ()=>jt, Am: ()=>Xt, u3: ()=>Jt, Ni: ()=>Yt, bq: ()=>$t, $f: ()=>Zt }); var e, t, s, n = i(539), o = { d: (e,t)=>{ for (var s in t) o.o(t, s) && !o.o(e, s) && Object.defineProperty(e, s, { enumerable: !0, get: t[s] }) } , o: (e,t)=>Object.prototype.hasOwnProperty.call(e, t) }, a = {}; o.d(a, { Dz: ()=>fe, g$: ()=>I, Lt: ()=>A, Q9: ()=>O, qf: ()=>F, hV: ()=>Pe, z$: ()=>we, J0: ()=>Ee, De: ()=>pe, $C: ()=>ve, al: ()=>W, _W: ()=>V, tz: ()=>H, Nu: ()=>be, zg: ()=>Ue, vp: ()=>oe, vU: ()=>ae, wF: ()=>Y, rv: ()=>Ce, Nh: ()=>ye, ss: ()=>De, qW: ()=>te, QL: ()=>ee, cf: ()=>Be, eM: ()=>J, Yd: ()=>l, iM: ()=>C, qy: ()=>c, ce: ()=>y, sK: ()=>ce, Ok: ()=>me, q5: ()=>Te, g: ()=>yt, xl: ()=>K, I: ()=>j, bx: ()=>X, Ib: ()=>M, Az: ()=>P, Iw: ()=>k, qY: ()=>L, db: ()=>R, mR: ()=>se, Tn: ()=>b, rV: ()=>$, gh: ()=>q, i5: ()=>Z, x_: ()=>he, Am: ()=>ut, eR: ()=>D, r8: ()=>Q, u3: ()=>ze, vd: ()=>U, iV: ()=>B, jZ: ()=>z, SW: ()=>G, ZH: ()=>N, Ni: ()=>gt, lh: ()=>_, bq: ()=>E, $f: ()=>St, eu: ()=>re, Ax: ()=>ie, Mc: ()=>ne }); class l { static GetStackTrace() { const e = new Error; let t = "No Stack Available for this browser"; return e.stack && (t = e.stack.toString().replace(/Error/g, "")), t } static SetLoggerVerbosity(e) { null != this.verboseLogLevel && (this.verboseLogLevel = e) } static Log(e, t, s) { if (s > this.verboseLogLevel) return; const n = `Level: Log\nMsg: ${t}\nCaller: ${e}`; console.log(n) } static Info(e, t, s) { if (s > this.verboseLogLevel) return; const n = `Level: Info\nMsg: ${t}`; console.info(n) } static Error(e, t) { const s = `Level: Error\nMsg: ${t}\nCaller: ${e}`; console.error(s) } static Warning(e, t) { const s = `Level: Warning\nCaller: ${e}\nMsg: ${t}`; console.warn(s) } } l.verboseLogLevel = 5, (s = e || (e = {})).LIST_STREAMERS = "listStreamers", s.SUBSCRIBE = "subscribe", s.UNSUBSCRIBE = "unsubscribe", s.ICE_CANDIDATE = "iceCandidate", s.OFFER = "offer", s.ANSWER = "answer", s.DATACHANNELREQUEST = "dataChannelRequest", s.SFURECVDATACHANNELREADY = "peerDataChannelsReady", s.PONG = "pong"; class c { payload() { return l.Log(l.GetStackTrace(), "Sending => \n" + JSON.stringify(this, void 0, 4), 6), JSON.stringify(this) } } class d extends c { constructor() { super(), this.type = e.LIST_STREAMERS } } class h extends c { constructor(t) { super(), this.type = e.SUBSCRIBE, this.streamerId = t } } class u extends c { constructor() { super(), this.type = e.UNSUBSCRIBE } } class m extends c { constructor(t) { super(), this.type = e.PONG, this.time = t } } class g extends c { constructor(t) { super(), this.type = e.OFFER, t && (this.type = t.type, this.sdp = t.sdp) } } class p extends c { constructor(t) { super(), this.type = e.ANSWER, t && (this.type = t.type, this.sdp = t.sdp) } } class v extends c { constructor() { super(), this.type = e.DATACHANNELREQUEST } } class f extends c { constructor() { super(), this.type = e.SFURECVDATACHANNELREADY } } class S { constructor(t) { this.type = e.ICE_CANDIDATE, this.candidate = t } payload() { return l.Log(l.GetStackTrace(), "Sending => \n" + JSON.stringify(this, void 0, 4), 6), JSON.stringify(this) } } !function(e) { e.CONFIG = "config", e.STREAMER_LIST = "streamerList", e.PLAYER_COUNT = "playerCount", e.OFFER = "offer", e.ANSWER = "answer", e.ICE_CANDIDATE = "iceCandidate", e.PEER_DATA_CHANNELS = "peerDataChannels", e.PING = "ping", e.WARNING = "warning" }(t || (t = {})); class C { } class y extends C { } class b { constructor() { this.FromUEMessageHandlers = new Map } addMessageHandler(e, t) { this.FromUEMessageHandlers.set(e, t) } handleMessage(e, t) { this.FromUEMessageHandlers.has(e) ? this.FromUEMessageHandlers.get(e)(t) : l.Error(l.GetStackTrace(), `Message type of ${e} does not have a message handler registered on the frontend - ignoring message.`) } static setupDefaultHandlers(e) { e.signallingProtocol.addMessageHandler(t.PING, (s=>{ const n = new m((new Date).getTime()).payload(); l.Log(l.GetStackTrace(), t.PING + ": " + s, 6), e.webSocket.send(n) } )), e.signallingProtocol.addMessageHandler(t.CONFIG, (s=>{ l.Log(l.GetStackTrace(), t.CONFIG, 6); const n = JSON.parse(s); e.onConfig(n) } )), e.signallingProtocol.addMessageHandler(t.STREAMER_LIST, (s=>{ l.Log(l.GetStackTrace(), t.STREAMER_LIST, 6); const n = JSON.parse(s); e.onStreamerList(n) } )), e.signallingProtocol.addMessageHandler(t.PLAYER_COUNT, (e=>{ l.Log(l.GetStackTrace(), t.PLAYER_COUNT, 6); const s = JSON.parse(e); l.Log(l.GetStackTrace(), "Player Count: " + s.count, 6) } )), e.signallingProtocol.addMessageHandler(t.ANSWER, (s=>{ l.Log(l.GetStackTrace(), t.ANSWER, 6); const n = JSON.parse(s); e.onWebRtcAnswer(n) } )), e.signallingProtocol.addMessageHandler(t.OFFER, (s=>{ l.Log(l.GetStackTrace(), t.OFFER, 6); const n = JSON.parse(s); e.onWebRtcOffer(n) } )), e.signallingProtocol.addMessageHandler(t.ICE_CANDIDATE, (s=>{ l.Log(l.GetStackTrace(), t.ICE_CANDIDATE, 6); const n = JSON.parse(s); e.onIceCandidate(n.candidate) } )), e.signallingProtocol.addMessageHandler(t.WARNING, (e=>{ l.Warning(l.GetStackTrace(), `Warning received: ${e}`) } )), e.signallingProtocol.addMessageHandler(t.PEER_DATA_CHANNELS, (s=>{ l.Log(l.GetStackTrace(), t.PEER_DATA_CHANNELS, 6); const n = JSON.parse(s); e.onWebRtcPeerDataChannels(n) } )) } } class E { constructor() { this.WS_OPEN_STATE = 1, this.onOpen = new EventTarget, this.onClose = new EventTarget, this.signallingProtocol = new b, b.setupDefaultHandlers(this) } connect(e) { l.Log(l.GetStackTrace(), e, 6); try { return this.webSocket = new WebSocket(e), this.webSocket.onopen = e=>this.handleOnOpen(e), this.webSocket.onerror = ()=>this.handleOnError(), this.webSocket.onclose = e=>this.handleOnClose(e), this.webSocket.onmessage = e=>this.handleOnMessage(e), this.webSocket.onmessagebinary = e=>this.handleOnMessageBinary(e), !0 } catch (e) { return l.Error(e, e), !1 } } handleOnMessageBinary(e) { e && e.data && e.data.text().then((e=>{ const t = new MessageEvent("messageFromBinary",{ data: e }); this.handleOnMessage(t) } )).catch((e=>{ l.Error(l.GetStackTrace(), `Failed to parse binary blob from websocket, reason: ${e}`) } )) } handleOnMessage(e) { if (e.data && e.data instanceof Blob) return void this.handleOnMessageBinary(e); const t = JSON.parse(e.data); l.Log(l.GetStackTrace(), "received => \n" + JSON.stringify(JSON.parse(e.data), void 0, 4), 6), this.signallingProtocol.handleMessage(t.type, e.data) } handleOnOpen(e) { l.Log(l.GetStackTrace(), "Connected to the signalling server via WebSocket", 6), this.onOpen.dispatchEvent(new Event("open")) } handleOnError() { l.Error(l.GetStackTrace(), "WebSocket error") } handleOnClose(e) { this.onWebSocketOncloseOverlayMessage(e), l.Log(l.GetStackTrace(), "Disconnected to the signalling server via WebSocket: " + JSON.stringify(e.code) + " - " + e.reason), this.onClose.dispatchEvent(new Event("close")) } requestStreamerList() { const e = new d; this.webSocket.send(e.payload()) } sendSubscribe(e) { const t = new h(e); this.webSocket.send(t.payload()) } sendUnsubscribe() { const e = new u; this.webSocket.send(e.payload()) } sendWebRtcOffer(e) { const t = new g(e); this.webSocket.send(t.payload()) } sendWebRtcAnswer(e) { const t = new p(e); this.webSocket.send(t.payload()) } sendWebRtcDatachannelRequest() { const e = new v; this.webSocket.send(e.payload()) } sendSFURecvDataChannelReady() { const e = new f; this.webSocket.send(e.payload()) } sendIceCandidate(e) { if (l.Log(l.GetStackTrace(), "Sending Ice Candidate"), this.webSocket && this.webSocket.readyState === this.WS_OPEN_STATE) { const t = new S(e); this.webSocket.send(t.payload()) } } close() { var e; null === (e = this.webSocket) || void 0 === e || e.close() } onWebSocketOncloseOverlayMessage(e) {} onConfig(e) {} onStreamerList(e) {} onIceCandidate(e) {} onWebRtcAnswer(e) {} onWebRtcOffer(e) {} onWebRtcPeerDataChannels(e) {} } class w { constructor(e) { this.videoElementProvider = e, this.audioElement = document.createElement("Audio") } handleOnTrack(e) { l.Log(l.GetStackTrace(), "handleOnTrack " + JSON.stringify(e.streams), 6); const t = this.videoElementProvider.getVideoElement(); if (e.track && l.Log(l.GetStackTrace(), "Got track - " + e.track.kind + " id=" + e.track.id + " readyState=" + e.track.readyState, 6), "audio" != e.track.kind) return "video" == e.track.kind && t.srcObject !== e.streams[0] ? (t.srcObject = e.streams[0], void l.Log(l.GetStackTrace(), "Set video source from video track ontrack.")) : void 0; this.CreateAudioTrack(e.streams[0]) } CreateAudioTrack(e) { const t = this.videoElementProvider.getVideoElement(); t.srcObject != e && t.srcObject && t.srcObject !== e && (this.audioElement.srcObject = e, l.Log(l.GetStackTrace(), "Created new audio element to play separate audio stream.")) } } class T { constructor(e) { this.freezeFrameHeight = 0, this.freezeFrameWidth = 0, this.rootDiv = e, this.rootElement = document.createElement("div"), this.rootElement.id = "freezeFrame", this.rootElement.style.display = "none", this.rootElement.style.pointerEvents = "none", this.rootElement.style.position = "absolute", this.rootElement.style.zIndex = "20", this.imageElement = document.createElement("img"), this.imageElement.style.position = "absolute", this.rootElement.appendChild(this.imageElement), this.rootDiv.appendChild(this.rootElement) } setElementForShow() { this.rootElement.style.display = "block" } setElementForHide() { this.rootElement.style.display = "none" } updateImageElementSource(e) { const t = btoa(e.reduce(((e,t)=>e + String.fromCharCode(t)), "")); this.imageElement.src = "data:image/jpeg;base64," + t } setDimensionsFromElementAndResize() { this.freezeFrameHeight = this.imageElement.naturalHeight, this.freezeFrameWidth = this.imageElement.naturalWidth, this.resize() } resize() { if (0 !== this.freezeFrameWidth && 0 !== this.freezeFrameHeight) { let e = 0 , t = 0 , s = 0 , n = 0; const i = this.rootDiv.clientWidth / this.rootDiv.clientHeight , r = this.freezeFrameWidth / this.freezeFrameHeight; i < r ? (e = this.rootDiv.clientWidth, t = Math.floor(this.rootDiv.clientWidth / r), s = Math.floor(.5 * (this.rootDiv.clientHeight - t)), n = 0) : (e = Math.floor(this.rootDiv.clientHeight * r), t = this.rootDiv.clientHeight, s = 0, n = Math.floor(.5 * (this.rootDiv.clientWidth - e))), this.rootElement.style.width = this.rootDiv.offsetWidth + "px", this.rootElement.style.height = this.rootDiv.offsetHeight + "px", this.rootElement.style.left = "0px", this.rootElement.style.top = "0px", this.imageElement.style.width = e + "px", this.imageElement.style.height = t + "px", this.imageElement.style.left = n + "px", this.imageElement.style.top = s + "px" } } } class x { constructor(e) { this.receiving = !1, this.size = 0, this.jpeg = void 0, this.valid = !1, this.freezeFrameDelay = 50, this.freezeFrame = new T(e) } showFreezeFrame() { this.valid && this.freezeFrame.setElementForShow() } hideFreezeFrame() { this.valid = !1, this.freezeFrame.setElementForHide() } updateFreezeFrameAndShow(e, t) { this.freezeFrame.updateImageElementSource(e), this.freezeFrame.imageElement.onload = ()=>{ this.freezeFrame.setDimensionsFromElementAndResize(), t() } } processFreezeFrameMessage(e, t) { this.receiving || (this.receiving = !0, this.valid = !1, this.size = 0, this.jpeg = void 0), this.size = new DataView(e.slice(1, 5).buffer).getInt32(0, !0); const s = e.slice(5); if (this.jpeg) { const e = new Uint8Array(this.jpeg.length + s.length); e.set(this.jpeg, 0), e.set(s, this.jpeg.length), this.jpeg = e } else this.jpeg = s, this.receiving = !0, l.Log(l.GetStackTrace(), `received first chunk of freeze frame: ${this.jpeg.length}/${this.size}`, 6); this.jpeg.length === this.size ? (this.receiving = !1, this.valid = !0, l.Log(l.GetStackTrace(), `received complete freeze frame ${this.size}`, 6), this.updateFreezeFrameAndShow(this.jpeg, t)) : this.jpeg.length > this.size && (l.Error(l.GetStackTrace(), `received bigger freeze frame than advertised: ${this.jpeg.length}/${this.size}`), this.jpeg = void 0, this.receiving = !1) } } class M { constructor(e, t, s, n, i=(()=>{} )) { this.onChange = i, this.onChangeEmit = ()=>{} , this.id = e, this.description = s, this.label = t, this.value = n } set label(e) { this._label = e, this.onChangeEmit(this._value) } get label() { return this._label } get value() { return this._value } set value(e) { this._value = e, this.onChange(this._value, this), this.onChangeEmit(this._value) } } class P extends M { constructor(e, t, s, n, i, r=(()=>{} )) { super(e, t, s, n, r); const o = new URLSearchParams(window.location.search); if (i && o.has(this.id)) { const e = this.getUrlParamFlag(); this.flag = e } else this.flag = n; this.useUrlParams = i } getUrlParamFlag() { const e = new URLSearchParams(window.location.search); return !!e.has(this.id) && "false" !== e.get(this.id) && "False" !== e.get(this.id) } updateURLParams() { if (this.useUrlParams) { const e = new URLSearchParams(window.location.search); !0 === this.flag ? e.set(this.id, "true") : e.set(this.id, "false"), window.history.replaceState({}, "", "" !== e.toString() ? `${location.pathname}?${e}` : `${location.pathname}`) } } enable() { this.flag = !0 } get flag() { return !!this.value } set flag(e) { this.value = e } } class k extends M { constructor(e, t, s, n, i, r, o, a=(()=>{} )) { super(e, t, s, r, a), this._min = n, this._max = i; const l = new URLSearchParams(window.location.search); if (o && l.has(this.id)) { const e = Number.parseInt(l.get(this.id)); this.number = Number.isNaN(e) ? r : e } else this.number = r; this.useUrlParams = o } updateURLParams() { if (this.useUrlParams) { const e = new URLSearchParams(window.location.search); e.set(this.id, this.number.toString()), window.history.replaceState({}, "", "" !== e.toString() ? `${location.pathname}?${e}` : `${location.pathname}`) } } set number(e) { this.value = this.clamp(e) } get number() { return this.value } clamp(e) { return Math.max(Math.min(this._max, e), this._min) } get min() { return this._min } get max() { return this._max } addOnChangedListener(e) { this.onChange = e } } class R extends M { constructor(e, t, s, n, i, r=(()=>{} )) { super(e, t, s, n, r); const o = new URLSearchParams(window.location.search); if (i && o.has(this.id)) { const e = this.getUrlParamText(); this.text = e } else this.text = n; this.useUrlParams = i } getUrlParamText() { var e; const t = new URLSearchParams(window.location.search); return t.has(this.id) && null !== (e = t.get(this.id)) && void 0 !== e ? e : "" } updateURLParams() { if (this.useUrlParams) { const e = new URLSearchParams(window.location.search); e.set(this.id, this.text), window.history.replaceState({}, "", "" !== e.toString() ? `${location.pathname}?${e}` : `${location.pathname}`) } } get text() { return this.value } set text(e) { this.value = e } } class L extends M { constructor(e, t, s, n, i, r, o=(()=>{} )) { super(e, t, s, [n, n], o), this.options = i; const a = new URLSearchParams(window.location.search) , l = r && a.has(this.id) ? this.getUrlParamText() : n; this.selected = l, this.useUrlParams = r } getUrlParamText() { var e; const t = new URLSearchParams(window.location.search); return t.has(this.id) && null !== (e = t.get(this.id)) && void 0 !== e ? e : "" } updateURLParams() { if (this.useUrlParams) { const e = new URLSearchParams(window.location.search); e.set(this.id, this.selected), window.history.replaceState({}, "", "" !== e.toString() ? `${location.pathname}?${e}` : `${location.pathname}`) } } addOnChangedListener(e) { this.onChange = e } get options() { return this._options } set options(e) { this._options = e, this.onChangeEmit(this.selected) } get selected() { return this.value } set selected(e) { const t = this.options.filter((t=>-1 !== t.indexOf(e))); t.length && (this.value = t[0]) } } class A extends Event { constructor(e) { super("afkWarningActivate"), this.data = e } } class F extends Event { constructor(e) { super("afkWarningUpdate"), this.data = e } } class O extends Event { constructor() { super("afkWarningDeactivate") } } class I extends Event { constructor() { super("afkTimedOut") } } class D extends Event { constructor(e) { super("videoEncoderAvgQP"), this.data = e } } class _ extends Event { constructor() { super("webRtcSdp") } } class U extends Event { constructor() { super("webRtcAutoConnect") } } class z extends Event { constructor() { super("webRtcConnecting") } } class B extends Event { constructor() { super("webRtcConnected") } } class N extends Event { constructor() { super("webRtcFailed") } } class G extends Event { constructor(e) { super("webRtcDisconnected"), this.data = e } } class H extends Event { constructor(e) { super("dataChannelOpen"), this.data = e } } class W extends Event { constructor(e) { super("dataChannelClose"), this.data = e } } class V extends Event { constructor(e) { super("dataChannelError"), this.data = e } } class Q extends Event { constructor() { super("videoInitialized") } } class q extends Event { constructor() { super("streamLoading") } } class K extends Event { constructor(e) { super("playStreamError"), this.data = e } } class j extends Event { constructor() { super("playStream") } } class X extends Event { constructor(e) { super("playStreamRejected"), this.data = e } } class J extends Event { constructor(e) { super("loadFreezeFrame"), this.data = e } } class Y extends Event { constructor() { super("hideFreezeFrame") } } class $ extends Event { constructor(e) { super("statsReceived"), this.data = e } } class Z extends Event { constructor(e) { super("streamerListMessage"), this.data = e } } class ee extends Event { constructor(e) { super("latencyTestResult"), this.data = e } } class te extends Event { constructor(e) { super("initialSettings"), this.data = e } } class se extends Event { constructor(e) { super("settingsChanged"), this.data = e } } class ne extends Event { constructor() { super("xrSessionStarted") } } class ie extends Event { constructor() { super("xrSessionEnded") } } class re extends Event { constructor(e) { super("xrFrame"), this.data = e } } class oe extends EventTarget { dispatchEvent(e) { return super.dispatchEvent(e) } addEventListener(e, t) { super.addEventListener(e, t) } removeEventListener(e, t) { super.removeEventListener(e, t) } } class ae { } ae.AutoConnect = "AutoConnect", ae.AutoPlayVideo = "AutoPlayVideo", ae.AFKDetection = "TimeoutIfIdle", ae.BrowserSendOffer = "OfferToReceive", ae.HoveringMouseMode = "HoveringMouse", ae.ForceMonoAudio = "ForceMonoAudio", ae.ForceTURN = "ForceTURN", ae.FakeMouseWithTouches = "FakeMouseWithTouches", ae.IsQualityController = "ControlsQuality", ae.MatchViewportResolution = "MatchViewportRes", ae.PreferSFU = "preferSFU", ae.StartVideoMuted = "StartVideoMuted", ae.SuppressBrowserKeys = "SuppressBrowserKeys", ae.UseMic = "UseMic", ae.KeyboardInput = "KeyboardInput", ae.MouseInput = "MouseInput", ae.TouchInput = "TouchInput", ae.GamepadInput = "GamepadInput", ae.XRControllerInput = "XRControllerInput"; const le = e=>Object.getOwnPropertyNames(ae).some((t=>ae[t] === e)); class ce { } ce.AFKTimeoutSecs = "AFKTimeout", ce.MinQP = "MinQP", ce.MaxQP = "MaxQP", ce.WebRTCFPS = "WebRTCFPS", ce.WebRTCMinBitrate = "WebRTCMinBitrate", ce.WebRTCMaxBitrate = "WebRTCMaxBitrate", ce.MaxReconnectAttempts = "MaxReconnectAttempts"; const de = e=>Object.getOwnPropertyNames(ce).some((t=>ce[t] === e)); class he { } he.SignallingServerUrl = "ss"; const ue = e=>Object.getOwnPropertyNames(he).some((t=>he[t] === e)); class me { } me.PreferredCodec = "PreferredCodec", me.StreamerId = "StreamerId"; const ge = e=>Object.getOwnPropertyNames(me).some((t=>me[t] === e)); class pe { constructor(e={}) { this.flags = new Map, this.numericParameters = new Map, this.textParameters = new Map, this.optionParameters = new Map; const {initialSettings: t, useUrlParams: s} = e; this._useUrlParams = !!s, this.populateDefaultSettings(this._useUrlParams), t && this.setSettings(t) } get useUrlParams() { return this._useUrlParams } populateDefaultSettings(e) { this.textParameters.set(he.SignallingServerUrl, new R(he.SignallingServerUrl,"Signalling url","Url of the signalling server",("https:" === location.protocol ? "wss://" : "ws://") + window.location.hostname + ("80" === window.location.port || "" === window.location.port ? "" : `:${window.location.port}`),e)), this.optionParameters.set(me.StreamerId, new L(me.StreamerId,"Streamer ID","The ID of the streamer to stream.","",[],e)), this.optionParameters.set(me.PreferredCodec, new L(me.PreferredCodec,"Preferred Codec","The preferred codec to be used during codec negotiation","H264 level-asymmetry-allowed=1;packetization-mode=1;profile-level-id=42e01f",function() { const e = []; if (!RTCRtpReceiver.getCapabilities) return e.push("Only available on Chrome"), e; const t = /(VP\d|H26\d|AV1).*/; return RTCRtpReceiver.getCapabilities("video").codecs.forEach((s=>{ const n = s.mimeType.split("/")[1] + " " + (s.sdpFmtpLine || ""); null !== t.exec(n) && e.push(n) } )), e }(),e)), this.flags.set(ae.AutoConnect, new P(ae.AutoConnect,"Auto connect to stream","Whether we should attempt to auto connect to the signalling server or show a click to start prompt.",!1,e)), this.flags.set(ae.AutoPlayVideo, new P(ae.AutoPlayVideo,"Auto play video","When video is ready automatically start playing it as opposed to showing a play button.",!0,e)), this.flags.set(ae.BrowserSendOffer, new P(ae.BrowserSendOffer,"Browser send offer","Browser will initiate the WebRTC handshake by sending the offer to the streamer",!1,e)), this.flags.set(ae.UseMic, new P(ae.UseMic,"Use microphone","Make browser request microphone access and open an input audio track.",!1,e)), this.flags.set(ae.StartVideoMuted, new P(ae.StartVideoMuted,"Start video muted","Video will start muted if true.",!1,e)), this.flags.set(ae.SuppressBrowserKeys, new P(ae.SuppressBrowserKeys,"Suppress browser keys","Suppress certain browser keys that we use in UE, for example F5 to show shader complexity instead of refresh the page.",!0,e)), this.flags.set(ae.PreferSFU, new P(ae.PreferSFU,"Prefer SFU","Try to connect to the SFU instead of P2P.",!1,e)), this.flags.set(ae.IsQualityController, new P(ae.IsQualityController,"Is quality controller?","True if this peer controls stream quality",!0,e)), this.flags.set(ae.ForceMonoAudio, new P(ae.ForceMonoAudio,"Force mono audio","Force browser to request mono audio in the SDP",!1,e)), this.flags.set(ae.ForceTURN, new P(ae.ForceTURN,"Force TURN","Only generate TURN/Relayed ICE candidates.",!1,e)), this.flags.set(ae.AFKDetection, new P(ae.AFKDetection,"AFK if idle","Timeout the experience if user is AFK for a period.",!1,e)), this.flags.set(ae.MatchViewportResolution, new P(ae.MatchViewportResolution,"Match viewport resolution","Pixel Streaming will be instructed to dynamically resize the video stream to match the size of the video element.",!1,e)), this.flags.set(ae.HoveringMouseMode, new P(ae.HoveringMouseMode,"Control Scheme: Locked Mouse","Either locked mouse, where the pointer is consumed by the video and locked to it, or hovering mouse, where the mouse is not consumed.",!1,e,((e,t)=>{ t.label = `Control Scheme: ${e ? "Hovering" : "Locked"} Mouse` } ))), this.flags.set(ae.FakeMouseWithTouches, new P(ae.FakeMouseWithTouches,"Fake mouse with touches","A single finger touch is converted into a mouse event. This allows a non-touch application to be controlled partially via a touch device.",!1,e)), this.flags.set(ae.KeyboardInput, new P(ae.KeyboardInput,"Keyboard input","If enabled, send keyboard events to streamer",!0,e)), this.flags.set(ae.MouseInput, new P(ae.MouseInput,"Mouse input","If enabled, send mouse events to streamer",!0,e)), this.flags.set(ae.TouchInput, new P(ae.TouchInput,"Touch input","If enabled, send touch events to streamer",!0,e)), this.flags.set(ae.GamepadInput, new P(ae.GamepadInput,"Gamepad input","If enabled, send gamepad events to streamer",!0,e)), this.flags.set(ae.XRControllerInput, new P(ae.XRControllerInput,"XR controller input","If enabled, send XR controller events to streamer",!0,e)), this.numericParameters.set(ce.AFKTimeoutSecs, new k(ce.AFKTimeoutSecs,"AFK timeout","The time (in seconds) it takes for the application to time out if AFK timeout is enabled.",0,600,120,e)), this.numericParameters.set(ce.MaxReconnectAttempts, new k(ce.MaxReconnectAttempts,"Max Reconnects","Maximum number of reconnects the application will attempt when a streamer disconnects.",0,999,3,e)), this.numericParameters.set(ce.MinQP, new k(ce.MinQP,"Min QP","The lower bound for the quantization parameter (QP) of the encoder. 0 = Best quality, 51 = worst quality.",0,51,0,e)), this.numericParameters.set(ce.MaxQP, new k(ce.MaxQP,"Max QP","The upper bound for the quantization parameter (QP) of the encoder. 0 = Best quality, 51 = worst quality.",0,51,51,e)), this.numericParameters.set(ce.WebRTCFPS, new k(ce.WebRTCFPS,"Max FPS","The maximum FPS that WebRTC will try to transmit frames at.",1,999,60,e)), this.numericParameters.set(ce.WebRTCMinBitrate, new k(ce.WebRTCMinBitrate,"Min Bitrate (kbps)","The minimum bitrate that WebRTC should use.",0,5e5,0,e)), this.numericParameters.set(ce.WebRTCMaxBitrate, new k(ce.WebRTCMaxBitrate,"Max Bitrate (kbps)","The maximum bitrate that WebRTC should use.",0,5e5,0,e)) } _addOnNumericSettingChangedListener(e, t) { this.numericParameters.has(e) && this.numericParameters.get(e).addOnChangedListener(t) } _addOnOptionSettingChangedListener(e, t) { this.optionParameters.has(e) && this.optionParameters.get(e).addOnChangedListener(t) } getNumericSettingValue(e) { if (this.numericParameters.has(e)) return this.numericParameters.get(e).number; throw new Error(`There is no numeric setting with the id of ${e}`) } getTextSettingValue(e) { if (this.textParameters.has(e)) return this.textParameters.get(e).value; throw new Error(`There is no numeric setting with the id of ${e}`) } setNumericSetting(e, t) { if (!this.numericParameters.has(e)) throw new Error(`There is no numeric setting with the id of ${e}`); this.numericParameters.get(e).number = t } _addOnSettingChangedListener(e, t) { this.flags.has(e) && (this.flags.get(e).onChange = t) } _addOnTextSettingChangedListener(e, t) { this.textParameters.has(e) && (this.textParameters.get(e).onChange = t) } getSettingOption(e) { return this.optionParameters.get(e) } isFlagEnabled(e) { return this.flags.get(e).flag } setFlagEnabled(e, t) { this.flags.has(e) ? this.flags.get(e).flag = t : l.Warning(l.GetStackTrace(), `Cannot toggle flag called ${e} - it does not exist in the Config.flags map.`) } setTextSetting(e, t) { this.textParameters.has(e) ? this.textParameters.get(e).text = t : l.Warning(l.GetStackTrace(), `Cannot set text setting called ${e} - it does not exist in the Config.textParameters map.`) } setOptionSettingOptions(e, t) { this.optionParameters.has(e) ? this.optionParameters.get(e).options = t : l.Warning(l.GetStackTrace(), `Cannot set text setting called ${e} - it does not exist in the Config.optionParameters map.`) } setOptionSettingValue(e, t) { this.optionParameters.has(e) ? this.optionParameters.get(e).selected = t : l.Warning(l.GetStackTrace(), `Cannot set text setting called ${e} - it does not exist in the Config.enumParameters map.`) } setFlagLabel(e, t) { this.flags.has(e) ? this.flags.get(e).label = t : l.Warning(l.GetStackTrace(), `Cannot set label for flag called ${e} - it does not exist in the Config.flags map.`) } setSettings(e) { for (const t of Object.keys(e)) le(t) ? this.setFlagEnabled(t, e[t]) : de(t) ? this.setNumericSetting(t, e[t]) : ue(t) ? this.setTextSetting(t, e[t]) : ge(t) && this.setOptionSettingValue(t, e[t]) } getSettings() { const e = {}; for (const [t,s] of this.flags.entries()) e[t] = s.flag; for (const [t,s] of this.numericParameters.entries()) e[t] = s.number; for (const [t,s] of this.textParameters.entries()) e[t] = s.text; for (const [t,s] of this.optionParameters.entries()) e[t] = s.selected; return e } getFlags() { return Array.from(this.flags.values()) } getTextSettings() { return Array.from(this.textParameters.values()) } getNumericSettings() { return Array.from(this.numericParameters.values()) } getOptionSettings() { return Array.from(this.optionParameters.values()) } _registerOnChangeEvents(e) { for (const t of this.flags.keys()) { const s = this.flags.get(t); s && (s.onChangeEmit = t=>e.dispatchEvent(new se({ id: s.id, type: "flag", value: t, target: s }))) } for (const t of this.numericParameters.keys()) { const s = this.numericParameters.get(t); s && (s.onChangeEmit = t=>e.dispatchEvent(new se({ id: s.id, type: "number", value: t, target: s }))) } for (const t of this.textParameters.keys()) { const s = this.textParameters.get(t); s && (s.onChangeEmit = t=>e.dispatchEvent(new se({ id: s.id, type: "text", value: t, target: s }))) } for (const t of this.optionParameters.keys()) { const s = this.optionParameters.get(t); s && (s.onChangeEmit = t=>e.dispatchEvent(new se({ id: s.id, type: "option", value: t, target: s }))) } } } var ve; !function(e) { e[e.LockedMouse = 0] = "LockedMouse", e[e.HoveringMouse = 1] = "HoveringMouse" }(ve || (ve = {})); class fe { constructor(e, t, s) { this.closeTimeout = 10, this.active = !1, this.countdownActive = !1, this.warnTimer = void 0, this.countDown = 0, this.countDownTimer = void 0, this.config = e, this.pixelStreaming = t, this.onDismissAfk = s, this.onAFKTimedOutCallback = ()=>{ console.log("AFK timed out, did you want to override this callback?") } } onAfkClick() { clearInterval(this.countDownTimer), (this.active || this.countdownActive) && (this.startAfkWarningTimer(), this.pixelStreaming.dispatchEvent(new O)) } startAfkWarningTimer() { this.config.getNumericSettingValue(ce.AFKTimeoutSecs) > 0 && this.config.isFlagEnabled(ae.AFKDetection) ? this.active = !0 : this.active = !1, this.resetAfkWarningTimer() } stopAfkWarningTimer() { this.active = !1, this.countdownActive = !1, clearTimeout(this.warnTimer), clearInterval(this.countDownTimer) } pauseAfkWarningTimer() { this.active = !1 } resetAfkWarningTimer() { this.active && this.config.isFlagEnabled(ae.AFKDetection) && (clearTimeout(this.warnTimer), this.warnTimer = setTimeout((()=>this.activateAfkEvent()), 1e3 * this.config.getNumericSettingValue(ce.AFKTimeoutSecs))) } activateAfkEvent() { this.pauseAfkWarningTimer(), this.pixelStreaming.dispatchEvent(new A({ countDown: this.countDown, dismissAfk: this.onDismissAfk })), this.countDown = this.closeTimeout, this.countdownActive = !0, this.pixelStreaming.dispatchEvent(new F({ countDown: this.countDown })), this.config.isFlagEnabled(ae.HoveringMouseMode) || document.exitPointerLock && document.exitPointerLock(), this.countDownTimer = setInterval((()=>{ this.countDown--, 0 == this.countDown ? (this.pixelStreaming.dispatchEvent(new I), this.onAFKTimedOutCallback(), l.Log(l.GetStackTrace(), "You have been disconnected due to inactivity"), this.stopAfkWarningTimer()) : this.pixelStreaming.dispatchEvent(new F({ countDown: this.countDown })) } ), 1e3) } } class Se { constructor() { this.isReceivingFreezeFrame = !1 } getDataChannelInstance() { return this } createDataChannel(e, t, s) { this.peerConnection = e, this.label = t, this.datachannelOptions = s, null == s && (this.datachannelOptions = {}, this.datachannelOptions.ordered = !0), this.dataChannel = this.peerConnection.createDataChannel(this.label, this.datachannelOptions), this.setupDataChannel() } setupDataChannel() { this.dataChannel.binaryType = "arraybuffer", this.dataChannel.onopen = e=>this.handleOnOpen(e), this.dataChannel.onclose = e=>this.handleOnClose(e), this.dataChannel.onmessage = e=>this.handleOnMessage(e), this.dataChannel.onerror = e=>this.handleOnError(e) } handleOnOpen(e) { var t; l.Log(l.GetStackTrace(), `Data Channel (${this.label}) opened.`, 7), this.onOpen(null === (t = this.dataChannel) || void 0 === t ? void 0 : t.label, e) } handleOnClose(e) { var t; l.Log(l.GetStackTrace(), `Data Channel (${this.label}) closed.`, 7), this.onClose(null === (t = this.dataChannel) || void 0 === t ? void 0 : t.label, e) } handleOnMessage(e) { l.Log(l.GetStackTrace(), `Data Channel (${this.label}) message: ${e}`, 8) } handleOnError(e) { var t; l.Log(l.GetStackTrace(), `Data Channel (${this.label}) error: ${e}`, 7), this.onError(null === (t = this.dataChannel) || void 0 === t ? void 0 : t.label, e) } onOpen(e, t) {} onClose(e, t) {} onError(e, t) {} } class Ce { } class ye { } class be { } class Ee { } class we { } class Te { } class xe { } class Me { } class Pe { constructor() { this.inboundVideoStats = new ye, this.inboundAudioStats = new Ce, this.candidatePair = new we, this.DataChannelStats = new be, this.outBoundVideoStats = new Te, this.sessionStats = new xe, this.streamStats = new Me, this.codecs = new Map } processStats(e) { this.localCandidates = new Array, this.remoteCandidates = new Array, e.forEach((e=>{ switch (e.type) { case "candidate-pair": this.handleCandidatePair(e); break; case "certificate": case "media-source": case "outbound-rtp": case "peer-connection": case "remote-inbound-rtp": case "transport": break; case "codec": this.handleCodec(e); break; case "data-channel": this.handleDataChannel(e); break; case "inbound-rtp": this.handleInBoundRTP(e); break; case "local-candidate": this.handleLocalCandidate(e); break; case "remote-candidate": this.handleRemoteCandidate(e); break; case "remote-outbound-rtp": this.handleRemoteOutBound(e); break; case "track": this.handleTrack(e); break; case "stream": this.handleStream(e); break; default: l.Error(l.GetStackTrace(), "unhandled Stat Type"), l.Log(l.GetStackTrace(), e) } } )) } handleStream(e) { this.streamStats = e } handleCandidatePair(e) { this.candidatePair.bytesReceived = e.bytesReceived, this.candidatePair.bytesSent = e.bytesSent, this.candidatePair.localCandidateId = e.localCandidateId, this.candidatePair.remoteCandidateId = e.remoteCandidateId, this.candidatePair.nominated = e.nominated, this.candidatePair.readable = e.readable, this.candidatePair.selected = e.selected, this.candidatePair.writable = e.writable, this.candidatePair.state = e.state, this.candidatePair.currentRoundTripTime = e.currentRoundTripTime } handleDataChannel(e) { this.DataChannelStats.bytesReceived = e.bytesReceived, this.DataChannelStats.bytesSent = e.bytesSent, this.DataChannelStats.dataChannelIdentifier = e.dataChannelIdentifier, this.DataChannelStats.id = e.id, this.DataChannelStats.label = e.label, this.DataChannelStats.messagesReceived = e.messagesReceived, this.DataChannelStats.messagesSent = e.messagesSent, this.DataChannelStats.protocol = e.protocol, this.DataChannelStats.state = e.state, this.DataChannelStats.timestamp = e.timestamp } handleLocalCandidate(e) { const t = new Ee; t.label = "local-candidate", t.address = e.address, t.port = e.port, t.protocol = e.protocol, t.candidateType = e.candidateType, t.id = e.id, this.localCandidates.push(t) } handleRemoteCandidate(e) { const t = new Ee; t.label = "local-candidate", t.address = e.address, t.port = e.port, t.protocol = e.protocol, t.id = e.id, t.candidateType = e.candidateType, this.remoteCandidates.push(t) } handleInBoundRTP(e) { switch (e.kind) { case "video": this.inboundVideoStats = e, null != this.lastVideoStats && (this.inboundVideoStats.bitrate = 8 * (this.inboundVideoStats.bytesReceived - this.lastVideoStats.bytesReceived) / (this.inboundVideoStats.timestamp - this.lastVideoStats.timestamp), this.inboundVideoStats.bitrate = Math.floor(this.inboundVideoStats.bitrate)), this.lastVideoStats = Object.assign({}, this.inboundVideoStats); break; case "audio": this.inboundAudioStats = e, null != this.lastAudioStats && (this.inboundAudioStats.bitrate = 8 * (this.inboundAudioStats.bytesReceived - this.lastAudioStats.bytesReceived) / (this.inboundAudioStats.timestamp - this.lastAudioStats.timestamp), this.inboundAudioStats.bitrate = Math.floor(this.inboundAudioStats.bitrate)), this.lastAudioStats = Object.assign({}, this.inboundAudioStats); break; default: l.Log(l.GetStackTrace(), "Kind is not handled") } } handleRemoteOutBound(e) { "video" === e.kind && (this.outBoundVideoStats.bytesSent = e.bytesSent, this.outBoundVideoStats.id = e.id, this.outBoundVideoStats.localId = e.localId, this.outBoundVideoStats.packetsSent = e.packetsSent, this.outBoundVideoStats.remoteTimestamp = e.remoteTimestamp, this.outBoundVideoStats.timestamp = e.timestamp) } handleTrack(e) { "track" !== e.type || "video_label" !== e.trackIdentifier && "video" !== e.kind || (this.inboundVideoStats.framesDropped = e.framesDropped, this.inboundVideoStats.framesReceived = e.framesReceived, this.inboundVideoStats.frameHeight = e.frameHeight, this.inboundVideoStats.frameWidth = e.frameWidth) } handleCodec(e) { const t = e.id , s = `${e.mimeType.replace("video/", "").replace("audio/", "")}${e.sdpFmtpLine ? ` ${e.sdpFmtpLine}` : ""}`; this.codecs.set(t, s) } handleSessionStatistics(e, t, s) { const n = Date.now() - e; this.sessionStats.runTime = new Date(n).toISOString().substr(11, 8).toString(); const i = null === t ? "Not sent yet" : t ? "true" : "false"; this.sessionStats.controlsStreamInput = i, this.sessionStats.videoEncoderAvgQP = s } isNumber(e) { return "number" == typeof e && isFinite(e) } } const ke = (Re = { parseRtpParameters: ()=>n.parseRtpParameters, splitSections: ()=>n.splitSections }, Le = {}, o.d(Le, Re), Le); var Re, Le, Ae, Fe, Oe = function(e, t, s, n) { return new (s || (s = Promise))((function(i, r) { function o(e) { try { l(n.next(e)) } catch (e) { r(e) } } function a(e) { try { l(n.throw(e)) } catch (e) { r(e) } } function l(e) { var t; e.done ? i(e.value) : (t = e.value, t instanceof s ? t : new s((function(e) { e(t) } ))).then(o, a) } l((n = n.apply(e, t || [])).next()) } )) }; class Ie { constructor(e, t, s) { this.config = t, this.createPeerConnection(e, s) } createPeerConnection(e, t) { this.config.isFlagEnabled(ae.ForceTURN) && (e.iceTransportPolicy = "relay", l.Log(l.GetStackTrace(), "Forcing TURN usage by setting ICE Transport Policy in peer connection config.")), this.peerConnection = new RTCPeerConnection(e), this.peerConnection.onsignalingstatechange = e=>this.handleSignalStateChange(e), this.peerConnection.oniceconnectionstatechange = e=>this.handleIceConnectionStateChange(e), this.peerConnection.onicegatheringstatechange = e=>this.handleIceGatheringStateChange(e), this.peerConnection.ontrack = e=>this.handleOnTrack(e), this.peerConnection.onicecandidate = e=>this.handleIceCandidate(e), this.peerConnection.ondatachannel = e=>this.handleDataChannel(e), this.aggregatedStats = new Pe, this.preferredCodec = t, this.updateCodecSelection = !0 } createOffer(e, t) { return Oe(this, void 0, void 0, (function*() { l.Log(l.GetStackTrace(), "Create Offer", 6); const s = "localhost" === location.hostname || "127.0.0.1" === location.hostname , n = "https:" === location.protocol; let i = t.isFlagEnabled(ae.UseMic); !i || s || n || (i = !1, l.Error(l.GetStackTrace(), "Microphone access in the browser will not work if you are not on HTTPS or localhost. Disabling mic access."), l.Error(l.GetStackTrace(), "For testing you can enable HTTP microphone access Chrome by visiting chrome://flags/ and enabling 'unsafely-treat-insecure-origin-as-secure'")), this.setupTransceiversAsync(i).finally((()=>{ var t; null === (t = this.peerConnection) || void 0 === t || t.createOffer(e).then((e=>{ var t; this.showTextOverlayConnecting(), e.sdp = this.mungeSDP(e.sdp, i), null === (t = this.peerConnection) || void 0 === t || t.setLocalDescription(e), this.onSendWebRTCOffer(e) } )).catch((()=>{ this.showTextOverlaySetupFailure() } )) } )) } )) } receiveOffer(e, t) { var s; return Oe(this, void 0, void 0, (function*() { l.Log(l.GetStackTrace(), "Receive Offer", 6), null === (s = this.peerConnection) || void 0 === s || s.setRemoteDescription(e).then((()=>{ const e = "localhost" === location.hostname || "127.0.0.1" === location.hostname , s = "https:" === location.protocol; let n = t.isFlagEnabled(ae.UseMic); !n || e || s || (n = !1, l.Error(l.GetStackTrace(), "Microphone access in the browser will not work if you are not on HTTPS or localhost. Disabling mic access."), l.Error(l.GetStackTrace(), "For testing you can enable HTTP microphone access Chrome by visiting chrome://flags/ and enabling 'unsafely-treat-insecure-origin-as-secure'")), this.setupTransceiversAsync(n).finally((()=>{ var e; null === (e = this.peerConnection) || void 0 === e || e.createAnswer().then((e=>{ var t; return e.sdp = this.mungeSDP(e.sdp, n), null === (t = this.peerConnection) || void 0 === t ? void 0 : t.setLocalDescription(e) } )).then((()=>{ var e; this.onSendWebRTCAnswer(null === (e = this.peerConnection) || void 0 === e ? void 0 : e.currentLocalDescription) } )).catch((()=>{ l.Error(l.GetStackTrace(), "createAnswer() failed") } )) } )) } )), this.config.setOptionSettingOptions(me.PreferredCodec, this.parseAvailableCodecs(e).filter((e=>this.config.getSettingOption(me.PreferredCodec).options.includes(e)))) } )) } receiveAnswer(e) { var t; null === (t = this.peerConnection) || void 0 === t || t.setRemoteDescription(e), this.config.setOptionSettingOptions(me.PreferredCodec, this.parseAvailableCodecs(e).filter((e=>this.config.getSettingOption(me.PreferredCodec).options.includes(e)))) } generateStats() { var e; null === (e = this.peerConnection) || void 0 === e || e.getStats(null).then((e=>{ this.aggregatedStats.processStats(e), this.onVideoStats(this.aggregatedStats), this.updateCodecSelection && this.config.setOptionSettingValue(me.PreferredCodec, this.aggregatedStats.codecs.get(this.aggregatedStats.inboundVideoStats.codecId)) } )) } close() { this.peerConnection && (this.peerConnection.close(), this.peerConnection = null) } mungeSDP(e, t) { const s = e; s.replace(/(a=fmtp:\d+ .*level-asymmetry-allowed=.*)\r\n/gm, "$1;x-google-start-bitrate=10000;x-google-max-bitrate=100000\r\n"); let n = ""; return n += "maxaveragebitrate=510000;", t && (n += "sprop-maxcapturerate=48000;"), n += this.config.isFlagEnabled(ae.ForceMonoAudio) ? "stereo=0;" : "stereo=1;", n += "useinbandfec=1", s.replace("useinbandfec=1", n), s } handleOnIce(e) { var t; l.Log(l.GetStackTrace(), "peerconnection handleOnIce", 6), this.config.isFlagEnabled(ae.ForceTURN) && e.candidate.indexOf("relay") < 0 ? l.Info(l.GetStackTrace(), `Dropping candidate because it was not TURN relay. | Type= ${e.type} | Protocol= ${e.protocol} | Address=${e.address} | Port=${e.port} |`, 6) : null === (t = this.peerConnection) || void 0 === t || t.addIceCandidate(e) } handleSignalStateChange(e) { l.Log(l.GetStackTrace(), "signaling state change: " + e, 6) } handleIceConnectionStateChange(e) { l.Log(l.GetStackTrace(), "ice connection state change: " + e, 6), this.onIceConnectionStateChange(e) } handleIceGatheringStateChange(e) { l.Log(l.GetStackTrace(), "ice gathering state change: " + JSON.stringify(e), 6) } handleOnTrack(e) { this.onTrack(e) } handleIceCandidate(e) { this.onPeerIceCandidate(e) } handleDataChannel(e) { this.onDataChannel(e) } onTrack(e) {} onIceConnectionStateChange(e) {} onPeerIceCandidate(e) {} onDataChannel(e) {} setupTransceiversAsync(e) { var t, s, n, i, r, o, a, l, c; return Oe(this, void 0, void 0, (function*() { const d = (null === (t = this.peerConnection) || void 0 === t ? void 0 : t.getTransceivers().length) > 0; if (null === (s = this.peerConnection) || void 0 === s || s.addTransceiver("video", { direction: "recvonly" }), RTCRtpReceiver.getCapabilities && "" != this.preferredCodec) for (const e of null !== (i = null === (n = this.peerConnection) || void 0 === n ? void 0 : n.getTransceivers()) && void 0 !== i ? i : []) if (e && e.receiver && e.receiver.track && "video" === e.receiver.track.kind) { const t = this.preferredCodec.split(" ") , s = [{ mimeType: "video/" + t[0], clockRate: 9e4, sdpFmtpLine: t[1] ? t[1] : "" }]; this.config.getSettingOption(me.PreferredCodec).options.filter((e=>e != this.preferredCodec)).forEach((e=>{ const t = e.split(" "); s.push({ mimeType: "video/" + t[0], clockRate: 9e4, sdpFmtpLine: t[1] ? t[1] : "" }) } )); for (const e of s) "" === e.sdpFmtpLine && delete e.sdpFmtpLine; e.setCodecPreferences(s) } if (e) { const t = { video: !1, audio: !!e && { autoGainControl: !1, channelCount: 1, echoCancellation: !1, latency: 0, noiseSuppression: !1, sampleRate: 48e3, sampleSize: 16, volume: 1 } } , s = yield navigator.mediaDevices.getUserMedia(t); if (s) if (d) { for (const e of null !== (a = null === (o = this.peerConnection) || void 0 === o ? void 0 : o.getTransceivers()) && void 0 !== a ? a : []) if (e && e.receiver && e.receiver.track && "audio" === e.receiver.track.kind) for (const t of s.getTracks()) t.kind && "audio" == t.kind && (e.sender.replaceTrack(t), e.direction = "sendrecv") } else for (const e of s.getTracks()) e.kind && "audio" == e.kind && (null === (l = this.peerConnection) || void 0 === l || l.addTransceiver(e, { direction: "sendrecv" })); else null === (c = this.peerConnection) || void 0 === c || c.addTransceiver("audio", { direction: "recvonly" }) } else null === (r = this.peerConnection) || void 0 === r || r.addTransceiver("audio", { direction: "recvonly" }) } )) } onVideoStats(e) {} onSendWebRTCOffer(e) {} onSendWebRTCAnswer(e) {} showTextOverlayConnecting() {} showTextOverlaySetupFailure() {} parseAvailableCodecs(e) { if (!RTCRtpReceiver.getCapabilities) return ["Only available on Chrome"]; const t = [] , s = (0, ke.splitSections)(e.sdp); return s.shift(), s.forEach((e=>{ const {codecs: s} = (0, ke.parseRtpParameters)(e) , n = /(VP\d|H26\d|AV1).*/; s.forEach((e=>{ const s = e.name + " " + Object.keys(e.parameters || {}).map((t=>t + "=" + e.parameters[t])).join(";"); if (null !== n.exec(s)) { "VP9" == e.name && (e.parameters = { "profile-id": "0" }); const s = e.name + " " + Object.keys(e.parameters || {}).map((t=>t + "=" + e.parameters[t])).join(";"); t.push(s) } } )) } )), t } } class De { constructor() { this.PixelStreamingSettings = new _e, this.EncoderSettings = new Ue, this.WebRTCSettings = new ze } ueCompatible() { null != this.WebRTCSettings.MaxFPS && (this.WebRTCSettings.FPS = this.WebRTCSettings.MaxFPS) } } class _e { } class Ue { } class ze { } class Be { constructor() { this.ReceiptTimeMs = null, this.TransmissionTimeMs = null, this.PreCaptureTimeMs = null, this.PostCaptureTimeMs = null, this.PreEncodeTimeMs = null, this.PostEncodeTimeMs = null, this.EncodeMs = null, this.CaptureToSendMs = null, this.testStartTimeMs = 0, this.browserReceiptTimeMs = 0, this.latencyExcludingDecode = 0, this.testDuration = 0, this.networkLatency = 0, this.browserSendLatency = 0, this.frameDisplayDeltaTimeMs = 0, this.endToEndLatency = 0, this.encodeLatency = 0 } setFrameDisplayDeltaTime(e) { 0 == this.frameDisplayDeltaTimeMs && (this.frameDisplayDeltaTimeMs = Math.round(e)) } processFields() { null != this.EncodeMs || null == this.PreEncodeTimeMs && null == this.PostEncodeTimeMs || (l.Log(l.GetStackTrace(), `Setting Encode Ms \n ${this.PostEncodeTimeMs} \n ${this.PreEncodeTimeMs}`, 6), this.EncodeMs = this.PostEncodeTimeMs - this.PreEncodeTimeMs), null != this.CaptureToSendMs || null == this.PreCaptureTimeMs && null == this.PostCaptureTimeMs || (l.Log(l.GetStackTrace(), `Setting CaptureToSendMs Ms \n ${this.PostCaptureTimeMs} \n ${this.PreCaptureTimeMs}`, 6), this.CaptureToSendMs = this.PostCaptureTimeMs - this.PreCaptureTimeMs) } } class Ne { static setExtensionFromBytes(e, t) { t.receiving || (t.mimetype = "", t.extension = "", t.receiving = !0, t.valid = !1, t.size = 0, t.data = [], t.timestampStart = (new Date).getTime(), l.Log(l.GetStackTrace(), "Received first chunk of file", 6)); const s = new TextDecoder("utf-16").decode(e.slice(1)); l.Log(l.GetStackTrace(), s, 6), t.extension = s } static setMimeTypeFromBytes(e, t) { t.receiving || (t.mimetype = "", t.extension = "", t.receiving = !0, t.valid = !1, t.size = 0, t.data = [], t.timestampStart = (new Date).getTime(), l.Log(l.GetStackTrace(), "Received first chunk of file", 6)); const s = new TextDecoder("utf-16").decode(e.slice(1)); l.Log(l.GetStackTrace(), s, 6), t.mimetype = s } static setContentsFromBytes(e, t) { if (!t.receiving) return; t.size = Math.ceil(new DataView(e.slice(1, 5).buffer).getInt32(0, !0) / 16379); const s = e.slice(5); if (t.data.push(s), l.Log(l.GetStackTrace(), `Received file chunk: ${t.data.length}/${t.size}`, 6), t.data.length === t.size) { t.receiving = !1, t.valid = !0, l.Log(l.GetStackTrace(), "Received complete file", 6); const e = (new Date).getTime() - t.timestampStart , s = Math.round(16 * t.size * 1024 / e); l.Log(l.GetStackTrace(), `Average transfer bitrate: ${s}kb/s over ${e / 1e3} seconds`, 6); const n = new Blob(t.data,{ type: t.mimetype }) , i = document.createElement("a"); i.setAttribute("href", URL.createObjectURL(n)), i.setAttribute("download", `transfer.${t.extension}`), document.body.append(i), i.remove() } else t.data.length > t.size && (t.receiving = !1, l.Error(l.GetStackTrace(), `Received bigger file than advertised: ${t.data.length}/${t.size}`)) } } class Ge { constructor() { this.mimetype = "", this.extension = "", this.receiving = !1, this.size = 0, this.data = [], this.valid = !1 } } class He { } He.mainButton = 0, He.auxiliaryButton = 1, He.secondaryButton = 2, He.fourthButton = 3, He.fifthButton = 4; class We { } We.primaryButton = 1, We.secondaryButton = 2, We.auxiliaryButton = 4, We.fourthButton = 8, We.fifthButton = 16; class Ve { constructor() { this.unregisterCallbacks = [] } addUnregisterCallback(e) { this.unregisterCallbacks.push(e) } unregisterAll() { for (const e of this.unregisterCallbacks) e(); this.unregisterCallbacks = [] } } class Qe { constructor(e, t, s) { this.touchEventListenerTracker = new Ve, this.toStreamerMessagesProvider = e, this.videoElementProvider = t, this.coordinateConverter = s; const n = e=>this.onTouchStart(e) , i = e=>this.onTouchEnd(e) , r = e=>this.onTouchMove(e); document.addEventListener("touchstart", n, { passive: !1 }), document.addEventListener("touchend", i, { passive: !1 }), document.addEventListener("touchmove", r, { passive: !1 }), this.touchEventListenerTracker.addUnregisterCallback((()=>document.removeEventListener("touchstart", n))), this.touchEventListenerTracker.addUnregisterCallback((()=>document.removeEventListener("touchend", i))), this.touchEventListenerTracker.addUnregisterCallback((()=>document.removeEventListener("touchmove", r))) } unregisterTouchEvents() { this.touchEventListenerTracker.unregisterAll() } setVideoElementParentClientRect(e) { this.videoElementParentClientRect = e } onTouchStart(e) { if (this.videoElementProvider.isVideoReady()) { if (null == this.fakeTouchFinger) { const t = e.changedTouches[0]; this.fakeTouchFinger = new qe(t.identifier,t.clientX - this.videoElementParentClientRect.left,t.clientY - this.videoElementParentClientRect.top); const s = this.videoElementProvider.getVideoParentElement() , n = new MouseEvent("mouseenter",t); s.dispatchEvent(n); const i = this.coordinateConverter.normalizeAndQuantizeUnsigned(this.fakeTouchFinger.x, this.fakeTouchFinger.y); this.toStreamerMessagesProvider.toStreamerHandlers.get("MouseDown")([He.mainButton, i.x, i.y]) } e.preventDefault() } } onTouchEnd(e) { if (!this.videoElementProvider.isVideoReady()) return; const t = this.videoElementProvider.getVideoParentElement() , s = this.toStreamerMessagesProvider.toStreamerHandlers; for (let n = 0; n < e.changedTouches.length; n++) { const i = e.changedTouches[n]; if (i.identifier === this.fakeTouchFinger.id) { const e = i.clientX - this.videoElementParentClientRect.left , n = i.clientY - this.videoElementParentClientRect.top , r = this.coordinateConverter.normalizeAndQuantizeUnsigned(e, n); s.get("MouseUp")([He.mainButton, r.x, r.y]); const o = new MouseEvent("mouseleave",i); t.dispatchEvent(o), this.fakeTouchFinger = null; break } } e.preventDefault() } onTouchMove(e) { if (!this.videoElementProvider.isVideoReady()) return; const t = this.toStreamerMessagesProvider.toStreamerHandlers; for (let s = 0; s < e.touches.length; s++) { const n = e.touches[s]; if (n.identifier === this.fakeTouchFinger.id) { const e = n.clientX - this.videoElementParentClientRect.left , s = n.clientY - this.videoElementParentClientRect.top , i = this.coordinateConverter.normalizeAndQuantizeUnsigned(e, s) , r = this.coordinateConverter.normalizeAndQuantizeSigned(e - this.fakeTouchFinger.x, s - this.fakeTouchFinger.y); t.get("MouseMove")([i.x, i.y, r.x, r.y]), this.fakeTouchFinger.x = e, this.fakeTouchFinger.y = s; break } } e.preventDefault() } } class qe { constructor(e, t, s) { this.id = e, this.x = t, this.y = s } } class Ke { } Ke.backSpace = 8, Ke.shift = 16, Ke.control = 17, Ke.alt = 18, Ke.rightShift = 253, Ke.rightControl = 254, Ke.rightAlt = 255; class je { constructor(e, t, s) { this.keyboardEventListenerTracker = new Ve, this.CodeToKeyCode = { Escape: 27, Digit0: 48, Digit1: 49, Digit2: 50, Digit3: 51, Digit4: 52, Digit5: 53, Digit6: 54, Digit7: 55, Digit8: 56, Digit9: 57, Minus: 173, Equal: 187, Backspace: 8, Tab: 9, KeyQ: 81, KeyW: 87, KeyE: 69, KeyR: 82, KeyT: 84, KeyY: 89, KeyU: 85, KeyI: 73, KeyO: 79, KeyP: 80, BracketLeft: 219, BracketRight: 221, Enter: 13, ControlLeft: 17, KeyA: 65, KeyS: 83, KeyD: 68, KeyF: 70, KeyG: 71, KeyH: 72, KeyJ: 74, KeyK: 75, KeyL: 76, Semicolon: 186, Quote: 222, Backquote: 192, ShiftLeft: 16, Backslash: 220, KeyZ: 90, KeyX: 88, KeyC: 67, KeyV: 86, KeyB: 66, KeyN: 78, KeyM: 77, Comma: 188, Period: 190, Slash: 191, ShiftRight: 253, AltLeft: 18, Space: 32, CapsLock: 20, F1: 112, F2: 113, F3: 114, F4: 115, F5: 116, F6: 117, F7: 118, F8: 119, F9: 120, F10: 121, F11: 122, F12: 123, Pause: 19, ScrollLock: 145, NumpadDivide: 111, NumpadMultiply: 106, NumpadSubtract: 109, NumpadAdd: 107, NumpadDecimal: 110, Numpad9: 105, Numpad8: 104, Numpad7: 103, Numpad6: 102, Numpad5: 101, Numpad4: 100, Numpad3: 99, Numpad2: 98, Numpad1: 97, Numpad0: 96, NumLock: 144, ControlRight: 254, AltRight: 255, Home: 36, End: 35, ArrowUp: 38, ArrowLeft: 37, ArrowRight: 39, ArrowDown: 40, PageUp: 33, PageDown: 34, Insert: 45, Delete: 46, ContextMenu: 93 }, this.toStreamerMessagesProvider = e, this.config = t, this.activeKeysProvider = s } registerKeyBoardEvents() { const e = e=>this.handleOnKeyDown(e) , t = e=>this.handleOnKeyUp(e) , s = e=>this.handleOnKeyPress(e); document.addEventListener("keydown", e), document.addEventListener("keyup", t), document.addEventListener("keypress", s), this.keyboardEventListenerTracker.addUnregisterCallback((()=>document.removeEventListener("keydown", e))), this.keyboardEventListenerTracker.addUnregisterCallback((()=>document.removeEventListener("keyup", t))), this.keyboardEventListenerTracker.addUnregisterCallback((()=>document.removeEventListener("keypress", s))) } unregisterKeyBoardEvents() { this.keyboardEventListenerTracker.unregisterAll() } handleOnKeyDown(e) { const t = this.getKeycode(e); t && (l.Log(l.GetStackTrace(), `key down ${t}, repeat = ${e.repeat}`, 6), this.toStreamerMessagesProvider.toStreamerHandlers.get("KeyDown")([this.getKeycode(e), e.repeat ? 1 : 0]), this.activeKeysProvider.getActiveKeys().push(t), t === Ke.backSpace && document.dispatchEvent(new KeyboardEvent("keypress",{ charCode: Ke.backSpace })), this.config.isFlagEnabled(ae.SuppressBrowserKeys) && this.isKeyCodeBrowserKey(t) && e.preventDefault()) } handleOnKeyUp(e) { const t = this.getKeycode(e); t && (l.Log(l.GetStackTrace(), `key up ${t}`, 6), this.toStreamerMessagesProvider.toStreamerHandlers.get("KeyUp")([t, e.repeat ? 1 : 0]), this.config.isFlagEnabled(ae.SuppressBrowserKeys) && this.isKeyCodeBrowserKey(t) && e.preventDefault()) } handleOnKeyPress(e) { if (!("charCode"in e)) return void l.Warning(l.GetStackTrace(), "KeyboardEvent.charCode is deprecated in this browser, cannot send key press."); const t = e.charCode; l.Log(l.GetStackTrace(), `key press ${t}`, 6), this.toStreamerMessagesProvider.toStreamerHandlers.get("KeyPress")([t]) } getKeycode(e) { if (!("keyCode"in e)) { const t = e; return t.code in this.CodeToKeyCode ? this.CodeToKeyCode[t.code] : (l.Warning(l.GetStackTrace(), `Keyboard code of ${t.code} is not supported in our mapping, ignoring this key.`), null) } return e.keyCode === Ke.shift && "ShiftRight" === e.code ? Ke.rightShift : e.keyCode === Ke.control && "ControlRight" === e.code ? Ke.rightControl : e.keyCode === Ke.alt && "AltRight" === e.code ? Ke.rightAlt : e.keyCode } isKeyCodeBrowserKey(e) { return e >= 112 && e <= 123 || 9 === e } } class Xe { constructor(e, t, s) { this.x = 0, this.y = 0, this.updateMouseMovePositionEvent = e=>{ this.updateMouseMovePosition(e) } , this.mouseEventListenerTracker = new Ve, this.videoElementProvider = e, this.mouseController = t, this.activeKeysProvider = s; const n = this.videoElementProvider.getVideoParentElement(); this.x = n.getBoundingClientRect().width / 2, this.y = n.getBoundingClientRect().height / 2, this.coord = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(this.x, this.y) } unregisterMouseEvents() { this.mouseEventListenerTracker.unregisterAll() } lockStateChange() { const e = this.videoElementProvider.getVideoParentElement() , t = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers; if (document.pointerLockElement === e || document.mozPointerLockElement === e) l.Log(l.GetStackTrace(), "Pointer locked", 6), document.addEventListener("mousemove", this.updateMouseMovePositionEvent, !1), this.mouseEventListenerTracker.addUnregisterCallback((()=>document.removeEventListener("mousemove", this.updateMouseMovePositionEvent, !1))); else { l.Log(l.GetStackTrace(), "The pointer lock status is now unlocked", 6), document.removeEventListener("mousemove", this.updateMouseMovePositionEvent, !1); let e = this.activeKeysProvider.getActiveKeys(); const s = new Set(e) , n = []; s.forEach((e=>{ n[e] } )), n.forEach((e=>{ t.get("KeyUp")([e]) } )), e = [] } } updateMouseMovePosition(e) { if (!this.videoElementProvider.isVideoReady()) return; const t = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers , s = this.videoElementProvider.getVideoParentElement().clientWidth , n = this.videoElementProvider.getVideoParentElement().clientHeight; this.x += e.movementX, this.y += e.movementY, this.x > s && (this.x -= s), this.y > n && (this.y -= n), this.x < 0 && (this.x = s + this.x), this.y < 0 && (this.y = n - this.y), this.coord = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(this.x, this.y); const i = this.mouseController.coordinateConverter.normalizeAndQuantizeSigned(e.movementX, e.movementY); t.get("MouseMove")([this.coord.x, this.coord.y, i.x, i.y]) } handleMouseDown(e) { this.videoElementProvider.isVideoReady() && this.mouseController.toStreamerMessagesProvider.toStreamerHandlers.get("MouseDown")([e.button, this.coord.x, this.coord.y]) } handleMouseUp(e) { this.videoElementProvider.isVideoReady() && this.mouseController.toStreamerMessagesProvider.toStreamerHandlers.get("MouseUp")([e.button, this.coord.x, this.coord.y]) } handleMouseWheel(e) { this.videoElementProvider.isVideoReady() && this.mouseController.toStreamerMessagesProvider.toStreamerHandlers.get("MouseWheel")([e.wheelDelta, this.coord.x, this.coord.y]) } handleMouseDouble(e) { this.videoElementProvider.isVideoReady() && this.mouseController.toStreamerMessagesProvider.toStreamerHandlers.get("MouseDouble")([e.button, this.coord.x, this.coord.y]) } handlePressMouseButtons(e) { this.videoElementProvider.isVideoReady() && this.mouseController.pressMouseButtons(e.buttons, this.x, this.y) } handleReleaseMouseButtons(e) { this.videoElementProvider.isVideoReady() && this.mouseController.releaseMouseButtons(e.buttons, this.x, this.y) } } class Je { constructor(e) { this.mouseController = e } unregisterMouseEvents() {} updateMouseMovePosition(e) { if (!this.mouseController.videoElementProvider.isVideoReady()) return; l.Log(l.GetStackTrace(), "MouseMove", 6); const t = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(e.offsetX, e.offsetY) , s = this.mouseController.coordinateConverter.normalizeAndQuantizeSigned(e.movementX, e.movementY); this.mouseController.toStreamerMessagesProvider.toStreamerHandlers.get("MouseMove")([t.x, t.y, s.x, s.y]), e.preventDefault() } handleMouseDown(e) { if (!this.mouseController.videoElementProvider.isVideoReady()) return; l.Log(l.GetStackTrace(), "onMouse Down", 6); const t = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(e.offsetX, e.offsetY); this.mouseController.toStreamerMessagesProvider.toStreamerHandlers.get("MouseDown")([e.button, t.x, t.y]), e.preventDefault() } handleMouseUp(e) { if (!this.mouseController.videoElementProvider.isVideoReady()) return; const t = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(e.offsetX, e.offsetY); this.mouseController.toStreamerMessagesProvider.toStreamerHandlers.get("MouseUp")([e.button, t.x, t.y]), e.preventDefault() } handleContextMenu(e) { if (!this.mouseController.videoElementProvider.isVideoReady()) return; const t = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(e.offsetX, e.offsetY); this.mouseController.toStreamerMessagesProvider.toStreamerHandlers.get("MouseUp")([e.button, t.x, t.y]), e.preventDefault() } handleMouseWheel(e) { if (!this.mouseController.videoElementProvider.isVideoReady()) return; const t = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(e.offsetX, e.offsetY); this.mouseController.toStreamerMessagesProvider.toStreamerHandlers.get("MouseWheel")([e.wheelDelta, t.x, t.y]), e.preventDefault() } handleMouseDouble(e) { if (!this.mouseController.videoElementProvider.isVideoReady()) return; const t = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(e.offsetX, e.offsetY); this.mouseController.toStreamerMessagesProvider.toStreamerHandlers.get("MouseDouble")([e.button, t.x, t.y]) } handlePressMouseButtons(e) { this.mouseController.videoElementProvider.isVideoReady() && this.mouseController.pressMouseButtons(e.buttons, e.offsetX, e.offsetY) } handleReleaseMouseButtons(e) { this.mouseController.videoElementProvider.isVideoReady() && this.mouseController.releaseMouseButtons(e.buttons, e.offsetX, e.offsetY) } } class Ye { constructor(e, t, s, n) { this.mouseEventListenerTracker = new Ve, this.toStreamerMessagesProvider = e, this.coordinateConverter = s, this.videoElementProvider = t, this.activeKeysProvider = n, this.registerMouseEnterAndLeaveEvents() } unregisterMouseEvents() { this.mouseEventListenerTracker.unregisterAll() } registerLockedMouseEvents(e) { const t = this.videoElementProvider.getVideoParentElement() , s = new Xe(this.videoElementProvider,e,this.activeKeysProvider); if (t.requestPointerLock = t.requestPointerLock || t.mozRequestPointerLock, document.exitPointerLock = document.exitPointerLock || document.mozExitPointerLock, t.requestPointerLock) { const e = ()=>{ t.requestPointerLock() } ; t.addEventListener("click", e), this.mouseEventListenerTracker.addUnregisterCallback((()=>t.removeEventListener("click", e))) } const n = ()=>s.lockStateChange(); document.addEventListener("pointerlockchange", n, !1), document.addEventListener("mozpointerlockchange", n, !1), this.mouseEventListenerTracker.addUnregisterCallback((()=>document.removeEventListener("pointerlockchange", n, !1))), this.mouseEventListenerTracker.addUnregisterCallback((()=>document.removeEventListener("mozpointerlockchange", n, !1))); const i = e=>s.handleMouseDown(e) , r = e=>s.handleMouseUp(e) , o = e=>s.handleMouseWheel(e) , a = e=>s.handleMouseDouble(e); t.addEventListener("mousedown", i), t.addEventListener("mouseup", r), t.addEventListener("wheel", o), t.addEventListener("dblclick", a), this.mouseEventListenerTracker.addUnregisterCallback((()=>t.removeEventListener("mousedown", i))), this.mouseEventListenerTracker.addUnregisterCallback((()=>t.removeEventListener("mouseup", r))), this.mouseEventListenerTracker.addUnregisterCallback((()=>t.removeEventListener("wheel", o))), this.mouseEventListenerTracker.addUnregisterCallback((()=>t.removeEventListener("dblclick", a))), this.mouseEventListenerTracker.addUnregisterCallback((()=>s.unregisterMouseEvents())), this.mouseEventListenerTracker.addUnregisterCallback((()=>{ !document.exitPointerLock || document.pointerLockElement !== t && document.mozPointerLockElement !== t || document.exitPointerLock() } )) } registerHoveringMouseEvents(e) { const t = this.videoElementProvider.getVideoParentElement() , s = new Je(e) , n = e=>s.updateMouseMovePosition(e) , i = e=>s.handleMouseDown(e) , r = e=>s.handleMouseUp(e) , o = e=>s.handleContextMenu(e) , a = e=>s.handleMouseWheel(e) , l = e=>s.handleMouseDouble(e); t.addEventListener("mousemove", n), t.addEventListener("mousedown", i), t.addEventListener("mouseup", r), t.addEventListener("contextmenu", o), t.addEventListener("wheel", a), t.addEventListener("dblclick", l), this.mouseEventListenerTracker.addUnregisterCallback((()=>t.removeEventListener("mousemove", n))), this.mouseEventListenerTracker.addUnregisterCallback((()=>t.removeEventListener("mousedown", i))), this.mouseEventListenerTracker.addUnregisterCallback((()=>t.removeEventListener("mouseup", r))), this.mouseEventListenerTracker.addUnregisterCallback((()=>t.removeEventListener("contextmenu", o))), this.mouseEventListenerTracker.addUnregisterCallback((()=>t.removeEventListener("wheel", a))), this.mouseEventListenerTracker.addUnregisterCallback((()=>t.removeEventListener("dblclick", l))), this.mouseEventListenerTracker.addUnregisterCallback((()=>s.unregisterMouseEvents())) } registerMouseEnterAndLeaveEvents() { const e = this.videoElementProvider.getVideoParentElement() , t = e=>{ this.videoElementProvider.isVideoReady() && (l.Log(l.GetStackTrace(), "Mouse Entered", 6), this.sendMouseEnter(), this.pressMouseButtons(e.buttons, e.x, e.y)) } , s = e=>{ this.videoElementProvider.isVideoReady() && (l.Log(l.GetStackTrace(), "Mouse Left", 6), this.sendMouseLeave(), this.releaseMouseButtons(e.buttons, e.x, e.y)) } ; e.addEventListener("mouseenter", t), e.addEventListener("mouseleave", s), this.mouseEventListenerTracker.addUnregisterCallback((()=>e.removeEventListener("mouseenter", t))), this.mouseEventListenerTracker.addUnregisterCallback((()=>e.removeEventListener("mouseleave", s))) } releaseMouseButtons(e, t, s) { const n = this.coordinateConverter.normalizeAndQuantizeUnsigned(t, s); e & We.primaryButton && this.sendMouseUp(He.mainButton, n.x, n.y), e & We.secondaryButton && this.sendMouseUp(He.secondaryButton, n.x, n.y), e & We.auxiliaryButton && this.sendMouseUp(He.auxiliaryButton, n.x, n.y), e & We.fourthButton && this.sendMouseUp(He.fourthButton, n.x, n.y), e & We.fifthButton && this.sendMouseUp(He.fifthButton, n.x, n.y) } pressMouseButtons(e, t, s) { if (!this.videoElementProvider.isVideoReady()) return; const n = this.coordinateConverter.normalizeAndQuantizeUnsigned(t, s); e & We.primaryButton && this.sendMouseDown(He.mainButton, n.x, n.y), e & We.secondaryButton && this.sendMouseDown(He.secondaryButton, n.x, n.y), e & We.auxiliaryButton && this.sendMouseDown(He.auxiliaryButton, n.x, n.y), e & We.fourthButton && this.sendMouseDown(He.fourthButton, n.x, n.y), e & We.fifthButton && this.sendMouseDown(He.fifthButton, n.x, n.y) } sendMouseEnter() { this.videoElementProvider.isVideoReady() && this.toStreamerMessagesProvider.toStreamerHandlers.get("MouseEnter")() } sendMouseLeave() { this.videoElementProvider.isVideoReady() && this.toStreamerMessagesProvider.toStreamerHandlers.get("MouseLeave")() } sendMouseDown(e, t, s) { this.videoElementProvider.isVideoReady() && (l.Log(l.GetStackTrace(), `mouse button ${e} down at (${t}, ${s})`, 6), this.toStreamerMessagesProvider.toStreamerHandlers.get("MouseDown")([e, t, s])) } sendMouseUp(e, t, s) { if (!this.videoElementProvider.isVideoReady()) return; l.Log(l.GetStackTrace(), `mouse button ${e} up at (${t}, ${s})`, 6); const n = this.coordinateConverter.normalizeAndQuantizeUnsigned(t, s); this.toStreamerMessagesProvider.toStreamerHandlers.get("MouseUp")([e, n.x, n.y]) } } class $e { constructor(e, t, s) { this.fingers = [9, 8, 7, 6, 5, 4, 3, 2, 1, 0], this.fingerIds = new Map, this.maxByteValue = 255, this.touchEventListenerTracker = new Ve, this.toStreamerMessagesProvider = e, this.videoElementProvider = t, this.coordinateConverter = s, this.videoElementParent = t.getVideoElement(); const n = e=>this.onTouchStart(e) , i = e=>this.onTouchEnd(e) , r = e=>this.onTouchMove(e); this.videoElementParent.addEventListener("touchstart", n, { passive: !1 }), this.videoElementParent.addEventListener("touchend", i, { passive: !1 }), this.videoElementParent.addEventListener("touchmove", r, { passive: !1 }), this.touchEventListenerTracker.addUnregisterCallback((()=>this.videoElementParent.removeEventListener("touchstart", n))), this.touchEventListenerTracker.addUnregisterCallback((()=>this.videoElementParent.removeEventListener("touchend", i))), this.touchEventListenerTracker.addUnregisterCallback((()=>this.videoElementParent.removeEventListener("touchmove", r))), l.Log(l.GetStackTrace(), "Touch Events Registered", 6); const o = e=>{ e.preventDefault() } ; document.addEventListener("touchmove", o, { passive: !1 }), this.touchEventListenerTracker.addUnregisterCallback((()=>document.removeEventListener("touchmove", o))) } unregisterTouchEvents() { this.touchEventListenerTracker.unregisterAll() } rememberTouch(e) { const t = this.fingers.pop(); void 0 === t && l.Log(l.GetStackTrace(), "exhausted touch identifiers", 6), this.fingerIds.set(e.identifier, t) } forgetTouch(e) { this.fingers.push(this.fingerIds.get(e.identifier)), this.fingers.sort((function(e, t) { return t - e } )), this.fingerIds.delete(e.identifier) } onTouchStart(e) { if (this.videoElementProvider.isVideoReady()) { for (let t = 0; t < e.changedTouches.length; t++) this.rememberTouch(e.changedTouches[t]); l.Log(l.GetStackTrace(), "touch start", 6), this.emitTouchData("TouchStart", e.changedTouches), e.preventDefault() } } onTouchEnd(e) { if (this.videoElementProvider.isVideoReady()) { l.Log(l.GetStackTrace(), "touch end", 6), this.emitTouchData("TouchEnd", e.changedTouches); for (let t = 0; t < e.changedTouches.length; t++) this.forgetTouch(e.changedTouches[t]); e.preventDefault() } } onTouchMove(e) { this.videoElementProvider.isVideoReady() && (l.Log(l.GetStackTrace(), "touch move", 6), this.emitTouchData("TouchMove", e.touches), e.preventDefault()) } emitTouchData(e, t) { if (!this.videoElementProvider.isVideoReady()) return; const s = this.videoElementProvider.getVideoParentElement() , n = this.toStreamerMessagesProvider.toStreamerHandlers; for (let i = 0; i < t.length; i++) { const r = 1 , o = t[i] , a = o.clientX - s.offsetLeft , c = o.clientY - s.offsetTop; l.Log(l.GetStackTrace(), `F ${this.fingerIds.get(o.identifier)}=(${a}, ${c})`, 6); const d = this.coordinateConverter.normalizeAndQuantizeUnsigned(a, c); switch (e) { case "TouchStart": n.get("TouchStart")([r, d.x, d.y, this.fingerIds.get(o.identifier), this.maxByteValue * o.force, d.inRange ? 1 : 0]); break; case "TouchEnd": n.get("TouchEnd")([r, d.x, d.y, this.fingerIds.get(o.identifier), this.maxByteValue * o.force, d.inRange ? 1 : 0]); break; case "TouchMove": n.get("TouchMove")([r, d.x, d.y, this.fingerIds.get(o.identifier), this.maxByteValue * o.force, d.inRange ? 1 : 0]) } } } } class Ze { constructor(e) { this.gamePadEventListenerTracker = new Ve, this.toStreamerMessagesProvider = e, this.requestAnimationFrame = (window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || window.requestAnimationFrame).bind(window); const t = window; if ("GamepadEvent"in t) { const e = e=>this.gamePadConnectHandler(e) , t = e=>this.gamePadDisconnectHandler(e); window.addEventListener("gamepadconnected", e), window.addEventListener("gamepaddisconnected", t), this.gamePadEventListenerTracker.addUnregisterCallback((()=>window.removeEventListener("gamepadconnected", e))), this.gamePadEventListenerTracker.addUnregisterCallback((()=>window.removeEventListener("gamepaddisconnected", t))) } else if ("WebKitGamepadEvent"in t) { const e = e=>this.gamePadConnectHandler(e) , t = e=>this.gamePadDisconnectHandler(e); window.addEventListener("webkitgamepadconnected", e), window.addEventListener("webkitgamepaddisconnected", t), this.gamePadEventListenerTracker.addUnregisterCallback((()=>window.removeEventListener("webkitgamepadconnected", e))), this.gamePadEventListenerTracker.addUnregisterCallback((()=>window.removeEventListener("webkitgamepaddisconnected", t))) } if (this.controllers = [], navigator.getGamepads) for (const e of navigator.getGamepads()) e && this.gamePadConnectHandler(new GamepadEvent("gamepadconnected",{ gamepad: e })) } unregisterGamePadEvents() { this.gamePadEventListenerTracker.unregisterAll(); for (const e of this.controllers) void 0 !== e.id && this.onGamepadDisconnected(e.id); this.controllers = [], this.onGamepadConnected = ()=>{} , this.onGamepadDisconnected = ()=>{} } gamePadConnectHandler(e) { l.Log(l.GetStackTrace(), "Gamepad connect handler", 6); const t = e.gamepad , s = { currentState: t, prevState: t, id: void 0 }; this.controllers.push(s), this.controllers[t.index].currentState = t, this.controllers[t.index].prevState = t, l.Log(l.GetStackTrace(), "gamepad: " + t.id + " connected", 6), window.requestAnimationFrame((()=>this.updateStatus())), this.onGamepadConnected() } gamePadDisconnectHandler(e) { l.Log(l.GetStackTrace(), "Gamepad disconnect handler", 6), l.Log(l.GetStackTrace(), "gamepad: " + e.gamepad.id + " disconnected", 6); const t = this.controllers[e.gamepad.index]; delete this.controllers[e.gamepad.index], this.controllers = this.controllers.filter((e=>void 0 !== e)), this.onGamepadDisconnected(t.id) } scanGamePads() { const e = navigator.getGamepads ? navigator.getGamepads() : navigator.webkitGetGamepads ? navigator.webkitGetGamepads() : []; for (let t = 0; t < e.length; t++) e[t] && e[t].index in this.controllers && (this.controllers[e[t].index].currentState = e[t]) } updateStatus() { this.scanGamePads(); const e = this.toStreamerMessagesProvider.toStreamerHandlers; for (const t of this.controllers) { const s = void 0 === t.id ? this.controllers.indexOf(t) : t.id , n = t.currentState; for (let n = 0; n < t.currentState.buttons.length; n++) { const i = t.currentState.buttons[n] , r = t.prevState.buttons[n]; i.pressed ? n == Ae.LeftTrigger ? e.get("GamepadAnalog")([s, 5, i.value]) : n == Ae.RightTrigger ? e.get("GamepadAnalog")([s, 6, i.value]) : e.get("GamepadButtonPressed")([s, n, r.pressed ? 1 : 0]) : !i.pressed && r.pressed && (n == Ae.LeftTrigger ? e.get("GamepadAnalog")([s, 5, 0]) : n == Ae.RightTrigger ? e.get("GamepadAnalog")([s, 6, 0]) : e.get("GamepadButtonReleased")([s, n])) } for (let t = 0; t < n.axes.length; t += 2) { const i = parseFloat(n.axes[t].toFixed(4)) , r = -parseFloat(n.axes[t + 1].toFixed(4)); e.get("GamepadAnalog")([s, t + 1, i]), e.get("GamepadAnalog")([s, t + 2, r]) } this.controllers[s].prevState = n } this.controllers.length > 0 && this.requestAnimationFrame((()=>this.updateStatus())) } onGamepadResponseReceived(e) { for (const t of this.controllers) if (void 0 === t.id) { t.id = e; break } } onGamepadConnected() {} onGamepadDisconnected(e) {} } !function(e) { e[e.RightClusterBottomButton = 0] = "RightClusterBottomButton", e[e.RightClusterRightButton = 1] = "RightClusterRightButton", e[e.RightClusterLeftButton = 2] = "RightClusterLeftButton", e[e.RightClusterTopButton = 3] = "RightClusterTopButton", e[e.LeftShoulder = 4] = "LeftShoulder", e[e.RightShoulder = 5] = "RightShoulder", e[e.LeftTrigger = 6] = "LeftTrigger", e[e.RightTrigger = 7] = "RightTrigger", e[e.SelectOrBack = 8] = "SelectOrBack", e[e.StartOrForward = 9] = "StartOrForward", e[e.LeftAnalogPress = 10] = "LeftAnalogPress", e[e.RightAnalogPress = 11] = "RightAnalogPress", e[e.LeftClusterTopButton = 12] = "LeftClusterTopButton", e[e.LeftClusterBottomButton = 13] = "LeftClusterBottomButton", e[e.LeftClusterLeftButton = 14] = "LeftClusterLeftButton", e[e.LeftClusterRightButton = 15] = "LeftClusterRightButton", e[e.CentreButton = 16] = "CentreButton", e[e.LeftStickHorizontal = 0] = "LeftStickHorizontal", e[e.LeftStickVertical = 1] = "LeftStickVertical", e[e.RightStickHorizontal = 2] = "RightStickHorizontal", e[e.RightStickVertical = 3] = "RightStickVertical" }(Ae || (Ae = {})); class et { constructor(e, t, s) { this.activeKeys = new tt, this.toStreamerMessagesProvider = e, this.videoElementProvider = t, this.coordinateConverter = s } registerKeyBoard(e) { l.Log(l.GetStackTrace(), "Register Keyboard Events", 7); const t = new je(this.toStreamerMessagesProvider,e,this.activeKeys); return t.registerKeyBoardEvents(), t } registerMouse(e) { l.Log(l.GetStackTrace(), "Register Mouse Events", 7); const t = new Ye(this.toStreamerMessagesProvider,this.videoElementProvider,this.coordinateConverter,this.activeKeys); switch (e) { case ve.LockedMouse: t.registerLockedMouseEvents(t); break; case ve.HoveringMouse: t.registerHoveringMouseEvents(t); break; default: l.Info(l.GetStackTrace(), "unknown Control Scheme Type Defaulting to Locked Mouse Events"), t.registerLockedMouseEvents(t) } return t } registerTouch(e, t) { if (l.Log(l.GetStackTrace(), "Registering Touch", 6), e) { const e = new Qe(this.toStreamerMessagesProvider,this.videoElementProvider,this.coordinateConverter); return e.setVideoElementParentClientRect(t), e } return new $e(this.toStreamerMessagesProvider,this.videoElementProvider,this.coordinateConverter) } registerGamePad() { return l.Log(l.GetStackTrace(), "Register Game Pad", 7), new Ze(this.toStreamerMessagesProvider) } } class tt { constructor() { this.activeKeys = [], this.activeKeys = [] } getActiveKeys() { return this.activeKeys } } class st { constructor(e, t) { this.lastTimeResized = (new Date).getTime(), this.videoElement = document.createElement("video"), this.config = t, this.videoElement.id = "streamingVideo", this.videoElement.disablePictureInPicture = !0, this.videoElement.playsInline = !0, this.videoElement.style.width = "100%", this.videoElement.style.height = "100%", this.videoElement.style.position = "absolute", this.videoElement.style.pointerEvents = "all", e.appendChild(this.videoElement), this.onResizePlayerCallback = ()=>{ console.log("Resolution changed, restyling player, did you forget to override this function?") } , this.onMatchViewportResolutionCallback = ()=>{ console.log("Resolution changed and match viewport resolution is turned on, did you forget to override this function?") } , this.videoElement.onclick = ()=>{ this.videoElement.paused && this.videoElement.play() } , this.videoElement.onloadedmetadata = ()=>{ this.onVideoInitialized() } , window.addEventListener("resize", (()=>this.resizePlayerStyle()), !0), window.addEventListener("orientationchange", (()=>this.onOrientationChange())) } play() { return this.videoElement.muted = this.config.isFlagEnabled(ae.StartVideoMuted), this.videoElement.autoplay = this.config.isFlagEnabled(ae.AutoPlayVideo), this.videoElement.play() } isPaused() { return this.videoElement.paused } isVideoReady() { return void 0 !== this.videoElement.readyState && this.videoElement.readyState > 0 } hasVideoSource() { return void 0 !== this.videoElement.srcObject && null !== this.videoElement.srcObject } getVideoElement() { return this.videoElement } getVideoParentElement() { return this.videoElement.parentElement } setVideoEnabled(e) { this.videoElement.srcObject.getTracks().forEach((t=>t.enabled = e)) } onVideoInitialized() {} onOrientationChange() { clearTimeout(this.orientationChangeTimeout), this.orientationChangeTimeout = window.setTimeout((()=>{ this.resizePlayerStyle() } ), 500) } resizePlayerStyle() { const e = this.getVideoParentElement(); e && (this.updateVideoStreamSize(), e.classList.contains("fixed-size") || this.resizePlayerStyleToFillParentElement(), this.onResizePlayerCallback()) } resizePlayerStyleToFillParentElement() { this.getVideoParentElement().setAttribute("style", "top: 0px; left: 0px; width: 100%; height: 100%; cursor: default;") } updateVideoStreamSize() { if (this.config.isFlagEnabled(ae.MatchViewportResolution)) if ((new Date).getTime() - this.lastTimeResized > 1e3) { const e = this.getVideoParentElement(); if (!e) return; this.onMatchViewportResolutionCallback(e.clientWidth, e.clientHeight), this.lastTimeResized = (new Date).getTime() } else l.Log(l.GetStackTrace(), "Resizing too often - skipping", 6), clearTimeout(this.resizeTimeoutHandle), this.resizeTimeoutHandle = window.setTimeout((()=>this.updateVideoStreamSize), 1e3) } } class nt { constructor() { this.map = new Map, this.reverseMap = new Map } getFromKey(e) { return this.map.get(e) } getFromValue(e) { return this.reverseMap.get(e) } add(e, t) { this.map.set(e, t), this.reverseMap.set(t, e) } remove(e, t) { this.map.delete(e), this.reverseMap.delete(t) } } class it { constructor() { this.toStreamerHandlers = new Map, this.fromStreamerHandlers = new Map, this.toStreamerMessages = new nt, this.fromStreamerMessages = new nt } populateDefaultProtocol() { this.toStreamerMessages.add("IFrameRequest", { id: 0, byteLength: 0, structure: [] }), this.toStreamerMessages.add("RequestQualityControl", { id: 1, byteLength: 0, structure: [] }), this.toStreamerMessages.add("FpsRequest", { id: 2, byteLength: 0, structure: [] }), this.toStreamerMessages.add("AverageBitrateRequest", { id: 3, byteLength: 0, structure: [] }), this.toStreamerMessages.add("StartStreaming", { id: 4, byteLength: 0, structure: [] }), this.toStreamerMessages.add("StopStreaming", { id: 5, byteLength: 0, structure: [] }), this.toStreamerMessages.add("LatencyTest", { id: 6, byteLength: 0, structure: [] }), this.toStreamerMessages.add("RequestInitialSettings", { id: 7, byteLength: 0, structure: [] }), this.toStreamerMessages.add("TestEcho", { id: 8, byteLength: 0, structure: [] }), this.toStreamerMessages.add("UIInteraction", { id: 50, byteLength: 0, structure: [] }), this.toStreamerMessages.add("Command", { id: 51, byteLength: 0, structure: [] }), this.toStreamerMessages.add("KeyDown", { id: 60, byteLength: 2, structure: ["uint8", "uint8"] }), this.toStreamerMessages.add("KeyUp", { id: 61, byteLength: 1, structure: ["uint8"] }), this.toStreamerMessages.add("KeyPress", { id: 62, byteLength: 2, structure: ["uint16"] }), this.toStreamerMessages.add("MouseEnter", { id: 70, byteLength: 0, structure: [] }), this.toStreamerMessages.add("MouseLeave", { id: 71, byteLength: 0, structure: [] }), this.toStreamerMessages.add("MouseDown", { id: 72, byteLength: 5, structure: ["uint8", "uint16", "uint16"] }), this.toStreamerMessages.add("MouseUp", { id: 73, byteLength: 5, structure: ["uint8", "uint16", "uint16"] }), this.toStreamerMessages.add("MouseMove", { id: 74, byteLength: 8, structure: ["uint16", "uint16", "int16", "int16"] }), this.toStreamerMessages.add("MouseWheel", { id: 75, byteLength: 6, structure: ["int16", "uint16", "uint16"] }), this.toStreamerMessages.add("MouseDouble", { id: 76, byteLength: 5, structure: ["uint8", "uint16", "uint16"] }), this.toStreamerMessages.add("TouchStart", { id: 80, byteLength: 8, structure: ["uint8", "uint16", "uint16", "uint8", "uint8", "uint8"] }), this.toStreamerMessages.add("TouchEnd", { id: 81, byteLength: 8, structure: ["uint8", "uint16", "uint16", "uint8", "uint8", "uint8"] }), this.toStreamerMessages.add("TouchMove", { id: 82, byteLength: 8, structure: ["uint8", "uint16", "uint16", "uint8", "uint8", "uint8"] }), this.toStreamerMessages.add("GamepadConnected", { id: 93, byteLength: 0, structure: [] }), this.toStreamerMessages.add("GamepadButtonPressed", { id: 90, byteLength: 3, structure: ["uint8", "uint8", "uint8"] }), this.toStreamerMessages.add("GamepadButtonReleased", { id: 91, byteLength: 3, structure: ["uint8", "uint8", "uint8"] }), this.toStreamerMessages.add("GamepadAnalog", { id: 92, byteLength: 10, structure: ["uint8", "uint8", "double"] }), this.toStreamerMessages.add("GamepadDisconnected", { id: 94, byteLength: 1, structure: ["uint8"] }), this.fromStreamerMessages.add("QualityControlOwnership", 0), this.fromStreamerMessages.add("Response", 1), this.fromStreamerMessages.add("Command", 2), this.fromStreamerMessages.add("FreezeFrame", 3), this.fromStreamerMessages.add("UnfreezeFrame", 4), this.fromStreamerMessages.add("VideoEncoderAvgQP", 5), this.fromStreamerMessages.add("LatencyTest", 6), this.fromStreamerMessages.add("InitialSettings", 7), this.fromStreamerMessages.add("FileExtension", 8), this.fromStreamerMessages.add("FileMimeType", 9), this.fromStreamerMessages.add("FileContents", 10), this.fromStreamerMessages.add("TestEcho", 11), this.fromStreamerMessages.add("InputControlOwnership", 12), this.fromStreamerMessages.add("GamepadResponse", 13), this.fromStreamerMessages.add("Protocol", 255) } registerMessageHandler(e, t, s) { switch (e) { case Fe.ToStreamer: this.toStreamerHandlers.set(t, s); break; case Fe.FromStreamer: this.fromStreamerHandlers.set(t, s); break; default: l.Log(l.GetStackTrace(), `Unknown message direction ${e}`) } } } !function(e) { e[e.ToStreamer = 0] = "ToStreamer", e[e.FromStreamer = 1] = "FromStreamer" }(Fe || (Fe = {})); class rt { constructor() { this.responseEventListeners = new Map } addResponseEventListener(e, t) { this.responseEventListeners.set(e, t) } removeResponseEventListener(e) { this.responseEventListeners.delete(e) } onResponse(e) { l.Log(l.GetStackTrace(), "DataChannelReceiveMessageType.Response", 6); const t = new TextDecoder("utf-16").decode(e.slice(1)); l.Log(l.GetStackTrace(), t, 6), this.responseEventListeners.forEach((e=>{ e(t) } )) } } class ot { constructor(e, t) { this.dataChannelSender = e, this.toStreamerMessagesMapProvider = t } sendLatencyTest(e) { this.sendDescriptor("LatencyTest", e) } emitCommand(e) { this.sendDescriptor("Command", e) } emitUIInteraction(e) { this.sendDescriptor("UIInteraction", e) } sendDescriptor(e, t) { const s = JSON.stringify(t) , n = this.toStreamerMessagesMapProvider.toStreamerMessages.getFromKey(e); void 0 === n && l.Error(l.GetStackTrace(), `Attempted to emit descriptor with message type: ${e}, but the frontend hasn't been configured to send such a message. Check you've added the message type in your cpp`), l.Log(l.GetStackTrace(), "Sending: " + t, 6); const i = new DataView(new ArrayBuffer(3 + 2 * s.length)); let r = 0; i.setUint8(r, n.id), r++, i.setUint16(r, s.length, !0), r += 2; for (let e = 0; e < s.length; e++) i.setUint16(r, s.charCodeAt(e), !0), r += 2; this.dataChannelSender.canSend() ? this.dataChannelSender.sendData(i.buffer) : l.Info(l.GetStackTrace(), `Data channel cannot send yet, skipping sending descriptor message: ${e} - ${s}`) } } class at { constructor(e, t) { this.dataChannelSender = e, this.toStreamerMessagesMapProvider = t } sendMessageToStreamer(e, t) { void 0 === t && (t = []); const s = this.toStreamerMessagesMapProvider.toStreamerMessages.getFromKey(e); if (void 0 === s) return void l.Error(l.GetStackTrace(), `Attempted to send a message to the streamer with message type: ${e}, but the frontend hasn't been configured to send such a message. Check you've added the message type in your cpp`); const n = new DataView(new ArrayBuffer(s.byteLength + 1)); n.setUint8(0, s.id); let i = 1; t.forEach(((e,t)=>{ switch (s.structure[t]) { case "uint8": n.setUint8(i, e), i += 1; break; case "uint16": n.setUint16(i, e, !0), i += 2; break; case "int16": n.setInt16(i, e, !0), i += 2; break; case "float": n.setFloat32(i, e, !0), i += 4; break; case "double": n.setFloat64(i, e, !0), i += 8 } } )), this.dataChannelSender.canSend() ? this.dataChannelSender.sendData(n.buffer) : l.Info(l.GetStackTrace(), `Data channel cannot send yet, skipping sending message: ${e} - ${new Uint8Array(n.buffer)}`) } } class lt { constructor(e) { this.sendMessageController = e } SendRequestQualityControl() { this.sendMessageController.sendMessageToStreamer("RequestQualityControl") } SendMaxFpsRequest() { this.sendMessageController.sendMessageToStreamer("FpsRequest") } SendAverageBitrateRequest() { this.sendMessageController.sendMessageToStreamer("AverageBitrateRequest") } SendStartStreaming() { this.sendMessageController.sendMessageToStreamer("StartStreaming") } SendStopStreaming() { this.sendMessageController.sendMessageToStreamer("StopStreaming") } SendRequestInitialSettings() { this.sendMessageController.sendMessageToStreamer("RequestInitialSettings") } } class ct { constructor(e) { this.dataChannelProvider = e } canSend() { return void 0 !== this.dataChannelProvider.getDataChannelInstance().dataChannel && "open" == this.dataChannelProvider.getDataChannelInstance().dataChannel.readyState } sendData(e) { const t = this.dataChannelProvider.getDataChannelInstance(); "open" == t.dataChannel.readyState ? (t.dataChannel.send(e), l.Log(l.GetStackTrace(), `Message Sent: ${new Uint8Array(e)}`, 6), this.resetAfkWarningTimerOnDataSend()) : l.Error(l.GetStackTrace(), `Message Failed: ${new Uint8Array(e)}`) } resetAfkWarningTimerOnDataSend() {} } class dt { constructor(e) { this.videoElementProvider = e, this.normalizeAndQuantizeUnsignedFunc = ()=>{ throw new Error("Normalize and quantize unsigned, method not implemented.") } , this.normalizeAndQuantizeSignedFunc = ()=>{ throw new Error("Normalize and unquantize signed, method not implemented.") } , this.denormalizeAndUnquantizeUnsignedFunc = ()=>{ throw new Error("Denormalize and unquantize unsigned, method not implemented.") } } normalizeAndQuantizeUnsigned(e, t) { return this.normalizeAndQuantizeUnsignedFunc(e, t) } unquantizeAndDenormalizeUnsigned(e, t) { return this.denormalizeAndUnquantizeUnsignedFunc(e, t) } normalizeAndQuantizeSigned(e, t) { return this.normalizeAndQuantizeSignedFunc(e, t) } setupNormalizeAndQuantize() { if (this.videoElementParent = this.videoElementProvider.getVideoParentElement(), this.videoElement = this.videoElementProvider.getVideoElement(), this.videoElementParent && this.videoElement) { const e = this.videoElementParent.clientHeight / this.videoElementParent.clientWidth , t = this.videoElement.videoHeight / this.videoElement.videoWidth; e > t ? (l.Log(l.GetStackTrace(), "Setup Normalize and Quantize for playerAspectRatio > videoAspectRatio", 6), this.ratio = e / t, this.normalizeAndQuantizeUnsignedFunc = (e,t)=>this.normalizeAndQuantizeUnsignedPlayerBigger(e, t), this.normalizeAndQuantizeSignedFunc = (e,t)=>this.normalizeAndQuantizeSignedPlayerBigger(e, t), this.denormalizeAndUnquantizeUnsignedFunc = (e,t)=>this.denormalizeAndUnquantizeUnsignedPlayerBigger(e, t)) : (l.Log(l.GetStackTrace(), "Setup Normalize and Quantize for playerAspectRatio <= videoAspectRatio", 6), this.ratio = t / e, this.normalizeAndQuantizeUnsignedFunc = (e,t)=>this.normalizeAndQuantizeUnsignedPlayerSmaller(e, t), this.normalizeAndQuantizeSignedFunc = (e,t)=>this.normalizeAndQuantizeSignedPlayerSmaller(e, t), this.denormalizeAndUnquantizeUnsignedFunc = (e,t)=>this.denormalizeAndUnquantizeUnsignedPlayerSmaller(e, t)) } } normalizeAndQuantizeUnsignedPlayerBigger(e, t) { const s = e / this.videoElementParent.clientWidth , n = this.ratio * (t / this.videoElementParent.clientHeight - .5) + .5; return s < 0 || s > 1 || n < 0 || n > 1 ? new ht(!1,65535,65535) : new ht(!0,65536 * s,65536 * n) } denormalizeAndUnquantizeUnsignedPlayerBigger(e, t) { const s = e / 65536 , n = (t / 65536 - .5) / this.ratio + .5; return new ut(s * this.videoElementParent.clientWidth,n * this.videoElementParent.clientHeight) } normalizeAndQuantizeSignedPlayerBigger(e, t) { const s = e / (.5 * this.videoElementParent.clientWidth) , n = this.ratio * t / (.5 * this.videoElementParent.clientHeight); return new mt(32767 * s,32767 * n) } normalizeAndQuantizeUnsignedPlayerSmaller(e, t) { const s = this.ratio * (e / this.videoElementParent.clientWidth - .5) + .5 , n = t / this.videoElementParent.clientHeight; return s < 0 || s > 1 || n < 0 || n > 1 ? new ht(!1,65535,65535) : new ht(!0,65536 * s,65536 * n) } denormalizeAndUnquantizeUnsignedPlayerSmaller(e, t) { const s = (e / 65536 - .5) / this.ratio + .5 , n = t / 65536; return new ut(s * this.videoElementParent.clientWidth,n * this.videoElementParent.clientHeight) } normalizeAndQuantizeSignedPlayerSmaller(e, t) { const s = this.ratio * e / (.5 * this.videoElementParent.clientWidth) , n = t / (.5 * this.videoElementParent.clientHeight); return new mt(32767 * s,32767 * n) } } class ht { constructor(e, t, s) { this.inRange = e, this.x = t, this.y = s } } class ut { constructor(e, t) { this.x = e, this.y = t } } class mt { constructor(e, t) { this.x = e, this.y = t } } class gt { constructor(e, t) { this.shouldShowPlayOverlay = !0, this.config = e, this.pixelStreaming = t, this.responseController = new rt, this.file = new Ge, this.sdpConstraints = { offerToReceiveAudio: !0, offerToReceiveVideo: !0 }, this.afkController = new fe(this.config,this.pixelStreaming,this.onAfkTriggered.bind(this)), this.afkController.onAFKTimedOutCallback = ()=>{ this.setDisconnectMessageOverride("You have been disconnected due to inactivity"), this.closeSignalingServer() } , this.freezeFrameController = new x(this.pixelStreaming.videoElementParent), this.videoPlayer = new st(this.pixelStreaming.videoElementParent,this.config), this.videoPlayer.onVideoInitialized = ()=>this.handleVideoInitialized(), this.videoPlayer.onMatchViewportResolutionCallback = (e,t)=>{ const s = { "Resolution.Width": e, "Resolution.Height": t }; this.sendDescriptorController.emitCommand(s) } , this.videoPlayer.onResizePlayerCallback = ()=>{ this.setUpMouseAndFreezeFrame() } , this.streamController = new w(this.videoPlayer), this.coordinateConverter = new dt(this.videoPlayer), this.sendrecvDataChannelController = new Se, this.recvDataChannelController = new Se, this.registerDataChannelEventEmitters(this.sendrecvDataChannelController), this.registerDataChannelEventEmitters(this.recvDataChannelController), this.dataChannelSender = new ct(this.sendrecvDataChannelController), this.dataChannelSender.resetAfkWarningTimerOnDataSend = ()=>this.afkController.resetAfkWarningTimer(), this.streamMessageController = new it, this.webSocketController = new E, this.webSocketController.onConfig = e=>this.handleOnConfigMessage(e), this.webSocketController.onStreamerList = e=>this.handleStreamerListMessage(e), this.webSocketController.onWebSocketOncloseOverlayMessage = e=>{ this.pixelStreaming._onDisconnect(`Websocket disconnect (${e.code}) ${"" != e.reason ? "- " + e.reason : ""}`), this.setVideoEncoderAvgQP(0) } , this.webSocketController.onOpen.addEventListener("open", (()=>{ this.config.isFlagEnabled(ae.BrowserSendOffer) || this.webSocketController.requestStreamerList() } )), this.webSocketController.onClose.addEventListener("close", (()=>{ this.afkController.stopAfkWarningTimer(), this.statsTimerHandle && void 0 !== this.statsTimerHandle && window.clearInterval(this.statsTimerHandle), this.setTouchInputEnabled(!1), this.setMouseInputEnabled(!1), this.setKeyboardInputEnabled(!1), this.setGamePadInputEnabled(!1), this.shouldReconnect && this.config.getNumericSettingValue(ce.MaxReconnectAttempts) > 0 && (this.isReconnecting = !0, this.reconnectAttempt++, this.restartStreamAutomatically()) } )), this.sendDescriptorController = new ot(this.dataChannelSender,this.streamMessageController), this.sendMessageController = new at(this.dataChannelSender,this.streamMessageController), this.toStreamerMessagesController = new lt(this.sendMessageController), this.registerMessageHandlers(), this.streamMessageController.populateDefaultProtocol(), this.inputClassesFactory = new et(this.streamMessageController,this.videoPlayer,this.coordinateConverter), this.isUsingSFU = !1, this.isQualityController = !1, this.preferredCodec = "", this.shouldReconnect = !0, this.isReconnecting = !1, this.reconnectAttempt = 0, this.config._addOnOptionSettingChangedListener(me.StreamerId, (e=>{ "" !== e && (this.peerConnectionController.peerConnection.close(), this.peerConnectionController.createPeerConnection(this.peerConfig, this.preferredCodec), this.subscribedStream = e, this.webSocketController.sendSubscribe(e)) } )), this.setVideoEncoderAvgQP(-1), this.signallingUrlBuilder = ()=>{ let e = this.config.getTextSettingValue(he.SignallingServerUrl); return this.config.isFlagEnabled(ae.BrowserSendOffer) && (e += "?" + ae.BrowserSendOffer + "=true"), e } } requestUnquantizedAndDenormalizeUnsigned(e, t) { return this.coordinateConverter.unquantizeAndDenormalizeUnsigned(e, t) } handleOnMessage(e) { const t = new Uint8Array(e.data); l.Log(l.GetStackTrace(), "Message incoming:" + t, 6); const s = this.streamMessageController.fromStreamerMessages.getFromValue(t[0]); this.streamMessageController.fromStreamerHandlers.get(s)(e.data) } registerMessageHandlers() { this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "QualityControlOwnership", (e=>this.onQualityControlOwnership(e))), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "Response", (e=>this.responseController.onResponse(e))), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "Command", (e=>{ this.onCommand(e) } )), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "FreezeFrame", (e=>this.onFreezeFrameMessage(e))), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "UnfreezeFrame", (()=>this.invalidateFreezeFrameAndEnableVideo())), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "VideoEncoderAvgQP", (e=>this.handleVideoEncoderAvgQP(e))), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "LatencyTest", (e=>this.handleLatencyTestResult(e))), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "InitialSettings", (e=>this.handleInitialSettings(e))), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "FileExtension", (e=>this.onFileExtension(e))), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "FileMimeType", (e=>this.onFileMimeType(e))), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "FileContents", (e=>this.onFileContents(e))), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "TestEcho", (()=>{} )), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "InputControlOwnership", (e=>this.onInputControlOwnership(e))), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "GamepadResponse", (e=>this.onGamepadResponse(e))), this.streamMessageController.registerMessageHandler(Fe.FromStreamer, "Protocol", (e=>this.onProtocolMessage(e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "IFrameRequest", (()=>this.sendMessageController.sendMessageToStreamer("IFrameRequest"))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "RequestQualityControl", (()=>this.sendMessageController.sendMessageToStreamer("RequestQualityControl"))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "FpsRequest", (()=>this.sendMessageController.sendMessageToStreamer("FpsRequest"))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "AverageBitrateRequest", (()=>this.sendMessageController.sendMessageToStreamer("AverageBitrateRequest"))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "StartStreaming", (()=>this.sendMessageController.sendMessageToStreamer("StartStreaming"))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "StopStreaming", (()=>this.sendMessageController.sendMessageToStreamer("StopStreaming"))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "LatencyTest", (()=>this.sendMessageController.sendMessageToStreamer("LatencyTest"))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "RequestInitialSettings", (()=>this.sendMessageController.sendMessageToStreamer("RequestInitialSettings"))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "TestEcho", (()=>{} )), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "UIInteraction", (e=>this.sendDescriptorController.emitUIInteraction(e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "Command", (e=>this.sendDescriptorController.emitCommand(e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "KeyDown", (e=>this.sendMessageController.sendMessageToStreamer("KeyDown", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "KeyUp", (e=>this.sendMessageController.sendMessageToStreamer("KeyUp", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "KeyPress", (e=>this.sendMessageController.sendMessageToStreamer("KeyPress", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "MouseEnter", (e=>this.sendMessageController.sendMessageToStreamer("MouseEnter", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "MouseLeave", (e=>this.sendMessageController.sendMessageToStreamer("MouseLeave", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "MouseDown", (e=>this.sendMessageController.sendMessageToStreamer("MouseDown", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "MouseUp", (e=>this.sendMessageController.sendMessageToStreamer("MouseUp", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "MouseMove", (e=>this.sendMessageController.sendMessageToStreamer("MouseMove", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "MouseWheel", (e=>this.sendMessageController.sendMessageToStreamer("MouseWheel", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "MouseDouble", (e=>this.sendMessageController.sendMessageToStreamer("MouseDouble", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "TouchStart", (e=>this.sendMessageController.sendMessageToStreamer("TouchStart", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "TouchEnd", (e=>this.sendMessageController.sendMessageToStreamer("TouchEnd", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "TouchMove", (e=>this.sendMessageController.sendMessageToStreamer("TouchMove", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "GamepadConnected", (()=>this.sendMessageController.sendMessageToStreamer("GamepadConnected"))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "GamepadButtonPressed", (e=>this.sendMessageController.sendMessageToStreamer("GamepadButtonPressed", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "GamepadButtonReleased", (e=>this.sendMessageController.sendMessageToStreamer("GamepadButtonReleased", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "GamepadAnalog", (e=>this.sendMessageController.sendMessageToStreamer("GamepadAnalog", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "GamepadDisconnected", (e=>this.sendMessageController.sendMessageToStreamer("GamepadDisconnected", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "XRHMDTransform", (e=>this.sendMessageController.sendMessageToStreamer("XRHMDTransform", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "XRControllerTransform", (e=>this.sendMessageController.sendMessageToStreamer("XRControllerTransform", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "XRSystem", (e=>this.sendMessageController.sendMessageToStreamer("XRSystem", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "XRButtonTouched", (e=>this.sendMessageController.sendMessageToStreamer("XRButtonTouched", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "XRButtonPressed", (e=>this.sendMessageController.sendMessageToStreamer("XRButtonPressed", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "XRButtonReleased", (e=>this.sendMessageController.sendMessageToStreamer("XRButtonReleased", e))), this.streamMessageController.registerMessageHandler(Fe.ToStreamer, "XRAnalog", (e=>this.sendMessageController.sendMessageToStreamer("XRAnalog", e))) } onCommand(e) { l.Log(l.GetStackTrace(), "DataChannelReceiveMessageType.Command", 6); const t = new TextDecoder("utf-16").decode(e.slice(1)); l.Log(l.GetStackTrace(), "Data Channel Command: " + t, 6); const s = JSON.parse(t); "onScreenKeyboard" === s.command && this.pixelStreaming._activateOnScreenKeyboard(s) } onProtocolMessage(e) { try { const t = new TextDecoder("utf-16").decode(e.slice(1)) , s = JSON.parse(t); Object.prototype.hasOwnProperty.call(s, "Direction") || l.Error(l.GetStackTrace(), "Malformed protocol received. Ensure the protocol message contains a direction"); const n = s.Direction; delete s.Direction, l.Log(l.GetStackTrace(), `Received new ${n == Fe.FromStreamer ? "FromStreamer" : "ToStreamer"} protocol. Updating existing protocol...`), Object.keys(s).forEach((e=>{ const t = s[e]; switch (n) { case Fe.ToStreamer: if (!Object.prototype.hasOwnProperty.call(t, "id") || !Object.prototype.hasOwnProperty.call(t, "byteLength")) return void l.Error(l.GetStackTrace(), `ToStreamer->${e} protocol definition was malformed as it didn't contain at least an id and a byteLength\n\n Definition was: ${JSON.stringify(t, null, 2)}`); if (t.byteLength > 0 && !Object.prototype.hasOwnProperty.call(t, "structure")) return void l.Error(l.GetStackTrace(), `ToStreamer->${e} protocol definition was malformed as it specified a byteLength but no accompanying structure`); this.streamMessageController.toStreamerHandlers.get(e) ? this.streamMessageController.toStreamerMessages.add(e, t) : l.Error(l.GetStackTrace(), `There was no registered handler for "${e}" - try adding one using registerMessageHandler(MessageDirection.ToStreamer, "${e}", myHandler)`); break; case Fe.FromStreamer: if (!Object.prototype.hasOwnProperty.call(t, "id")) return void l.Error(l.GetStackTrace(), `FromStreamer->${e} protocol definition was malformed as it didn't contain at least an id\n\n Definition was: ${JSON.stringify(t, null, 2)}`); this.streamMessageController.fromStreamerHandlers.get(e) ? this.streamMessageController.fromStreamerMessages.add(e, t.id) : l.Error(l.GetStackTrace(), `There was no registered handler for "${t}" - try adding one using registerMessageHandler(MessageDirection.FromStreamer, "${e}", myHandler)`); break; default: l.Error(l.GetStackTrace(), `Unknown direction: ${n}`) } } )), this.toStreamerMessagesController.SendRequestInitialSettings(), this.toStreamerMessagesController.SendRequestQualityControl() } catch (e) { l.Log(l.GetStackTrace(), e) } } onInputControlOwnership(e) { const t = new Uint8Array(e); l.Log(l.GetStackTrace(), "DataChannelReceiveMessageType.InputControlOwnership", 6); const s = new Boolean(t[1]).valueOf(); l.Log(l.GetStackTrace(), `Received input controller message - will your input control the stream: ${s}`), this.pixelStreaming._onInputControlOwnership(s) } onGamepadResponse(e) { const t = new TextDecoder("utf-16").decode(e.slice(1)) , s = JSON.parse(t); this.gamePadController.onGamepadResponseReceived(s.controllerId) } onAfkTriggered() { this.afkController.onAfkClick(), this.videoPlayer.isPaused() && this.videoPlayer.hasVideoSource() && this.playStream() } setAfkEnabled(e) { e ? this.onAfkTriggered() : this.afkController.stopAfkWarningTimer() } restartStreamAutomatically() { if (this.webSocketController) if (this.webSocketController.webSocket && this.webSocketController.webSocket.readyState !== WebSocket.CLOSED) { this.pixelStreaming._showActionOrErrorOnDisconnect = !1, this.setDisconnectMessageOverride("Restarting stream..."), this.closeSignalingServer(); const e = setTimeout((()=>{ this.pixelStreaming._onWebRtcAutoConnect(), this.connectToSignallingServer(), clearTimeout(e) } ), 3e3) } else l.Log(l.GetStackTrace(), "A websocket connection has not been made yet so we will start the stream"), this.pixelStreaming._onWebRtcAutoConnect(), this.connectToSignallingServer(); else l.Log(l.GetStackTrace(), "The Web Socket Controller does not exist so this will not work right now.") } loadFreezeFrameOrShowPlayOverlay() { this.pixelStreaming.dispatchEvent(new J({ shouldShowPlayOverlay: this.shouldShowPlayOverlay, isValid: this.freezeFrameController.valid, jpegData: this.freezeFrameController.jpeg })), !0 === this.shouldShowPlayOverlay ? (l.Log(l.GetStackTrace(), "showing play overlay"), this.resizePlayerStyle()) : (l.Log(l.GetStackTrace(), "showing freeze frame"), this.freezeFrameController.showFreezeFrame()), setTimeout((()=>{ this.videoPlayer.setVideoEnabled(!1) } ), this.freezeFrameController.freezeFrameDelay) } onFreezeFrameMessage(e) { l.Log(l.GetStackTrace(), "DataChannelReceiveMessageType.FreezeFrame", 6); const t = new Uint8Array(e); this.freezeFrameController.processFreezeFrameMessage(t, (()=>this.loadFreezeFrameOrShowPlayOverlay())) } invalidateFreezeFrameAndEnableVideo() { l.Log(l.GetStackTrace(), "DataChannelReceiveMessageType.FreezeFrame", 6), setTimeout((()=>{ this.pixelStreaming.dispatchEvent(new Y), this.freezeFrameController.hideFreezeFrame() } ), this.freezeFrameController.freezeFrameDelay), this.videoPlayer.getVideoElement() && this.videoPlayer.setVideoEnabled(!0) } onFileExtension(e) { const t = new Uint8Array(e); Ne.setExtensionFromBytes(t, this.file) } onFileMimeType(e) { const t = new Uint8Array(e); Ne.setMimeTypeFromBytes(t, this.file) } onFileContents(e) { const t = new Uint8Array(e); Ne.setContentsFromBytes(t, this.file) } playStream() { if (!this.videoPlayer.getVideoElement()) { const e = "Could not play video stream because the video player was not initialized correctly."; return this.pixelStreaming.dispatchEvent(new K({ message: e })), l.Error(l.GetStackTrace(), e), this.setDisconnectMessageOverride("Stream not initialized correctly"), void this.closeSignalingServer() } this.videoPlayer.hasVideoSource() ? (this.setTouchInputEnabled(this.config.isFlagEnabled(ae.TouchInput)), this.pixelStreaming.dispatchEvent(new j), this.streamController.audioElement.srcObject ? (this.streamController.audioElement.muted = this.config.isFlagEnabled(ae.StartVideoMuted), this.streamController.audioElement.play().then((()=>{ this.playVideo() } )).catch((e=>{ l.Log(l.GetStackTrace(), e), l.Log(l.GetStackTrace(), "Browser does not support autoplaying video without interaction - to resolve this we are going to show the play button overlay."), this.pixelStreaming.dispatchEvent(new X({ reason: e })) } ))) : this.playVideo(), this.shouldShowPlayOverlay = !1, this.freezeFrameController.showFreezeFrame()) : l.Warning(l.GetStackTrace(), "Cannot play stream, the video element has no srcObject to play.") } playVideo() { this.videoPlayer.play().catch((e=>{ this.streamController.audioElement.srcObject && this.streamController.audioElement.pause(), l.Log(l.GetStackTrace(), e), l.Log(l.GetStackTrace(), "Browser does not support autoplaying video without interaction - to resolve this we are going to show the play button overlay."), this.pixelStreaming.dispatchEvent(new X({ reason: e })) } )) } autoPlayVideoOrSetUpPlayOverlay() { this.config.isFlagEnabled(ae.AutoPlayVideo) && this.playStream(), this.resizePlayerStyle() } connectToSignallingServer() { const e = this.signallingUrlBuilder(); this.webSocketController.connect(e) } startSession(e) { if (this.peerConfig = e, this.config.isFlagEnabled(ae.ForceTURN) && !this.checkTurnServerAvailability(e)) return l.Info(l.GetStackTrace(), "No turn server was found in the Peer Connection Options. TURN cannot be forced, closing connection. Please use STUN instead"), this.setDisconnectMessageOverride("TURN cannot be forced, closing connection. Please use STUN instead."), void this.closeSignalingServer(); this.peerConnectionController = new Ie(this.peerConfig,this.config,this.preferredCodec), this.peerConnectionController.onVideoStats = e=>this.handleVideoStats(e), this.peerConnectionController.onSendWebRTCOffer = e=>this.handleSendWebRTCOffer(e), this.peerConnectionController.onSendWebRTCAnswer = e=>this.handleSendWebRTCAnswer(e), this.peerConnectionController.onPeerIceCandidate = e=>this.handleSendIceCandidate(e), this.peerConnectionController.onDataChannel = e=>this.handleDataChannel(e), this.peerConnectionController.showTextOverlayConnecting = ()=>this.pixelStreaming._onWebRtcConnecting(), this.peerConnectionController.showTextOverlaySetupFailure = ()=>this.pixelStreaming._onWebRtcFailed(); let t = !1; this.peerConnectionController.onIceConnectionStateChange = ()=>{ !t && ["connected", "completed"].includes(this.peerConnectionController.peerConnection.iceConnectionState) && (this.pixelStreaming._onWebRtcConnected(), t = !0) } , this.peerConnectionController.onTrack = e=>this.streamController.handleOnTrack(e), this.config.isFlagEnabled(ae.BrowserSendOffer) && (this.sendrecvDataChannelController.createDataChannel(this.peerConnectionController.peerConnection, "cirrus", this.datachannelOptions), this.sendrecvDataChannelController.handleOnMessage = e=>this.handleOnMessage(e), this.peerConnectionController.createOffer(this.sdpConstraints, this.config)) } checkTurnServerAvailability(e) { if (!e.iceServers) return l.Info(l.GetStackTrace(), "A turn sever was not found"), !1; for (const t of e.iceServers) for (const e of t.urls) if (e.includes("turn")) return l.Log(l.GetStackTrace(), `A turn sever was found at ${e}`), !0; return l.Info(l.GetStackTrace(), "A turn sever was not found"), !1 } handleOnConfigMessage(e) { this.resizePlayerStyle(), this.startSession(e.peerConnectionOptions), this.webSocketController.onWebRtcAnswer = e=>this.handleWebRtcAnswer(e), this.webSocketController.onWebRtcOffer = e=>this.handleWebRtcOffer(e), this.webSocketController.onWebRtcPeerDataChannels = e=>this.handleWebRtcSFUPeerDatachannels(e), this.webSocketController.onIceCandidate = e=>this.handleIceCandidate(e) } handleStreamerListMessage(e) { if (l.Log(l.GetStackTrace(), `Got streamer list ${e.ids}`, 6), this.isReconnecting) e.ids.includes(this.subscribedStream) ? (this.isReconnecting = !1, this.reconnectAttempt = 0, this.webSocketController.sendSubscribe(this.subscribedStream)) : this.reconnectAttempt < this.config.getNumericSettingValue(ce.MaxReconnectAttempts) ? (this.reconnectAttempt++, setTimeout((()=>{ this.webSocketController.requestStreamerList() } ), 2e3)) : (this.reconnectAttempt = 0, this.isReconnecting = !1, this.shouldReconnect = !1, this.webSocketController.close(), this.config.setOptionSettingValue(me.StreamerId, ""), this.config.setOptionSettingOptions(me.StreamerId, [])); else { const t = [...e.ids]; t.unshift(""), this.config.setOptionSettingOptions(me.StreamerId, t); const s = new URLSearchParams(window.location.search); let n = null; 1 == e.ids.length ? n = e.ids[0] : this.config.isFlagEnabled(ae.PreferSFU) && e.ids.includes("SFU") ? n = "SFU" : s.has(me.StreamerId) && e.ids.includes(s.get(me.StreamerId)) && (n = s.get(me.StreamerId)), null !== n && this.config.setOptionSettingValue(me.StreamerId, n), this.pixelStreaming.dispatchEvent(new Z({ messageStreamerList: e, autoSelectedStreamerId: n })) } } handleWebRtcAnswer(e) { l.Log(l.GetStackTrace(), `Got answer sdp ${e.sdp}`, 6); const t = { sdp: e.sdp, type: "answer" }; this.peerConnectionController.receiveAnswer(t), this.handlePostWebrtcNegotiation() } handleWebRtcOffer(e) { l.Log(l.GetStackTrace(), `Got offer sdp ${e.sdp}`, 6), this.isUsingSFU = !!e.sfu && e.sfu, this.isUsingSFU && (this.peerConnectionController.preferredCodec = ""); const t = { sdp: e.sdp, type: "offer" }; this.peerConnectionController.receiveOffer(t, this.config), this.handlePostWebrtcNegotiation() } handleWebRtcSFUPeerDatachannels(e) { const t = { ordered: !0, negotiated: !0, id: e.sendStreamId } , s = e.sendStreamId != e.recvStreamId; if (this.sendrecvDataChannelController.createDataChannel(this.peerConnectionController.peerConnection, s ? "send-datachannel" : "datachannel", t), s) { const t = { ordered: !0, negotiated: !0, id: e.recvStreamId }; this.recvDataChannelController.createDataChannel(this.peerConnectionController.peerConnection, "recv-datachannel", t), this.recvDataChannelController.handleOnOpen = ()=>this.webSocketController.sendSFURecvDataChannelReady(), this.recvDataChannelController.handleOnMessage = e=>this.handleOnMessage(e) } else this.sendrecvDataChannelController.handleOnMessage = e=>this.handleOnMessage(e) } handlePostWebrtcNegotiation() { this.afkController.startAfkWarningTimer(), this.pixelStreaming._onWebRtcSdp(), this.statsTimerHandle && void 0 !== this.statsTimerHandle && window.clearInterval(this.statsTimerHandle), this.statsTimerHandle = window.setInterval((()=>this.getStats()), 1e3), this.setMouseInputEnabled(this.config.isFlagEnabled(ae.MouseInput)), this.setKeyboardInputEnabled(this.config.isFlagEnabled(ae.KeyboardInput)), this.setGamePadInputEnabled(this.config.isFlagEnabled(ae.GamepadInput)) } handleIceCandidate(e) { l.Log(l.GetStackTrace(), "Web RTC Controller: onWebRtcIce", 6); const t = new RTCIceCandidate(e); this.peerConnectionController.handleOnIce(t) } handleSendIceCandidate(e) { l.Log(l.GetStackTrace(), "OnIceCandidate", 6), e.candidate && e.candidate.candidate && this.webSocketController.sendIceCandidate(e.candidate) } handleDataChannel(e) { l.Log(l.GetStackTrace(), "Data channel created for us by browser as we are a receiving peer.", 6), this.sendrecvDataChannelController.dataChannel = e.channel, this.sendrecvDataChannelController.setupDataChannel(), this.sendrecvDataChannelController.handleOnMessage = e=>this.handleOnMessage(e) } handleSendWebRTCOffer(e) { l.Log(l.GetStackTrace(), "Sending the offer to the Server", 6), this.webSocketController.sendWebRtcOffer(e) } handleSendWebRTCAnswer(e) { l.Log(l.GetStackTrace(), "Sending the answer to the Server", 6), this.webSocketController.sendWebRtcAnswer(e), this.isUsingSFU && this.webSocketController.sendWebRtcDatachannelRequest() } setUpMouseAndFreezeFrame() { this.videoElementParentClientRect = this.videoPlayer.getVideoParentElement().getBoundingClientRect(), this.coordinateConverter.setupNormalizeAndQuantize(), this.freezeFrameController.freezeFrame.resize() } closeSignalingServer() { var e; this.shouldReconnect = !1, null === (e = this.webSocketController) || void 0 === e || e.close() } closePeerConnection() { var e; null === (e = this.peerConnectionController) || void 0 === e || e.close() } close() { this.closeSignalingServer(), this.closePeerConnection() } getStats() { this.peerConnectionController.generateStats() } sendLatencyTest() { this.latencyStartTime = Date.now(), this.sendDescriptorController.sendLatencyTest({ StartTime: this.latencyStartTime }) } sendEncoderMinQP(e) { l.Log(l.GetStackTrace(), `MinQP=${e}\n`, 6), null != e && this.sendDescriptorController.emitCommand({ "Encoder.MinQP": e }) } sendEncoderMaxQP(e) { l.Log(l.GetStackTrace(), `MaxQP=${e}\n`, 6), null != e && this.sendDescriptorController.emitCommand({ "Encoder.MaxQP": e }) } sendWebRTCMinBitrate(e) { l.Log(l.GetStackTrace(), `WebRTC Min Bitrate=${e}`, 6), null != e && this.sendDescriptorController.emitCommand({ "WebRTC.MinBitrate": e }) } sendWebRTCMaxBitrate(e) { l.Log(l.GetStackTrace(), `WebRTC Max Bitrate=${e}`, 6), null != e && this.sendDescriptorController.emitCommand({ "WebRTC.MaxBitrate": e }) } sendWebRTCFps(e) { l.Log(l.GetStackTrace(), `WebRTC FPS=${e}`, 6), null != e && (this.sendDescriptorController.emitCommand({ "WebRTC.Fps": e }), this.sendDescriptorController.emitCommand({ "WebRTC.MaxFps": e })) } sendShowFps() { l.Log(l.GetStackTrace(), "---- Sending show stat to UE ----", 6), this.sendDescriptorController.emitCommand({ "stat.fps": "" }) } sendIframeRequest() { l.Log(l.GetStackTrace(), "---- Sending Request for an IFrame ----", 6), this.streamMessageController.toStreamerHandlers.get("IFrameRequest")() } emitUIInteraction(e) { l.Log(l.GetStackTrace(), "---- Sending custom UIInteraction message ----", 6), this.sendDescriptorController.emitUIInteraction(e) } emitCommand(e) { l.Log(l.GetStackTrace(), "---- Sending custom Command message ----", 6), this.sendDescriptorController.emitCommand(e) } emitConsoleCommand(e) { l.Log(l.GetStackTrace(), "---- Sending custom Command:ConsoleCommand message ----", 6), this.sendDescriptorController.emitCommand({ ConsoleCommand: e }) } sendRequestQualityControlOwnership() { l.Log(l.GetStackTrace(), "---- Sending Request to Control Quality ----", 6), this.toStreamerMessagesController.SendRequestQualityControl() } handleLatencyTestResult(e) { l.Log(l.GetStackTrace(), "DataChannelReceiveMessageType.latencyTest", 6); const t = new TextDecoder("utf-16").decode(e.slice(1)) , s = new Be; Object.assign(s, JSON.parse(t)), s.processFields(), s.testStartTimeMs = this.latencyStartTime, s.browserReceiptTimeMs = Date.now(), s.latencyExcludingDecode = ~~(s.browserReceiptTimeMs - s.testStartTimeMs), s.testDuration = ~~(s.TransmissionTimeMs - s.ReceiptTimeMs), s.networkLatency = ~~(s.latencyExcludingDecode - s.testDuration), s.frameDisplayDeltaTimeMs && s.browserReceiptTimeMs && (s.endToEndLatency = (s.frameDisplayDeltaTimeMs, s.networkLatency, ~~+s.CaptureToSendMs)), this.pixelStreaming._onLatencyTestResult(s) } handleInitialSettings(e) { l.Log(l.GetStackTrace(), "DataChannelReceiveMessageType.InitialSettings", 6); const t = new TextDecoder("utf-16").decode(e.slice(1)) , s = JSON.parse(t) , n = new De; s.Encoder && (n.EncoderSettings = s.Encoder), s.WebRTC && (n.WebRTCSettings = s.WebRTC), s.PixelStreaming && (n.PixelStreamingSettings = s.PixelStreaming), s.ConfigOptions && this.config.setFlagEnabled(ae.HoveringMouseMode, !!s.ConfigOptions.DefaultToHover), n.ueCompatible(), l.Log(l.GetStackTrace(), t, 6), this.pixelStreaming._onInitialSettings(n) } handleVideoEncoderAvgQP(e) { l.Log(l.GetStackTrace(), "DataChannelReceiveMessageType.VideoEncoderAvgQP", 6); const t = Number(new TextDecoder("utf-16").decode(e.slice(1))); this.setVideoEncoderAvgQP(t) } handleVideoInitialized() { this.pixelStreaming._onVideoInitialized(), this.autoPlayVideoOrSetUpPlayOverlay(), this.resizePlayerStyle(), this.videoPlayer.updateVideoStreamSize() } onQualityControlOwnership(e) { const t = new Uint8Array(e); l.Log(l.GetStackTrace(), "DataChannelReceiveMessageType.QualityControlOwnership", 6), this.isQualityController = new Boolean(t[1]).valueOf(), l.Log(l.GetStackTrace(), `Received quality controller message, will control quality: ${this.isQualityController}`), this.pixelStreaming._onQualityControlOwnership(this.isQualityController) } handleVideoStats(e) { this.pixelStreaming._onVideoStats(e) } resizePlayerStyle() { this.videoPlayer.resizePlayerStyle() } getDisconnectMessageOverride() { return this.disconnectMessageOverride } setDisconnectMessageOverride(e) { this.disconnectMessageOverride = e } setPreferredCodec(e) { this.preferredCodec = e, this.peerConnectionController && (this.peerConnectionController.preferredCodec = e, this.peerConnectionController.updateCodecSelection = !1) } setVideoEncoderAvgQP(e) { this.videoAvgQp = e, this.pixelStreaming._onVideoEncoderAvgQP(this.videoAvgQp) } setKeyboardInputEnabled(e) { var t; null === (t = this.keyboardController) || void 0 === t || t.unregisterKeyBoardEvents(), e && (this.keyboardController = this.inputClassesFactory.registerKeyBoard(this.config)) } setMouseInputEnabled(e) { var t; if (null === (t = this.mouseController) || void 0 === t || t.unregisterMouseEvents(), e) { const e = this.config.isFlagEnabled(ae.HoveringMouseMode) ? ve.HoveringMouse : ve.LockedMouse; this.mouseController = this.inputClassesFactory.registerMouse(e) } } setTouchInputEnabled(e) { var t; null === (t = this.touchController) || void 0 === t || t.unregisterTouchEvents(), e && (this.touchController = this.inputClassesFactory.registerTouch(this.config.isFlagEnabled(ae.FakeMouseWithTouches), this.videoElementParentClientRect)) } setGamePadInputEnabled(e) { var t; null === (t = this.gamePadController) || void 0 === t || t.unregisterGamePadEvents(), e && (this.gamePadController = this.inputClassesFactory.registerGamePad(), this.gamePadController.onGamepadConnected = ()=>{ this.streamMessageController.toStreamerHandlers.get("GamepadConnected")() } , this.gamePadController.onGamepadDisconnected = e=>{ this.streamMessageController.toStreamerHandlers.get("GamepadDisconnected")([e]) } ) } registerDataChannelEventEmitters(e) { e.onOpen = (e,t)=>this.pixelStreaming.dispatchEvent(new H({ label: e, event: t })), e.onClose = (e,t)=>this.pixelStreaming.dispatchEvent(new W({ label: e, event: t })), e.onError = (e,t)=>this.pixelStreaming.dispatchEvent(new V({ label: e, event: t })) } } class pt { static vertexShader() { return "\n\t\tattribute vec2 a_position;\n\t\tattribute vec2 a_texCoord;\n\n\t\t// input\n\t\tuniform vec2 u_resolution;\n\t\tuniform vec4 u_offset;\n\n\t\t//\n\t\tvarying vec2 v_texCoord;\n\n\t\tvoid main() {\n\t\t // convert the rectangle from pixels to 0.0 to 1.0\n\t\t vec2 zeroToOne = a_position / u_resolution;\n\n\t\t // convert from 0->1 to 0->2\n\t\t vec2 zeroToTwo = zeroToOne * 2.0;\n\n\t\t // convert from 0->2 to -1->+1 (clipspace)\n\t\t vec2 clipSpace = zeroToTwo - 1.0;\n\n\t\t gl_Position = vec4(clipSpace * vec2(1, -1), 0, 1);\n\t\t // pass the texCoord to the fragment shader\n\t\t // The GPU will interpolate this value between points.\n\t\t v_texCoord = (a_texCoord * u_offset.xy) + u_offset.zw;\n\t\t}\n\t\t" } static fragmentShader() { return "\n\t\tprecision mediump float;\n\n\t\t// our texture\n\t\tuniform sampler2D u_image;\n\n\t\t// the texCoords passed in from the vertex shader.\n\t\tvarying vec2 v_texCoord;\n\n\t\tvoid main() {\n\t\t gl_FragColor = texture2D(u_image, v_texCoord);\n\t\t}\n\t\t" } } class vt { static deepCopyGamepad(e) { return JSON.parse(JSON.stringify({ buttons: e.buttons.map((e=>JSON.parse(JSON.stringify({ pressed: e.pressed, touched: e.touched })))), axes: e.axes })) } } class ft { constructor(e) { this.toStreamerMessagesProvider = e, this.controllers = [] } updateStatus(e, t, s) { if (e.gamepad) { const n = t.getPose(e.gripSpace, s); if (!n) return; let i = 0; e.profiles.includes("htc-vive") ? i = 1 : e.profiles.includes("oculus-touch") && (i = 2), this.toStreamerMessagesProvider.toStreamerHandlers.get("XRSystem")([i]); let r = 2; switch (e.handedness) { case "left": r = 0; break; case "right": r = 1 } const o = n.transform.matrix , a = []; for (let e = 0; e < 16; e++) a[e] = new Float32Array([o[e]])[0]; this.toStreamerMessagesProvider.toStreamerHandlers.get("XRControllerTransform")([a[0], a[4], a[8], a[12], a[1], a[5], a[9], a[13], a[2], a[6], a[10], a[14], a[3], a[7], a[11], a[15], r]), void 0 === this.controllers[r] && (this.controllers[r] = { prevState: void 0, currentState: void 0, id: void 0 }, this.controllers[r].prevState = vt.deepCopyGamepad(e.gamepad)), this.controllers[r].currentState = vt.deepCopyGamepad(e.gamepad); const l = this.controllers[r] , c = l.currentState , d = l.prevState; for (let e = 0; e < c.buttons.length; e++) { const t = c.buttons[e] , s = d.buttons[e]; t.pressed ? this.toStreamerMessagesProvider.toStreamerHandlers.get("XRButtonPressed")([r, e, s.pressed ? 1 : 0]) : !t.pressed && s.pressed && this.toStreamerMessagesProvider.toStreamerHandlers.get("XRButtonReleased")([r, e, 0]), t.touched && !t.pressed ? this.toStreamerMessagesProvider.toStreamerHandlers.get("XRButtonPressed")([r, 3, s.touched ? 1 : 0]) : !t.touched && s.touched && this.toStreamerMessagesProvider.toStreamerHandlers.get("XRButtonReleased")([r, 3, 0]) } for (let e = 0; e < c.axes.length; e++) this.toStreamerMessagesProvider.toStreamerHandlers.get("XRAnalog")([r, e, c.axes[e]]); this.controllers[r].prevState = c } } } class St { constructor(e) { this.xrSession = null, this.webRtcController = e, this.xrControllers = [], this.xrGamepadController = new ft(this.webRtcController.streamMessageController), this.onSessionEnded = new EventTarget, this.onSessionStarted = new EventTarget, this.onFrame = new EventTarget } xrClicked() { this.xrSession ? this.xrSession.end() : navigator.xr.requestSession("immersive-vr").then((e=>{ this.onXrSessionStarted(e) } )) } onXrSessionEnded() { l.Log(l.GetStackTrace(), "XR Session ended"), this.xrSession = null, this.onSessionEnded.dispatchEvent(new Event("xrSessionEnded")) } onXrSessionStarted(e) { l.Log(l.GetStackTrace(), "XR Session started"), this.xrSession = e, this.xrSession.addEventListener("end", (()=>{ this.onXrSessionEnded() } )); const t = document.createElement("canvas"); this.gl = t.getContext("webgl2", { xrCompatible: !0 }), this.xrSession.updateRenderState({ baseLayer: new XRWebGLLayer(this.xrSession,this.gl) }); const s = this.gl.createShader(this.gl.VERTEX_SHADER); this.gl.shaderSource(s, pt.vertexShader()), this.gl.compileShader(s); const n = this.gl.createShader(this.gl.FRAGMENT_SHADER); this.gl.shaderSource(n, pt.fragmentShader()), this.gl.compileShader(n); const i = this.gl.createProgram(); this.gl.attachShader(i, s), this.gl.attachShader(i, n), this.gl.linkProgram(i), this.gl.useProgram(i), this.positionLocation = this.gl.getAttribLocation(i, "a_position"), this.texcoordLocation = this.gl.getAttribLocation(i, "a_texCoord"), this.positionBuffer = this.gl.createBuffer(), this.gl.bindBuffer(this.gl.ARRAY_BUFFER, this.positionBuffer), this.gl.enableVertexAttribArray(this.positionLocation); const r = this.gl.createTexture(); this.gl.bindTexture(this.gl.TEXTURE_2D, r), this.gl.texParameteri(this.gl.TEXTURE_2D, this.gl.TEXTURE_WRAP_S, this.gl.CLAMP_TO_EDGE), this.gl.texParameteri(this.gl.TEXTURE_2D, this.gl.TEXTURE_WRAP_T, this.gl.CLAMP_TO_EDGE), this.gl.texParameteri(this.gl.TEXTURE_2D, this.gl.TEXTURE_MIN_FILTER, this.gl.NEAREST), this.gl.texParameteri(this.gl.TEXTURE_2D, this.gl.TEXTURE_MAG_FILTER, this.gl.NEAREST), this.texcoordBuffer = this.gl.createBuffer(), this.resolutionLocation = this.gl.getUniformLocation(i, "u_resolution"), this.offsetLocation = this.gl.getUniformLocation(i, "u_offset"), e.requestReferenceSpace("local").then((e=>{ this.xrRefSpace = e, this.xrSession.requestAnimationFrame(((e,t)=>this.onXrFrame(e, t))) } )), this.onSessionStarted.dispatchEvent(new Event("xrSessionStarted")) } onXrFrame(e, t) { const s = t.getViewerPose(this.xrRefSpace); if (s) { const e = s.transform.matrix , t = []; for (let s = 0; s < 16; s++) t[s] = new Float32Array([e[s]])[0]; this.webRtcController.streamMessageController.toStreamerHandlers.get("XRHMDTransform")([t[0], t[4], t[8], t[12], t[1], t[5], t[9], t[13], t[2], t[6], t[10], t[14], t[3], t[7], t[11], t[15]]); const n = this.xrSession.renderState.baseLayer; this.gl.bindFramebuffer(this.gl.FRAMEBUFFER, n.framebuffer), this.gl.texImage2D(this.gl.TEXTURE_2D, 0, this.gl.RGBA, this.gl.RGBA, this.gl.UNSIGNED_BYTE, this.webRtcController.videoPlayer.getVideoElement()), this.render(this.webRtcController.videoPlayer.getVideoElement()) } this.webRtcController.config.isFlagEnabled(ae.XRControllerInput) && this.xrSession.inputSources.forEach(((e,s,n)=>{ this.xrGamepadController.updateStatus(e, t, this.xrRefSpace) } ), this), this.xrSession.requestAnimationFrame(((e,t)=>this.onXrFrame(e, t))), this.onFrame.dispatchEvent(new re({ time: e, frame: t })) } render(e) { if (!this.gl) return; const t = this.xrSession.renderState.baseLayer; let s, n, i, r, o; this.gl.viewport(0, 0, t.framebufferWidth, t.framebufferHeight), this.gl.uniform4f(this.offsetLocation, 1, 1, 0, 0), this.gl.bufferData(this.gl.ARRAY_BUFFER, new Float32Array([0, 0, e.videoWidth, 0, 0, e.videoHeight, 0, e.videoHeight, e.videoWidth, 0, e.videoWidth, e.videoHeight]), this.gl.STATIC_DRAW), this.gl.bindBuffer(this.gl.ARRAY_BUFFER, this.texcoordBuffer), this.gl.bufferData(this.gl.ARRAY_BUFFER, new Float32Array([0, 0, 1, 0, 0, 1, 0, 1, 1, 0, 1, 1]), this.gl.STATIC_DRAW), this.gl.bindBuffer(this.gl.ARRAY_BUFFER, this.positionBuffer), s = 2, n = this.gl.FLOAT, i = !1, r = 0, o = 0, this.gl.vertexAttribPointer(this.positionLocation, s, n, i, r, o), this.gl.enableVertexAttribArray(this.texcoordLocation), this.gl.bindBuffer(this.gl.ARRAY_BUFFER, this.texcoordBuffer), s = 2, n = this.gl.FLOAT, i = !1, r = 0, o = 0, this.gl.vertexAttribPointer(this.texcoordLocation, s, n, i, r, o), this.gl.uniform2f(this.resolutionLocation, e.videoWidth, e.videoHeight); const a = this.gl.TRIANGLES; o = 0, this.gl.drawArrays(a, o, 6) } static isSessionSupported(e) { return navigator.xr ? navigator.xr.isSessionSupported(e) : new Promise((()=>!1)) } } class Ct { constructor(e) { this.editTextButton = null, this.hiddenInput = null, "ontouchstart"in document.documentElement && this.createOnScreenKeyboardHelpers(e) } unquantizeAndDenormalizeUnsigned(e, t) { return null } createOnScreenKeyboardHelpers(e) { this.hiddenInput || (this.hiddenInput = document.createElement("input"), this.hiddenInput.id = "hiddenInput", this.hiddenInput.maxLength = 0, e.appendChild(this.hiddenInput)), this.editTextButton || (this.editTextButton = document.createElement("button"), this.editTextButton.id = "editTextButton", this.editTextButton.innerHTML = "edit text", e.appendChild(this.editTextButton), this.editTextButton.classList.add("hiddenState"), this.editTextButton.addEventListener("touchend", (e=>{ this.hiddenInput.focus(), e.preventDefault() } ))) } showOnScreenKeyboard(e) { if (e.showOnScreenKeyboard) { this.editTextButton.classList.remove("hiddenState"); const t = this.unquantizeAndDenormalizeUnsigned(e.x, e.y); this.editTextButton.style.top = t.y.toString() + "px", this.editTextButton.style.left = (t.x - 40).toString() + "px" } else this.editTextButton.classList.add("hiddenState"), this.hiddenInput.blur() } } class yt { constructor(e, t) { this._showActionOrErrorOnDisconnect = !0, this.allowConsoleCommands = !1, this.config = e, (null == t ? void 0 : t.videoElementParent) && (this._videoElementParent = t.videoElementParent), this._eventEmitter = new oe, this.configureSettings(), this.setWebRtcPlayerController(new gt(this.config,this)), this.onScreenKeyboardHelper = new Ct(this.videoElementParent), this.onScreenKeyboardHelper.unquantizeAndDenormalizeUnsigned = (e,t)=>this._webRtcController.requestUnquantizedAndDenormalizeUnsigned(e, t), this._activateOnScreenKeyboard = e=>this.onScreenKeyboardHelper.showOnScreenKeyboard(e), this._webXrController = new St(this._webRtcController) } get videoElementParent() { return this._videoElementParent || (this._videoElementParent = document.createElement("div"), this._videoElementParent.id = "videoElementParent"), this._videoElementParent } configureSettings() { this.config._addOnSettingChangedListener(ae.IsQualityController, (e=>{ !0 !== e || this._webRtcController.isQualityController || this._webRtcController.sendRequestQualityControlOwnership() } )), this.config._addOnSettingChangedListener(ae.AFKDetection, (e=>{ this._webRtcController.setAfkEnabled(e) } )), this.config._addOnSettingChangedListener(ae.MatchViewportResolution, (()=>{ this._webRtcController.videoPlayer.updateVideoStreamSize() } )), this.config._addOnSettingChangedListener(ae.HoveringMouseMode, (e=>{ this.config.setFlagLabel(ae.HoveringMouseMode, `Control Scheme: ${e ? "Hovering" : "Locked"} Mouse`), this._webRtcController.setMouseInputEnabled(this.config.isFlagEnabled(ae.MouseInput)) } )), this.config._addOnSettingChangedListener(ae.KeyboardInput, (e=>{ this._webRtcController.setKeyboardInputEnabled(e) } )), this.config._addOnSettingChangedListener(ae.MouseInput, (e=>{ this._webRtcController.setMouseInputEnabled(e) } )), this.config._addOnSettingChangedListener(ae.TouchInput, (e=>{ this._webRtcController.setTouchInputEnabled(e) } )), this.config._addOnSettingChangedListener(ae.GamepadInput, (e=>{ this._webRtcController.setGamePadInputEnabled(e) } )), this.config._addOnNumericSettingChangedListener(ce.MinQP, (e=>{ l.Log(l.GetStackTrace(), "-------- Sending MinQP --------", 7), this._webRtcController.sendEncoderMinQP(e), l.Log(l.GetStackTrace(), "-------------------------------------------", 7) } )), this.config._addOnNumericSettingChangedListener(ce.MaxQP, (e=>{ l.Log(l.GetStackTrace(), "-------- Sending encoder settings --------", 7), this._webRtcController.sendEncoderMaxQP(e), l.Log(l.GetStackTrace(), "-------------------------------------------", 7) } )), this.config._addOnNumericSettingChangedListener(ce.WebRTCMinBitrate, (e=>{ l.Log(l.GetStackTrace(), "-------- Sending web rtc settings --------", 7), this._webRtcController.sendWebRTCMinBitrate(1e3 * e), l.Log(l.GetStackTrace(), "-------------------------------------------", 7) } )), this.config._addOnNumericSettingChangedListener(ce.WebRTCMaxBitrate, (e=>{ l.Log(l.GetStackTrace(), "-------- Sending web rtc settings --------", 7), this._webRtcController.sendWebRTCMaxBitrate(1e3 * e), l.Log(l.GetStackTrace(), "-------------------------------------------", 7) } )), this.config._addOnNumericSettingChangedListener(ce.WebRTCFPS, (e=>{ l.Log(l.GetStackTrace(), "-------- Sending web rtc settings --------", 7), this._webRtcController.sendWebRTCFps(e), l.Log(l.GetStackTrace(), "-------------------------------------------", 7) } )), this.config._addOnOptionSettingChangedListener(me.PreferredCodec, (e=>{ this._webRtcController && this._webRtcController.setPreferredCodec(e) } )), this.config._registerOnChangeEvents(this._eventEmitter) } _activateOnScreenKeyboard(e) { throw new Error("Method not implemented.") } _onInputControlOwnership(e) { this._inputController = e } setWebRtcPlayerController(e) { this._webRtcController = e, this._webRtcController.setPreferredCodec(this.config.getSettingOption(me.PreferredCodec).selected), this._webRtcController.resizePlayerStyle(), this.checkForAutoConnect() } connect() { this._webRtcController.connectToSignallingServer() } reconnect() { this._webRtcController.restartStreamAutomatically() } disconnect() { this._webRtcController.close() } play() { this._onStreamLoading(), this._webRtcController.playStream() } checkForAutoConnect() { this.config.isFlagEnabled(ae.AutoConnect) && (this._onWebRtcAutoConnect(), this._webRtcController.connectToSignallingServer()) } _onWebRtcAutoConnect() { this._eventEmitter.dispatchEvent(new U), this._showActionOrErrorOnDisconnect = !0 } _onWebRtcSdp() { this._eventEmitter.dispatchEvent(new _) } _onStreamLoading() { this._eventEmitter.dispatchEvent(new q) } _onDisconnect(e) { "" != this._webRtcController.getDisconnectMessageOverride() && void 0 !== this._webRtcController.getDisconnectMessageOverride() && null != this._webRtcController.getDisconnectMessageOverride() && (e = this._webRtcController.getDisconnectMessageOverride(), this._webRtcController.setDisconnectMessageOverride("")), this._eventEmitter.dispatchEvent(new G({ eventString: e, showActionOrErrorOnDisconnect: this._showActionOrErrorOnDisconnect })), 0 == this._showActionOrErrorOnDisconnect && (this._showActionOrErrorOnDisconnect = !0) } _onWebRtcConnecting() { this._eventEmitter.dispatchEvent(new z) } _onWebRtcConnected() { this._eventEmitter.dispatchEvent(new B) } _onWebRtcFailed() { this._eventEmitter.dispatchEvent(new N) } _onVideoInitialized() { this._eventEmitter.dispatchEvent(new Q), this._videoStartTime = Date.now() } _onLatencyTestResult(e) { this._eventEmitter.dispatchEvent(new ee({ latencyTimings: e })) } _onVideoStats(e) { this._videoStartTime && void 0 !== this._videoStartTime || (this._videoStartTime = Date.now()), e.handleSessionStatistics(this._videoStartTime, this._inputController, this._webRtcController.videoAvgQp), this._eventEmitter.dispatchEvent(new $({ aggregatedStats: e })) } _onVideoEncoderAvgQP(e) { this._eventEmitter.dispatchEvent(new D({ avgQP: e })) } _onInitialSettings(e) { var t; this._eventEmitter.dispatchEvent(new te({ settings: e })), e.PixelStreamingSettings && (this.allowConsoleCommands = null !== (t = e.PixelStreamingSettings.AllowPixelStreamingCommands) && void 0 !== t && t, !1 === this.allowConsoleCommands && l.Info(l.GetStackTrace(), "-AllowPixelStreamingCommands=false, sending arbitrary console commands from browser to UE is disabled.")); const s = this.config.useUrlParams , n = new URLSearchParams(window.location.search); e.EncoderSettings && (this.config.setNumericSetting(ce.MinQP, s && n.has(ce.MinQP) ? Number.parseInt(n.get(ce.MinQP)) : e.EncoderSettings.MinQP), this.config.setNumericSetting(ce.MaxQP, s && n.has(ce.MaxQP) ? Number.parseInt(n.get(ce.MaxQP)) : e.EncoderSettings.MaxQP)), e.WebRTCSettings && (this.config.setNumericSetting(ce.WebRTCMinBitrate, s && n.has(ce.WebRTCMinBitrate) ? Number.parseInt(n.get(ce.WebRTCMinBitrate)) / 1e3 : e.WebRTCSettings.MinBitrate / 1e3), this.config.setNumericSetting(ce.WebRTCMaxBitrate, s && n.has(ce.WebRTCMaxBitrate) ? Number.parseInt(n.get(ce.WebRTCMaxBitrate)) / 1e3 : e.WebRTCSettings.MaxBitrate / 1e3), this.config.setNumericSetting(ce.WebRTCFPS, s && n.has(ce.WebRTCFPS) ? Number.parseInt(n.get(ce.WebRTCFPS)) : e.WebRTCSettings.FPS)) } _onQualityControlOwnership(e) { this.config.setFlagEnabled(ae.IsQualityController, e) } requestLatencyTest() { return !!this._webRtcController.videoPlayer.isVideoReady() && (this._webRtcController.sendLatencyTest(), !0) } requestShowFps() { return !!this._webRtcController.videoPlayer.isVideoReady() && (this._webRtcController.sendShowFps(), !0) } requestIframe() { return !!this._webRtcController.videoPlayer.isVideoReady() && (this._webRtcController.sendIframeRequest(), !0) } emitUIInteraction(e) { return !!this._webRtcController.videoPlayer.isVideoReady() && (this._webRtcController.emitUIInteraction(e), !0) } emitCommand(e) { return !(!this._webRtcController.videoPlayer.isVideoReady() || !this.allowConsoleCommands && "ConsoleCommand"in e || (this._webRtcController.emitCommand(e), 0)) } emitConsoleCommand(e) { return !(!this.allowConsoleCommands || !this._webRtcController.videoPlayer.isVideoReady() || (this._webRtcController.emitConsoleCommand(e), 0)) } addResponseEventListener(e, t) { this._webRtcController.responseController.addResponseEventListener(e, t) } removeResponseEventListener(e) { this._webRtcController.responseController.removeResponseEventListener(e) } dispatchEvent(e) { return this._eventEmitter.dispatchEvent(e) } addEventListener(e, t) { this._eventEmitter.addEventListener(e, t) } removeEventListener(e, t) { this._eventEmitter.removeEventListener(e, t) } toggleXR() { this.webXrController.xrClicked() } setSignallingUrlBuilder(e) { this._webRtcController.signallingUrlBuilder = e } get webSocketController() { return this._webRtcController.webSocketController } get webXrController() { return this._webXrController } } var bt = a.Dz , Et = a.hV , wt = a.z$ , Tt = a.J0 , xt = a.De , Mt = a.$C , Pt = a.Nu , kt = a.zg , Rt = a.vU , Lt = a.rv , At = a.Nh , Ft = a.ss , Ot = a.cf , It = a.Yd , Dt = a.iM , _t = a.qy , Ut = a.ce , zt = a.sK , Bt = a.Ok , Nt = a.q5 , Gt = a.g , Ht = a.Ib , Wt = a.Az , Vt = a.Iw , Qt = a.qY , qt = a.db , Kt = a.Tn , jt = a.x_ , Xt = a.Am , Jt = a.u3 , Yt = a.Ni , $t = a.bq , Zt = a.$f; function es() { return es = Object.assign ? Object.assign.bind() : function(e) { for (var t = 1; t < arguments.length; t++) { var s = arguments[t]; for (var n in s) Object.prototype.hasOwnProperty.call(s, n) && (e[n] = s[n]) } return e } , es.apply(this, arguments) } var ts = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function(e) { return typeof e } : function(e) { return e && "function" == typeof Symbol && e.constructor === Symbol && e !== Symbol.prototype ? "symbol" : typeof e } ; const ss = "object" === ("undefined" == typeof window ? "undefined" : ts(window)) && "object" === ("undefined" == typeof document ? "undefined" : ts(document)) && 9 === document.nodeType; function ns(e) { return ns = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function(e) { return typeof e } : function(e) { return e && "function" == typeof Symbol && e.constructor === Symbol && e !== Symbol.prototype ? "symbol" : typeof e } , ns(e) } function is(e, t) { for (var s = 0; s < t.length; s++) { var n = t[s]; n.enumerable = n.enumerable || !1, n.configurable = !0, "value"in n && (n.writable = !0), Object.defineProperty(e, (i = function(e, t) { if ("object" !== ns(e) || null === e) return e; var s = e[Symbol.toPrimitive]; if (void 0 !== s) { var n = s.call(e, "string"); if ("object" !== ns(n)) return n; throw new TypeError("@@toPrimitive must return a primitive value.") } return String(e) }(n.key), "symbol" === ns(i) ? i : String(i)), n) } var i } function rs(e, t, s) { return t && is(e.prototype, t), s && is(e, s), Object.defineProperty(e, "prototype", { writable: !1 }), e } function os(e, t) { return os = Object.setPrototypeOf ? Object.setPrototypeOf.bind() : function(e, t) { return e.__proto__ = t, e } , os(e, t) } function as(e, t) { e.prototype = Object.create(t.prototype), e.prototype.constructor = e, os(e, t) } function ls(e) { if (void 0 === e) throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); return e } var cs = {}.constructor; function ds(e) { if (null == e || "object" != typeof e) return e; if (Array.isArray(e)) return e.map(ds); if (e.constructor !== cs) return e; var t = {}; for (var s in e) t[s] = ds(e[s]); return t } function hs(e, t, s) { void 0 === e && (e = "unnamed"); var n = s.jss , i = ds(t); return n.plugins.onCreateRule(e, i, s) || (e[0], null) } var us = function(e, t) { for (var s = "", n = 0; n < e.length && "!important" !== e[n]; n++) s && (s += t), s += e[n]; return s } , ms = function(e) { if (!Array.isArray(e)) return e; var t = ""; if (Array.isArray(e[0])) for (var s = 0; s < e.length && "!important" !== e[s]; s++) t && (t += ", "), t += us(e[s], " "); else t = us(e, ", "); return "!important" === e[e.length - 1] && (t += " !important"), t }; function gs(e) { return e && !1 === e.format ? { linebreak: "", space: "" } : { linebreak: "\n", space: " " } } function ps(e, t) { for (var s = "", n = 0; n < t; n++) s += " "; return s + e } function vs(e, t, s) { void 0 === s && (s = {}); var n = ""; if (!t) return n; var i = s.indent , r = void 0 === i ? 0 : i , o = t.fallbacks; !1 === s.format && (r = -1 / 0); var a = gs(s) , l = a.linebreak , c = a.space; if (e && r++, o) if (Array.isArray(o)) for (var d = 0; d < o.length; d++) { var h = o[d]; for (var u in h) { var m = h[u]; null != m && (n && (n += l), n += ps(u + ":" + c + ms(m) + ";", r)) } } else for (var g in o) { var p = o[g]; null != p && (n && (n += l), n += ps(g + ":" + c + ms(p) + ";", r)) } for (var v in t) { var f = t[v]; null != f && "fallbacks" !== v && (n && (n += l), n += ps(v + ":" + c + ms(f) + ";", r)) } return (n || s.allowEmpty) && e ? (n && (n = "" + l + n + l), ps("" + e + c + "{" + n, --r) + ps("}", r)) : n } var fs = /([[\].#*$><+~=|^:(),"'`\s])/g , Ss = "undefined" != typeof CSS && CSS.escape , Cs = function(e) { return Ss ? Ss(e) : e.replace(fs, "\\$1") } , ys = function() { function e(e, t, s) { this.type = "style", this.isProcessed = !1; var n = s.sheet , i = s.Renderer; this.key = e, this.options = s, this.style = t, n ? this.renderer = n.renderer : i && (this.renderer = new i) } return e.prototype.prop = function(e, t, s) { if (void 0 === t) return this.style[e]; var n = !!s && s.force; if (!n && this.style[e] === t) return this; var i = t; s && !1 === s.process || (i = this.options.jss.plugins.onChangeValue(t, e, this)); var r = null == i || !1 === i , o = e in this.style; if (r && !o && !n) return this; var a = r && o; if (a ? delete this.style[e] : this.style[e] = i, this.renderable && this.renderer) return a ? this.renderer.removeProperty(this.renderable, e) : this.renderer.setProperty(this.renderable, e, i), this; var l = this.options.sheet; return l && l.attached, this } , e }() , bs = function(e) { function t(t, s, n) { var i; i = e.call(this, t, s, n) || this; var r = n.selector , o = n.scoped , a = n.sheet , l = n.generateId; return r ? i.selectorText = r : !1 !== o && (i.id = l(ls(ls(i)), a), i.selectorText = "." + Cs(i.id)), i } as(t, e); var s = t.prototype; return s.applyTo = function(e) { var t = this.renderer; if (t) { var s = this.toJSON(); for (var n in s) t.setProperty(e, n, s[n]) } return this } , s.toJSON = function() { var e = {}; for (var t in this.style) { var s = this.style[t]; "object" != typeof s ? e[t] = s : Array.isArray(s) && (e[t] = ms(s)) } return e } , s.toString = function(e) { var t = this.options.sheet , s = t && t.options.link ? es({}, e, { allowEmpty: !0 }) : e; return vs(this.selectorText, this.style, s) } , rs(t, [{ key: "selector", set: function(e) { if (e !== this.selectorText) { this.selectorText = e; var t = this.renderer , s = this.renderable; s && t && (t.setSelector(s, e) || t.replaceRule(s, this)) } }, get: function() { return this.selectorText } }]), t }(ys) , Es = { onCreateRule: function(e, t, s) { return "@" === e[0] || s.parent && "keyframes" === s.parent.type ? null : new bs(e,t,s) } } , ws = { indent: 1, children: !0 } , Ts = /@([\w-]+)/ , xs = function() { function e(e, t, s) { this.type = "conditional", this.isProcessed = !1, this.key = e; var n = e.match(Ts); for (var i in this.at = n ? n[1] : "unknown", this.query = s.name || "@" + this.at, this.options = s, this.rules = new Xs(es({}, s, { parent: this })), t) this.rules.add(i, t[i]); this.rules.process() } var t = e.prototype; return t.getRule = function(e) { return this.rules.get(e) } , t.indexOf = function(e) { return this.rules.indexOf(e) } , t.addRule = function(e, t, s) { var n = this.rules.add(e, t, s); return n ? (this.options.jss.plugins.onProcessRule(n), n) : null } , t.replaceRule = function(e, t, s) { var n = this.rules.replace(e, t, s); return n && this.options.jss.plugins.onProcessRule(n), n } , t.toString = function(e) { void 0 === e && (e = ws); var t = gs(e).linebreak; if (null == e.indent && (e.indent = ws.indent), null == e.children && (e.children = ws.children), !1 === e.children) return this.query + " {}"; var s = this.rules.toString(e); return s ? this.query + " {" + t + s + t + "}" : "" } , e }() , Ms = /@container|@media|@supports\s+/ , Ps = { onCreateRule: function(e, t, s) { return Ms.test(e) ? new xs(e,t,s) : null } } , ks = { indent: 1, children: !0 } , Rs = /@keyframes\s+([\w-]+)/ , Ls = function() { function e(e, t, s) { this.type = "keyframes", this.at = "@keyframes", this.isProcessed = !1; var n = e.match(Rs); n && n[1] ? this.name = n[1] : this.name = "noname", this.key = this.type + "-" + this.name, this.options = s; var i = s.scoped , r = s.sheet , o = s.generateId; for (var a in this.id = !1 === i ? this.name : Cs(o(this, r)), this.rules = new Xs(es({}, s, { parent: this })), t) this.rules.add(a, t[a], es({}, s, { parent: this })); this.rules.process() } return e.prototype.toString = function(e) { void 0 === e && (e = ks); var t = gs(e).linebreak; if (null == e.indent && (e.indent = ks.indent), null == e.children && (e.children = ks.children), !1 === e.children) return this.at + " " + this.id + " {}"; var s = this.rules.toString(e); return s && (s = "" + t + s + t), this.at + " " + this.id + " {" + s + "}" } , e }() , As = /@keyframes\s+/ , Fs = /\$([\w-]+)/g , Os = function(e, t) { return "string" == typeof e ? e.replace(Fs, (function(e, s) { return s in t ? t[s] : e } )) : e } , Is = function(e, t, s) { var n = e[t] , i = Os(n, s); i !== n && (e[t] = i) } , Ds = { onCreateRule: function(e, t, s) { return "string" == typeof e && As.test(e) ? new Ls(e,t,s) : null }, onProcessStyle: function(e, t, s) { return "style" === t.type && s ? ("animation-name"in e && Is(e, "animation-name", s.keyframes), "animation"in e && Is(e, "animation", s.keyframes), e) : e }, onChangeValue: function(e, t, s) { var n = s.options.sheet; if (!n) return e; switch (t) { case "animation": case "animation-name": return Os(e, n.keyframes); default: return e } } } , _s = function(e) { function t() { return e.apply(this, arguments) || this } return as(t, e), t.prototype.toString = function(e) { var t = this.options.sheet , s = t && t.options.link ? es({}, e, { allowEmpty: !0 }) : e; return vs(this.key, this.style, s) } , t }(ys) , Us = { onCreateRule: function(e, t, s) { return s.parent && "keyframes" === s.parent.type ? new _s(e,t,s) : null } } , zs = function() { function e(e, t, s) { this.type = "font-face", this.at = "@font-face", this.isProcessed = !1, this.key = e, this.style = t, this.options = s } return e.prototype.toString = function(e) { var t = gs(e).linebreak; if (Array.isArray(this.style)) { for (var s = "", n = 0; n < this.style.length; n++) s += vs(this.at, this.style[n]), this.style[n + 1] && (s += t); return s } return vs(this.at, this.style, e) } , e }() , Bs = /@font-face/ , Ns = { onCreateRule: function(e, t, s) { return Bs.test(e) ? new zs(e,t,s) : null } } , Gs = function() { function e(e, t, s) { this.type = "viewport", this.at = "@viewport", this.isProcessed = !1, this.key = e, this.style = t, this.options = s } return e.prototype.toString = function(e) { return vs(this.key, this.style, e) } , e }() , Hs = { onCreateRule: function(e, t, s) { return "@viewport" === e || "@-ms-viewport" === e ? new Gs(e,t,s) : null } } , Ws = function() { function e(e, t, s) { this.type = "simple", this.isProcessed = !1, this.key = e, this.value = t, this.options = s } return e.prototype.toString = function(e) { if (Array.isArray(this.value)) { for (var t = "", s = 0; s < this.value.length; s++) t += this.key + " " + this.value[s] + ";", this.value[s + 1] && (t += "\n"); return t } return this.key + " " + this.value + ";" } , e }() , Vs = { "@charset": !0, "@import": !0, "@namespace": !0 } , Qs = { onCreateRule: function(e, t, s) { return e in Vs ? new Ws(e,t,s) : null } } , qs = [Es, Ps, Ds, Us, Ns, Hs, Qs] , Ks = { process: !0 } , js = { force: !0, process: !0 } , Xs = function() { function e(e) { this.map = {}, this.raw = {}, this.index = [], this.counter = 0, this.options = e, this.classes = e.classes, this.keyframes = e.keyframes } var t = e.prototype; return t.add = function(e, t, s) { var n = this.options , i = n.parent , r = n.sheet , o = n.jss , a = n.Renderer , l = n.generateId , c = n.scoped , d = es({ classes: this.classes, parent: i, sheet: r, jss: o, Renderer: a, generateId: l, scoped: c, name: e, keyframes: this.keyframes, selector: void 0 }, s) , h = e; e in this.raw && (h = e + "-d" + this.counter++), this.raw[h] = t, h in this.classes && (d.selector = "." + Cs(this.classes[h])); var u = hs(h, t, d); if (!u) return null; this.register(u); var m = void 0 === d.index ? this.index.length : d.index; return this.index.splice(m, 0, u), u } , t.replace = function(e, t, s) { var n = this.get(e) , i = this.index.indexOf(n); n && this.remove(n); var r = s; return -1 !== i && (r = es({}, s, { index: i })), this.add(e, t, r) } , t.get = function(e) { return this.map[e] } , t.remove = function(e) { this.unregister(e), delete this.raw[e.key], this.index.splice(this.index.indexOf(e), 1) } , t.indexOf = function(e) { return this.index.indexOf(e) } , t.process = function() { var e = this.options.jss.plugins; this.index.slice(0).forEach(e.onProcessRule, e) } , t.register = function(e) { this.map[e.key] = e, e instanceof bs ? (this.map[e.selector] = e, e.id && (this.classes[e.key] = e.id)) : e instanceof Ls && this.keyframes && (this.keyframes[e.name] = e.id) } , t.unregister = function(e) { delete this.map[e.key], e instanceof bs ? (delete this.map[e.selector], delete this.classes[e.key]) : e instanceof Ls && delete this.keyframes[e.name] } , t.update = function() { var e, t, s; if ("string" == typeof (arguments.length <= 0 ? void 0 : arguments[0]) ? (e = arguments.length <= 0 ? void 0 : arguments[0], t = arguments.length <= 1 ? void 0 : arguments[1], s = arguments.length <= 2 ? void 0 : arguments[2]) : (t = arguments.length <= 0 ? void 0 : arguments[0], s = arguments.length <= 1 ? void 0 : arguments[1], e = null), e) this.updateOne(this.get(e), t, s); else for (var n = 0; n < this.index.length; n++) this.updateOne(this.index[n], t, s) } , t.updateOne = function(t, s, n) { void 0 === n && (n = Ks); var i = this.options , r = i.jss.plugins , o = i.sheet; if (t.rules instanceof e) t.rules.update(s, n); else { var a = t.style; if (r.onUpdate(s, t, o, n), n.process && a && a !== t.style) { for (var l in r.onProcessStyle(t.style, t, o), t.style) { var c = t.style[l]; c !== a[l] && t.prop(l, c, js) } for (var d in a) { var h = t.style[d] , u = a[d]; null == h && h !== u && t.prop(d, null, js) } } } } , t.toString = function(e) { for (var t = "", s = this.options.sheet, n = !!s && s.options.link, i = gs(e).linebreak, r = 0; r < this.index.length; r++) { var o = this.index[r].toString(e); (o || n) && (t && (t += i), t += o) } return t } , e }() , Js = function() { function e(e, t) { for (var s in this.attached = !1, this.deployed = !1, this.classes = {}, this.keyframes = {}, this.options = es({}, t, { sheet: this, parent: this, classes: this.classes, keyframes: this.keyframes }), t.Renderer && (this.renderer = new t.Renderer(this)), this.rules = new Xs(this.options), e) this.rules.add(s, e[s]); this.rules.process() } var t = e.prototype; return t.attach = function() { return this.attached || (this.renderer && this.renderer.attach(), this.attached = !0, this.deployed || this.deploy()), this } , t.detach = function() { return this.attached ? (this.renderer && this.renderer.detach(), this.attached = !1, this) : this } , t.addRule = function(e, t, s) { var n = this.queue; this.attached && !n && (this.queue = []); var i = this.rules.add(e, t, s); return i ? (this.options.jss.plugins.onProcessRule(i), this.attached ? this.deployed ? (n ? n.push(i) : (this.insertRule(i), this.queue && (this.queue.forEach(this.insertRule, this), this.queue = void 0)), i) : i : (this.deployed = !1, i)) : null } , t.replaceRule = function(e, t, s) { var n = this.rules.get(e); if (!n) return this.addRule(e, t, s); var i = this.rules.replace(e, t, s); return i && this.options.jss.plugins.onProcessRule(i), this.attached ? this.deployed ? (this.renderer && (i ? n.renderable && this.renderer.replaceRule(n.renderable, i) : this.renderer.deleteRule(n)), i) : i : (this.deployed = !1, i) } , t.insertRule = function(e) { this.renderer && this.renderer.insertRule(e) } , t.addRules = function(e, t) { var s = []; for (var n in e) { var i = this.addRule(n, e[n], t); i && s.push(i) } return s } , t.getRule = function(e) { return this.rules.get(e) } , t.deleteRule = function(e) { var t = "object" == typeof e ? e : this.rules.get(e); return !(!t || this.attached && !t.renderable) && (this.rules.remove(t), !(this.attached && t.renderable && this.renderer) || this.renderer.deleteRule(t.renderable)) } , t.indexOf = function(e) { return this.rules.indexOf(e) } , t.deploy = function() { return this.renderer && this.renderer.deploy(), this.deployed = !0, this } , t.update = function() { var e; return (e = this.rules).update.apply(e, arguments), this } , t.updateOne = function(e, t, s) { return this.rules.updateOne(e, t, s), this } , t.toString = function(e) { return this.rules.toString(e) } , e }() , Ys = function() { function e() { this.plugins = { internal: [], external: [] }, this.registry = {} } var t = e.prototype; return t.onCreateRule = function(e, t, s) { for (var n = 0; n < this.registry.onCreateRule.length; n++) { var i = this.registry.onCreateRule[n](e, t, s); if (i) return i } return null } , t.onProcessRule = function(e) { if (!e.isProcessed) { for (var t = e.options.sheet, s = 0; s < this.registry.onProcessRule.length; s++) this.registry.onProcessRule[s](e, t); e.style && this.onProcessStyle(e.style, e, t), e.isProcessed = !0 } } , t.onProcessStyle = function(e, t, s) { for (var n = 0; n < this.registry.onProcessStyle.length; n++) t.style = this.registry.onProcessStyle[n](t.style, t, s) } , t.onProcessSheet = function(e) { for (var t = 0; t < this.registry.onProcessSheet.length; t++) this.registry.onProcessSheet[t](e) } , t.onUpdate = function(e, t, s, n) { for (var i = 0; i < this.registry.onUpdate.length; i++) this.registry.onUpdate[i](e, t, s, n) } , t.onChangeValue = function(e, t, s) { for (var n = e, i = 0; i < this.registry.onChangeValue.length; i++) n = this.registry.onChangeValue[i](n, t, s); return n } , t.use = function(e, t) { void 0 === t && (t = { queue: "external" }); var s = this.plugins[t.queue]; -1 === s.indexOf(e) && (s.push(e), this.registry = [].concat(this.plugins.external, this.plugins.internal).reduce((function(e, t) { for (var s in t) s in e && e[s].push(t[s]); return e } ), { onCreateRule: [], onProcessRule: [], onProcessStyle: [], onProcessSheet: [], onChangeValue: [], onUpdate: [] })) } , e }() , $s = function() { function e() { this.registry = [] } var t = e.prototype; return t.add = function(e) { var t = this.registry , s = e.options.index; if (-1 === t.indexOf(e)) if (0 === t.length || s >= this.index) t.push(e); else for (var n = 0; n < t.length; n++) if (t[n].options.index > s) return void t.splice(n, 0, e) } , t.reset = function() { this.registry = [] } , t.remove = function(e) { var t = this.registry.indexOf(e); this.registry.splice(t, 1) } , t.toString = function(e) { for (var t = void 0 === e ? {} : e, s = t.attached, n = function(e, t) { if (null == e) return {}; var s, n, i = {}, r = Object.keys(e); for (n = 0; n < r.length; n++) s = r[n], t.indexOf(s) >= 0 || (i[s] = e[s]); return i }(t, ["attached"]), i = gs(n).linebreak, r = "", o = 0; o < this.registry.length; o++) { var a = this.registry[o]; null != s && a.attached !== s || (r && (r += i), r += a.toString(n)) } return r } , rs(e, [{ key: "index", get: function() { return 0 === this.registry.length ? 0 : this.registry[this.registry.length - 1].options.index } }]), e }() , Zs = new $s , en = "undefined" != typeof globalThis ? globalThis : "undefined" != typeof window && window.Math === Math ? window : "undefined" != typeof self && self.Math === Math ? self : Function("return this")() , tn = "2f1acc6c3a606b082e5eef5e54414ffb"; null == en[tn] && (en[tn] = 0); var sn = en[tn]++ , nn = function(e) { void 0 === e && (e = {}); var t = 0; return function(s, n) { t += 1; var i = "" , r = ""; return n && (n.options.classNamePrefix && (r = n.options.classNamePrefix), null != n.options.jss.id && (i = String(n.options.jss.id))), e.minify ? "" + (r || "c") + sn + i + t : r + s.key + "-" + sn + (i ? "-" + i : "") + "-" + t } } , rn = function(e) { var t; return function() { return t || (t = e()), t } } , on = function(e, t) { try { return e.attributeStyleMap ? e.attributeStyleMap.get(t) : e.style.getPropertyValue(t) } catch (e) { return "" } } , an = function(e, t, s) { try { var n = s; if (Array.isArray(s) && (n = ms(s)), e.attributeStyleMap) e.attributeStyleMap.set(t, n); else { var i = n ? n.indexOf("!important") : -1 , r = i > -1 ? n.substr(0, i - 1) : n; e.style.setProperty(t, r, i > -1 ? "important" : "") } } catch (e) { return !1 } return !0 } , ln = function(e, t) { try { e.attributeStyleMap ? e.attributeStyleMap.delete(t) : e.style.removeProperty(t) } catch (e) {} } , cn = function(e, t) { return e.selectorText = t, e.selectorText === t } , dn = rn((function() { return document.querySelector("head") } )) , hn = rn((function() { var e = document.querySelector('meta[property="csp-nonce"]'); return e ? e.getAttribute("content") : null } )) , un = function(e, t, s) { try { "insertRule"in e ? e.insertRule(t, s) : "appendRule"in e && e.appendRule(t) } catch (e) { return !1 } return e.cssRules[s] } , mn = function(e, t) { var s = e.cssRules.length; return void 0 === t || t > s ? s : t } , gn = function() { function e(e) { this.getPropertyValue = on, this.setProperty = an, this.removeProperty = ln, this.setSelector = cn, this.hasInsertedRules = !1, this.cssRules = [], e && Zs.add(e), this.sheet = e; var t, s = this.sheet ? this.sheet.options : {}, n = s.media, i = s.meta, r = s.element; this.element = r || ((t = document.createElement("style")).textContent = "\n", t), this.element.setAttribute("data-jss", ""), n && this.element.setAttribute("media", n), i && this.element.setAttribute("data-meta", i); var o = hn(); o && this.element.setAttribute("nonce", o) } var t = e.prototype; return t.attach = function() { if (!this.element.parentNode && this.sheet) { !function(e, t) { var s = t.insertionPoint , n = function(e) { var t = Zs.registry; if (t.length > 0) { var s = function(e, t) { for (var s = 0; s < e.length; s++) { var n = e[s]; if (n.attached && n.options.index > t.index && n.options.insertionPoint === t.insertionPoint) return n } return null }(t, e); if (s && s.renderer) return { parent: s.renderer.element.parentNode, node: s.renderer.element }; if (s = function(e, t) { for (var s = e.length - 1; s >= 0; s--) { var n = e[s]; if (n.attached && n.options.insertionPoint === t.insertionPoint) return n } return null }(t, e), s && s.renderer) return { parent: s.renderer.element.parentNode, node: s.renderer.element.nextSibling } } var n = e.insertionPoint; if (n && "string" == typeof n) { var i = function(e) { for (var t = dn(), s = 0; s < t.childNodes.length; s++) { var n = t.childNodes[s]; if (8 === n.nodeType && n.nodeValue.trim() === e) return n } return null }(n); if (i) return { parent: i.parentNode, node: i.nextSibling } } return !1 }(t); if (!1 !== n && n.parent) n.parent.insertBefore(e, n.node); else if (s && "number" == typeof s.nodeType) { var i = s , r = i.parentNode; r && r.insertBefore(e, i.nextSibling) } else dn().appendChild(e) }(this.element, this.sheet.options); var e = Boolean(this.sheet && this.sheet.deployed); this.hasInsertedRules && e && (this.hasInsertedRules = !1, this.deploy()) } } , t.detach = function() { if (this.sheet) { var e = this.element.parentNode; e && e.removeChild(this.element), this.sheet.options.link && (this.cssRules = [], this.element.textContent = "\n") } } , t.deploy = function() { var e = this.sheet; e && (e.options.link ? this.insertRules(e.rules) : this.element.textContent = "\n" + e.toString() + "\n") } , t.insertRules = function(e, t) { for (var s = 0; s < e.index.length; s++) this.insertRule(e.index[s], s, t) } , t.insertRule = function(e, t, s) { if (void 0 === s && (s = this.element.sheet), e.rules) { var n = e , i = s; if ("conditional" === e.type || "keyframes" === e.type) { var r = mn(s, t); if (!1 === (i = un(s, n.toString({ children: !1 }), r))) return !1; this.refCssRule(e, r, i) } return this.insertRules(n.rules, i), i } var o = e.toString(); if (!o) return !1; var a = mn(s, t) , l = un(s, o, a); return !1 !== l && (this.hasInsertedRules = !0, this.refCssRule(e, a, l), l) } , t.refCssRule = function(e, t, s) { e.renderable = s, e.options.parent instanceof Js && this.cssRules.splice(t, 0, s) } , t.deleteRule = function(e) { var t = this.element.sheet , s = this.indexOf(e); return -1 !== s && (t.deleteRule(s), this.cssRules.splice(s, 1), !0) } , t.indexOf = function(e) { return this.cssRules.indexOf(e) } , t.replaceRule = function(e, t) { var s = this.indexOf(e); return -1 !== s && (this.element.sheet.deleteRule(s), this.cssRules.splice(s, 1), this.insertRule(t, s)) } , t.getRules = function() { return this.element.sheet.cssRules } , e }() , pn = 0 , vn = function() { function e(e) { this.id = pn++, this.version = "10.10.0", this.plugins = new Ys, this.options = { id: { minify: !1 }, createGenerateId: nn, Renderer: ss ? gn : null, plugins: [] }, this.generateId = nn({ minify: !1 }); for (var t = 0; t < qs.length; t++) this.plugins.use(qs[t], { queue: "internal" }); this.setup(e) } var t = e.prototype; return t.setup = function(e) { return void 0 === e && (e = {}), e.createGenerateId && (this.options.createGenerateId = e.createGenerateId), e.id && (this.options.id = es({}, this.options.id, e.id)), (e.createGenerateId || e.id) && (this.generateId = this.options.createGenerateId(this.options.id)), null != e.insertionPoint && (this.options.insertionPoint = e.insertionPoint), "Renderer"in e && (this.options.Renderer = e.Renderer), e.plugins && this.use.apply(this, e.plugins), this } , t.createStyleSheet = function(e, t) { void 0 === t && (t = {}); var s = t.index; "number" != typeof s && (s = 0 === Zs.index ? 0 : Zs.index + 1); var n = new Js(e,es({}, t, { jss: this, generateId: t.generateId || this.generateId, insertionPoint: this.options.insertionPoint, Renderer: this.options.Renderer, index: s })); return this.plugins.onProcessSheet(n), n } , t.removeStyleSheet = function(e) { return e.detach(), Zs.remove(e), this } , t.createRule = function(e, t, s) { if (void 0 === t && (t = {}), void 0 === s && (s = {}), "object" == typeof e) return this.createRule(void 0, e, t); var n = es({}, s, { name: e, jss: this, Renderer: this.options.Renderer }); n.generateId || (n.generateId = this.generateId), n.classes || (n.classes = {}), n.keyframes || (n.keyframes = {}); var i = hs(e, t, n); return i && this.plugins.onProcessRule(i), i } , t.use = function() { for (var e = this, t = arguments.length, s = new Array(t), n = 0; n < t; n++) s[n] = arguments[n]; return s.forEach((function(t) { e.plugins.use(t) } )), this } , e }(); "object" == typeof CSS && null != CSS && CSS; const fn = new vn(void 0); var Sn = "@global" , Cn = "@global " , yn = function() { function e(e, t, s) { for (var n in this.type = "global", this.at = Sn, this.isProcessed = !1, this.key = e, this.options = s, this.rules = new Xs(es({}, s, { parent: this })), t) this.rules.add(n, t[n]); this.rules.process() } var t = e.prototype; return t.getRule = function(e) { return this.rules.get(e) } , t.addRule = function(e, t, s) { var n = this.rules.add(e, t, s); return n && this.options.jss.plugins.onProcessRule(n), n } , t.replaceRule = function(e, t, s) { var n = this.rules.replace(e, t, s); return n && this.options.jss.plugins.onProcessRule(n), n } , t.indexOf = function(e) { return this.rules.indexOf(e) } , t.toString = function(e) { return this.rules.toString(e) } , e }() , bn = function() { function e(e, t, s) { this.type = "global", this.at = Sn, this.isProcessed = !1, this.key = e, this.options = s; var n = e.substr(Cn.length); this.rule = s.jss.createRule(n, t, es({}, s, { parent: this })) } return e.prototype.toString = function(e) { return this.rule ? this.rule.toString(e) : "" } , e }() , En = /\s*,\s*/g; function wn(e, t) { for (var s = e.split(En), n = "", i = 0; i < s.length; i++) n += t + " " + s[i].trim(), s[i + 1] && (n += ", "); return n } const Tn = function() { return { onCreateRule: function(e, t, s) { if (!e) return null; if (e === Sn) return new yn(e,t,s); if ("@" === e[0] && e.substr(0, Cn.length) === Cn) return new bn(e,t,s); var n = s.parent; return n && ("global" === n.type || n.options.parent && "global" === n.options.parent.type) && (s.scoped = !1), s.selector || !1 !== s.scoped || (s.selector = e), null }, onProcessRule: function(e, t) { "style" === e.type && t && (function(e, t) { var s = e.options , n = e.style , i = n ? n[Sn] : null; if (i) { for (var r in i) t.addRule(r, i[r], es({}, s, { selector: wn(r, e.selector) })); delete n[Sn] } }(e, t), function(e, t) { var s = e.options , n = e.style; for (var i in n) if ("@" === i[0] && i.substr(0, Sn.length) === Sn) { var r = wn(i.substr(Sn.length), e.selector); t.addRule(r, n[i], es({}, s, { selector: r })), delete n[i] } }(e, t)) } } }; var xn = /[A-Z]/g , Mn = /^ms-/ , Pn = {}; function kn(e) { return "-" + e.toLowerCase() } const Rn = function(e) { if (Pn.hasOwnProperty(e)) return Pn[e]; var t = e.replace(xn, kn); return Pn[e] = Mn.test(t) ? "-" + t : t }; function Ln(e) { var t = {}; for (var s in e) t[0 === s.indexOf("--") ? s : Rn(s)] = e[s]; return e.fallbacks && (Array.isArray(e.fallbacks) ? t.fallbacks = e.fallbacks.map(Ln) : t.fallbacks = Ln(e.fallbacks)), t } const An = function() { return { onProcessStyle: function(e) { if (Array.isArray(e)) { for (var t = 0; t < e.length; t++) e[t] = Ln(e[t]); return e } return Ln(e) }, onChangeValue: function(e, t, s) { if (0 === t.indexOf("--")) return e; var n = Rn(t); return t === n ? e : (s.prop(n, e), null) } } }; var Fn = { d: (e,t)=>{ for (var s in t) Fn.o(t, s) && !Fn.o(e, s) && Object.defineProperty(e, s, { enumerable: !0, get: t[s] }) } , o: (e,t)=>Object.prototype.hasOwnProperty.call(e, t) } , On = {}; Fn.d(On, { ym: ()=>qn, EW: ()=>Bn, Mx: ()=>vi, L3: ()=>pi, rl: ()=>Nn, NR: ()=>Gn, f1: ()=>Qn, IY: ()=>Vn, RZ: ()=>zn, XY: ()=>yi, Fk: ()=>Hn, Zh: ()=>ci, ek: ()=>di, MG: ()=>hi, gE: ()=>mi, tW: ()=>ui, wr: ()=>Wn, QT: ()=>Un }); const In = (Dn = { Flags: ()=>Rt, Logger: ()=>It, NumericParameters: ()=>zt, OptionParameters: ()=>Bt, SettingFlag: ()=>Wt, TextParameters: ()=>jt, WebXRController: ()=>Zt }, _n = {}, Fn.d(_n, Dn), _n); var Dn, _n, Un; class zn { constructor(e, t, s) { this.rootDiv = e, this.rootElement = t, this.textElement = s, this.rootElement.appendChild(this.textElement), this.hide(), this.rootDiv.appendChild(this.rootElement) } show() { this.rootElement.classList.remove("hiddenState") } hide() { this.rootElement.classList.add("hiddenState") } } class Bn extends zn { constructor(e, t, s) { super(e, t, s), this.onActionCallback = ()=>{ In.Logger.Info(In.Logger.GetStackTrace(), "Did you forget to set the onAction callback in your overlay?") } } update(e) { null == e && null == e || (this.textElement.innerHTML = e) } onAction(e) { this.onActionCallback = e } activate() { this.onActionCallback() } } class Nn extends Bn { static createRootElement() { const e = document.createElement("div"); return e.id = "connectOverlay", e.className = "clickableState", e } static createContentElement() { const e = document.createElement("div"); return e.id = "connectButton", e.innerHTML = "Click to start", e } constructor(e) { super(e, Nn.createRootElement(), Nn.createContentElement()), this.rootElement.addEventListener("click", (()=>{ this.activate() } )) } } class Gn extends Bn { static createRootElement() { const e = document.createElement("div"); return e.id = "disconnectOverlay", e.className = "clickableState", e } static createContentElement() { const e = document.createElement("div"); return e.id = "disconnectButton", e.innerHTML = "Click To Restart", e } constructor(e) { super(e, Gn.createRootElement(), Gn.createContentElement()), this.rootElement.addEventListener("click", (()=>{ this.activate() } )) } } class Hn extends Bn { static createRootElement() { const e = document.createElement("div"); return e.id = "playOverlay", e.className = "clickableState", e } static createContentElement() { const e = document.createElement("img"); return e.id = "playButton", e.src = "", e.alt = "Start Streaming", e } constructor(e) { super(e, Hn.createRootElement(), Hn.createContentElement()), this.rootElement.addEventListener("click", (()=>{ this.activate() } )) } } class Wn extends zn { constructor(e, t, s) { super(e, t, s) } update(e) { null == e && null == e || (this.textElement.innerHTML = e) } } class Vn extends Wn { static createRootElement() { const e = document.createElement("div"); return e.id = "infoOverlay", e.className = "textDisplayState", e } static createContentElement() { const e = document.createElement("div"); return e.id = "messageOverlayInner", e } constructor(e) { super(e, Vn.createRootElement(), Vn.createContentElement()) } } class Qn extends Wn { static createRootElement() { const e = document.createElement("div"); return e.id = "errorOverlay", e.className = "textDisplayState", e } static createContentElement() { const e = document.createElement("div"); return e.id = "errorOverlayInner", e } constructor(e) { super(e, Qn.createRootElement(), Qn.createContentElement()) } } class qn extends Bn { static createRootElement() { const e = document.createElement("div"); return e.id = "afkOverlay", e.className = "clickableState", e } static createContentElement() { const e = document.createElement("div"); return e.id = "afkOverlayInner", e.innerHTML = '