(function webpackUniversalModuleDefinition(root, factory) { if(typeof exports === 'object' && typeof module === 'object') module.exports = factory(require("sdp")); else if(typeof define === 'function' && define.amd) define(["sdp"], factory); else if(typeof exports === 'object') exports["lib-pixelstreamingfrontend"] = factory(require("sdp")); else root["lib-pixelstreamingfrontend"] = factory(root["sdp"]); })(this, (__WEBPACK_EXTERNAL_MODULE_sdp__) => { return /******/ (() => { // webpackBootstrap /******/ "use strict"; /******/ var __webpack_modules__ = ({ /***/ "./src/AFK/AFKController.ts": /*!**********************************!*\ !*** ./src/AFK/AFKController.ts ***! \**********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "AFKController": () => (/* binding */ AFKController) /* harmony export */ }); /* harmony import */ var _Config_Config__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Config/Config */ "./src/Config/Config.ts"); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Util/EventEmitter */ "./src/Util/EventEmitter.ts"); // Copyright Epic Games, Inc. All Rights Reserved. class AFKController { constructor(config, pixelStreaming, onDismissAfk) { // time out logic details this.closeTimeout = 10; this.active = false; this.countdownActive = false; this.warnTimer = undefined; this.countDown = 0; this.countDownTimer = undefined; this.config = config; this.pixelStreaming = pixelStreaming; this.onDismissAfk = onDismissAfk; this.onAFKTimedOutCallback = () => { console.log('AFK timed out, did you want to override this callback?'); }; } /** * The methods that occur when an afk event listener is clicked */ onAfkClick() { clearInterval(this.countDownTimer); if (this.active || this.countdownActive) { this.startAfkWarningTimer(); this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.AfkWarningDeactivateEvent()); } } /** * Start the warning timer if a timeout is set greater that 0 seconds */ startAfkWarningTimer() { if (this.config.getNumericSettingValue(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.NumericParameters.AFKTimeoutSecs) > 0 && this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.Flags.AFKDetection)) { this.active = true; } else { this.active = false; } this.resetAfkWarningTimer(); } /** * Stop the afk warning timer */ stopAfkWarningTimer() { this.active = false; this.countdownActive = false; clearTimeout(this.warnTimer); clearInterval(this.countDownTimer); } /** * Pause the timer which when elapsed will warn the user they are inactive. */ pauseAfkWarningTimer() { this.active = false; } /** * If the user interacts then reset the warning timer. */ resetAfkWarningTimer() { if (this.active && this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.Flags.AFKDetection)) { clearTimeout(this.warnTimer); this.warnTimer = setTimeout(() => this.activateAfkEvent(), this.config.getNumericSettingValue(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.NumericParameters.AFKTimeoutSecs) * 1000); } } /** * Show the AFK overlay and begin the countDown */ activateAfkEvent() { // Pause the timer while the user is looking at the inactivity warning overlay this.pauseAfkWarningTimer(); // instantiate a new overlay this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.AfkWarningActivateEvent({ countDown: this.countDown, dismissAfk: this.onDismissAfk })); // update our countDown timer and overlay contents this.countDown = this.closeTimeout; this.countdownActive = true; this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.AfkWarningUpdateEvent({ countDown: this.countDown })); // if we are in locked mouse exit pointerlock if (!this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.Flags.HoveringMouseMode)) { // minor hack to alleviate ios not supporting pointerlock if (document.exitPointerLock) { document.exitPointerLock(); } } // reset our countDown interval accordingly this.countDownTimer = setInterval(() => { this.countDown--; if (this.countDown == 0) { // The user failed to click so hide the overlay and disconnect them. this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.AfkTimedOutEvent()); this.onAFKTimedOutCallback(); _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), 'You have been disconnected due to inactivity'); // switch off the afk feature as stream has closed this.stopAfkWarningTimer(); } else { this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.AfkWarningUpdateEvent({ countDown: this.countDown })); } }, 1000); } } /***/ }), /***/ "./src/Config/Config.ts": /*!******************************!*\ !*** ./src/Config/Config.ts ***! \******************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Config": () => (/* binding */ Config), /* harmony export */ "ControlSchemeType": () => (/* binding */ ControlSchemeType), /* harmony export */ "Flags": () => (/* binding */ Flags), /* harmony export */ "NumericParameters": () => (/* binding */ NumericParameters), /* harmony export */ "OptionParameters": () => (/* binding */ OptionParameters), /* harmony export */ "TextParameters": () => (/* binding */ TextParameters) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _SettingFlag__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./SettingFlag */ "./src/Config/SettingFlag.ts"); /* harmony import */ var _SettingNumber__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./SettingNumber */ "./src/Config/SettingNumber.ts"); /* harmony import */ var _SettingText__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./SettingText */ "./src/Config/SettingText.ts"); /* harmony import */ var _SettingOption__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./SettingOption */ "./src/Config/SettingOption.ts"); /* harmony import */ var _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../Util/EventEmitter */ "./src/Util/EventEmitter.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * A collection of flags that can be toggled and are core to all Pixel Streaming experiences. * These are used in the `Config.Flags` map. */ class Flags { } Flags.AutoConnect = 'AutoConnect'; Flags.AutoPlayVideo = 'AutoPlayVideo'; Flags.AFKDetection = 'TimeoutIfIdle'; Flags.BrowserSendOffer = 'OfferToReceive'; Flags.HoveringMouseMode = 'HoveringMouse'; Flags.ForceMonoAudio = 'ForceMonoAudio'; Flags.ForceTURN = 'ForceTURN'; Flags.FakeMouseWithTouches = 'FakeMouseWithTouches'; Flags.IsQualityController = 'ControlsQuality'; Flags.MatchViewportResolution = 'MatchViewportRes'; Flags.StartVideoMuted = 'StartVideoMuted'; Flags.SuppressBrowserKeys = 'SuppressBrowserKeys'; Flags.UseMic = 'UseMic'; Flags.KeyboardInput = 'KeyboardInput'; Flags.MouseInput = 'MouseInput'; Flags.TouchInput = 'TouchInput'; Flags.GamepadInput = 'GamepadInput'; Flags.XRControllerInput = 'XRControllerInput'; Flags.WaitForStreamer = "WaitForStreamer"; const isFlagId = (id) => Object.getOwnPropertyNames(Flags).some((name) => Flags[name] === id); /** * A collection of numeric parameters that are core to all Pixel Streaming experiences. * */ class NumericParameters { } NumericParameters.AFKTimeoutSecs = 'AFKTimeout'; NumericParameters.MinQP = 'MinQP'; NumericParameters.MaxQP = 'MaxQP'; NumericParameters.WebRTCFPS = 'WebRTCFPS'; NumericParameters.WebRTCMinBitrate = 'WebRTCMinBitrate'; NumericParameters.WebRTCMaxBitrate = 'WebRTCMaxBitrate'; NumericParameters.MaxReconnectAttempts = 'MaxReconnectAttempts'; NumericParameters.StreamerAutoJoinInterval = 'StreamerAutoJoinInterval'; const isNumericId = (id) => Object.getOwnPropertyNames(NumericParameters).some((name) => NumericParameters[name] === id); /** * A collection of textual parameters that are core to all Pixel Streaming experiences. * */ class TextParameters { } TextParameters.SignallingServerUrl = 'ss'; const isTextId = (id) => Object.getOwnPropertyNames(TextParameters).some((name) => TextParameters[name] === id); /** * A collection of enum based parameters that are core to all Pixel Streaming experiences. * */ class OptionParameters { } OptionParameters.PreferredCodec = 'PreferredCodec'; OptionParameters.StreamerId = 'StreamerId'; const isOptionId = (id) => Object.getOwnPropertyNames(OptionParameters).some((name) => OptionParameters[name] === id); class Config { // ------------ Settings ----------------- constructor(config = {}) { /* A map of flags that can be toggled - options that can be set in the application - e.g. Use Mic? */ this.flags = new Map(); /* A map of numerical settings - options that can be in the application - e.g. MinBitrate */ this.numericParameters = new Map(); /* A map of text settings - e.g. signalling server url */ this.textParameters = new Map(); /* A map of enum based settings - e.g. preferred codec */ this.optionParameters = new Map(); const { initialSettings, useUrlParams } = config; this._useUrlParams = !!useUrlParams; this.populateDefaultSettings(this._useUrlParams); if (initialSettings) { this.setSettings(initialSettings); } } /** * True if reading configuration initial values from URL parameters, and * persisting changes in URL when changed. */ get useUrlParams() { return this._useUrlParams; } /** * Populate the default settings for a Pixel Streaming application */ populateDefaultSettings(useUrlParams) { /** * Text Parameters */ this.textParameters.set(TextParameters.SignallingServerUrl, new _SettingText__WEBPACK_IMPORTED_MODULE_0__.SettingText(TextParameters.SignallingServerUrl, 'Signalling url', 'Url of the signalling server', (location.protocol === 'https:' ? 'wss://' : 'ws://') + window.location.hostname + // for readability, we omit the port if it's 80 (window.location.port === '80' || window.location.port === '' ? '' : `:${window.location.port}`), useUrlParams)); this.optionParameters.set(OptionParameters.StreamerId, new _SettingOption__WEBPACK_IMPORTED_MODULE_1__.SettingOption(OptionParameters.StreamerId, 'Streamer ID', 'The ID of the streamer to stream.', '', [], useUrlParams)); /** * Enum Parameters */ this.optionParameters.set(OptionParameters.PreferredCodec, new _SettingOption__WEBPACK_IMPORTED_MODULE_1__.SettingOption(OptionParameters.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 browserSupportedCodecs = []; // Try get the info needed from the RTCRtpReceiver. This is only available on chrome if (!RTCRtpReceiver.getCapabilities) { browserSupportedCodecs.push('Only available on Chrome'); return browserSupportedCodecs; } const matcher = /(VP\d|H26\d|AV1).*/; const codecs = RTCRtpReceiver.getCapabilities('video').codecs; codecs.forEach((codec) => { const str = codec.mimeType.split('/')[1] + ' ' + (codec.sdpFmtpLine || ''); const match = matcher.exec(str); if (match !== null) { browserSupportedCodecs.push(str); } }); return browserSupportedCodecs; })(), useUrlParams)); /** * Boolean parameters */ this.flags.set(Flags.AutoConnect, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.AutoConnect, 'Auto connect to stream', 'Whether we should attempt to auto connect to the signalling server or show a click to start prompt.', false, useUrlParams)); this.flags.set(Flags.AutoPlayVideo, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.AutoPlayVideo, 'Auto play video', 'When video is ready automatically start playing it as opposed to showing a play button.', true, useUrlParams)); this.flags.set(Flags.BrowserSendOffer, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.BrowserSendOffer, 'Browser send offer', 'Browser will initiate the WebRTC handshake by sending the offer to the streamer', false, useUrlParams)); this.flags.set(Flags.UseMic, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.UseMic, 'Use microphone', 'Make browser request microphone access and open an input audio track.', false, useUrlParams)); this.flags.set(Flags.StartVideoMuted, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.StartVideoMuted, 'Start video muted', 'Video will start muted if true.', false, useUrlParams)); this.flags.set(Flags.SuppressBrowserKeys, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.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.', true, useUrlParams)); this.flags.set(Flags.IsQualityController, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.IsQualityController, 'Is quality controller?', 'True if this peer controls stream quality', true, useUrlParams)); this.flags.set(Flags.ForceMonoAudio, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.ForceMonoAudio, 'Force mono audio', 'Force browser to request mono audio in the SDP', false, useUrlParams)); this.flags.set(Flags.ForceTURN, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.ForceTURN, 'Force TURN', 'Only generate TURN/Relayed ICE candidates.', false, useUrlParams)); this.flags.set(Flags.AFKDetection, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.AFKDetection, 'AFK if idle', 'Timeout the experience if user is AFK for a period.', false, useUrlParams)); this.flags.set(Flags.MatchViewportResolution, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.MatchViewportResolution, 'Match viewport resolution', 'Pixel Streaming will be instructed to dynamically resize the video stream to match the size of the video element.', false, useUrlParams)); this.flags.set(Flags.HoveringMouseMode, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.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.', false, useUrlParams, (isHoveringMouse, setting) => { setting.label = `Control Scheme: ${isHoveringMouse ? 'Hovering' : 'Locked'} Mouse`; })); this.flags.set(Flags.FakeMouseWithTouches, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.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.', false, useUrlParams)); this.flags.set(Flags.KeyboardInput, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.KeyboardInput, 'Keyboard input', 'If enabled, send keyboard events to streamer', true, useUrlParams)); this.flags.set(Flags.MouseInput, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.MouseInput, 'Mouse input', 'If enabled, send mouse events to streamer', true, useUrlParams)); this.flags.set(Flags.TouchInput, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.TouchInput, 'Touch input', 'If enabled, send touch events to streamer', true, useUrlParams)); this.flags.set(Flags.GamepadInput, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.GamepadInput, 'Gamepad input', 'If enabled, send gamepad events to streamer', true, useUrlParams)); this.flags.set(Flags.XRControllerInput, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.XRControllerInput, 'XR controller input', 'If enabled, send XR controller events to streamer', true, useUrlParams)); this.flags.set(Flags.WaitForStreamer, new _SettingFlag__WEBPACK_IMPORTED_MODULE_2__.SettingFlag(Flags.WaitForStreamer, 'Wait for streamer', 'Will continue trying to connect to the first streamer available.', true, useUrlParams)); /** * Numeric parameters */ this.numericParameters.set(NumericParameters.AFKTimeoutSecs, new _SettingNumber__WEBPACK_IMPORTED_MODULE_3__.SettingNumber(NumericParameters.AFKTimeoutSecs, 'AFK timeout', 'The time (in seconds) it takes for the application to time out if AFK timeout is enabled.', 0 /*min*/, 600 /*max*/, 120 /*value*/, useUrlParams)); this.numericParameters.set(NumericParameters.MaxReconnectAttempts, new _SettingNumber__WEBPACK_IMPORTED_MODULE_3__.SettingNumber(NumericParameters.MaxReconnectAttempts, 'Max Reconnects', 'Maximum number of reconnects the application will attempt when a streamer disconnects.', 0 /*min*/, 999 /*max*/, 3 /*value*/, useUrlParams)); this.numericParameters.set(NumericParameters.MinQP, new _SettingNumber__WEBPACK_IMPORTED_MODULE_3__.SettingNumber(NumericParameters.MinQP, 'Min QP', 'The lower bound for the quantization parameter (QP) of the encoder. 0 = Best quality, 51 = worst quality.', 0 /*min*/, 51 /*max*/, 0 /*value*/, useUrlParams)); this.numericParameters.set(NumericParameters.MaxQP, new _SettingNumber__WEBPACK_IMPORTED_MODULE_3__.SettingNumber(NumericParameters.MaxQP, 'Max QP', 'The upper bound for the quantization parameter (QP) of the encoder. 0 = Best quality, 51 = worst quality.', 0 /*min*/, 51 /*max*/, 51 /*value*/, useUrlParams)); this.numericParameters.set(NumericParameters.WebRTCFPS, new _SettingNumber__WEBPACK_IMPORTED_MODULE_3__.SettingNumber(NumericParameters.WebRTCFPS, 'Max FPS', 'The maximum FPS that WebRTC will try to transmit frames at.', 1 /*min*/, 999 /*max*/, 60 /*value*/, useUrlParams)); this.numericParameters.set(NumericParameters.WebRTCMinBitrate, new _SettingNumber__WEBPACK_IMPORTED_MODULE_3__.SettingNumber(NumericParameters.WebRTCMinBitrate, 'Min Bitrate (kbps)', 'The minimum bitrate that WebRTC should use.', 0 /*min*/, 500000 /*max*/, 0 /*value*/, useUrlParams)); this.numericParameters.set(NumericParameters.WebRTCMaxBitrate, new _SettingNumber__WEBPACK_IMPORTED_MODULE_3__.SettingNumber(NumericParameters.WebRTCMaxBitrate, 'Max Bitrate (kbps)', 'The maximum bitrate that WebRTC should use.', 0 /*min*/, 500000 /*max*/, 0 /*value*/, useUrlParams)); this.numericParameters.set(NumericParameters.StreamerAutoJoinInterval, new _SettingNumber__WEBPACK_IMPORTED_MODULE_3__.SettingNumber(NumericParameters.StreamerAutoJoinInterval, 'Streamer Auto Join Interval (ms)', 'Delay between retries when waiting for an available streamer.', 500 /*min*/, 900000 /*max*/, 3000 /*value*/, useUrlParams)); } /** * Add a callback to fire when the numeric setting is toggled. * @param id The id of the flag. * @param onChangedListener The callback to fire when the numeric value changes. */ _addOnNumericSettingChangedListener(id, onChangedListener) { if (this.numericParameters.has(id)) { this.numericParameters .get(id) .addOnChangedListener(onChangedListener); } } _addOnOptionSettingChangedListener(id, onChangedListener) { if (this.optionParameters.has(id)) { this.optionParameters .get(id) .addOnChangedListener(onChangedListener); } } /** * @param id The id of the numeric setting we are interested in getting a value for. * @returns The numeric value stored in the parameter with the passed id. */ getNumericSettingValue(id) { if (this.numericParameters.has(id)) { return this.numericParameters.get(id).number; } else { throw new Error(`There is no numeric setting with the id of ${id}`); } } /** * @param id The id of the text setting we are interested in getting a value for. * @returns The text value stored in the parameter with the passed id. */ getTextSettingValue(id) { if (this.textParameters.has(id)) { return this.textParameters.get(id).value; } else { throw new Error(`There is no numeric setting with the id of ${id}`); } } /** * Set number in the setting. * @param id The id of the numeric setting we are interested in. * @param value The numeric value to set. */ setNumericSetting(id, value) { if (this.numericParameters.has(id)) { this.numericParameters.get(id).number = value; } else { throw new Error(`There is no numeric setting with the id of ${id}`); } } /** * Add a callback to fire when the flag is toggled. * @param id The id of the flag. * @param onChangeListener The callback to fire when the value changes. */ _addOnSettingChangedListener(id, onChangeListener) { if (this.flags.has(id)) { this.flags.get(id).onChange = onChangeListener; } } /** * Add a callback to fire when the text is changed. * @param id The id of the flag. * @param onChangeListener The callback to fire when the value changes. */ _addOnTextSettingChangedListener(id, onChangeListener) { if (this.textParameters.has(id)) { this.textParameters.get(id).onChange = onChangeListener; } } /** * Get the option which has the given id. * @param id The id of the option. * @returns The SettingOption object matching id */ getSettingOption(id) { return this.optionParameters.get(id); } /** * Get the value of the configuration flag which has the given id. * @param id The unique id for the flag. * @returns True if the flag is enabled. */ isFlagEnabled(id) { return this.flags.get(id).flag; } /** * Set flag to be enabled/disabled. * @param id The id of the flag to toggle. * @param flagEnabled True if the flag should be enabled. */ setFlagEnabled(id, flagEnabled) { if (!this.flags.has(id)) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_4__.Logger.Warning(_Logger_Logger__WEBPACK_IMPORTED_MODULE_4__.Logger.GetStackTrace(), `Cannot toggle flag called ${id} - it does not exist in the Config.flags map.`); } else { this.flags.get(id).flag = flagEnabled; } } /** * Set the text setting. * @param id The id of the setting * @param settingValue The value to set in the setting. */ setTextSetting(id, settingValue) { if (!this.textParameters.has(id)) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_4__.Logger.Warning(_Logger_Logger__WEBPACK_IMPORTED_MODULE_4__.Logger.GetStackTrace(), `Cannot set text setting called ${id} - it does not exist in the Config.textParameters map.`); } else { this.textParameters.get(id).text = settingValue; } } /** * Set the option setting list of options. * @param id The id of the setting * @param settingOptions The values the setting could take */ setOptionSettingOptions(id, settingOptions) { if (!this.optionParameters.has(id)) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_4__.Logger.Warning(_Logger_Logger__WEBPACK_IMPORTED_MODULE_4__.Logger.GetStackTrace(), `Cannot set text setting called ${id} - it does not exist in the Config.optionParameters map.`); } else { this.optionParameters.get(id).options = settingOptions; } } /** * Set option enum settings selected option. * @param id The id of the setting * @param settingOptions The value to select out of all the options */ setOptionSettingValue(id, settingValue) { if (!this.optionParameters.has(id)) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_4__.Logger.Warning(_Logger_Logger__WEBPACK_IMPORTED_MODULE_4__.Logger.GetStackTrace(), `Cannot set text setting called ${id} - it does not exist in the Config.enumParameters map.`); } else { this.optionParameters.get(id).selected = settingValue; } } /** * Set the label for the flag. * @param id The id of the flag. * @param label The new label to use for the flag. */ setFlagLabel(id, label) { if (!this.flags.has(id)) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_4__.Logger.Warning(_Logger_Logger__WEBPACK_IMPORTED_MODULE_4__.Logger.GetStackTrace(), `Cannot set label for flag called ${id} - it does not exist in the Config.flags map.`); } else { this.flags.get(id).label = label; } } /** * Set a subset of all settings in one function call. * * @param settings A (partial) list of settings to set */ setSettings(settings) { for (const key of Object.keys(settings)) { if (isFlagId(key)) { this.setFlagEnabled(key, settings[key]); } else if (isNumericId(key)) { this.setNumericSetting(key, settings[key]); } else if (isTextId(key)) { this.setTextSetting(key, settings[key]); } else if (isOptionId(key)) { this.setOptionSettingValue(key, settings[key]); } } } /** * Get all settings * @returns All setting values as an object with setting ids as keys */ getSettings() { const settings = {}; for (const [key, value] of this.flags.entries()) { settings[key] = value.flag; } for (const [key, value] of this.numericParameters.entries()) { settings[key] = value.number; } for (const [key, value] of this.textParameters.entries()) { settings[key] = value.text; } for (const [key, value] of this.optionParameters.entries()) { settings[key] = value.selected; } return settings; } /** * Get all Flag settings as an array. * @returns All SettingFlag objects */ getFlags() { return Array.from(this.flags.values()); } /** * Get all Text settings as an array. * @returns All SettingText objects */ getTextSettings() { return Array.from(this.textParameters.values()); } /** * Get all Number settings as an array. * @returns All SettingNumber objects */ getNumericSettings() { return Array.from(this.numericParameters.values()); } /** * Get all Option settings as an array. * @returns All SettingOption objects */ getOptionSettings() { return Array.from(this.optionParameters.values()); } /** * Emit events when settings change. * @param eventEmitter */ _registerOnChangeEvents(eventEmitter) { for (const key of this.flags.keys()) { const flag = this.flags.get(key); if (flag) { flag.onChangeEmit = (newValue) => eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_5__.SettingsChangedEvent({ id: flag.id, type: 'flag', value: newValue, target: flag })); } } for (const key of this.numericParameters.keys()) { const number = this.numericParameters.get(key); if (number) { number.onChangeEmit = (newValue) => eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_5__.SettingsChangedEvent({ id: number.id, type: 'number', value: newValue, target: number })); } } for (const key of this.textParameters.keys()) { const text = this.textParameters.get(key); if (text) { text.onChangeEmit = (newValue) => eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_5__.SettingsChangedEvent({ id: text.id, type: 'text', value: newValue, target: text })); } } for (const key of this.optionParameters.keys()) { const option = this.optionParameters.get(key); if (option) { option.onChangeEmit = (newValue) => eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_5__.SettingsChangedEvent({ id: option.id, type: 'option', value: newValue, target: option })); } } } } /** * The enum associated with the mouse being locked or hovering */ var ControlSchemeType; (function (ControlSchemeType) { ControlSchemeType[ControlSchemeType["LockedMouse"] = 0] = "LockedMouse"; ControlSchemeType[ControlSchemeType["HoveringMouse"] = 1] = "HoveringMouse"; })(ControlSchemeType || (ControlSchemeType = {})); /***/ }), /***/ "./src/Config/SettingBase.ts": /*!***********************************!*\ !*** ./src/Config/SettingBase.ts ***! \***********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SettingBase": () => (/* binding */ SettingBase) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Base class for a setting that has a text label and an arbitrary setting value it stores. */ class SettingBase { constructor(id, label, description, defaultSettingValue, // eslint-disable-next-line @typescript-eslint/no-empty-function defaultOnChangeListener = () => { }) { this.onChange = defaultOnChangeListener; this.onChangeEmit = () => { /* Do nothing, to be overridden. */ }; this.id = id; this.description = description; this.label = label; this.value = defaultSettingValue; } /** * Set the label text for the setting. * @param label setting label. */ set label(inLabel) { this._label = inLabel; this.onChangeEmit(this._value); } /** * @returns The label text for the setting. */ get label() { return this._label; } /** * @return The setting's value. */ get value() { return this._value; } /** * Update the setting's stored value. * @param inValue The new value for the setting. */ set value(inValue) { this._value = inValue; this.onChange(this._value, this); this.onChangeEmit(this._value); } } /***/ }), /***/ "./src/Config/SettingFlag.ts": /*!***********************************!*\ !*** ./src/Config/SettingFlag.ts ***! \***********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SettingFlag": () => (/* binding */ SettingFlag) /* harmony export */ }); /* harmony import */ var _SettingBase__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./SettingBase */ "./src/Config/SettingBase.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * A boolean flag setting object with a text label. */ class SettingFlag extends _SettingBase__WEBPACK_IMPORTED_MODULE_0__.SettingBase { constructor(id, label, description, defaultFlagValue, useUrlParams, // eslint-disable-next-line @typescript-eslint/no-empty-function defaultOnChangeListener = () => { }) { super(id, label, description, defaultFlagValue, defaultOnChangeListener); const urlParams = new URLSearchParams(window.location.search); if (!useUrlParams || !urlParams.has(this.id)) { this.flag = defaultFlagValue; } else { // parse flag from url parameters const urlParamFlag = this.getUrlParamFlag(); this.flag = urlParamFlag; } this.useUrlParams = useUrlParams; } /** * Parse the flag value from the url parameters. * @returns True if the url parameters contains /?id, but False if /?id=false */ getUrlParamFlag() { const urlParams = new URLSearchParams(window.location.search); if (urlParams.has(this.id)) { if (urlParams.get(this.id) === 'false' || urlParams.get(this.id) === 'False') { return false; } return true; } return false; } /** * Persist the setting value in URL. */ updateURLParams() { if (this.useUrlParams) { // set url params const urlParams = new URLSearchParams(window.location.search); if (this.flag === true) { urlParams.set(this.id, 'true'); } else { urlParams.set(this.id, 'false'); } window.history.replaceState({}, '', urlParams.toString() !== '' ? `${location.pathname}?${urlParams}` : `${location.pathname}`); } } /** * Enables this flag. */ enable() { this.flag = true; } /** * @return The setting's value. */ get flag() { return !!this.value; } /** * Update the setting's stored value. * @param inValue The new value for the setting. */ set flag(inValue) { this.value = inValue; } } /***/ }), /***/ "./src/Config/SettingNumber.ts": /*!*************************************!*\ !*** ./src/Config/SettingNumber.ts ***! \*************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SettingNumber": () => (/* binding */ SettingNumber) /* harmony export */ }); /* harmony import */ var _SettingBase__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./SettingBase */ "./src/Config/SettingBase.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * A number setting object with a text label. Min and max limit the range of allowed values. */ class SettingNumber extends _SettingBase__WEBPACK_IMPORTED_MODULE_0__.SettingBase { constructor(id, label, description, min, max, defaultNumber, useUrlParams, // eslint-disable-next-line @typescript-eslint/no-empty-function defaultOnChangeListener = () => { }) { super(id, label, description, defaultNumber, defaultOnChangeListener); this._min = min; this._max = max; // attempt to read the number from the url params const urlParams = new URLSearchParams(window.location.search); if (!useUrlParams || !urlParams.has(this.id)) { this.number = defaultNumber; } else { const parsedValue = Number.parseInt(urlParams.get(this.id)); this.number = Number.isNaN(parsedValue) ? defaultNumber : parsedValue; } this.useUrlParams = useUrlParams; } /** * Persist the setting value in URL. */ updateURLParams() { if (this.useUrlParams) { // set url params like ?id=number const urlParams = new URLSearchParams(window.location.search); urlParams.set(this.id, this.number.toString()); window.history.replaceState({}, '', urlParams.toString() !== '' ? `${location.pathname}?${urlParams}` : `${location.pathname}`); } } /** * Set the number value (will be clamped within range). */ set number(newNumber) { this.value = this.clamp(newNumber); } /** * @returns The number stored. */ get number() { return this.value; } /** * Clamps a number between the min and max values (inclusive). * @param inNumber The number to clamp. * @returns The clamped number. */ clamp(inNumber) { return Math.max(Math.min(this._max, inNumber), this._min); } /** * Returns the minimum value * @returns The minimum value */ get min() { return this._min; } /** * Returns the maximum value * @returns The maximum value */ get max() { return this._max; } /** * Add a change listener to the number object. */ addOnChangedListener(onChangedFunc) { this.onChange = onChangedFunc; } } /***/ }), /***/ "./src/Config/SettingOption.ts": /*!*************************************!*\ !*** ./src/Config/SettingOption.ts ***! \*************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SettingOption": () => (/* binding */ SettingOption) /* harmony export */ }); /* harmony import */ var _SettingBase__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./SettingBase */ "./src/Config/SettingBase.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * An Option setting object with a text label. Allows you to specify an array of options and select one of them. */ class SettingOption extends _SettingBase__WEBPACK_IMPORTED_MODULE_0__.SettingBase { constructor(id, label, description, defaultTextValue, options, useUrlParams, // eslint-disable-next-line @typescript-eslint/no-empty-function defaultOnChangeListener = () => { }) { super(id, label, description, [defaultTextValue, defaultTextValue], defaultOnChangeListener); this.options = options; const urlParams = new URLSearchParams(window.location.search); const stringToMatch = useUrlParams && urlParams.has(this.id) ? this.getUrlParamText() : defaultTextValue; this.selected = stringToMatch; this.useUrlParams = useUrlParams; } /** * Parse the text value from the url parameters. * @returns The text value parsed from the url if the url parameters contains /?id=value, but empty string if just /?id or no url param found. */ getUrlParamText() { var _a; const urlParams = new URLSearchParams(window.location.search); if (urlParams.has(this.id)) { return (_a = urlParams.get(this.id)) !== null && _a !== void 0 ? _a : ''; } return ''; } /** * Persist the setting value in URL. */ updateURLParams() { if (this.useUrlParams) { // set url params const urlParams = new URLSearchParams(window.location.search); urlParams.set(this.id, this.selected); window.history.replaceState({}, '', urlParams.toString() !== '' ? `${location.pathname}?${urlParams}` : `${location.pathname}`); } } /** * Add a change listener to the select element. */ addOnChangedListener(onChangedFunc) { this.onChange = onChangedFunc; } /** * @returns All available options as an array */ get options() { return this._options; } /** * Set options * @param values Array of options */ set options(values) { this._options = values; this.onChangeEmit(this.selected); } /** * @returns Selected option as a string */ get selected() { return this.value; } /** * Set selected option if it matches one of the available options * @param value Selected option */ set selected(value) { // A user may not specify the full possible value so we instead use the closest match. // eg ?xxx=H264 would select 'H264 level-asymmetry-allowed=1;packetization-mode=1;profile-level-id=42001f' let filteredList = this.options.filter((option) => option.indexOf(value) !== -1); if (filteredList.length) { this.value = filteredList[0]; return; } // A user has specified a codec with a fmtp string but this codec + fmtp line isn't available. // in that case, just use the codec filteredList = this.options.filter((option) => option.indexOf(value.split(' ')[0]) !== -1); if (filteredList.length) { this.value = filteredList[0]; return; } } } /***/ }), /***/ "./src/Config/SettingText.ts": /*!***********************************!*\ !*** ./src/Config/SettingText.ts ***! \***********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SettingText": () => (/* binding */ SettingText) /* harmony export */ }); /* harmony import */ var _SettingBase__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./SettingBase */ "./src/Config/SettingBase.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * A text setting object with a text label. */ class SettingText extends _SettingBase__WEBPACK_IMPORTED_MODULE_0__.SettingBase { constructor(id, label, description, defaultTextValue, useUrlParams, // eslint-disable-next-line @typescript-eslint/no-empty-function defaultOnChangeListener = () => { }) { super(id, label, description, defaultTextValue, defaultOnChangeListener); const urlParams = new URLSearchParams(window.location.search); if (!useUrlParams || !urlParams.has(this.id)) { this.text = defaultTextValue; } else { // parse flag from url parameters const urlParamFlag = this.getUrlParamText(); this.text = urlParamFlag; } this.useUrlParams = useUrlParams; } /** * Parse the text value from the url parameters. * @returns The text value parsed from the url if the url parameters contains /?id=value, but empty string if just /?id or no url param found. */ getUrlParamText() { var _a; const urlParams = new URLSearchParams(window.location.search); if (urlParams.has(this.id)) { return (_a = urlParams.get(this.id)) !== null && _a !== void 0 ? _a : ''; } return ''; } /** * Persist the setting value in URL. */ updateURLParams() { if (this.useUrlParams) { // set url params const urlParams = new URLSearchParams(window.location.search); urlParams.set(this.id, this.text); window.history.replaceState({}, '', urlParams.toString() !== '' ? `${location.pathname}?${urlParams}` : `${location.pathname}`); } } /** * @return The setting's value. */ get text() { return this.value; } /** * Update the setting's stored value. * @param inValue The new value for the setting. */ set text(inValue) { this.value = inValue; } } /***/ }), /***/ "./src/DataChannel/DataChannelController.ts": /*!**************************************************!*\ !*** ./src/DataChannel/DataChannelController.ts ***! \**************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "DataChannelController": () => (/* binding */ DataChannelController) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Handles the Sending and Receiving of messages to the UE Instance via the Data Channel */ class DataChannelController { constructor() { this.isReceivingFreezeFrame = false; } /** * return the current state of a datachannel controller instance * @returns the current DataChannelController instance */ getDataChannelInstance() { return this; } /** * To Create and Set up a Data Channel * @param peerConnection - The RTC Peer Connection * @param label - Label of the Data Channel * @param datachannelOptions - Optional RTC DataChannel options */ createDataChannel(peerConnection, label, datachannelOptions) { this.peerConnection = peerConnection; this.label = label; this.datachannelOptions = datachannelOptions; if (datachannelOptions == null) { this.datachannelOptions = {}; this.datachannelOptions.ordered = true; } this.dataChannel = this.peerConnection.createDataChannel(this.label, this.datachannelOptions); this.setupDataChannel(); } setupDataChannel() { //We Want an Array Buffer not a blob this.dataChannel.binaryType = 'arraybuffer'; this.dataChannel.onopen = (ev) => this.handleOnOpen(ev); this.dataChannel.onclose = (ev) => this.handleOnClose(ev); this.dataChannel.onmessage = (ev) => this.handleOnMessage(ev); this.dataChannel.onerror = (ev) => this.handleOnError(ev); } /** * Handles when the Data Channel is opened */ handleOnOpen(ev) { var _a; _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Data Channel (${this.label}) opened.`, 7); this.onOpen((_a = this.dataChannel) === null || _a === void 0 ? void 0 : _a.label, ev); } /** * Handles when the Data Channel is closed */ handleOnClose(ev) { var _a; _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Data Channel (${this.label}) closed.`, 7); this.onClose((_a = this.dataChannel) === null || _a === void 0 ? void 0 : _a.label, ev); } /** * Handles when a message is received * @param event - Message Event */ handleOnMessage(event) { // Higher log level to prevent log spam with messages received _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Data Channel (${this.label}) message: ${event}`, 8); } /** * Handles when an error is thrown * @param event - Error Event */ handleOnError(event) { var _a; _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Data Channel (${this.label}) error: ${event}`, 7); this.onError((_a = this.dataChannel) === null || _a === void 0 ? void 0 : _a.label, event); } /** * Override to register onOpen handler * @param label Data channel label ("datachannel", "send-datachannel", "recv-datachannel") * @param ev event */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onOpen(label, ev) { // empty default implementation } /** * Override to register onClose handler * @param label Data channel label ("datachannel", "send-datachannel", "recv-datachannel") * @param ev event */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onClose(label, ev) { // empty default implementation } /** * Override to register onError handler * @param label Data channel label ("datachannel", "send-datachannel", "recv-datachannel") * @param ev event */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onError(label, ev) { // empty default implementation } } /***/ }), /***/ "./src/DataChannel/DataChannelLatencyTestController.ts": /*!*************************************************************!*\ !*** ./src/DataChannel/DataChannelLatencyTestController.ts ***! \*************************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "DataChannelLatencyTestController": () => (/* binding */ DataChannelLatencyTestController) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _DataChannelLatencyTestResults__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./DataChannelLatencyTestResults */ "./src/DataChannel/DataChannelLatencyTestResults.ts"); // Copyright Epic Games, Inc. All Rights Reserved. class DataChannelLatencyTestController { constructor(sink, callback) { this.sink = sink; this.callback = callback; this.records = new Map(); this.seq = 0; } start(config) { if (this.isRunning()) { return false; } this.startTime = Date.now(); this.records.clear(); this.interval = setInterval((() => { if (Date.now() - this.startTime >= config.duration) { this.stop(); } else { this.sendRequest(config.requestSize, config.responseSize); } }).bind(this), Math.floor(1000 / config.rps)); return true; } stop() { if (this.interval) { clearInterval(this.interval); this.interval = undefined; this.callback(this.produceResult()); } } produceResult() { const resultRecords = new Map(this.records); return { records: resultRecords, dataChannelRtt: Math.ceil(Array.from(this.records.values()).reduce((acc, next) => { return acc + (next.playerReceivedTimestamp - next.playerSentTimestamp); }, 0) / this.records.size), playerToStreamerTime: Math.ceil(Array.from(this.records.values()).reduce((acc, next) => { return acc + (next.streamerReceivedTimestamp - next.playerSentTimestamp); }, 0) / this.records.size), streamerToPlayerTime: Math.ceil(Array.from(this.records.values()).reduce((acc, next) => { return acc + (next.playerReceivedTimestamp - next.streamerSentTimestamp); }, 0) / this.records.size), exportLatencyAsCSV: () => { let csv = "Timestamp;RTT;PlayerToStreamer;StreamerToPlayer;\n"; resultRecords.forEach((record) => { csv += record.playerSentTimestamp + ";"; csv += (record.playerReceivedTimestamp - record.playerSentTimestamp) + ";"; csv += (record.streamerReceivedTimestamp - record.playerSentTimestamp) + ";"; csv += (record.playerReceivedTimestamp - record.streamerSentTimestamp) + ";"; csv += "\n"; }); return csv; } }; } isRunning() { return !!this.interval; } receive(response) { if (!this.isRunning()) { return; } if (!response) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), "Undefined response from server"); return; } let record = this.records.get(response.Seq); if (record) { record.update(response); } } sendRequest(requestSize, responseSize) { let request = this.createRequest(requestSize, responseSize); let record = new _DataChannelLatencyTestResults__WEBPACK_IMPORTED_MODULE_1__.DataChannelLatencyTestRecord(request); this.records.set(record.seq, record); this.sink(request); } createRequest(requestSize, responseSize) { return { Seq: this.seq++, FillResponseSize: responseSize, Filler: requestSize ? "A".repeat(requestSize) : "" }; } } /***/ }), /***/ "./src/DataChannel/DataChannelLatencyTestResults.ts": /*!**********************************************************!*\ !*** ./src/DataChannel/DataChannelLatencyTestResults.ts ***! \**********************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "DataChannelLatencyTestRecord": () => (/* binding */ DataChannelLatencyTestRecord) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. class DataChannelLatencyTestRecord { constructor(request) { this.seq = request.Seq; this.playerSentTimestamp = Date.now(); this.requestFillerSize = request.Filler ? request.Filler.length : 0; } update(response) { this.playerReceivedTimestamp = Date.now(); this.streamerReceivedTimestamp = response.ReceivedTimestamp; this.streamerSentTimestamp = response.SentTimestamp; this.responseFillerSize = response.Filler ? response.Filler.length : 0; } } /***/ }), /***/ "./src/DataChannel/DataChannelSender.ts": /*!**********************************************!*\ !*** ./src/DataChannel/DataChannelSender.ts ***! \**********************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "DataChannelSender": () => (/* binding */ DataChannelSender) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * A class for sending data channel messages */ class DataChannelSender { /** * @param dataChannelProvider - Data channel object type */ constructor(dataChannelProvider) { this.dataChannelProvider = dataChannelProvider; } canSend() { return (this.dataChannelProvider.getDataChannelInstance().dataChannel !== undefined && this.dataChannelProvider.getDataChannelInstance().dataChannel .readyState == 'open'); } /** * Send Data over the Data channel to the UE Instance * @param data - Message Data Array Buffer */ sendData(data) { // reset the afk inactivity const dataChannelInstance = this.dataChannelProvider.getDataChannelInstance(); if (dataChannelInstance.dataChannel.readyState == 'open') { dataChannelInstance.dataChannel.send(data); _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Message Sent: ${new Uint8Array(data)}`, 6); this.resetAfkWarningTimerOnDataSend(); } else { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Message Failed: ${new Uint8Array(data)}`); } } /** * An override method for resetting the Afk warning timer when data is sent over the data channel */ resetAfkWarningTimerOnDataSend() { // Base Functionality: Do Nothing } } /***/ }), /***/ "./src/DataChannel/InitialSettings.ts": /*!********************************************!*\ !*** ./src/DataChannel/InitialSettings.ts ***! \********************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "EncoderSettings": () => (/* binding */ EncoderSettings), /* harmony export */ "InitialSettings": () => (/* binding */ InitialSettings), /* harmony export */ "PixelStreamingSettings": () => (/* binding */ PixelStreamingSettings), /* harmony export */ "WebRTCSettings": () => (/* binding */ WebRTCSettings) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Latency Test Results Data */ class InitialSettings { constructor() { this.PixelStreamingSettings = new PixelStreamingSettings(); this.EncoderSettings = new EncoderSettings(); this.WebRTCSettings = new WebRTCSettings(); } /** * Checks for compatibility with the FPS and MaxFPS stats between 4.27 and 5 */ ueCompatible() { if (this.WebRTCSettings.MaxFPS != null) { this.WebRTCSettings.FPS = this.WebRTCSettings.MaxFPS; } } } /** * A class for handling Pixel Streaming details */ class PixelStreamingSettings { } /** * A class for handling encoder stats */ class EncoderSettings { } /** * A class for handling web rtc stats */ class WebRTCSettings { } /***/ }), /***/ "./src/DataChannel/LatencyTestResults.ts": /*!***********************************************!*\ !*** ./src/DataChannel/LatencyTestResults.ts ***! \***********************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "LatencyTestResults": () => (/* binding */ LatencyTestResults) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Latency Test Results Data */ class LatencyTestResults { constructor() { //Fields Set from the latency payload regardless of version this.ReceiptTimeMs = null; this.TransmissionTimeMs = null; //Fields Set from the latency payload from 4.27.2 this.PreCaptureTimeMs = null; this.PostCaptureTimeMs = null; this.PreEncodeTimeMs = null; this.PostEncodeTimeMs = null; //Fields Set from the latency payload from 5.0 this.EncodeMs = null; this.CaptureToSendMs = null; //Fields Set when processed this.testStartTimeMs = 0; this.browserReceiptTimeMs = 0; //Fields set from calculations this.latencyExcludingDecode = 0; this.testDuration = 0; //ueLatency: number = 0; this.networkLatency = 0; this.browserSendLatency = 0; this.frameDisplayDeltaTimeMs = 0; this.endToEndLatency = 0; //uePixelStreamLatency: number = 0; this.encodeLatency = 0; } /** * Sets the Delta Time Milliseconds * @param DeltaTimeMs - Delta Time Milliseconds */ setFrameDisplayDeltaTime(DeltaTimeMs) { if (this.frameDisplayDeltaTimeMs == 0) { this.frameDisplayDeltaTimeMs = Math.round(DeltaTimeMs); } } /** * Process the encoder times and set them */ processFields() { if (this.EncodeMs == null && (this.PreEncodeTimeMs != null || this.PostEncodeTimeMs != null)) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Setting Encode Ms \n ${this.PostEncodeTimeMs} \n ${this.PreEncodeTimeMs}`, 6); this.EncodeMs = this.PostEncodeTimeMs - this.PreEncodeTimeMs; } if (this.CaptureToSendMs == null && (this.PreCaptureTimeMs != null || this.PostCaptureTimeMs != null)) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Setting CaptureToSendMs Ms \n ${this.PostCaptureTimeMs} \n ${this.PreCaptureTimeMs}`, 6); this.CaptureToSendMs = this.PostCaptureTimeMs - this.PreCaptureTimeMs; } } } /***/ }), /***/ "./src/FreezeFrame/FreezeFrame.ts": /*!****************************************!*\ !*** ./src/FreezeFrame/FreezeFrame.ts ***! \****************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "FreezeFrame": () => (/* binding */ FreezeFrame) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * A class for managing the freeze frame object */ class FreezeFrame { /** * Construct a freeze frame * @param rootDiv the div that a freeze frame element will be injected into */ constructor(rootDiv) { this.freezeFrameHeight = 0; this.freezeFrameWidth = 0; this.rootDiv = rootDiv; // create the overlay 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'; // create the image place holder this.imageElement = document.createElement('img'); this.imageElement.style.position = 'absolute'; // append the image into the root element and append the element to the root div this.rootElement.appendChild(this.imageElement); this.rootDiv.appendChild(this.rootElement); } /** * Set the freeze frame element for showing */ setElementForShow() { this.rootElement.style.display = 'block'; } /** * Set the freeze frame element for hiding */ setElementForHide() { this.rootElement.style.display = 'none'; } /** * Update the freeze frames image source * @param jpeg - the freeze frame image as a byte array data */ updateImageElementSource(jpeg) { const base64 = btoa(jpeg.reduce((data, byte) => data + String.fromCharCode(byte), '')); this.imageElement.src = 'data:image/jpeg;base64,' + base64; } /** * Set the dimensions for the freeze frame from the element and resize it */ setDimensionsFromElementAndResize() { this.freezeFrameHeight = this.imageElement.naturalHeight; this.freezeFrameWidth = this.imageElement.naturalWidth; this.resize(); } /** * Resize a freeze frame element */ resize() { if (this.freezeFrameWidth !== 0 && this.freezeFrameHeight !== 0) { let displayWidth = 0; let displayHeight = 0; let displayTop = 0; let displayLeft = 0; const parentAspectRatio = this.rootDiv.clientWidth / this.rootDiv.clientHeight; const videoAspectRatio = this.freezeFrameWidth / this.freezeFrameHeight; if (parentAspectRatio < videoAspectRatio) { displayWidth = this.rootDiv.clientWidth; displayHeight = Math.floor(this.rootDiv.clientWidth / videoAspectRatio); displayTop = Math.floor((this.rootDiv.clientHeight - displayHeight) * 0.5); displayLeft = 0; } else { displayWidth = Math.floor(this.rootDiv.clientHeight * videoAspectRatio); displayHeight = this.rootDiv.clientHeight; displayTop = 0; displayLeft = Math.floor((this.rootDiv.clientWidth - displayWidth) * 0.5); } this.rootElement.style.width = this.rootDiv.offsetWidth + 'px'; this.rootElement.style.height = this.rootDiv.offsetHeight + 'px'; this.rootElement.style.left = 0 + 'px'; this.rootElement.style.top = 0 + 'px'; this.imageElement.style.width = displayWidth + 'px'; this.imageElement.style.height = displayHeight + 'px'; this.imageElement.style.left = displayLeft + 'px'; this.imageElement.style.top = displayTop + 'px'; } } } /***/ }), /***/ "./src/FreezeFrame/FreezeFrameController.ts": /*!**************************************************!*\ !*** ./src/FreezeFrame/FreezeFrameController.ts ***! \**************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "FreezeFrameController": () => (/* binding */ FreezeFrameController) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _FreezeFrame__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./FreezeFrame */ "./src/FreezeFrame/FreezeFrame.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * A class for controlling freeze frame functionality */ class FreezeFrameController { /** * Construct a freeze frame controller * @param rootDiv - the div that a freeze frame element will be injected into */ constructor(rootDiv) { this.receiving = false; this.size = 0; this.jpeg = undefined; this.valid = false; this.freezeFrameDelay = 50; this.freezeFrame = new _FreezeFrame__WEBPACK_IMPORTED_MODULE_0__.FreezeFrame(rootDiv); } /** * Show the freeze frame if it is valid */ showFreezeFrame() { if (this.valid) { this.freezeFrame.setElementForShow(); } } /** * Hide the freeze frame and set the validity to false */ hideFreezeFrame() { this.valid = false; this.freezeFrame.setElementForHide(); } /** * Update the freeze frames image source and load it * @param jpeg - the freeze frame image as a byte array data * @param onLoadCallBack - a call back for managing if the play overlay needs to be shown or not */ updateFreezeFrameAndShow(jpeg, onLoadCallBack) { this.freezeFrame.updateImageElementSource(jpeg); this.freezeFrame.imageElement.onload = () => { this.freezeFrame.setDimensionsFromElementAndResize(); onLoadCallBack(); }; } /** * Process the new freeze frame image and update it * @param view - the freeze frame image as a byte array data * @param onLoadCallBack - a call back for managing if the play overlay needs to be shown or not */ processFreezeFrameMessage(view, onLoadCallBack) { // Reset freeze frame if we got a freeze frame message and we are not "receiving" yet. if (!this.receiving) { this.receiving = true; this.valid = false; this.size = 0; this.jpeg = undefined; } // Extract total size of freeze frame (across all chunks) this.size = new DataView(view.slice(1, 5).buffer).getInt32(0, true); // Get the jpeg part of the payload const jpegBytes = view.slice(1 + 4); // Append to existing jpeg that holds the freeze frame if (this.jpeg) { const jpeg = new Uint8Array(this.jpeg.length + jpegBytes.length); jpeg.set(this.jpeg, 0); jpeg.set(jpegBytes, this.jpeg.length); this.jpeg = jpeg; } // No existing freeze frame jpeg, make one else { this.jpeg = jpegBytes; this.receiving = true; _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), `received first chunk of freeze frame: ${this.jpeg.length}/${this.size}`, 6); } // Finished receiving freeze frame, we can show it now if (this.jpeg.length === this.size) { this.receiving = false; this.valid = true; _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), `received complete freeze frame ${this.size}`, 6); this.updateFreezeFrameAndShow(this.jpeg, onLoadCallBack); } // We received more data than the freeze frame payload message indicate (this is an error) else if (this.jpeg.length > this.size) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), `received bigger freeze frame than advertised: ${this.jpeg.length}/${this.size}`); this.jpeg = undefined; this.receiving = false; } } } /***/ }), /***/ "./src/Inputs/FakeTouchController.ts": /*!*******************************************!*\ !*** ./src/Inputs/FakeTouchController.ts ***! \*******************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "FakeTouchController": () => (/* binding */ FakeTouchController), /* harmony export */ "FakeTouchFinger": () => (/* binding */ FakeTouchFinger) /* harmony export */ }); /* harmony import */ var _MouseButtons__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./MouseButtons */ "./src/Inputs/MouseButtons.ts"); /* harmony import */ var _Util_EventListenerTracker__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Util/EventListenerTracker */ "./src/Util/EventListenerTracker.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Allows for the usage of fake touch events and implements ITouchController * @param dataChannelController - The controller for the Data channel * @param videoElementParent - The video player DOM element */ class FakeTouchController { /** * @param toStreamerMessagesProvider - Stream message instance * @param videoElementProvider - Video element instance * @param coordinateConverter - A coordinate converter instance */ constructor(toStreamerMessagesProvider, videoElementProvider, coordinateConverter) { // Utility for keeping track of event handlers and unregistering them this.touchEventListenerTracker = new _Util_EventListenerTracker__WEBPACK_IMPORTED_MODULE_0__.EventListenerTracker(); this.toStreamerMessagesProvider = toStreamerMessagesProvider; this.videoElementProvider = videoElementProvider; this.coordinateConverter = coordinateConverter; const ontouchstart = (ev) => this.onTouchStart(ev); const ontouchend = (ev) => this.onTouchEnd(ev); const ontouchmove = (ev) => this.onTouchMove(ev); document.addEventListener('touchstart', ontouchstart, { passive: false }); document.addEventListener('touchend', ontouchend, { passive: false }); document.addEventListener('touchmove', ontouchmove, { passive: false }); this.touchEventListenerTracker.addUnregisterCallback(() => document.removeEventListener('touchstart', ontouchstart)); this.touchEventListenerTracker.addUnregisterCallback(() => document.removeEventListener('touchend', ontouchend)); this.touchEventListenerTracker.addUnregisterCallback(() => document.removeEventListener('touchmove', ontouchmove)); } /** * Unregister all touch events */ unregisterTouchEvents() { this.touchEventListenerTracker.unregisterAll(); } /** * Sets the video Element Parent Client Rect numbers for this class * @param videoElementParentClientRect - a html ElementParentClientRect object */ setVideoElementParentClientRect(videoElementParentClientRect) { this.videoElementParentClientRect = videoElementParentClientRect; } /** * When a touch event begins * @param touch - the activating touch event */ onTouchStart(touch) { if (!this.videoElementProvider.isVideoReady()) { return; } if (this.fakeTouchFinger == null) { const first_touch = touch.changedTouches[0]; this.fakeTouchFinger = new FakeTouchFinger(first_touch.identifier, first_touch.clientX - this.videoElementParentClientRect.left, first_touch.clientY - this.videoElementParentClientRect.top); const videoElementParent = this.videoElementProvider.getVideoParentElement(); const mouseEvent = new MouseEvent('mouseenter', first_touch); videoElementParent.dispatchEvent(mouseEvent); const coord = this.coordinateConverter.normalizeAndQuantizeUnsigned(this.fakeTouchFinger.x, this.fakeTouchFinger.y); const toStreamerHandlers = this.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseDown')([ _MouseButtons__WEBPACK_IMPORTED_MODULE_1__.MouseButton.mainButton, coord.x, coord.y ]); } touch.preventDefault(); } /** * When a touch event ends * @param touchEvent - the activating touch event */ onTouchEnd(touchEvent) { if (!this.videoElementProvider.isVideoReady()) { return; } const videoElementParent = this.videoElementProvider.getVideoParentElement(); const toStreamerHandlers = this.toStreamerMessagesProvider.toStreamerHandlers; for (let t = 0; t < touchEvent.changedTouches.length; t++) { const touch = touchEvent.changedTouches[t]; if (touch.identifier === this.fakeTouchFinger.id) { const x = touch.clientX - this.videoElementParentClientRect.left; const y = touch.clientY - this.videoElementParentClientRect.top; const coord = this.coordinateConverter.normalizeAndQuantizeUnsigned(x, y); toStreamerHandlers.get('MouseUp')([ _MouseButtons__WEBPACK_IMPORTED_MODULE_1__.MouseButton.mainButton, coord.x, coord.y ]); const mouseEvent = new MouseEvent('mouseleave', touch); videoElementParent.dispatchEvent(mouseEvent); this.fakeTouchFinger = null; break; } } touchEvent.preventDefault(); } /** * On a Move touch event * @param touchEvent - the activating touch event */ onTouchMove(touchEvent) { if (!this.videoElementProvider.isVideoReady()) { return; } const toStreamerHandlers = this.toStreamerMessagesProvider.toStreamerHandlers; for (let t = 0; t < touchEvent.touches.length; t++) { const touch = touchEvent.touches[t]; if (touch.identifier === this.fakeTouchFinger.id) { const x = touch.clientX - this.videoElementParentClientRect.left; const y = touch.clientY - this.videoElementParentClientRect.top; const coord = this.coordinateConverter.normalizeAndQuantizeUnsigned(x, y); const delta = this.coordinateConverter.normalizeAndQuantizeSigned(x - this.fakeTouchFinger.x, y - this.fakeTouchFinger.y); toStreamerHandlers.get('MouseMove')([ coord.x, coord.y, delta.x, delta.y ]); this.fakeTouchFinger.x = x; this.fakeTouchFinger.y = y; break; } } touchEvent.preventDefault(); } } /** * The interface for finger position mapping */ class FakeTouchFinger { /** * @param id - the button id * @param x - the x axis value * @param y - the y axis value */ constructor(id, x, y) { this.id = id; this.x = x; this.y = y; } } /***/ }), /***/ "./src/Inputs/GamepadController.ts": /*!*****************************************!*\ !*** ./src/Inputs/GamepadController.ts ***! \*****************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "GamePadController": () => (/* binding */ GamePadController), /* harmony export */ "gamepadLayout": () => (/* binding */ gamepadLayout) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _Util_EventListenerTracker__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Util/EventListenerTracker */ "./src/Util/EventListenerTracker.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * The class that handles the functionality of gamepads and controllers */ class GamePadController { /** * @param toStreamerMessagesProvider - Stream message instance */ constructor(toStreamerMessagesProvider) { // Utility for keeping track of event handlers and unregistering them this.gamePadEventListenerTracker = new _Util_EventListenerTracker__WEBPACK_IMPORTED_MODULE_0__.EventListenerTracker(); this.toStreamerMessagesProvider = toStreamerMessagesProvider; this.requestAnimationFrame = (window.mozRequestAnimationFrame || window.webkitRequestAnimationFrame || window.requestAnimationFrame).bind(window); const browserWindow = window; if ('GamepadEvent' in browserWindow) { const onGamePadConnected = (ev) => this.gamePadConnectHandler(ev); const onGamePadDisconnected = (ev) => this.gamePadDisconnectHandler(ev); window.addEventListener('gamepadconnected', onGamePadConnected); window.addEventListener('gamepaddisconnected', onGamePadDisconnected); this.gamePadEventListenerTracker.addUnregisterCallback(() => window.removeEventListener('gamepadconnected', onGamePadConnected)); this.gamePadEventListenerTracker.addUnregisterCallback(() => window.removeEventListener('gamepaddisconnected', onGamePadDisconnected)); } else if ('WebKitGamepadEvent' in browserWindow) { const onWebkitGamePadConnected = (ev) => this.gamePadConnectHandler(ev); const onWebkitGamePadDisconnected = (ev) => this.gamePadDisconnectHandler(ev); window.addEventListener('webkitgamepadconnected', onWebkitGamePadConnected); window.addEventListener('webkitgamepaddisconnected', onWebkitGamePadDisconnected); this.gamePadEventListenerTracker.addUnregisterCallback(() => window.removeEventListener('webkitgamepadconnected', onWebkitGamePadConnected)); this.gamePadEventListenerTracker.addUnregisterCallback(() => window.removeEventListener('webkitgamepaddisconnected', onWebkitGamePadDisconnected)); } this.controllers = []; if (navigator.getGamepads) { for (const gamepad of navigator.getGamepads()) { if (gamepad) { this.gamePadConnectHandler(new GamepadEvent('gamepadconnected', { gamepad })); } } } } /** * Unregisters all event handlers */ unregisterGamePadEvents() { this.gamePadEventListenerTracker.unregisterAll(); for (const controller of this.controllers) { if (controller.id !== undefined) { this.onGamepadDisconnected(controller.id); } } this.controllers = []; this.onGamepadConnected = () => { }; this.onGamepadDisconnected = () => { }; } /** * Connects the gamepad handler * @param gamePadEvent - the activating gamepad event */ gamePadConnectHandler(gamePadEvent) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'Gamepad connect handler', 6); const gamepad = gamePadEvent.gamepad; const temp = { currentState: gamepad, prevState: gamepad, id: undefined }; this.controllers.push(temp); this.controllers[gamepad.index].currentState = gamepad; this.controllers[gamepad.index].prevState = gamepad; _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'gamepad: ' + gamepad.id + ' connected', 6); window.requestAnimationFrame(() => this.updateStatus()); this.onGamepadConnected(); } /** * Disconnects the gamepad handler * @param gamePadEvent - the activating gamepad event */ gamePadDisconnectHandler(gamePadEvent) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'Gamepad disconnect handler', 6); _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'gamepad: ' + gamePadEvent.gamepad.id + ' disconnected', 6); const deletedController = this.controllers[gamePadEvent.gamepad.index]; delete this.controllers[gamePadEvent.gamepad.index]; this.controllers = this.controllers.filter((controller) => controller !== undefined); this.onGamepadDisconnected(deletedController.id); } /** * Scan for connected gamepads */ scanGamePads() { const gamepads = navigator.getGamepads ? navigator.getGamepads() : navigator.webkitGetGamepads ? navigator.webkitGetGamepads() : []; for (let i = 0; i < gamepads.length; i++) { if (gamepads[i] && gamepads[i].index in this.controllers) { this.controllers[gamepads[i].index].currentState = gamepads[i]; } } } /** * Updates the status of the gamepad and sends the inputs */ updateStatus() { this.scanGamePads(); const toStreamerHandlers = this.toStreamerMessagesProvider.toStreamerHandlers; // Iterate over multiple controllers in the case the multiple gamepads are connected for (const controller of this.controllers) { // If we haven't received an id (possible if using an older version of UE), return to original functionality const controllerIndex = (controller.id === undefined) ? this.controllers.indexOf(controller) : controller.id; const currentState = controller.currentState; for (let i = 0; i < controller.currentState.buttons.length; i++) { const currentButton = controller.currentState.buttons[i]; const previousButton = controller.prevState.buttons[i]; if (currentButton.pressed) { // press if (i == gamepadLayout.LeftTrigger) { // UEs left analog has a button index of 5 toStreamerHandlers.get('GamepadAnalog')([ controllerIndex, 5, currentButton.value ]); } else if (i == gamepadLayout.RightTrigger) { // UEs right analog has a button index of 6 toStreamerHandlers.get('GamepadAnalog')([ controllerIndex, 6, currentButton.value ]); } else { toStreamerHandlers.get('GamepadButtonPressed')([ controllerIndex, i, previousButton.pressed ? 1 : 0 ]); } } else if (!currentButton.pressed && previousButton.pressed) { // release if (i == gamepadLayout.LeftTrigger) { // UEs left analog has a button index of 5 toStreamerHandlers.get('GamepadAnalog')([ controllerIndex, 5, 0 ]); } else if (i == gamepadLayout.RightTrigger) { // UEs right analog has a button index of 6 toStreamerHandlers.get('GamepadAnalog')([ controllerIndex, 6, 0 ]); } else { toStreamerHandlers.get('GamepadButtonReleased')([ controllerIndex, i ]); } } } // Iterate over gamepad axes (we will increment in lots of 2 as there is 2 axes per stick) for (let i = 0; i < currentState.axes.length; i += 2) { // Horizontal axes are even numbered const x = parseFloat(currentState.axes[i].toFixed(4)); // Vertical axes are odd numbered // https://w3c.github.io/gamepad/#remapping Gamepad browser side standard mapping has positive down, negative up. This is downright disgusting. So we fix it. const y = -parseFloat(currentState.axes[i + 1].toFixed(4)); // UE's analog axes follow the same order as the browsers, but start at index 1 so we will offset as such toStreamerHandlers.get('GamepadAnalog')([ controllerIndex, i + 1, x ]); // Horizontal axes, only offset by 1 toStreamerHandlers.get('GamepadAnalog')([ controllerIndex, i + 2, y ]); // Vertical axes, offset by two (1 to match UEs axes convention and then another 1 for the vertical axes) } this.controllers[controllerIndex].prevState = currentState; } if (this.controllers.length > 0) { this.requestAnimationFrame(() => this.updateStatus()); } } onGamepadResponseReceived(gamepadId) { for (const controller of this.controllers) { if (controller.id === undefined) { controller.id = gamepadId; break; } } } /** * Event to send the gamepadconnected message to the application */ onGamepadConnected() { // Default Functionality: Do Nothing } /** * Event to send the gamepaddisconnected message to the application */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onGamepadDisconnected(controllerIdx) { // Default Functionality: Do Nothing } } /** * Gamepad layout codes enum */ var gamepadLayout; (function (gamepadLayout) { gamepadLayout[gamepadLayout["RightClusterBottomButton"] = 0] = "RightClusterBottomButton"; gamepadLayout[gamepadLayout["RightClusterRightButton"] = 1] = "RightClusterRightButton"; gamepadLayout[gamepadLayout["RightClusterLeftButton"] = 2] = "RightClusterLeftButton"; gamepadLayout[gamepadLayout["RightClusterTopButton"] = 3] = "RightClusterTopButton"; gamepadLayout[gamepadLayout["LeftShoulder"] = 4] = "LeftShoulder"; gamepadLayout[gamepadLayout["RightShoulder"] = 5] = "RightShoulder"; gamepadLayout[gamepadLayout["LeftTrigger"] = 6] = "LeftTrigger"; gamepadLayout[gamepadLayout["RightTrigger"] = 7] = "RightTrigger"; gamepadLayout[gamepadLayout["SelectOrBack"] = 8] = "SelectOrBack"; gamepadLayout[gamepadLayout["StartOrForward"] = 9] = "StartOrForward"; gamepadLayout[gamepadLayout["LeftAnalogPress"] = 10] = "LeftAnalogPress"; gamepadLayout[gamepadLayout["RightAnalogPress"] = 11] = "RightAnalogPress"; gamepadLayout[gamepadLayout["LeftClusterTopButton"] = 12] = "LeftClusterTopButton"; gamepadLayout[gamepadLayout["LeftClusterBottomButton"] = 13] = "LeftClusterBottomButton"; gamepadLayout[gamepadLayout["LeftClusterLeftButton"] = 14] = "LeftClusterLeftButton"; gamepadLayout[gamepadLayout["LeftClusterRightButton"] = 15] = "LeftClusterRightButton"; gamepadLayout[gamepadLayout["CentreButton"] = 16] = "CentreButton"; // Axes gamepadLayout[gamepadLayout["LeftStickHorizontal"] = 0] = "LeftStickHorizontal"; gamepadLayout[gamepadLayout["LeftStickVertical"] = 1] = "LeftStickVertical"; gamepadLayout[gamepadLayout["RightStickHorizontal"] = 2] = "RightStickHorizontal"; gamepadLayout[gamepadLayout["RightStickVertical"] = 3] = "RightStickVertical"; })(gamepadLayout || (gamepadLayout = {})); /***/ }), /***/ "./src/Inputs/HoveringMouseEvents.ts": /*!*******************************************!*\ !*** ./src/Inputs/HoveringMouseEvents.ts ***! \*******************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "HoveringMouseEvents": () => (/* binding */ HoveringMouseEvents) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Video Player mouse Hover handler */ class HoveringMouseEvents { /** * @param mouseController - Mouse Controller instance */ constructor(mouseController) { this.mouseController = mouseController; } /** * Unregister event handlers */ unregisterMouseEvents() { // empty for HoveringMouseEvents implementation } /** * Handle the mouse move event, sends the mouse data to the UE Instance * @param mouseEvent - Mouse Event */ updateMouseMovePosition(mouseEvent) { if (!this.mouseController.videoElementProvider.isVideoReady()) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'MouseMove', 6); const coord = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(mouseEvent.offsetX, mouseEvent.offsetY); const delta = this.mouseController.coordinateConverter.normalizeAndQuantizeSigned(mouseEvent.movementX, mouseEvent.movementY); const toStreamerHandlers = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseMove')([ coord.x, coord.y, delta.x, delta.y ]); mouseEvent.preventDefault(); } /** * Handle the mouse Down event, sends the mouse data to the UE Instance * @param mouseEvent - Mouse Event */ handleMouseDown(mouseEvent) { if (!this.mouseController.videoElementProvider.isVideoReady()) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'onMouse Down', 6); const coord = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(mouseEvent.offsetX, mouseEvent.offsetY); const toStreamerHandlers = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseDown')([ mouseEvent.button, coord.x, coord.y ]); mouseEvent.preventDefault(); } /** * Handle the mouse Up event, sends the mouse data to the UE Instance * @param mouseEvent - Mouse Event */ handleMouseUp(mouseEvent) { if (!this.mouseController.videoElementProvider.isVideoReady()) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'onMouse Up', 6); const coord = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(mouseEvent.offsetX, mouseEvent.offsetY); const toStreamerHandlers = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseUp')([ mouseEvent.button, coord.x, coord.y ]); mouseEvent.preventDefault(); } /** * Consumes the mouse context event. The UE instance has no equivalent and doesn't need to be informed. * @param mouseEvent - Mouse Event */ handleContextMenu(mouseEvent) { if (!this.mouseController.videoElementProvider.isVideoReady()) { return; } mouseEvent.preventDefault(); } /** * Handle the mouse wheel event, sends the mouse wheel data to the UE Instance * @param wheelEvent - Mouse Event */ handleMouseWheel(wheelEvent) { if (!this.mouseController.videoElementProvider.isVideoReady()) { return; } const coord = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(wheelEvent.offsetX, wheelEvent.offsetY); const toStreamerHandlers = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseWheel')([ wheelEvent.wheelDelta, coord.x, coord.y ]); wheelEvent.preventDefault(); } /** * Handle the mouse double click event, sends the mouse data to the UE Instance * @param mouseEvent - Mouse Event */ handleMouseDouble(mouseEvent) { if (!this.mouseController.videoElementProvider.isVideoReady()) { return; } const coord = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(mouseEvent.offsetX, mouseEvent.offsetY); const toStreamerHandlers = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseDouble')([ mouseEvent.button, coord.x, coord.y ]); } /** * Handle the press mouse buttons event, sends the mouse data to the UE Instance * @param mouseEvent - Mouse Event */ handlePressMouseButtons(mouseEvent) { if (!this.mouseController.videoElementProvider.isVideoReady()) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'onMouse press', 6); this.mouseController.pressMouseButtons(mouseEvent.buttons, mouseEvent.offsetX, mouseEvent.offsetY); } /** * Handle the release mouse buttons event, sends the mouse data to the UE Instance * @param mouseEvent - Mouse Event */ handleReleaseMouseButtons(mouseEvent) { if (!this.mouseController.videoElementProvider.isVideoReady()) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'onMouse release', 6); this.mouseController.releaseMouseButtons(mouseEvent.buttons, mouseEvent.offsetX, mouseEvent.offsetY); } } /***/ }), /***/ "./src/Inputs/InputClassesFactory.ts": /*!*******************************************!*\ !*** ./src/Inputs/InputClassesFactory.ts ***! \*******************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "ActiveKeys": () => (/* binding */ ActiveKeys), /* harmony export */ "InputClassesFactory": () => (/* binding */ InputClassesFactory) /* harmony export */ }); /* harmony import */ var _FakeTouchController__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./FakeTouchController */ "./src/Inputs/FakeTouchController.ts"); /* harmony import */ var _KeyboardController__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./KeyboardController */ "./src/Inputs/KeyboardController.ts"); /* harmony import */ var _MouseController__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./MouseController */ "./src/Inputs/MouseController.ts"); /* harmony import */ var _TouchController__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./TouchController */ "./src/Inputs/TouchController.ts"); /* harmony import */ var _GamepadController__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./GamepadController */ "./src/Inputs/GamepadController.ts"); /* harmony import */ var _Config_Config__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../Config/Config */ "./src/Config/Config.ts"); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Class for making and setting up input class types */ class InputClassesFactory { /** * @param toStreamerMessagesProvider - Stream message instance * @param videoElementProvider - Video Player instance * @param coordinateConverter - A coordinateConverter instance */ constructor(toStreamerMessagesProvider, videoElementProvider, coordinateConverter) { this.activeKeys = new ActiveKeys(); this.toStreamerMessagesProvider = toStreamerMessagesProvider; this.videoElementProvider = videoElementProvider; this.coordinateConverter = coordinateConverter; } /** * Registers browser key events. */ registerKeyBoard(config) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Register Keyboard Events', 7); const keyboardController = new _KeyboardController__WEBPACK_IMPORTED_MODULE_1__.KeyboardController(this.toStreamerMessagesProvider, config, this.activeKeys); keyboardController.registerKeyBoardEvents(); return keyboardController; } /** * register mouse events based on a control type * @param controlScheme - if the mouse is either hovering or locked */ registerMouse(controlScheme) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Register Mouse Events', 7); const mouseController = new _MouseController__WEBPACK_IMPORTED_MODULE_2__.MouseController(this.toStreamerMessagesProvider, this.videoElementProvider, this.coordinateConverter, this.activeKeys); switch (controlScheme) { case _Config_Config__WEBPACK_IMPORTED_MODULE_3__.ControlSchemeType.LockedMouse: mouseController.registerLockedMouseEvents(mouseController); break; case _Config_Config__WEBPACK_IMPORTED_MODULE_3__.ControlSchemeType.HoveringMouse: mouseController.registerHoveringMouseEvents(mouseController); break; default: _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Info(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'unknown Control Scheme Type Defaulting to Locked Mouse Events'); mouseController.registerLockedMouseEvents(mouseController); break; } return mouseController; } /** * register touch events * @param fakeMouseTouch - the faked mouse touch event */ registerTouch(fakeMouseTouch, videoElementParentClientRect) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Registering Touch', 6); if (fakeMouseTouch) { const fakeTouchController = new _FakeTouchController__WEBPACK_IMPORTED_MODULE_4__.FakeTouchController(this.toStreamerMessagesProvider, this.videoElementProvider, this.coordinateConverter); fakeTouchController.setVideoElementParentClientRect(videoElementParentClientRect); return fakeTouchController; } else { return new _TouchController__WEBPACK_IMPORTED_MODULE_5__.TouchController(this.toStreamerMessagesProvider, this.videoElementProvider, this.coordinateConverter); } } /** * registers a gamepad */ registerGamePad() { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Register Game Pad', 7); const gamePadController = new _GamepadController__WEBPACK_IMPORTED_MODULE_6__.GamePadController(this.toStreamerMessagesProvider); return gamePadController; } } /** * A class that keeps track of current active keys */ class ActiveKeys { constructor() { this.activeKeys = []; this.activeKeys = []; } /** * Get the current array of active keys * @returns - an array of active keys */ getActiveKeys() { return this.activeKeys; } } /***/ }), /***/ "./src/Inputs/KeyboardController.ts": /*!******************************************!*\ !*** ./src/Inputs/KeyboardController.ts ***! \******************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "KeyboardController": () => (/* binding */ KeyboardController) /* harmony export */ }); /* harmony import */ var _SpecialKeyCodes__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./SpecialKeyCodes */ "./src/Inputs/SpecialKeyCodes.ts"); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _Config_Config__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../Config/Config */ "./src/Config/Config.ts"); /* harmony import */ var _Util_EventListenerTracker__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Util/EventListenerTracker */ "./src/Util/EventListenerTracker.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Handles the Keyboard Inputs for the document */ class KeyboardController { /** * @param toStreamerMessagesProvider Stream message provider class object * @param config The applications configuration. We're interested in the suppress browser keys option * @param activeKeysProvider Active keys provider class object */ constructor(toStreamerMessagesProvider, config, activeKeysProvider) { // Utility for keeping track of event handlers and unregistering them this.keyboardEventListenerTracker = new _Util_EventListenerTracker__WEBPACK_IMPORTED_MODULE_0__.EventListenerTracker(); /* * New browser APIs have moved away from KeyboardEvent.keyCode to KeyboardEvent.Code. * For details see: https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/keyCode#constants_for_keycode_value * We still use old KeyboardEvent.keyCode integers in the UE C++ side, so we need a way to map the new * string-based KeyboardEvent.Code to the old integers. */ 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 = toStreamerMessagesProvider; this.config = config; this.activeKeysProvider = activeKeysProvider; } /** * Registers document keyboard events with the controller */ registerKeyBoardEvents() { const keyDownHandler = (ev) => this.handleOnKeyDown(ev); const keyUpHandler = (ev) => this.handleOnKeyUp(ev); const keyPressHandler = (ev) => this.handleOnKeyPress(ev); document.addEventListener("keydown", keyDownHandler); document.addEventListener("keyup", keyUpHandler); //This has been deprecated as at Jun 13 2021 document.addEventListener("keypress", keyPressHandler); this.keyboardEventListenerTracker.addUnregisterCallback(() => document.removeEventListener("keydown", keyDownHandler)); this.keyboardEventListenerTracker.addUnregisterCallback(() => document.removeEventListener("keyup", keyUpHandler)); this.keyboardEventListenerTracker.addUnregisterCallback(() => document.removeEventListener("keypress", keyPressHandler)); } /** * Unregisters document keyboard events */ unregisterKeyBoardEvents() { this.keyboardEventListenerTracker.unregisterAll(); } /** * Handles When a key is down * @param keyboardEvent - Keyboard event */ handleOnKeyDown(keyboardEvent) { const keyCode = this.getKeycode(keyboardEvent); if (!keyCode) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), `key down ${keyCode}, repeat = ${keyboardEvent.repeat}`, 6); const toStreamerHandlers = this.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('KeyDown')([ this.getKeycode(keyboardEvent), keyboardEvent.repeat ? 1 : 0 ]); const activeKeys = this.activeKeysProvider.getActiveKeys(); activeKeys.push(keyCode); // Backspace is not considered a keypress in JavaScript but we need it // to be so characters may be deleted in a UE text entry field. if (keyCode === _SpecialKeyCodes__WEBPACK_IMPORTED_MODULE_2__.SpecialKeyCodes.backSpace) { document.dispatchEvent(new KeyboardEvent('keypress', { charCode: _SpecialKeyCodes__WEBPACK_IMPORTED_MODULE_2__.SpecialKeyCodes.backSpace })); } if (this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_3__.Flags.SuppressBrowserKeys) && this.isKeyCodeBrowserKey(keyCode)) { keyboardEvent.preventDefault(); } } /** * handles when a key is up * @param keyboardEvent - Keyboard event */ handleOnKeyUp(keyboardEvent) { const keyCode = this.getKeycode(keyboardEvent); if (!keyCode) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), `key up ${keyCode}`, 6); const toStreamerHandlers = this.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('KeyUp')([keyCode]); if (this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_3__.Flags.SuppressBrowserKeys) && this.isKeyCodeBrowserKey(keyCode)) { keyboardEvent.preventDefault(); } } /** * Handles when a key is press * @param keyboard - Keyboard Event */ handleOnKeyPress(keyboard) { if (!('charCode' in keyboard)) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Warning(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'KeyboardEvent.charCode is deprecated in this browser, cannot send key press.'); return; } const charCode = keyboard.charCode; _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), `key press ${charCode}`, 6); const toStreamerHandlers = this.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('KeyPress')([charCode]); } /** * Gets the Keycode of the Key pressed * @param keyboardEvent - Key board Event * @returns - the key code of the Key */ getKeycode(keyboardEvent) { // If we don't have keyCode property because browser API is deprecated then use KeyboardEvent.code instead. // See: https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/keyCode#constants_for_keycode_value if (!('keyCode' in keyboardEvent)) { // Convert KeyboardEvent.code string into integer-based key code for backwards compatibility reasons. const event = keyboardEvent; if (event.code in this.CodeToKeyCode) { return this.CodeToKeyCode[event.code]; } else { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Warning(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), `Keyboard code of ${event.code} is not supported in our mapping, ignoring this key.`); return null; } } // If we made it here KeyboardEvent.keyCode is still supported so we can safely use it. if (keyboardEvent.keyCode === _SpecialKeyCodes__WEBPACK_IMPORTED_MODULE_2__.SpecialKeyCodes.shift && keyboardEvent.code === 'ShiftRight') { return _SpecialKeyCodes__WEBPACK_IMPORTED_MODULE_2__.SpecialKeyCodes.rightShift; } else if (keyboardEvent.keyCode === _SpecialKeyCodes__WEBPACK_IMPORTED_MODULE_2__.SpecialKeyCodes.control && keyboardEvent.code === 'ControlRight') { return _SpecialKeyCodes__WEBPACK_IMPORTED_MODULE_2__.SpecialKeyCodes.rightControl; } else if (keyboardEvent.keyCode === _SpecialKeyCodes__WEBPACK_IMPORTED_MODULE_2__.SpecialKeyCodes.alt && keyboardEvent.code === 'AltRight') { return _SpecialKeyCodes__WEBPACK_IMPORTED_MODULE_2__.SpecialKeyCodes.rightAlt; } else { return keyboardEvent.keyCode; } } /** * Browser keys do not have a charCode so we only need to test keyCode. * @param keyCode - the browser keycode number */ isKeyCodeBrowserKey(keyCode) { // Function keys or tab key are considered "browser keys" that we may wish to suppress by preventing them being process by browser. return (keyCode >= 112 && keyCode <= 123) || keyCode === 9; } } /***/ }), /***/ "./src/Inputs/LockedMouseEvents.ts": /*!*****************************************!*\ !*** ./src/Inputs/LockedMouseEvents.ts ***! \*****************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "LockedMouseEvents": () => (/* binding */ LockedMouseEvents) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _Util_EventListenerTracker__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Util/EventListenerTracker */ "./src/Util/EventListenerTracker.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Handle the mouse locked events */ class LockedMouseEvents { /** * @param videoElementProvider - Video Player instance * @param mouseController - Mouse controller instance * @param activeKeysProvider - Active keys provider instance * @param playerStyleAttributesProvider - Player style attributes instance */ constructor(videoElementProvider, mouseController, activeKeysProvider) { this.x = 0; this.y = 0; this.updateMouseMovePositionEvent = (mouseEvent) => { this.updateMouseMovePosition(mouseEvent); }; // Utility for keeping track of event handlers and unregistering them this.mouseEventListenerTracker = new _Util_EventListenerTracker__WEBPACK_IMPORTED_MODULE_0__.EventListenerTracker(); this.videoElementProvider = videoElementProvider; this.mouseController = mouseController; this.activeKeysProvider = activeKeysProvider; const videoElementParent = this.videoElementProvider.getVideoParentElement(); this.x = videoElementParent.getBoundingClientRect().width / 2; this.y = videoElementParent.getBoundingClientRect().height / 2; this.coord = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(this.x, this.y); } /** * Unregisters all event handlers */ unregisterMouseEvents() { this.mouseEventListenerTracker.unregisterAll(); } /** * Handle when the locked state Changed */ lockStateChange() { const videoElementParent = this.videoElementProvider.getVideoParentElement(); const toStreamerHandlers = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers; if (document.pointerLockElement === videoElementParent || document.mozPointerLockElement === videoElementParent) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'Pointer locked', 6); document.addEventListener('mousemove', this.updateMouseMovePositionEvent, false); this.mouseEventListenerTracker.addUnregisterCallback(() => document.removeEventListener('mousemove', this.updateMouseMovePositionEvent, false)); } else { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'The pointer lock status is now unlocked', 6); // !a new arrow function must not be used here as it will be counted as a new function that cannot be removed document.removeEventListener('mousemove', this.updateMouseMovePositionEvent, false); // If mouse loses focus, send a key up for all of the currently held-down keys // This is necessary as when the mouse loses focus, the windows stops listening for events and as such // the keyup listener won't get fired let activeKeys = this.activeKeysProvider.getActiveKeys(); const setKeys = new Set(activeKeys); const newKeysIterable = []; setKeys.forEach((setKey) => { newKeysIterable[setKey]; }); newKeysIterable.forEach((uniqueKeycode) => { toStreamerHandlers.get('KeyUp')([uniqueKeycode]); }); // Reset the active keys back to nothing activeKeys = []; } } /** * Handle the mouse move event, sends the mouse data to the UE Instance * @param mouseEvent - Mouse Event */ updateMouseMovePosition(mouseEvent) { if (!this.videoElementProvider.isVideoReady()) { return; } const toStreamerHandlers = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers; const styleWidth = this.videoElementProvider.getVideoParentElement().clientWidth; const styleHeight = this.videoElementProvider.getVideoParentElement().clientHeight; this.x += mouseEvent.movementX; this.y += mouseEvent.movementY; if (this.x > styleWidth) { this.x -= styleWidth; } if (this.y > styleHeight) { this.y -= styleHeight; } if (this.x < 0) { this.x = styleWidth + this.x; } if (this.y < 0) { this.y = styleHeight - this.y; } this.coord = this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(this.x, this.y); const delta = this.mouseController.coordinateConverter.normalizeAndQuantizeSigned(mouseEvent.movementX, mouseEvent.movementY); toStreamerHandlers.get('MouseMove')([ this.coord.x, this.coord.y, delta.x, delta.y ]); } /** * Handle the mouse Down event, sends the mouse data to the UE Instance * @param mouseEvent - Mouse Event */ handleMouseDown(mouseEvent) { if (!this.videoElementProvider.isVideoReady()) { return; } const toStreamerHandlers = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseDown')([ mouseEvent.button, // We use the store value of this.coord as opposed to the mouseEvent.x/y as the mouseEvent location // uses the system cursor location which hasn't moved this.coord.x, this.coord.y ]); } /** * Handle the mouse Up event, sends the mouse data to the UE Instance * @param mouseEvent - Mouse Event */ handleMouseUp(mouseEvent) { if (!this.videoElementProvider.isVideoReady()) { return; } const toStreamerHandlers = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseUp')([ mouseEvent.button, // We use the store value of this.coord as opposed to the mouseEvent.x/y as the mouseEvent location // uses the system cursor location which hasn't moved this.coord.x, this.coord.y ]); } /** * Handle the mouse wheel event, sends the mouse wheel data to the UE Instance * @param wheelEvent - Mouse Event */ handleMouseWheel(wheelEvent) { if (!this.videoElementProvider.isVideoReady()) { return; } const toStreamerHandlers = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseWheel')([ wheelEvent.wheelDelta, // We use the store value of this.coord as opposed to the mouseEvent.x/y as the mouseEvent location // uses the system cursor location which hasn't moved this.coord.x, this.coord.y ]); } /** * Handle the mouse double click event, sends the mouse data to the UE Instance * @param mouseEvent - Mouse Event */ handleMouseDouble(mouseEvent) { if (!this.videoElementProvider.isVideoReady()) { return; } const toStreamerHandlers = this.mouseController.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseDouble')([ mouseEvent.button, // We use the store value of this.coord as opposed to the mouseEvent.x/y as the mouseEvent location // uses the system cursor location which hasn't moved this.coord.x, this.coord.y ]); } /** * Handle the press mouse buttons event, sends the mouse data to the UE Instance * @param mouseEvent - Mouse Event */ handlePressMouseButtons(mouseEvent) { if (!this.videoElementProvider.isVideoReady()) { return; } this.mouseController.pressMouseButtons(mouseEvent.buttons, this.x, this.y); } /** * Handle the release mouse buttons event, sends the mouse data to the UE Instance * @param mouseEvent - Mouse Event */ handleReleaseMouseButtons(mouseEvent) { if (!this.videoElementProvider.isVideoReady()) { return; } this.mouseController.releaseMouseButtons(mouseEvent.buttons, this.x, this.y); } } /***/ }), /***/ "./src/Inputs/MouseButtons.ts": /*!************************************!*\ !*** ./src/Inputs/MouseButtons.ts ***! \************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "MouseButton": () => (/* binding */ MouseButton), /* harmony export */ "MouseButtonsMask": () => (/* binding */ MouseButtonsMask) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Mouse Button Data * {@link https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/button} */ class MouseButton { } MouseButton.mainButton = 0; // Left button. MouseButton.auxiliaryButton = 1; // Wheel button. MouseButton.secondaryButton = 2; // Right button. MouseButton.fourthButton = 3; // Browser Back button. MouseButton.fifthButton = 4; // Browser Forward button. /** * Mouse Button Mask Data * {@link https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/buttons} */ class MouseButtonsMask { } MouseButtonsMask.primaryButton = 1; // Left button. MouseButtonsMask.secondaryButton = 2; // Right button. MouseButtonsMask.auxiliaryButton = 4; // Wheel button. MouseButtonsMask.fourthButton = 8; // Browser Back button. MouseButtonsMask.fifthButton = 16; // Browser Forward button. /***/ }), /***/ "./src/Inputs/MouseController.ts": /*!***************************************!*\ !*** ./src/Inputs/MouseController.ts ***! \***************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "MouseController": () => (/* binding */ MouseController) /* harmony export */ }); /* harmony import */ var _MouseButtons__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./MouseButtons */ "./src/Inputs/MouseButtons.ts"); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _LockedMouseEvents__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./LockedMouseEvents */ "./src/Inputs/LockedMouseEvents.ts"); /* harmony import */ var _HoveringMouseEvents__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./HoveringMouseEvents */ "./src/Inputs/HoveringMouseEvents.ts"); /* harmony import */ var _Util_EventListenerTracker__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Util/EventListenerTracker */ "./src/Util/EventListenerTracker.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Handles the Mouse Inputs for the document */ class MouseController { /** * @param toStreamerMessagesProvider - Stream message instance * @param videoElementProvider - Video Player instance * @param normalizeAndQuantize - A normalize and quantize instance */ constructor(toStreamerMessagesProvider, videoElementProvider, coordinateConverter, activeKeysProvider) { // Utility for keeping track of event handlers and unregistering them this.mouseEventListenerTracker = new _Util_EventListenerTracker__WEBPACK_IMPORTED_MODULE_0__.EventListenerTracker(); this.toStreamerMessagesProvider = toStreamerMessagesProvider; this.coordinateConverter = coordinateConverter; this.videoElementProvider = videoElementProvider; this.activeKeysProvider = activeKeysProvider; this.registerMouseEnterAndLeaveEvents(); } /** * Clears all the click events on the current video element parent div */ unregisterMouseEvents() { this.mouseEventListenerTracker.unregisterAll(); } /** * Register a locked mouse class * @param mouseController - a mouse controller instance * @param playerStyleAttributesProvider - a player style attributes instance */ registerLockedMouseEvents(mouseController) { const videoElementParent = this.videoElementProvider.getVideoParentElement(); const lockedMouseEvents = new _LockedMouseEvents__WEBPACK_IMPORTED_MODULE_1__.LockedMouseEvents(this.videoElementProvider, mouseController, this.activeKeysProvider); videoElementParent.requestPointerLock = videoElementParent.requestPointerLock || videoElementParent.mozRequestPointerLock; document.exitPointerLock = document.exitPointerLock || document.mozExitPointerLock; // minor hack to alleviate ios not supporting pointerlock if (videoElementParent.requestPointerLock) { const onclick = () => { videoElementParent.requestPointerLock(); }; videoElementParent.addEventListener('click', onclick); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('click', onclick)); } const lockStateChangeListener = () => lockedMouseEvents.lockStateChange(); document.addEventListener('pointerlockchange', lockStateChangeListener, false); document.addEventListener('mozpointerlockchange', lockStateChangeListener, false); this.mouseEventListenerTracker.addUnregisterCallback(() => document.removeEventListener('pointerlockchange', lockStateChangeListener, false)); this.mouseEventListenerTracker.addUnregisterCallback(() => document.removeEventListener('mozpointerlockchange', lockStateChangeListener, false)); const onmousedown = (mouseEvent) => lockedMouseEvents.handleMouseDown(mouseEvent); const onmouseup = (mouseEvent) => lockedMouseEvents.handleMouseUp(mouseEvent); const onwheel = (wheelEvent) => lockedMouseEvents.handleMouseWheel(wheelEvent); const ondblclick = (mouseEvent) => lockedMouseEvents.handleMouseDouble(mouseEvent); videoElementParent.addEventListener('mousedown', onmousedown); videoElementParent.addEventListener('mouseup', onmouseup); videoElementParent.addEventListener('wheel', onwheel); videoElementParent.addEventListener('dblclick', ondblclick); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('mousedown', onmousedown)); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('mouseup', onmouseup)); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('wheel', onwheel)); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('dblclick', ondblclick)); this.mouseEventListenerTracker.addUnregisterCallback(() => lockedMouseEvents.unregisterMouseEvents()); this.mouseEventListenerTracker.addUnregisterCallback(() => { if (document.exitPointerLock && (document.pointerLockElement === videoElementParent || document.mozPointerLockElement === videoElementParent)) { document.exitPointerLock(); } }); } /** * Register a hovering mouse class * @param mouseController - A mouse controller object */ registerHoveringMouseEvents(mouseController) { const videoElementParent = this.videoElementProvider.getVideoParentElement(); const hoveringMouseEvents = new _HoveringMouseEvents__WEBPACK_IMPORTED_MODULE_2__.HoveringMouseEvents(mouseController); const onmousemove = (mouseEvent) => hoveringMouseEvents.updateMouseMovePosition(mouseEvent); const onmousedown = (mouseEvent) => hoveringMouseEvents.handleMouseDown(mouseEvent); const onmouseup = (mouseEvent) => hoveringMouseEvents.handleMouseUp(mouseEvent); const oncontextmenu = (mouseEvent) => hoveringMouseEvents.handleContextMenu(mouseEvent); const onwheel = (wheelEvent) => hoveringMouseEvents.handleMouseWheel(wheelEvent); const ondblclick = (mouseEvent) => hoveringMouseEvents.handleMouseDouble(mouseEvent); videoElementParent.addEventListener('mousemove', onmousemove); videoElementParent.addEventListener('mousedown', onmousedown); videoElementParent.addEventListener('mouseup', onmouseup); videoElementParent.addEventListener('contextmenu', oncontextmenu); videoElementParent.addEventListener('wheel', onwheel); videoElementParent.addEventListener('dblclick', ondblclick); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('mousemove', onmousemove)); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('mousedown', onmousedown)); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('mouseup', onmouseup)); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('contextmenu', oncontextmenu)); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('wheel', onwheel)); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('dblclick', ondblclick)); this.mouseEventListenerTracker.addUnregisterCallback(() => hoveringMouseEvents.unregisterMouseEvents()); } /** * Set the mouse enter and mouse leave events */ registerMouseEnterAndLeaveEvents() { const videoElementParent = this.videoElementProvider.getVideoParentElement(); // Handle when the Mouse has entered the element const onmouseenter = (event) => { if (!this.videoElementProvider.isVideoReady()) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_3__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_3__.Logger.GetStackTrace(), 'Mouse Entered', 6); this.sendMouseEnter(); this.pressMouseButtons(event.buttons, event.x, event.y); }; // Handles when the mouse has left the element const onmouseleave = (event) => { if (!this.videoElementProvider.isVideoReady()) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_3__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_3__.Logger.GetStackTrace(), 'Mouse Left', 6); this.sendMouseLeave(); this.releaseMouseButtons(event.buttons, event.x, event.y); }; videoElementParent.addEventListener('mouseenter', onmouseenter); videoElementParent.addEventListener('mouseleave', onmouseleave); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('mouseenter', onmouseenter)); this.mouseEventListenerTracker.addUnregisterCallback(() => videoElementParent.removeEventListener('mouseleave', onmouseleave)); } /** * Handle when a mouse button is released * @param buttons - Mouse Button * @param X - Mouse pointer X coordinate * @param Y - Mouse pointer Y coordinate */ releaseMouseButtons(buttons, X, Y) { const coord = this.coordinateConverter.normalizeAndQuantizeUnsigned(X, Y); if (buttons & _MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButtonsMask.primaryButton) { this.sendMouseUp(_MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButton.mainButton, coord.x, coord.y); } if (buttons & _MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButtonsMask.secondaryButton) { this.sendMouseUp(_MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButton.secondaryButton, coord.x, coord.y); } if (buttons & _MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButtonsMask.auxiliaryButton) { this.sendMouseUp(_MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButton.auxiliaryButton, coord.x, coord.y); } if (buttons & _MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButtonsMask.fourthButton) { this.sendMouseUp(_MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButton.fourthButton, coord.x, coord.y); } if (buttons & _MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButtonsMask.fifthButton) { this.sendMouseUp(_MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButton.fifthButton, coord.x, coord.y); } } /** * Handle when a mouse button is pressed * @param buttons - Mouse Button * @param X - Mouse pointer X coordinate * @param Y - Mouse pointer Y coordinate */ pressMouseButtons(buttons, X, Y) { if (!this.videoElementProvider.isVideoReady()) { return; } const coord = this.coordinateConverter.normalizeAndQuantizeUnsigned(X, Y); if (buttons & _MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButtonsMask.primaryButton) { this.sendMouseDown(_MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButton.mainButton, coord.x, coord.y); } if (buttons & _MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButtonsMask.secondaryButton) { this.sendMouseDown(_MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButton.secondaryButton, coord.x, coord.y); } if (buttons & _MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButtonsMask.auxiliaryButton) { this.sendMouseDown(_MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButton.auxiliaryButton, coord.x, coord.y); } if (buttons & _MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButtonsMask.fourthButton) { this.sendMouseDown(_MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButton.fourthButton, coord.x, coord.y); } if (buttons & _MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButtonsMask.fifthButton) { this.sendMouseDown(_MouseButtons__WEBPACK_IMPORTED_MODULE_4__.MouseButton.fifthButton, coord.x, coord.y); } } /** * Handles mouse enter */ sendMouseEnter() { if (!this.videoElementProvider.isVideoReady()) { return; } const toStreamerHandlers = this.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseEnter')(); } /** * Handles mouse Leave */ sendMouseLeave() { if (!this.videoElementProvider.isVideoReady()) { return; } const toStreamerHandlers = this.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseLeave')(); } /** * Handles when a mouse button is pressed down * @param button - Mouse Button Pressed * @param X - Mouse X Coordinate * @param Y - Mouse Y Coordinate */ sendMouseDown(button, X, Y) { if (!this.videoElementProvider.isVideoReady()) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_3__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_3__.Logger.GetStackTrace(), `mouse button ${button} down at (${X}, ${Y})`, 6); const toStreamerHandlers = this.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseDown')([button, X, Y]); } /** * Handles when a mouse button is pressed up * @param button - Mouse Button Pressed * @param X - Mouse X Coordinate * @param Y - Mouse Y Coordinate */ sendMouseUp(button, X, Y) { if (!this.videoElementProvider.isVideoReady()) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_3__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_3__.Logger.GetStackTrace(), `mouse button ${button} up at (${X}, ${Y})`, 6); const coord = this.coordinateConverter.normalizeAndQuantizeUnsigned(X, Y); const toStreamerHandlers = this.toStreamerMessagesProvider.toStreamerHandlers; toStreamerHandlers.get('MouseUp')([button, coord.x, coord.y]); } } /***/ }), /***/ "./src/Inputs/SpecialKeyCodes.ts": /*!***************************************!*\ !*** ./src/Inputs/SpecialKeyCodes.ts ***! \***************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SpecialKeyCodes": () => (/* binding */ SpecialKeyCodes) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Registers the Special Key codes * Must be kept in sync with JavaScriptKeyCodeToFKey C++ array. * The index of the entry in the array is the special key code given below. */ class SpecialKeyCodes { } SpecialKeyCodes.backSpace = 8; SpecialKeyCodes.shift = 16; SpecialKeyCodes.control = 17; SpecialKeyCodes.alt = 18; SpecialKeyCodes.rightShift = 253; SpecialKeyCodes.rightControl = 254; SpecialKeyCodes.rightAlt = 255; /***/ }), /***/ "./src/Inputs/TouchController.ts": /*!***************************************!*\ !*** ./src/Inputs/TouchController.ts ***! \***************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "TouchController": () => (/* binding */ TouchController) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _Util_EventListenerTracker__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Util/EventListenerTracker */ "./src/Util/EventListenerTracker.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Handles the Touch input Events */ class TouchController { /** * @param toStreamerMessagesProvider - Stream message instance * @param videoElementProvider - Video Player instance * @param coordinateConverter - A coordinate converter instance */ constructor(toStreamerMessagesProvider, videoElementProvider, coordinateConverter) { this.fingers = [9, 8, 7, 6, 5, 4, 3, 2, 1, 0]; this.fingerIds = new Map(); this.maxByteValue = 255; // Utility for keeping track of event handlers and unregistering them this.touchEventListenerTracker = new _Util_EventListenerTracker__WEBPACK_IMPORTED_MODULE_0__.EventListenerTracker(); this.toStreamerMessagesProvider = toStreamerMessagesProvider; this.videoElementProvider = videoElementProvider; this.coordinateConverter = coordinateConverter; this.videoElementParent = videoElementProvider.getVideoElement(); const ontouchstart = (ev) => this.onTouchStart(ev); const ontouchend = (ev) => this.onTouchEnd(ev); const ontouchmove = (ev) => this.onTouchMove(ev); this.videoElementParent.addEventListener('touchstart', ontouchstart); this.videoElementParent.addEventListener('touchend', ontouchend); this.videoElementParent.addEventListener('touchmove', ontouchmove); this.touchEventListenerTracker.addUnregisterCallback(() => this.videoElementParent.removeEventListener('touchstart', ontouchstart)); this.touchEventListenerTracker.addUnregisterCallback(() => this.videoElementParent.removeEventListener('touchend', ontouchend)); this.touchEventListenerTracker.addUnregisterCallback(() => this.videoElementParent.removeEventListener('touchmove', ontouchmove)); _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'Touch Events Registered', 6); // is this strictly necessary? const preventOnTouchMove = (event) => { event.preventDefault(); }; document.addEventListener('touchmove', preventOnTouchMove); this.touchEventListenerTracker.addUnregisterCallback(() => document.removeEventListener('touchmove', preventOnTouchMove)); } /** * Unregister all touch events */ unregisterTouchEvents() { this.touchEventListenerTracker.unregisterAll(); } /** * Remember a touch command * @param touch - the touch command */ rememberTouch(touch) { const finger = this.fingers.pop(); if (finger === undefined) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'exhausted touch identifiers', 6); } this.fingerIds.set(touch.identifier, finger); } /** * Forgets a touch command * @param touch - the touch command */ forgetTouch(touch) { this.fingers.push(this.fingerIds.get(touch.identifier)); // Sort array back into descending order. This means if finger '1' were to lift after finger '0', we would ensure that 0 will be the first index to pop this.fingers.sort(function (a, b) { return b - a; }); this.fingerIds.delete(touch.identifier); } /** * When a touch event starts * @param touchEvent - the touch event being intercepted */ onTouchStart(touchEvent) { if (!this.videoElementProvider.isVideoReady()) { return; } for (let t = 0; t < touchEvent.changedTouches.length; t++) { this.rememberTouch(touchEvent.changedTouches[t]); } _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'touch start', 6); this.emitTouchData('TouchStart', touchEvent.changedTouches); touchEvent.preventDefault(); } /** * When a touch event ends * @param touchEvent - the touch event being intercepted */ onTouchEnd(touchEvent) { if (!this.videoElementProvider.isVideoReady()) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'touch end', 6); this.emitTouchData('TouchEnd', touchEvent.changedTouches); // Re-cycle unique identifiers previously assigned to each touch. for (let t = 0; t < touchEvent.changedTouches.length; t++) { this.forgetTouch(touchEvent.changedTouches[t]); } touchEvent.preventDefault(); } /** * when a moving touch event occurs * @param touchEvent - the touch event being intercepted */ onTouchMove(touchEvent) { if (!this.videoElementProvider.isVideoReady()) { return; } _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'touch move', 6); this.emitTouchData('TouchMove', touchEvent.touches); touchEvent.preventDefault(); } emitTouchData(type, touches) { if (!this.videoElementProvider.isVideoReady()) { return; } const offset = this.videoElementProvider.getVideoParentElement().getBoundingClientRect(); const toStreamerHandlers = this.toStreamerMessagesProvider.toStreamerHandlers; for (let t = 0; t < touches.length; t++) { const numTouches = 1; // the number of touches to be sent this message const touch = touches[t]; const x = touch.clientX - offset.left; const y = touch.clientY - offset.top; _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), `F${this.fingerIds.get(touch.identifier)}=(${x}, ${y})`, 6); const coord = this.coordinateConverter.normalizeAndQuantizeUnsigned(x, y); switch (type) { case 'TouchStart': toStreamerHandlers.get('TouchStart')([ numTouches, coord.x, coord.y, this.fingerIds.get(touch.identifier), this.maxByteValue * touch.force, coord.inRange ? 1 : 0 ]); break; case 'TouchEnd': toStreamerHandlers.get('TouchEnd')([ numTouches, coord.x, coord.y, this.fingerIds.get(touch.identifier), this.maxByteValue * touch.force, coord.inRange ? 1 : 0 ]); break; case 'TouchMove': toStreamerHandlers.get('TouchMove')([ numTouches, coord.x, coord.y, this.fingerIds.get(touch.identifier), this.maxByteValue * touch.force, coord.inRange ? 1 : 0 ]); break; } } } } /***/ }), /***/ "./src/Inputs/XRGamepadController.ts": /*!*******************************************!*\ !*** ./src/Inputs/XRGamepadController.ts ***! \*******************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "XRGamepadController": () => (/* binding */ XRGamepadController) /* harmony export */ }); /* harmony import */ var _Util_WebXRUtils__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Util/WebXRUtils */ "./src/Util/WebXRUtils.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * The class that handles the functionality of xrgamepads and controllers */ class XRGamepadController { /** * @param toStreamerMessagesProvider - Stream message instance */ constructor(toStreamerMessagesProvider) { this.toStreamerMessagesProvider = toStreamerMessagesProvider; this.controllers = []; } updateStatus(source, frame, refSpace) { if (source.gamepad) { const gamepadPose = frame.getPose(source.gripSpace, refSpace); if (!gamepadPose) { return; } let system = 0; if (source.profiles.includes('htc-vive')) { system = 1; } else if (source.profiles.includes('oculus-touch')) { system = 2; } // TODO (william.belcher): Add other profiles (Quest, Microsoft Mixed Reality, etc) this.toStreamerMessagesProvider.toStreamerHandlers.get('XRSystem')([ system ]); // Default: AnyHand (2) let handedness = 2; switch (source.handedness) { case 'left': handedness = 0; break; case 'right': handedness = 1; break; } // Send controller transform const matrix = gamepadPose.transform.matrix; const mat = []; for (let i = 0; i < 16; i++) { mat[i] = new Float32Array([matrix[i]])[0]; } // prettier-ignore this.toStreamerMessagesProvider.toStreamerHandlers.get('XRControllerTransform')([ mat[0], mat[4], mat[8], mat[12], mat[1], mat[5], mat[9], mat[13], mat[2], mat[6], mat[10], mat[14], mat[3], mat[7], mat[11], mat[15], handedness ]); // Handle controller buttons and axes if (this.controllers[handedness] === undefined) { this.controllers[handedness] = { prevState: undefined, currentState: undefined, id: undefined }; this.controllers[handedness].prevState = _Util_WebXRUtils__WEBPACK_IMPORTED_MODULE_0__.WebXRUtils.deepCopyGamepad(source.gamepad); } this.controllers[handedness].currentState = _Util_WebXRUtils__WEBPACK_IMPORTED_MODULE_0__.WebXRUtils.deepCopyGamepad(source.gamepad); const controller = this.controllers[handedness]; const currState = controller.currentState; const prevState = controller.prevState; // Iterate over buttons for (let i = 0; i < currState.buttons.length; i++) { const currButton = currState.buttons[i]; const prevButton = prevState.buttons[i]; if (currButton.pressed) { // press this.toStreamerMessagesProvider.toStreamerHandlers.get('XRButtonPressed')([handedness, i, prevButton.pressed ? 1 : 0]); } else if (!currButton.pressed && prevButton.pressed) { this.toStreamerMessagesProvider.toStreamerHandlers.get('XRButtonReleased')([handedness, i, 0]); } if (currButton.touched && !currButton.pressed) { // press this.toStreamerMessagesProvider.toStreamerHandlers.get('XRButtonPressed')([handedness, 3, prevButton.touched ? 1 : 0]); } else if (!currButton.touched && prevButton.touched) { this.toStreamerMessagesProvider.toStreamerHandlers.get('XRButtonReleased')([handedness, 3, 0]); } } // Iterate over gamepad axes for (let i = 0; i < currState.axes.length; i++) { this.toStreamerMessagesProvider.toStreamerHandlers.get('XRAnalog')([handedness, i, currState.axes[i]]); } this.controllers[handedness].prevState = currState; } } } /***/ }), /***/ "./src/Logger/Logger.ts": /*!******************************!*\ !*** ./src/Logger/Logger.ts ***! \******************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Logger": () => (/* binding */ Logger) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. class Logger { /** * Captures the stack and returns it * @returns the current stack */ static GetStackTrace() { const error = new Error(); let formattedStack = 'No Stack Available for this browser'; // format the error if (error.stack) { formattedStack = error.stack.toString().replace(/Error/g, ''); } return formattedStack; } /** * Set the log verbosity level */ static SetLoggerVerbosity(verboseLogLevel) { if (this.verboseLogLevel != null) { this.verboseLogLevel = verboseLogLevel; } } /** * The standard logging output * @param stack - the stack trace * @param message - the message to be logged * @param verbosity - the verbosity level */ static Log(stack, message, verbosity) { if (verbosity > this.verboseLogLevel) { return; } const returnString = `Level: Log\nMsg: ${message}\nCaller: ${stack}`; console.log(returnString); } /** * The standard logging output * @param stack - the stack trace * @param message - the message to be logged * @param verbosity - the verbosity level */ static Info(stack, message, verbosity) { if (verbosity > this.verboseLogLevel) { return; } const returnString = `Level: Info\nMsg: ${message}`; console.info(returnString); } /** * The standard logging output * @param stack - the stack trace * @param message - the message to be logged */ static Error(stack, message) { const returnString = `Level: Error\nMsg: ${message}\nCaller: ${stack}`; console.error(returnString); } /** * The standard logging output * @param stack - the stack trace * @param message - the message to be logged */ static Warning(stack, message) { const returnString = `Level: Warning\nCaller: ${stack}\nMsg: ${message}`; console.warn(returnString); } } Logger.verboseLogLevel = 5; /***/ }), /***/ "./src/PeerConnectionController/AggregatedStats.ts": /*!*********************************************************!*\ !*** ./src/PeerConnectionController/AggregatedStats.ts ***! \*********************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "AggregatedStats": () => (/* binding */ AggregatedStats) /* harmony export */ }); /* harmony import */ var _InboundRTPStats__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./InboundRTPStats */ "./src/PeerConnectionController/InboundRTPStats.ts"); /* harmony import */ var _DataChannelStats__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./DataChannelStats */ "./src/PeerConnectionController/DataChannelStats.ts"); /* harmony import */ var _CandidateStat__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ./CandidateStat */ "./src/PeerConnectionController/CandidateStat.ts"); /* harmony import */ var _CandidatePairStats__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./CandidatePairStats */ "./src/PeerConnectionController/CandidatePairStats.ts"); /* harmony import */ var _OutBoundRTPStats__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./OutBoundRTPStats */ "./src/PeerConnectionController/OutBoundRTPStats.ts"); /* harmony import */ var _SessionStats__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./SessionStats */ "./src/PeerConnectionController/SessionStats.ts"); /* harmony import */ var _StreamStats__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./StreamStats */ "./src/PeerConnectionController/StreamStats.ts"); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. class AggregatedStats { constructor() { this.inboundVideoStats = new _InboundRTPStats__WEBPACK_IMPORTED_MODULE_0__.InboundVideoStats(); this.inboundAudioStats = new _InboundRTPStats__WEBPACK_IMPORTED_MODULE_0__.InboundAudioStats(); this.candidatePair = new _CandidatePairStats__WEBPACK_IMPORTED_MODULE_1__.CandidatePairStats(); this.DataChannelStats = new _DataChannelStats__WEBPACK_IMPORTED_MODULE_2__.DataChannelStats(); this.outBoundVideoStats = new _OutBoundRTPStats__WEBPACK_IMPORTED_MODULE_3__.OutBoundVideoStats(); this.sessionStats = new _SessionStats__WEBPACK_IMPORTED_MODULE_4__.SessionStats(); this.streamStats = new _StreamStats__WEBPACK_IMPORTED_MODULE_5__.StreamStats(); this.codecs = new Map(); } /** * Gather all the information from the RTC Peer Connection Report * @param rtcStatsReport - RTC Stats Report */ processStats(rtcStatsReport) { this.localCandidates = new Array(); this.remoteCandidates = new Array(); rtcStatsReport.forEach((stat) => { const type = stat.type; switch (type) { case 'candidate-pair': this.handleCandidatePair(stat); break; case 'certificate': break; case 'codec': this.handleCodec(stat); break; case 'data-channel': this.handleDataChannel(stat); break; case 'inbound-rtp': this.handleInBoundRTP(stat); break; case 'local-candidate': this.handleLocalCandidate(stat); break; case 'media-source': break; case 'media-playout': break; case 'outbound-rtp': break; case 'peer-connection': break; case 'remote-candidate': this.handleRemoteCandidate(stat); break; case 'remote-inbound-rtp': break; case 'remote-outbound-rtp': this.handleRemoteOutBound(stat); break; case 'track': this.handleTrack(stat); break; case 'transport': break; case 'stream': this.handleStream(stat); break; default: _Logger_Logger__WEBPACK_IMPORTED_MODULE_6__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_6__.Logger.GetStackTrace(), 'unhandled Stat Type'); _Logger_Logger__WEBPACK_IMPORTED_MODULE_6__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_6__.Logger.GetStackTrace(), stat); break; } }); } /** * Process stream stats data from webrtc * * @param stat - the stats coming in from webrtc */ handleStream(stat) { this.streamStats = stat; } /** * Process the Ice Candidate Pair Data * @param stat - the stats coming in from ice candidates */ handleCandidatePair(stat) { this.candidatePair.bytesReceived = stat.bytesReceived; this.candidatePair.bytesSent = stat.bytesSent; this.candidatePair.localCandidateId = stat.localCandidateId; this.candidatePair.remoteCandidateId = stat.remoteCandidateId; this.candidatePair.nominated = stat.nominated; this.candidatePair.readable = stat.readable; this.candidatePair.selected = stat.selected; this.candidatePair.writable = stat.writable; this.candidatePair.state = stat.state; this.candidatePair.currentRoundTripTime = stat.currentRoundTripTime; } /** * Process the Data Channel Data * @param stat - the stats coming in from the data channel */ handleDataChannel(stat) { this.DataChannelStats.bytesReceived = stat.bytesReceived; this.DataChannelStats.bytesSent = stat.bytesSent; this.DataChannelStats.dataChannelIdentifier = stat.dataChannelIdentifier; this.DataChannelStats.id = stat.id; this.DataChannelStats.label = stat.label; this.DataChannelStats.messagesReceived = stat.messagesReceived; this.DataChannelStats.messagesSent = stat.messagesSent; this.DataChannelStats.protocol = stat.protocol; this.DataChannelStats.state = stat.state; this.DataChannelStats.timestamp = stat.timestamp; } /** * Process the Local Ice Candidate Data * @param stat - local stats */ handleLocalCandidate(stat) { const localCandidate = new _CandidateStat__WEBPACK_IMPORTED_MODULE_7__.CandidateStat(); localCandidate.label = 'local-candidate'; localCandidate.address = stat.address; localCandidate.port = stat.port; localCandidate.protocol = stat.protocol; localCandidate.candidateType = stat.candidateType; localCandidate.id = stat.id; this.localCandidates.push(localCandidate); } /** * Process the Remote Ice Candidate Data * @param stat - ice candidate stats */ handleRemoteCandidate(stat) { const RemoteCandidate = new _CandidateStat__WEBPACK_IMPORTED_MODULE_7__.CandidateStat(); RemoteCandidate.label = 'local-candidate'; RemoteCandidate.address = stat.address; RemoteCandidate.port = stat.port; RemoteCandidate.protocol = stat.protocol; RemoteCandidate.id = stat.id; RemoteCandidate.candidateType = stat.candidateType; this.remoteCandidates.push(RemoteCandidate); } /** * Process the Inbound RTP Audio and Video Data * @param stat - inbound rtp stats */ handleInBoundRTP(stat) { switch (stat.kind) { case 'video': // Need to convert to unknown first to remove an error around // InboundVideoStats having the bitrate member which isn't found on // the InboundRTPStats this.inboundVideoStats = stat; if (this.lastVideoStats != undefined) { 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': // Need to convert to unknown first to remove an error around // InboundAudioStats having the bitrate member which isn't found on // the InboundRTPStats this.inboundAudioStats = stat; if (this.lastAudioStats != undefined) { 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: _Logger_Logger__WEBPACK_IMPORTED_MODULE_6__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_6__.Logger.GetStackTrace(), 'Kind is not handled'); break; } } /** * Process the outbound RTP Audio and Video Data * @param stat - remote outbound stats */ handleRemoteOutBound(stat) { switch (stat.kind) { case 'video': this.outBoundVideoStats.bytesSent = stat.bytesSent; this.outBoundVideoStats.id = stat.id; this.outBoundVideoStats.localId = stat.localId; this.outBoundVideoStats.packetsSent = stat.packetsSent; this.outBoundVideoStats.remoteTimestamp = stat.remoteTimestamp; this.outBoundVideoStats.timestamp = stat.timestamp; break; case 'audio': break; default: break; } } /** * Process the Inbound Video Track Data * @param stat - video track stats */ handleTrack(stat) { // we only want to extract stats from the video track if (stat.type === 'track' && (stat.trackIdentifier === 'video_label' || stat.kind === 'video')) { this.inboundVideoStats.framesDropped = stat.framesDropped; this.inboundVideoStats.framesReceived = stat.framesReceived; this.inboundVideoStats.frameHeight = stat.frameHeight; this.inboundVideoStats.frameWidth = stat.frameWidth; } } handleCodec(stat) { const codecId = stat.id; const codecType = `${stat.mimeType .replace('video/', '') .replace('audio/', '')}${stat.sdpFmtpLine ? ` ${stat.sdpFmtpLine}` : ''}`; this.codecs.set(codecId, codecType); } handleSessionStatistics(videoStartTime, inputController, videoEncoderAvgQP) { const deltaTime = Date.now() - videoStartTime; this.sessionStats.runTime = new Date(deltaTime) .toISOString() .substr(11, 8) .toString(); const controlsStreamInput = inputController === null ? 'Not sent yet' : inputController ? 'true' : 'false'; this.sessionStats.controlsStreamInput = controlsStreamInput; this.sessionStats.videoEncoderAvgQP = videoEncoderAvgQP; } /** * Check if a value coming in from our stats is actually a number * @param value - the number to be checked */ isNumber(value) { return typeof value === 'number' && isFinite(value); } } /***/ }), /***/ "./src/PeerConnectionController/CandidatePairStats.ts": /*!************************************************************!*\ !*** ./src/PeerConnectionController/CandidatePairStats.ts ***! \************************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "CandidatePairStats": () => (/* binding */ CandidatePairStats) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * ICE Candidate Pair Stats collected from the RTC Stats Report */ class CandidatePairStats { } /***/ }), /***/ "./src/PeerConnectionController/CandidateStat.ts": /*!*******************************************************!*\ !*** ./src/PeerConnectionController/CandidateStat.ts ***! \*******************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "CandidateStat": () => (/* binding */ CandidateStat) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * ICE Candidate Stat collected from the RTC Stats Report */ class CandidateStat { } /***/ }), /***/ "./src/PeerConnectionController/DataChannelStats.ts": /*!**********************************************************!*\ !*** ./src/PeerConnectionController/DataChannelStats.ts ***! \**********************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "DataChannelStats": () => (/* binding */ DataChannelStats) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Data Channel Stats collected from the RTC Stats Report */ class DataChannelStats { } /***/ }), /***/ "./src/PeerConnectionController/InboundRTPStats.ts": /*!*********************************************************!*\ !*** ./src/PeerConnectionController/InboundRTPStats.ts ***! \*********************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "InboundAudioStats": () => (/* binding */ InboundAudioStats), /* harmony export */ "InboundRTPStats": () => (/* binding */ InboundRTPStats), /* harmony export */ "InboundVideoStats": () => (/* binding */ InboundVideoStats) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Inbound Audio Stats collected from the RTC Stats Report */ class InboundAudioStats { } /** * Inbound Video Stats collected from the RTC Stats Report */ class InboundVideoStats { } /** * Inbound Stats collected from the RTC Stats Report */ class InboundRTPStats { } /***/ }), /***/ "./src/PeerConnectionController/OutBoundRTPStats.ts": /*!**********************************************************!*\ !*** ./src/PeerConnectionController/OutBoundRTPStats.ts ***! \**********************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "OutBoundRTPStats": () => (/* binding */ OutBoundRTPStats), /* harmony export */ "OutBoundVideoStats": () => (/* binding */ OutBoundVideoStats) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Outbound Video Stats collected from the RTC Stats Report */ class OutBoundVideoStats { } /** * Outbound Stats collected from the RTC Stats Report */ class OutBoundRTPStats { } /***/ }), /***/ "./src/PeerConnectionController/PeerConnectionController.ts": /*!******************************************************************!*\ !*** ./src/PeerConnectionController/PeerConnectionController.ts ***! \******************************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "PeerConnectionController": () => (/* binding */ PeerConnectionController) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _Config_Config__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Config/Config */ "./src/Config/Config.ts"); /* harmony import */ var _AggregatedStats__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./AggregatedStats */ "./src/PeerConnectionController/AggregatedStats.ts"); /* harmony import */ var sdp__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! sdp */ "sdp"); /* harmony import */ var sdp__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(sdp__WEBPACK_IMPORTED_MODULE_0__); /* harmony import */ var _Util_RTCUtils__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../Util/RTCUtils */ "./src/Util/RTCUtils.ts"); // Copyright Epic Games, Inc. All Rights Reserved. var __awaiter = (undefined && undefined.__awaiter) || function (thisArg, _arguments, P, generator) { function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); }; /** * Handles the Peer Connection */ class PeerConnectionController { /** * Create a new RTC Peer Connection client * @param options - Peer connection Options * @param config - The config for our PS experience. */ constructor(options, config, preferredCodec) { this.config = config; this.createPeerConnection(options, preferredCodec); } createPeerConnection(options, preferredCodec) { // Set the ICE transport to relay if TURN enabled if (this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.Flags.ForceTURN)) { options.iceTransportPolicy = 'relay'; _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), 'Forcing TURN usage by setting ICE Transport Policy in peer connection config.'); } // build a new peer connection with the options this.peerConnection = new RTCPeerConnection(options); this.peerConnection.onsignalingstatechange = (ev) => this.handleSignalStateChange(ev); this.peerConnection.oniceconnectionstatechange = (ev) => this.handleIceConnectionStateChange(ev); this.peerConnection.onicegatheringstatechange = (ev) => this.handleIceGatheringStateChange(ev); this.peerConnection.ontrack = (ev) => this.handleOnTrack(ev); this.peerConnection.onicecandidate = (ev) => this.handleIceCandidate(ev); this.peerConnection.ondatachannel = (ev) => this.handleDataChannel(ev); this.aggregatedStats = new _AggregatedStats__WEBPACK_IMPORTED_MODULE_3__.AggregatedStats(); this.preferredCodec = preferredCodec; this.updateCodecSelection = true; } /** * Create an offer for the Web RTC handshake and send the offer to the signaling server via websocket * @param offerOptions - RTC Offer Options */ createOffer(offerOptions, config) { return __awaiter(this, void 0, void 0, function* () { _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), 'Create Offer', 6); const isLocalhostConnection = location.hostname === 'localhost' || location.hostname === '127.0.0.1'; const isHttpsConnection = location.protocol === 'https:'; let useMic = config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.Flags.UseMic); if (useMic && !(isLocalhostConnection || isHttpsConnection)) { useMic = false; _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), 'Microphone access in the browser will not work if you are not on HTTPS or localhost. Disabling mic access.'); _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), "For testing you can enable HTTP microphone access Chrome by visiting chrome://flags/ and enabling 'unsafely-treat-insecure-origin-as-secure'"); } this.setupTransceiversAsync(useMic).finally(() => { var _a; (_a = this.peerConnection) === null || _a === void 0 ? void 0 : _a.createOffer(offerOptions).then((offer) => { var _a; this.showTextOverlayConnecting(); offer.sdp = this.mungeSDP(offer.sdp, useMic); (_a = this.peerConnection) === null || _a === void 0 ? void 0 : _a.setLocalDescription(offer); this.onSendWebRTCOffer(offer); }).catch(() => { this.showTextOverlaySetupFailure(); }); }); }); } /** * */ receiveOffer(offer, config) { var _a; return __awaiter(this, void 0, void 0, function* () { _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), 'Receive Offer', 6); (_a = this.peerConnection) === null || _a === void 0 ? void 0 : _a.setRemoteDescription(offer).then(() => { const isLocalhostConnection = location.hostname === 'localhost' || location.hostname === '127.0.0.1'; const isHttpsConnection = location.protocol === 'https:'; let useMic = config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.Flags.UseMic); if (useMic && !(isLocalhostConnection || isHttpsConnection)) { useMic = false; _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), 'Microphone access in the browser will not work if you are not on HTTPS or localhost. Disabling mic access.'); _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), "For testing you can enable HTTP microphone access Chrome by visiting chrome://flags/ and enabling 'unsafely-treat-insecure-origin-as-secure'"); } this.setupTransceiversAsync(useMic).finally(() => { var _a; (_a = this.peerConnection) === null || _a === void 0 ? void 0 : _a.createAnswer().then((Answer) => { var _a; Answer.sdp = this.mungeSDP(Answer.sdp, useMic); return (_a = this.peerConnection) === null || _a === void 0 ? void 0 : _a.setLocalDescription(Answer); }).then(() => { var _a; this.onSendWebRTCAnswer((_a = this.peerConnection) === null || _a === void 0 ? void 0 : _a.currentLocalDescription); }).catch(() => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), 'createAnswer() failed'); }); }); }); // Ugly syntax, but this achieves the intersection of the browser supported list and the UE supported list this.config.setOptionSettingOptions(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.OptionParameters.PreferredCodec, this.parseAvailableCodecs(offer).filter((value) => this.config .getSettingOption(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.OptionParameters.PreferredCodec) .options.includes(value))); }); } /** * Set the Remote Descriptor from the signaling server to the RTC Peer Connection * @param answer - RTC Session Descriptor from the Signaling Server */ receiveAnswer(answer) { var _a; (_a = this.peerConnection) === null || _a === void 0 ? void 0 : _a.setRemoteDescription(answer); // Ugly syntax, but this achieves the intersection of the browser supported list and the UE supported list this.config.setOptionSettingOptions(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.OptionParameters.PreferredCodec, this.parseAvailableCodecs(answer).filter((value) => this.config .getSettingOption(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.OptionParameters.PreferredCodec) .options.includes(value))); } /** * Generate Aggregated Stats and then fire a onVideo Stats event */ generateStats() { var _a; (_a = this.peerConnection) === null || _a === void 0 ? void 0 : _a.getStats(null).then((StatsData) => { this.aggregatedStats.processStats(StatsData); this.onVideoStats(this.aggregatedStats); // Update the preferred codec selection based on what was actually negotiated if (this.updateCodecSelection) { this.config.setOptionSettingValue(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.OptionParameters.PreferredCodec, this.aggregatedStats.codecs.get(this.aggregatedStats.inboundVideoStats.codecId)); } }); } /** * Close The Peer Connection */ close() { if (this.peerConnection) { this.peerConnection.close(); this.peerConnection = null; } } /** * Modify the Session Descriptor * @param sdp - Session Descriptor as a string * @param useMic - Is the microphone in use * @returns A modified Session Descriptor */ mungeSDP(sdp, useMic) { let mungedSDP = sdp.replace(/(a=fmtp:\d+ .*level-asymmetry-allowed=.*)\r\n/gm, '$1;x-google-start-bitrate=10000;x-google-max-bitrate=100000\r\n'); // set max bitrate to highest bitrate Opus supports let audioSDP = 'maxaveragebitrate=510000;'; if (useMic) { // set the max capture rate to 48khz (so we can send high quality audio from mic) audioSDP += 'sprop-maxcapturerate=48000;'; } // Force mono or stereo based on whether ?forceMono was passed or not audioSDP += this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.Flags.ForceMonoAudio) ? 'stereo=0;' : 'stereo=1;'; // enable in-band forward error correction for opus audio audioSDP += 'useinbandfec=1'; // We use the line 'useinbandfec=1' (which Opus uses) to set our Opus specific audio parameters. mungedSDP = mungedSDP.replace('useinbandfec=1', audioSDP); return mungedSDP; } /** * When a Ice Candidate is received add to the RTC Peer Connection * @param iceCandidate - RTC Ice Candidate from the Signaling Server */ handleOnIce(iceCandidate) { var _a; _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), 'peerconnection handleOnIce', 6); // // if forcing TURN, reject any candidates not relay if (this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.Flags.ForceTURN)) { // check if no relay address is found, if so, we are assuming it means no TURN server if (iceCandidate.candidate.indexOf('relay') < 0) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Info(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), `Dropping candidate because it was not TURN relay. | Type= ${iceCandidate.type} | Protocol= ${iceCandidate.protocol} | Address=${iceCandidate.address} | Port=${iceCandidate.port} |`, 6); return; } } (_a = this.peerConnection) === null || _a === void 0 ? void 0 : _a.addIceCandidate(iceCandidate); } /** * When the RTC Peer Connection Signaling server state Changes * @param state - Signaling Server State Change Event */ handleSignalStateChange(state) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), 'signaling state change: ' + state, 6); } /** * Handle when the Ice Connection State Changes * @param state - Ice Connection State */ handleIceConnectionStateChange(state) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), 'ice connection state change: ' + state, 6); this.onIceConnectionStateChange(state); } /** * Handle when the Ice Gathering State Changes * @param state - Ice Gathering State Change */ handleIceGatheringStateChange(state) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_2__.Logger.GetStackTrace(), 'ice gathering state change: ' + JSON.stringify(state), 6); } /** * Activates the onTrack method * @param event - The webRtc track event */ handleOnTrack(event) { this.onTrack(event); } /** * Activates the onPeerIceCandidate * @param event - The peer ice candidate */ handleIceCandidate(event) { this.onPeerIceCandidate(event); } /** * Activates the onDataChannel * @param event - The peer's data channel */ handleDataChannel(event) { this.onDataChannel(event); } /** * An override method for onTrack for use outside of the PeerConnectionController * @param trackEvent - The webRtc track event */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onTrack(trackEvent) { // Default Functionality: Do Nothing } /** * An override method for onIceConnectionStateChange for use outside of the PeerConnectionController * @param event - The webRtc iceconnectionstatechange event */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onIceConnectionStateChange(event) { // Default Functionality: Do Nothing } /** * An override method for onPeerIceCandidate for use outside of the PeerConnectionController * @param peerConnectionIceEvent - The peer ice candidate */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onPeerIceCandidate(peerConnectionIceEvent) { // Default Functionality: Do Nothing } /** * An override method for onDataChannel for use outside of the PeerConnectionController * @param datachannelEvent - The peer's data channel */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onDataChannel(datachannelEvent) { // Default Functionality: Do Nothing } /** * Setup tracks on the RTC Peer Connection * @param useMic - is mic in use */ setupTransceiversAsync(useMic) { var _a, _b, _c, _d, _e, _f, _g, _h, _j; return __awaiter(this, void 0, void 0, function* () { const hasTransceivers = ((_a = this.peerConnection) === null || _a === void 0 ? void 0 : _a.getTransceivers().length) > 0; // Setup a transceiver for getting UE video (_b = this.peerConnection) === null || _b === void 0 ? void 0 : _b.addTransceiver('video', { direction: 'recvonly' }); // We can only set preferrec codec on Chrome if (RTCRtpReceiver.getCapabilities && this.preferredCodec != '') { for (const transceiver of (_d = (_c = this.peerConnection) === null || _c === void 0 ? void 0 : _c.getTransceivers()) !== null && _d !== void 0 ? _d : []) { if (transceiver && transceiver.receiver && transceiver.receiver.track && transceiver.receiver.track.kind === 'video' && // As of 06/2023, FireFox has added RTCRtpReceiver.getCapabilities, but hasn't added the ability to set codec preferences transceiver.setCodecPreferences) { const preferredRTPCodec = this.preferredCodec.split(' '); const codecs = [ { mimeType: 'video/' + preferredRTPCodec[0] /* Name */, clockRate: 90000, sdpFmtpLine: preferredRTPCodec[1] /* sdpFmtpLine */ ? preferredRTPCodec[1] : '' } ]; this.config .getSettingOption(_Config_Config__WEBPACK_IMPORTED_MODULE_1__.OptionParameters.PreferredCodec) .options.filter((option) => { // Remove the preferred codec from the list of possible codecs as we've set it already return option != this.preferredCodec; }) .forEach((option) => { // Ammend the rest of the browsers supported codecs const altCodec = option.split(' '); codecs.push({ mimeType: 'video/' + altCodec[0] /* Name */, clockRate: 90000, sdpFmtpLine: altCodec[1] /* sdpFmtpLine */ ? altCodec[1] : '' }); }); for (const codec of codecs) { if (codec.sdpFmtpLine === '') { // We can't dynamically add members to the codec, so instead remove the field if it's empty delete codec.sdpFmtpLine; } } transceiver.setCodecPreferences(codecs); } } } // Setup a transceiver for sending mic audio to UE and receiving audio from UE if (!useMic) { (_e = this.peerConnection) === null || _e === void 0 ? void 0 : _e.addTransceiver('audio', { direction: 'recvonly' }); } else { // set the audio options based on mic usage const audioOptions = { autoGainControl: false, channelCount: 1, echoCancellation: false, latency: 0, noiseSuppression: false, sampleRate: 48000, sampleSize: 16, volume: 1.0 }; // set the media send options const mediaSendOptions = { video: false, audio: audioOptions }; // Note using mic on android chrome requires SSL or chrome://flags/ "unsafely-treat-insecure-origin-as-secure" const stream = yield navigator.mediaDevices.getUserMedia(mediaSendOptions); if (stream) { if (hasTransceivers) { for (const transceiver of (_g = (_f = this.peerConnection) === null || _f === void 0 ? void 0 : _f.getTransceivers()) !== null && _g !== void 0 ? _g : []) { if (_Util_RTCUtils__WEBPACK_IMPORTED_MODULE_4__.RTCUtils.canTransceiverReceiveAudio(transceiver)) { for (const track of stream.getTracks()) { if (track.kind && track.kind == 'audio') { transceiver.sender.replaceTrack(track); transceiver.direction = 'sendrecv'; } } } } } else { for (const track of stream.getTracks()) { if (track.kind && track.kind == 'audio') { (_h = this.peerConnection) === null || _h === void 0 ? void 0 : _h.addTransceiver(track, { direction: 'sendrecv' }); } } } } else { (_j = this.peerConnection) === null || _j === void 0 ? void 0 : _j.addTransceiver('audio', { direction: 'recvonly' }); } } }); } /** * And override event for when the video stats are fired * @param event - Aggregated Stats */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onVideoStats(event) { // Default Functionality: Do Nothing } /** * Event to send the RTC offer to the Signaling server * @param offer - RTC Offer */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onSendWebRTCOffer(offer) { // Default Functionality: Do Nothing } /** * Event to send the RTC Answer to the Signaling server * @param answer - RTC Answer */ // eslint-disable-next-line @typescript-eslint/no-unused-vars onSendWebRTCAnswer(answer) { // Default Functionality: Do Nothing } /** * An override for showing the Peer connection connecting Overlay */ showTextOverlayConnecting() { // Default Functionality: Do Nothing } /** * An override for showing the Peer connection Failed overlay */ showTextOverlaySetupFailure() { // Default Functionality: Do Nothing } parseAvailableCodecs(rtcSessionDescription) { // No point in updating the available codecs if on FF if (!RTCRtpReceiver.getCapabilities) return ['Only available on Chrome']; const ueSupportedCodecs = []; const sections = (0,sdp__WEBPACK_IMPORTED_MODULE_0__.splitSections)(rtcSessionDescription.sdp); // discard the session information as we only want media related info sections.shift(); sections.forEach((mediaSection) => { const { codecs } = (0,sdp__WEBPACK_IMPORTED_MODULE_0__.parseRtpParameters)(mediaSection); // Filter only for VPX / H26X / AV1 const matcher = /(VP\d|H26\d|AV1).*/; codecs.forEach((c) => { const str = c.name + ' ' + Object.keys(c.parameters || {}) .map((p) => p + '=' + c.parameters[p]) .join(';'); const match = matcher.exec(str); if (match !== null) { if (c.name == 'VP9') { // UE answers don't specify profile but we know we want profile 0 c.parameters = { 'profile-id': '0' }; } const codecStr = c.name + ' ' + Object.keys(c.parameters || {}) .map((p) => p + '=' + c.parameters[p]) .join(';'); ueSupportedCodecs.push(codecStr); } }); }); return ueSupportedCodecs; } } /***/ }), /***/ "./src/PeerConnectionController/SessionStats.ts": /*!******************************************************!*\ !*** ./src/PeerConnectionController/SessionStats.ts ***! \******************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SessionStats": () => (/* binding */ SessionStats) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Session statistics */ class SessionStats { } /***/ }), /***/ "./src/PeerConnectionController/StreamStats.ts": /*!*****************************************************!*\ !*** ./src/PeerConnectionController/StreamStats.ts ***! \*****************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "StreamStats": () => (/* binding */ StreamStats) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Class to hold the stream stats data coming in from webRtc */ class StreamStats { } /***/ }), /***/ "./src/PixelStreaming/PixelStreaming.ts": /*!**********************************************!*\ !*** ./src/PixelStreaming/PixelStreaming.ts ***! \**********************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "PixelStreaming": () => (/* binding */ PixelStreaming) /* harmony export */ }); /* harmony import */ var _Config_Config__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../Config/Config */ "./src/Config/Config.ts"); /* harmony import */ var _WebRtcPlayer_WebRtcPlayerController__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../WebRtcPlayer/WebRtcPlayerController */ "./src/WebRtcPlayer/WebRtcPlayerController.ts"); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _UI_OnScreenKeyboard__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../UI/OnScreenKeyboard */ "./src/UI/OnScreenKeyboard.ts"); /* harmony import */ var _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Util/EventEmitter */ "./src/Util/EventEmitter.ts"); /* harmony import */ var _WebXR_WebXRController__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../WebXR/WebXRController */ "./src/WebXR/WebXRController.ts"); /* harmony import */ var _UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ../UeInstanceMessage/StreamMessageController */ "./src/UeInstanceMessage/StreamMessageController.ts"); /* harmony import */ var _DataChannel_DataChannelLatencyTestController__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../DataChannel/DataChannelLatencyTestController */ "./src/DataChannel/DataChannelLatencyTestController.ts"); /* harmony import */ var _Util_RTCUtils__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ../Util/RTCUtils */ "./src/Util/RTCUtils.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * The key class for the browser side of a Pixel Streaming application, it includes: * WebRTC handling, XR support, input handling, and emitters for lifetime and state change events. * Users are encouraged to use this class as is, through composition, or extend it. In any case, * this will likely be the core of your Pixel Streaming experience in terms of functionality. */ class PixelStreaming { /** * @param config - A newly instantiated config object * @param overrides - Parameters to override default behaviour * returns the base Pixel streaming object */ constructor(config, overrides) { this.allowConsoleCommands = false; this.config = config; if (overrides === null || overrides === void 0 ? void 0 : overrides.videoElementParent) { this._videoElementParent = overrides.videoElementParent; } this._eventEmitter = new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.EventEmitter(); this.configureSettings(); // setup WebRTC this.setWebRtcPlayerController(new _WebRtcPlayer_WebRtcPlayerController__WEBPACK_IMPORTED_MODULE_1__.WebRtcPlayerController(this.config, this)); // Onscreen keyboard this.onScreenKeyboardHelper = new _UI_OnScreenKeyboard__WEBPACK_IMPORTED_MODULE_2__.OnScreenKeyboard(this.videoElementParent); this.onScreenKeyboardHelper.unquantizeAndDenormalizeUnsigned = (x, y) => this._webRtcController.requestUnquantizedAndDenormalizeUnsigned(x, y); this._activateOnScreenKeyboard = (command) => this.onScreenKeyboardHelper.showOnScreenKeyboard(command); this._webXrController = new _WebXR_WebXRController__WEBPACK_IMPORTED_MODULE_3__.WebXRController(this._webRtcController); } /** * Gets the element that contains the video stream element. */ get videoElementParent() { if (!this._videoElementParent) { this._videoElementParent = document.createElement('div'); this._videoElementParent.id = 'videoElementParent'; } return this._videoElementParent; } /** * Configure the settings with on change listeners and any additional per experience settings. */ configureSettings() { this.config._addOnSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.Flags.IsQualityController, (wantsQualityController) => { // If the setting has been set to true (either programatically or the user has flicked the toggle) // and we aren't currently quality controller, send the request if (wantsQualityController === true && !this._webRtcController.isQualityController) { this._webRtcController.sendRequestQualityControlOwnership(); } }); this.config._addOnSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.Flags.AFKDetection, (isAFKEnabled) => { this._webRtcController.setAfkEnabled(isAFKEnabled); }); this.config._addOnSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.Flags.MatchViewportResolution, () => { this._webRtcController.videoPlayer.updateVideoStreamSize(); }); this.config._addOnSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.Flags.HoveringMouseMode, (isHoveringMouse) => { this.config.setFlagLabel(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.Flags.HoveringMouseMode, `Control Scheme: ${isHoveringMouse ? 'Hovering' : 'Locked'} Mouse`); this._webRtcController.setMouseInputEnabled(this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.Flags.MouseInput)); }); // user input this.config._addOnSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.Flags.KeyboardInput, (isEnabled) => { this._webRtcController.setKeyboardInputEnabled(isEnabled); }); this.config._addOnSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.Flags.MouseInput, (isEnabled) => { this._webRtcController.setMouseInputEnabled(isEnabled); }); this.config._addOnSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.Flags.TouchInput, (isEnabled) => { this._webRtcController.setTouchInputEnabled(isEnabled); }); this.config._addOnSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.Flags.GamepadInput, (isEnabled) => { this._webRtcController.setGamePadInputEnabled(isEnabled); }); // encoder settings this.config._addOnNumericSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.MinQP, (newValue) => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), '-------- Sending MinQP --------', 7); this._webRtcController.sendEncoderMinQP(newValue); _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), '-------------------------------------------', 7); }); this.config._addOnNumericSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.MaxQP, (newValue) => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), '-------- Sending encoder settings --------', 7); this._webRtcController.sendEncoderMaxQP(newValue); _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), '-------------------------------------------', 7); }); // WebRTC settings this.config._addOnNumericSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.WebRTCMinBitrate, (newValue) => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), '-------- Sending web rtc settings --------', 7); this._webRtcController.sendWebRTCMinBitrate(newValue * 1000 /* kbps to bps */); _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), '-------------------------------------------', 7); }); this.config._addOnNumericSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.WebRTCMaxBitrate, (newValue) => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), '-------- Sending web rtc settings --------', 7); this._webRtcController.sendWebRTCMaxBitrate(newValue * 1000 /* kbps to bps */); _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), '-------------------------------------------', 7); }); this.config._addOnNumericSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.WebRTCFPS, (newValue) => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), '-------- Sending web rtc settings --------', 7); this._webRtcController.sendWebRTCFps(newValue); _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), '-------------------------------------------', 7); }); this.config._addOnOptionSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.OptionParameters.PreferredCodec, (newValue) => { if (this._webRtcController) { this._webRtcController.setPreferredCodec(newValue); } }); this.config._registerOnChangeEvents(this._eventEmitter); } /** * Activate the on screen keyboard when receiving the command from the streamer * @param command - the keyboard command */ // eslint-disable-next-line @typescript-eslint/no-unused-vars _activateOnScreenKeyboard(command) { throw new Error('Method not implemented.'); } /** * Set the input control ownership * @param inputControlOwnership - does the user have input control ownership */ _onInputControlOwnership(inputControlOwnership) { this._inputController = inputControlOwnership; } /** * Instantiate the WebRTCPlayerController interface to provide WebRTCPlayerController functionality within this class and set up anything that requires it * @param webRtcPlayerController - a WebRtcPlayerController controller instance */ setWebRtcPlayerController(webRtcPlayerController) { this._webRtcController = webRtcPlayerController; this._webRtcController.setPreferredCodec(this.config.getSettingOption(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.OptionParameters.PreferredCodec) .selected); this._webRtcController.resizePlayerStyle(); // connect if auto connect flag is enabled this.checkForAutoConnect(); } /** * Connect to signaling server. */ connect() { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.StreamPreConnectEvent()); this._webRtcController.connectToSignallingServer(); } /** * Reconnects to the signaling server. If connection is up, disconnects first * before establishing a new connection */ reconnect() { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.StreamReconnectEvent()); this._webRtcController.tryReconnect("Reconnecting..."); } /** * Disconnect from the signaling server and close open peer connections. */ disconnect() { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.StreamPreDisconnectEvent()); this._webRtcController.close(); } /** * Play the stream. Can be called only after a peer connection has been established. */ play() { this._onStreamLoading(); this._webRtcController.playStream(); } /** * Auto connect if AutoConnect flag is enabled */ checkForAutoConnect() { // set up if the auto play will be used or regular click to start if (this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.Flags.AutoConnect)) { // if autoplaying show an info overlay while while waiting for the connection to begin this._onWebRtcAutoConnect(); this._webRtcController.connectToSignallingServer(); } } /** * Will unmute the microphone track which is sent to Unreal Engine. * By default, will only unmute an existing mic track. * * @param forceEnable Can be used for cases when this object wasn't initialized with a mic track. * If this parameter is true, the connection will be restarted with a microphone. * Warning: this takes some time, as a full renegotiation and reconnection will happen. */ unmuteMicrophone(forceEnable = false) { // If there's an existing mic track, we just set muted state if (this.config.isFlagEnabled('UseMic')) { this.setMicrophoneMuted(false); return; } // If there's no pre-existing mic track, and caller is ok with full reset, we enable and reset if (forceEnable) { this.config.setFlagEnabled("UseMic", true); this.reconnect(); return; } // If we prefer not to force a reconnection, just warn the user that this operation didn't happen _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Warning(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), 'Trying to unmute mic, but PixelStreaming was initialized with no microphone track. Call with forceEnable == true to re-connect with a mic track.'); } muteMicrophone() { if (this.config.isFlagEnabled('UseMic')) { this.setMicrophoneMuted(true); return; } // If there wasn't a mic track, just let user know there's nothing to mute _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Info(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), 'Trying to mute mic, but PixelStreaming has no microphone track, so sending sound is already disabled.'); } setMicrophoneMuted(mute) { var _a, _b, _c, _d; for (const transceiver of (_d = (_c = (_b = (_a = this._webRtcController) === null || _a === void 0 ? void 0 : _a.peerConnectionController) === null || _b === void 0 ? void 0 : _b.peerConnection) === null || _c === void 0 ? void 0 : _c.getTransceivers()) !== null && _d !== void 0 ? _d : []) { if (_Util_RTCUtils__WEBPACK_IMPORTED_MODULE_6__.RTCUtils.canTransceiverSendAudio(transceiver)) { transceiver.sender.track.enabled = !mute; } } } /** * Emit an event on auto connecting */ _onWebRtcAutoConnect() { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.WebRtcAutoConnectEvent()); } /** * Set up functionality to happen when receiving a webRTC answer */ _onWebRtcSdp() { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.WebRtcSdpEvent()); } /** * Emits a StreamLoading event */ _onStreamLoading() { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.StreamLoadingEvent()); } /** * Event fired when the video is disconnected - emits given eventString or an override * message from webRtcController if one has been set * @param eventString - a string describing why the connection closed * @param allowClickToReconnect - true if we want to allow the user to retry the connection with a click */ _onDisconnect(eventString, allowClickToReconnect) { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.WebRtcDisconnectedEvent({ eventString: eventString, allowClickToReconnect: allowClickToReconnect })); } /** * Handles when Web Rtc is connecting */ _onWebRtcConnecting() { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.WebRtcConnectingEvent()); } /** * Handles when Web Rtc has connected */ _onWebRtcConnected() { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.WebRtcConnectedEvent()); } /** * Handles when Web Rtc fails to connect */ _onWebRtcFailed() { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.WebRtcFailedEvent()); } /** * Handle when the Video has been Initialized */ _onVideoInitialized() { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.VideoInitializedEvent()); this._videoStartTime = Date.now(); } /** * Set up functionality to happen when receiving latency test results * @param latency - latency test results object */ _onLatencyTestResult(latencyTimings) { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.LatencyTestResultEvent({ latencyTimings })); } _onDataChannelLatencyTestResponse(response) { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.DataChannelLatencyTestResponseEvent({ response })); } /** * Set up functionality to happen when receiving video statistics * @param videoStats - video statistics as a aggregate stats object */ _onVideoStats(videoStats) { // Duration if (!this._videoStartTime || this._videoStartTime === undefined) { this._videoStartTime = Date.now(); } videoStats.handleSessionStatistics(this._videoStartTime, this._inputController, this._webRtcController.videoAvgQp); this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.StatsReceivedEvent({ aggregatedStats: videoStats })); } /** * Set up functionality to happen when calculating the average video encoder qp * @param QP - the quality number of the stream */ _onVideoEncoderAvgQP(QP) { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.VideoEncoderAvgQPEvent({ avgQP: QP })); } /** * Set up functionality to happen when receiving and handling initial settings for the UE app * @param settings - initial UE app settings */ _onInitialSettings(settings) { var _a; this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.InitialSettingsEvent({ settings })); if (settings.PixelStreamingSettings) { this.allowConsoleCommands = (_a = settings.PixelStreamingSettings.AllowPixelStreamingCommands) !== null && _a !== void 0 ? _a : false; if (this.allowConsoleCommands === false) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Info(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), '-AllowPixelStreamingCommands=false, sending arbitrary console commands from browser to UE is disabled.'); } } const useUrlParams = this.config.useUrlParams; const urlParams = new URLSearchParams(window.location.search); if (settings.EncoderSettings) { this.config.setNumericSetting(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.MinQP, // If a setting is set in the URL, make sure we respect that value as opposed to what the application sends us (useUrlParams && urlParams.has(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.MinQP)) ? Number.parseInt(urlParams.get(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.MinQP)) : settings.EncoderSettings.MinQP); this.config.setNumericSetting(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.MaxQP, (useUrlParams && urlParams.has(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.MaxQP)) ? Number.parseInt(urlParams.get(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.MaxQP)) : settings.EncoderSettings.MaxQP); } if (settings.WebRTCSettings) { this.config.setNumericSetting(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.WebRTCMinBitrate, (useUrlParams && urlParams.has(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.WebRTCMinBitrate)) ? Number.parseInt(urlParams.get(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.WebRTCMinBitrate)) : settings.WebRTCSettings.MinBitrate / 1000 /* bps to kbps */); this.config.setNumericSetting(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.WebRTCMaxBitrate, (useUrlParams && urlParams.has(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.WebRTCMaxBitrate)) ? Number.parseInt(urlParams.get(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.WebRTCMaxBitrate)) : settings.WebRTCSettings.MaxBitrate / 1000 /* bps to kbps */); this.config.setNumericSetting(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.WebRTCFPS, (useUrlParams && urlParams.has(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.WebRTCFPS)) ? Number.parseInt(urlParams.get(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.NumericParameters.WebRTCFPS)) : settings.WebRTCSettings.FPS); } } /** * Set up functionality to happen when setting quality control ownership of a stream * @param hasQualityOwnership - does this user have quality ownership of the stream true / false */ _onQualityControlOwnership(hasQualityOwnership) { this.config.setFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_4__.Flags.IsQualityController, hasQualityOwnership); } _onPlayerCount(playerCount) { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.PlayerCountEvent({ count: playerCount })); } /** * Request a connection latency test. * NOTE: There are plans to refactor all request* functions. Expect changes if you use this! * @returns */ requestLatencyTest() { if (!this._webRtcController.videoPlayer.isVideoReady()) { return false; } this._webRtcController.sendLatencyTest(); return true; } /** * Request a data channel latency test. * NOTE: There are plans to refactor all request* functions. Expect changes if you use this! */ requestDataChannelLatencyTest(config) { if (!this._webRtcController.videoPlayer.isVideoReady()) { return false; } if (!this._dataChannelLatencyTestController) { this._dataChannelLatencyTestController = new _DataChannel_DataChannelLatencyTestController__WEBPACK_IMPORTED_MODULE_7__.DataChannelLatencyTestController(this._webRtcController.sendDataChannelLatencyTest.bind(this._webRtcController), (result) => { this._eventEmitter.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_0__.DataChannelLatencyTestResultEvent({ result })); }); this.addEventListener("dataChannelLatencyTestResponse", ({ data: { response } }) => { this._dataChannelLatencyTestController.receive(response); }); } return this._dataChannelLatencyTestController.start(config); } /** * Request for the UE application to show FPS counter. * NOTE: There are plans to refactor all request* functions. Expect changes if you use this! * @returns */ requestShowFps() { if (!this._webRtcController.videoPlayer.isVideoReady()) { return false; } this._webRtcController.sendShowFps(); return true; } /** * Request for a new IFrame from the UE application. * NOTE: There are plans to refactor all request* functions. Expect changes if you use this! * @returns */ requestIframe() { if (!this._webRtcController.videoPlayer.isVideoReady()) { return false; } this._webRtcController.sendIframeRequest(); return true; } /** * Send data to UE application. The data will be run through JSON.stringify() so e.g. strings * and any serializable plain JSON objects with no recurrence can be sent. * @returns true if succeeded, false if rejected */ emitUIInteraction(descriptor) { if (!this._webRtcController.videoPlayer.isVideoReady()) { return false; } this._webRtcController.emitUIInteraction(descriptor); return true; } /** * Send a command to UE application. Blocks ConsoleCommand descriptors unless UE * has signaled that it allows console commands. * @returns true if succeeded, false if rejected */ emitCommand(descriptor) { if (!this._webRtcController.videoPlayer.isVideoReady()) { return false; } if (!this.allowConsoleCommands && 'ConsoleCommand' in descriptor) { return false; } this._webRtcController.emitCommand(descriptor); return true; } /** * Send a console command to UE application. Only allowed if UE has signaled that it allows * console commands. * @returns true if succeeded, false if rejected */ emitConsoleCommand(command) { if (!this.allowConsoleCommands || !this._webRtcController.videoPlayer.isVideoReady()) { return false; } this._webRtcController.emitConsoleCommand(command); return true; } /** * Add a UE -> browser response event listener * @param name - The name of the response handler * @param listener - The method to be activated when a message is received */ addResponseEventListener(name, listener) { this._webRtcController.responseController.addResponseEventListener(name, listener); } /** * Remove a UE -> browser response event listener * @param name - The name of the response handler */ removeResponseEventListener(name) { this._webRtcController.responseController.removeResponseEventListener(name); } /** * Dispatch a new event. * @param e event * @returns */ dispatchEvent(e) { return this._eventEmitter.dispatchEvent(e); } /** * Register an event handler. * @param type event name * @param listener event handler function */ addEventListener(type, listener) { this._eventEmitter.addEventListener(type, listener); } /** * Remove an event handler. * @param type event name * @param listener event handler function */ removeEventListener(type, listener) { this._eventEmitter.removeEventListener(type, listener); } /** * Enable/disable XR mode. */ toggleXR() { this.webXrController.xrClicked(); } /** * Pass in a function to generate a signalling server URL. * This function is useful if you need to programmatically construct your signalling server URL. * @param signallingUrlBuilderFunc A function that generates a signalling server url. */ setSignallingUrlBuilder(signallingUrlBuilderFunc) { this._webRtcController.signallingUrlBuilder = signallingUrlBuilderFunc; } /** * Public getter for the websocket controller. Access to this property allows you to send * custom websocket messages. */ get webSocketController() { return this._webRtcController.webSocketController; } /** * Public getter for the webXrController controller. Used for all XR features. */ get webXrController() { return this._webXrController; } registerMessageHandler(name, direction, handler) { if (direction === _UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_8__.MessageDirection.FromStreamer && typeof handler === 'undefined') { _Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.Warning(_Logger_Logger__WEBPACK_IMPORTED_MODULE_5__.Logger.GetStackTrace(), `Unable to register an undefined handler for ${name}`); return; } if (direction === _UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_8__.MessageDirection.ToStreamer && typeof handler === 'undefined') { this._webRtcController.streamMessageController.registerMessageHandler(direction, name, (data) => this._webRtcController.sendMessageController.sendMessageToStreamer(name, data)); } else { this._webRtcController.streamMessageController.registerMessageHandler(direction, name, (data) => handler(data)); } } get toStreamerHandlers() { return this._webRtcController.streamMessageController.toStreamerHandlers; } isReconnecting() { return this._webRtcController.isReconnecting; } } /***/ }), /***/ "./src/UI/OnScreenKeyboard.ts": /*!************************************!*\ !*** ./src/UI/OnScreenKeyboard.ts ***! \************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "OnScreenKeyboard": () => (/* binding */ OnScreenKeyboard) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Class for handling on screen keyboard usage */ class OnScreenKeyboard { /** * * @param videoElementParent The div element the video player is injected into */ constructor(videoElementParent) { this.editTextButton = null; this.hiddenInput = null; if ('ontouchstart' in document.documentElement) { this.createOnScreenKeyboardHelpers(videoElementParent); } } /** * An override for unquantizeAndDenormalizeUnsigned * @param x the x axis point * @param y the y axis point * @returns unquantizeAndDenormalizeUnsigned object */ // eslint-disable-next-line @typescript-eslint/no-unused-vars unquantizeAndDenormalizeUnsigned(x, y) { return null; } /** * Creates on screen keyboard helpers * @param videoElementParent The div element the video player i injected into */ createOnScreenKeyboardHelpers(videoElementParent) { if (!this.hiddenInput) { this.hiddenInput = document.createElement('input'); this.hiddenInput.id = 'hiddenInput'; this.hiddenInput.maxLength = 0; videoElementParent.appendChild(this.hiddenInput); } if (!this.editTextButton) { this.editTextButton = document.createElement('button'); this.editTextButton.id = 'editTextButton'; this.editTextButton.innerHTML = 'edit text'; videoElementParent.appendChild(this.editTextButton); // Hide the 'edit text' button. this.editTextButton.classList.add('hiddenState'); this.editTextButton.addEventListener('touchend', (event) => { // Show the on-screen keyboard. this.hiddenInput.focus(); event.preventDefault(); }); } } /** * Shows the on screen keyboard * @param command the command received via the data channel containing keyboard positions */ showOnScreenKeyboard(command) { if (command.showOnScreenKeyboard) { // Show the 'edit text' button. this.editTextButton.classList.remove('hiddenState'); // Place the 'edit text' button near the UE input widget. const pos = this.unquantizeAndDenormalizeUnsigned(command.x, command.y); this.editTextButton.style.top = pos.y.toString() + 'px'; this.editTextButton.style.left = (pos.x - 40).toString() + 'px'; } else { // Hide the 'edit text' button. this.editTextButton.classList.add('hiddenState'); // Hide the on-screen keyboard. this.hiddenInput.blur(); } } } /***/ }), /***/ "./src/UeInstanceMessage/ResponseController.ts": /*!*****************************************************!*\ !*** ./src/UeInstanceMessage/ResponseController.ts ***! \*****************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "ResponseController": () => (/* binding */ ResponseController) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. class ResponseController { constructor() { this.responseEventListeners = new Map(); } /** * Add a response event listener to the response map * @param name - The name of the response * @param listener - The method to be activated when the response is selected */ addResponseEventListener(name, listener) { this.responseEventListeners.set(name, listener); } /** * Remove a response event listener to the response map * @param name - The name of the response */ removeResponseEventListener(name) { this.responseEventListeners.delete(name); } /** * Handle a response when receiving one form the streamer * @param message - Data received from the data channel with the command in question */ onResponse(message) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'DataChannelReceiveMessageType.Response', 6); const responses = new TextDecoder('utf-16').decode(message.slice(1)); _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), responses, 6); this.responseEventListeners.forEach((listener) => { listener(responses); }); } } /***/ }), /***/ "./src/UeInstanceMessage/SendMessageController.ts": /*!********************************************************!*\ !*** ./src/UeInstanceMessage/SendMessageController.ts ***! \********************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SendMessageController": () => (/* binding */ SendMessageController) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. class SendMessageController { /** * @param dataChannelSender - Data channel instance * @param toStreamerMessagesMapProvider - Stream Messages instance */ constructor(dataChannelSender, toStreamerMessagesMapProvider) { this.dataChannelSender = dataChannelSender; this.toStreamerMessagesMapProvider = toStreamerMessagesMapProvider; } /** * Send a message to the streamer through the data channel * @param messageType - the type of message we are sending * @param messageData - the message data we are sending over the data channel * @returns - nil */ sendMessageToStreamer(messageType, messageData) { if (messageData === undefined) { messageData = []; } const toStreamerMessages = this.toStreamerMessagesMapProvider.toStreamerMessages; const messageFormat = toStreamerMessages.get(messageType); if (messageFormat === undefined) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Attempted to send a message to the streamer with message type: ${messageType}, but the frontend hasn't been configured to send such a message. Check you've added the message type in your cpp`); return; } if (messageFormat.structure && messageData && messageFormat.structure.length !== messageData.length) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Provided message data doesn't match expected layout. Expected [ ${messageFormat.structure.map((element) => { switch (element) { case 'uint8': case 'uint16': case 'int16': case 'float': case 'double': return 'number'; case 'string': return 'string'; } }).toString()} ] but received [ ${messageData.map((element) => typeof element).toString()} ]`); return; } let byteLength = 0; const textEncoder = new TextEncoder(); // One loop to calculate the length in bytes of all of the provided data messageData.forEach((element, idx) => { const type = messageFormat.structure[idx]; switch (type) { case 'uint8': byteLength += 1; break; case 'uint16': byteLength += 2; break; case 'int16': byteLength += 2; break; case 'float': byteLength += 4; break; case 'double': byteLength += 8; break; case 'string': // 2 bytes for string length byteLength += 2; // 2 bytes per characters byteLength += 2 * textEncoder.encode(element).length; break; } }); const data = new DataView(new ArrayBuffer(byteLength + 1)); data.setUint8(0, messageFormat.id); let byteOffset = 1; messageData.forEach((element, idx) => { const type = messageFormat.structure[idx]; switch (type) { case 'uint8': data.setUint8(byteOffset, element); byteOffset += 1; break; case 'uint16': data.setUint16(byteOffset, element, true); byteOffset += 2; break; case 'int16': data.setInt16(byteOffset, element, true); byteOffset += 2; break; case 'float': data.setFloat32(byteOffset, element, true); byteOffset += 4; break; case 'double': data.setFloat64(byteOffset, element, true); byteOffset += 8; break; case 'string': data.setUint16(byteOffset, element.length, true); byteOffset += 2; for (let i = 0; i < element.length; i++) { data.setUint16(byteOffset, element.charCodeAt(i), true); byteOffset += 2; } break; } }); if (!this.dataChannelSender.canSend()) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Info(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Data channel cannot send yet, skipping sending message: ${messageType} - ${new Uint8Array(data.buffer)}`); return; } this.dataChannelSender.sendData(data.buffer); } } /***/ }), /***/ "./src/UeInstanceMessage/StreamMessageController.ts": /*!**********************************************************!*\ !*** ./src/UeInstanceMessage/StreamMessageController.ts ***! \**********************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "MessageDirection": () => (/* binding */ MessageDirection), /* harmony export */ "StreamMessageController": () => (/* binding */ StreamMessageController), /* harmony export */ "ToStreamerMessage": () => (/* binding */ ToStreamerMessage) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. class ToStreamerMessage { } class StreamMessageController { constructor() { this.toStreamerHandlers = new Map(); this.fromStreamerHandlers = new Map(); this.toStreamerMessages = new Map(); this.fromStreamerMessages = new Map(); } /** * Populate the Default message protocol */ populateDefaultProtocol() { /* * Control Messages. Range = 0..49. */ this.toStreamerMessages.set('IFrameRequest', { id: 0, structure: [] }); this.toStreamerMessages.set('RequestQualityControl', { id: 1, structure: [] }); this.toStreamerMessages.set('FpsRequest', { id: 2, structure: [] }); this.toStreamerMessages.set('AverageBitrateRequest', { id: 3, structure: [] }); this.toStreamerMessages.set('StartStreaming', { id: 4, structure: [] }); this.toStreamerMessages.set('StopStreaming', { id: 5, structure: [] }); this.toStreamerMessages.set('LatencyTest', { id: 6, structure: ['string'] }); this.toStreamerMessages.set('RequestInitialSettings', { id: 7, structure: [] }); this.toStreamerMessages.set('TestEcho', { id: 8, structure: [] }); this.toStreamerMessages.set('DataChannelLatencyTest', { id: 9, structure: [] }); /* * Input Messages. Range = 50..89. */ // Generic Input Messages. Range = 50..59. this.toStreamerMessages.set('UIInteraction', { id: 50, structure: ['string'] }); this.toStreamerMessages.set('Command', { id: 51, structure: ['string'] }); // Keyboard Input Message. Range = 60..69. this.toStreamerMessages.set('KeyDown', { id: 60, // keyCode isRepeat structure: ['uint8', 'uint8'] }); this.toStreamerMessages.set('KeyUp', { id: 61, // keyCode structure: ['uint8'] }); this.toStreamerMessages.set('KeyPress', { id: 62, // charcode structure: ['uint16'] }); // Mouse Input Messages. Range = 70..79. this.toStreamerMessages.set('MouseEnter', { id: 70, structure: [] }); this.toStreamerMessages.set('MouseLeave', { id: 71, structure: [] }); this.toStreamerMessages.set('MouseDown', { id: 72, // button x y structure: ['uint8', 'uint16', 'uint16'] }); this.toStreamerMessages.set('MouseUp', { id: 73, // button x y structure: ['uint8', 'uint16', 'uint16'] }); this.toStreamerMessages.set('MouseMove', { id: 74, // x y deltaX deltaY structure: ['uint16', 'uint16', 'int16', 'int16'] }); this.toStreamerMessages.set('MouseWheel', { id: 75, // delta x y structure: ['int16', 'uint16', 'uint16'] }); this.toStreamerMessages.set('MouseDouble', { id: 76, // button x y structure: ['uint8', 'uint16', 'uint16'] }); // Touch Input Messages. Range = 80..89. this.toStreamerMessages.set('TouchStart', { id: 80, // numtouches(1) x y idx force valid structure: ['uint8', 'uint16', 'uint16', 'uint8', 'uint8', 'uint8'] }); this.toStreamerMessages.set('TouchEnd', { id: 81, // numtouches(1) x y idx force valid structure: ['uint8', 'uint16', 'uint16', 'uint8', 'uint8', 'uint8'] }); this.toStreamerMessages.set('TouchMove', { id: 82, // numtouches(1) x y idx force valid structure: ['uint8', 'uint16', 'uint16', 'uint8', 'uint8', 'uint8'] }); // Gamepad Input Messages. Range = 90..99 this.toStreamerMessages.set('GamepadConnected', { id: 93, structure: [] }); this.toStreamerMessages.set('GamepadButtonPressed', { id: 90, // ctrlerId button isRepeat structure: ['uint8', 'uint8', 'uint8'] }); this.toStreamerMessages.set('GamepadButtonReleased', { id: 91, // ctrlerId button isRepeat(0) structure: ['uint8', 'uint8', 'uint8'] }); this.toStreamerMessages.set('GamepadAnalog', { id: 92, // ctrlerId button analogValue structure: ['uint8', 'uint8', 'double'] }); this.toStreamerMessages.set('GamepadDisconnected', { id: 94, // ctrlerId structure: ['uint8'] }); this.fromStreamerMessages.set(0, 'QualityControlOwnership'); this.fromStreamerMessages.set(1, 'Response'); this.fromStreamerMessages.set(2, 'Command'); this.fromStreamerMessages.set(3, 'FreezeFrame'); this.fromStreamerMessages.set(4, 'UnfreezeFrame'); this.fromStreamerMessages.set(5, 'VideoEncoderAvgQP'); this.fromStreamerMessages.set(6, 'LatencyTest'); this.fromStreamerMessages.set(7, 'InitialSettings'); this.fromStreamerMessages.set(8, 'FileExtension'); this.fromStreamerMessages.set(9, 'FileMimeType'); this.fromStreamerMessages.set(10, 'FileContents'); this.fromStreamerMessages.set(11, 'TestEcho'); this.fromStreamerMessages.set(12, 'InputControlOwnership'); this.fromStreamerMessages.set(13, 'GamepadResponse'); this.fromStreamerMessages.set(14, 'DataChannelLatencyTest'); this.fromStreamerMessages.set(255, 'Protocol'); } /** * Register a message handler * @param messageDirection - the direction of the message; toStreamer or fromStreamer * @param messageType - the type of the message * @param messageHandler - the function or method to be executed when this handler is called */ registerMessageHandler(messageDirection, messageType, messageHandler) { switch (messageDirection) { case MessageDirection.ToStreamer: this.toStreamerHandlers.set(messageType, messageHandler); break; case MessageDirection.FromStreamer: this.fromStreamerHandlers.set(messageType, messageHandler); break; default: _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Unknown message direction ${messageDirection}`); } } } /** * The enum for message directions */ var MessageDirection; (function (MessageDirection) { MessageDirection[MessageDirection["ToStreamer"] = 0] = "ToStreamer"; MessageDirection[MessageDirection["FromStreamer"] = 1] = "FromStreamer"; })(MessageDirection || (MessageDirection = {})); /***/ }), /***/ "./src/UeInstanceMessage/ToStreamerMessagesController.ts": /*!***************************************************************!*\ !*** ./src/UeInstanceMessage/ToStreamerMessagesController.ts ***! \***************************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "ToStreamerMessagesController": () => (/* binding */ ToStreamerMessagesController) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. class ToStreamerMessagesController { /** * @param sendMessageController - Stream message controller instance */ constructor(sendMessageController) { this.sendMessageController = sendMessageController; } /** * Send Request to Take Quality Control to the UE Instance */ SendRequestQualityControl() { this.sendMessageController.sendMessageToStreamer('RequestQualityControl'); } /** * Send Max FPS Request to the UE Instance */ SendMaxFpsRequest() { this.sendMessageController.sendMessageToStreamer('FpsRequest'); } /** * Send Average Bitrate Request to the UE Instance */ SendAverageBitrateRequest() { this.sendMessageController.sendMessageToStreamer('AverageBitrateRequest'); } /** * Send a Start Streaming Message to the UE Instance */ SendStartStreaming() { this.sendMessageController.sendMessageToStreamer('StartStreaming'); } /** * Send a Stop Streaming Message to the UE Instance */ SendStopStreaming() { this.sendMessageController.sendMessageToStreamer('StopStreaming'); } /** * Send a Request Initial Settings to the UE Instance */ SendRequestInitialSettings() { this.sendMessageController.sendMessageToStreamer('RequestInitialSettings'); } } /***/ }), /***/ "./src/Util/CoordinateConverter.ts": /*!*****************************************!*\ !*** ./src/Util/CoordinateConverter.ts ***! \*****************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "CoordinateConverter": () => (/* binding */ CoordinateConverter), /* harmony export */ "NormalizedQuantizedSignedCoord": () => (/* binding */ NormalizedQuantizedSignedCoord), /* harmony export */ "NormalizedQuantizedUnsignedCoord": () => (/* binding */ NormalizedQuantizedUnsignedCoord), /* harmony export */ "UnquantizedDenormalizedUnsignedCoord": () => (/* binding */ UnquantizedDenormalizedUnsignedCoord) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Converts coordinates from element relative coordinates to values normalized within the value range of a short (and back again) */ class CoordinateConverter { /** * @param videoElementProvider - the div element that the video player will be injected into */ constructor(videoElementProvider) { this.videoElementProvider = videoElementProvider; 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.'); }; } /** * The surface method for setterNormalizeAndQuantizeUnsigned * @param x - x axis point * @param y - y axis point */ normalizeAndQuantizeUnsigned(x, y) { return this.normalizeAndQuantizeUnsignedFunc(x, y); } /** * The surface method for setterUnquantizeAndDenormalizeUnsigned * @param x - x axis point * @param y - y axis point */ unquantizeAndDenormalizeUnsigned(x, y) { return this.denormalizeAndUnquantizeUnsignedFunc(x, y); } /** * The surface method for setterNormalizeAndQuantizeSigned * @param x - x axis point * @param y - y axis point */ normalizeAndQuantizeSigned(x, y) { return this.normalizeAndQuantizeSignedFunc(x, y); } /** * set up the Normalize And Quantize methods based on the aspect ratio and the video player ratio */ setupNormalizeAndQuantize() { this.videoElementParent = this.videoElementProvider.getVideoParentElement(); this.videoElement = this.videoElementProvider.getVideoElement(); if (this.videoElementParent && this.videoElement) { const playerAspectRatio = this.videoElementParent.clientHeight / this.videoElementParent.clientWidth; const videoAspectRatio = this.videoElement.videoHeight / this.videoElement.videoWidth; if (playerAspectRatio > videoAspectRatio) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Setup Normalize and Quantize for playerAspectRatio > videoAspectRatio', 6); this.ratio = playerAspectRatio / videoAspectRatio; this.normalizeAndQuantizeUnsignedFunc = (x, y) => this.normalizeAndQuantizeUnsignedPlayerBigger(x, y); this.normalizeAndQuantizeSignedFunc = (x, y) => this.normalizeAndQuantizeSignedPlayerBigger(x, y); this.denormalizeAndUnquantizeUnsignedFunc = (x, y) => this.denormalizeAndUnquantizeUnsignedPlayerBigger(x, y); } else { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Setup Normalize and Quantize for playerAspectRatio <= videoAspectRatio', 6); this.ratio = videoAspectRatio / playerAspectRatio; this.normalizeAndQuantizeUnsignedFunc = (x, y) => this.normalizeAndQuantizeUnsignedPlayerSmaller(x, y); this.normalizeAndQuantizeSignedFunc = (x, y) => this.normalizeAndQuantizeSignedPlayerSmaller(x, y); this.denormalizeAndUnquantizeUnsignedFunc = (x, y) => this.denormalizeAndUnquantizeUnsignedPlayerSmaller(x, y); } } } /** * normalizeAndQuantizeUnsigned for playerAspectRatio > videoAspectRatio * @param x - x axis point * @param y - y axis point */ normalizeAndQuantizeUnsignedPlayerBigger(x, y) { const normalizedX = x / this.videoElementParent.clientWidth; const normalizedY = this.ratio * (y / this.videoElementParent.clientHeight - 0.5) + 0.5; if (normalizedX < 0.0 || normalizedX > 1.0 || normalizedY < 0.0 || normalizedY > 1.0) { return new NormalizedQuantizedUnsignedCoord(false, 65535, 65535); } else { return new NormalizedQuantizedUnsignedCoord(true, normalizedX * 65536, normalizedY * 65536); } } /** * unquantizeAndDenormalizeUnsigned for playerAspectRatio > videoAspectRatio * @param x - x axis point * @param y - y axis point */ denormalizeAndUnquantizeUnsignedPlayerBigger(x, y) { const normalizedX = x / 65536; const normalizedY = (y / 65536 - 0.5) / this.ratio + 0.5; return new UnquantizedDenormalizedUnsignedCoord(normalizedX * this.videoElementParent.clientWidth, normalizedY * this.videoElementParent.clientHeight); } /** * normalizeAndQuantizeSigned for playerAspectRatio > videoAspectRatio * @param x - x axis point * @param y - y axis point */ normalizeAndQuantizeSignedPlayerBigger(x, y) { const normalizedX = x / (0.5 * this.videoElementParent.clientWidth); const normalizedY = (this.ratio * y) / (0.5 * this.videoElementParent.clientHeight); return new NormalizedQuantizedSignedCoord(normalizedX * 32767, normalizedY * 32767); } /** * normalizeAndQuantizeUnsigned for playerAspectRatio <= videoAspectRatio * @param x - x axis point * @param y - y axis point */ normalizeAndQuantizeUnsignedPlayerSmaller(x, y) { const normalizedX = this.ratio * (x / this.videoElementParent.clientWidth - 0.5) + 0.5; const normalizedY = y / this.videoElementParent.clientHeight; if (normalizedX < 0.0 || normalizedX > 1.0 || normalizedY < 0.0 || normalizedY > 1.0) { return new NormalizedQuantizedUnsignedCoord(false, 65535, 65535); } else { return new NormalizedQuantizedUnsignedCoord(true, normalizedX * 65536, normalizedY * 65536); } } /** * unquantizeAndDenormalizeUnsigned for playerAspectRatio <= videoAspectRatio * @param x - x axis point * @param y - y axis point */ denormalizeAndUnquantizeUnsignedPlayerSmaller(x, y) { const normalizedX = (x / 65536 - 0.5) / this.ratio + 0.5; const normalizedY = y / 65536; return new UnquantizedDenormalizedUnsignedCoord(normalizedX * this.videoElementParent.clientWidth, normalizedY * this.videoElementParent.clientHeight); } /** * normalizeAndQuantizeSigned for playerAspectRatio <= videoAspectRatio * @param x - x axis point * @param y - y axis point */ normalizeAndQuantizeSignedPlayerSmaller(x, y) { const normalizedX = (this.ratio * x) / (0.5 * this.videoElementParent.clientWidth); const normalizedY = y / (0.5 * this.videoElementParent.clientHeight); return new NormalizedQuantizedSignedCoord(normalizedX * 32767, normalizedY * 32767); } } /** * A class for NormalizeAndQuantizeUnsigned objects */ class NormalizedQuantizedUnsignedCoord { constructor(inRange, x, y) { this.inRange = inRange; this.x = x; this.y = y; } } /** * A class for UnquantizedAndDenormalizeUnsigned objects */ class UnquantizedDenormalizedUnsignedCoord { constructor(x, y) { this.x = x; this.y = y; } } /** * A class for NormalizedQuantizedSignedCoord objects */ class NormalizedQuantizedSignedCoord { constructor(x, y) { this.x = x; this.y = y; } } /***/ }), /***/ "./src/Util/EventEmitter.ts": /*!**********************************!*\ !*** ./src/Util/EventEmitter.ts ***! \**********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "AfkTimedOutEvent": () => (/* binding */ AfkTimedOutEvent), /* harmony export */ "AfkWarningActivateEvent": () => (/* binding */ AfkWarningActivateEvent), /* harmony export */ "AfkWarningDeactivateEvent": () => (/* binding */ AfkWarningDeactivateEvent), /* harmony export */ "AfkWarningUpdateEvent": () => (/* binding */ AfkWarningUpdateEvent), /* harmony export */ "DataChannelCloseEvent": () => (/* binding */ DataChannelCloseEvent), /* harmony export */ "DataChannelErrorEvent": () => (/* binding */ DataChannelErrorEvent), /* harmony export */ "DataChannelLatencyTestResponseEvent": () => (/* binding */ DataChannelLatencyTestResponseEvent), /* harmony export */ "DataChannelLatencyTestResultEvent": () => (/* binding */ DataChannelLatencyTestResultEvent), /* harmony export */ "DataChannelOpenEvent": () => (/* binding */ DataChannelOpenEvent), /* harmony export */ "EventEmitter": () => (/* binding */ EventEmitter), /* harmony export */ "HideFreezeFrameEvent": () => (/* binding */ HideFreezeFrameEvent), /* harmony export */ "InitialSettingsEvent": () => (/* binding */ InitialSettingsEvent), /* harmony export */ "LatencyTestResultEvent": () => (/* binding */ LatencyTestResultEvent), /* harmony export */ "LoadFreezeFrameEvent": () => (/* binding */ LoadFreezeFrameEvent), /* harmony export */ "PlayStreamErrorEvent": () => (/* binding */ PlayStreamErrorEvent), /* harmony export */ "PlayStreamEvent": () => (/* binding */ PlayStreamEvent), /* harmony export */ "PlayStreamRejectedEvent": () => (/* binding */ PlayStreamRejectedEvent), /* harmony export */ "PlayerCountEvent": () => (/* binding */ PlayerCountEvent), /* harmony export */ "SettingsChangedEvent": () => (/* binding */ SettingsChangedEvent), /* harmony export */ "StatsReceivedEvent": () => (/* binding */ StatsReceivedEvent), /* harmony export */ "StreamLoadingEvent": () => (/* binding */ StreamLoadingEvent), /* harmony export */ "StreamPreConnectEvent": () => (/* binding */ StreamPreConnectEvent), /* harmony export */ "StreamPreDisconnectEvent": () => (/* binding */ StreamPreDisconnectEvent), /* harmony export */ "StreamReconnectEvent": () => (/* binding */ StreamReconnectEvent), /* harmony export */ "StreamerListMessageEvent": () => (/* binding */ StreamerListMessageEvent), /* harmony export */ "VideoEncoderAvgQPEvent": () => (/* binding */ VideoEncoderAvgQPEvent), /* harmony export */ "VideoInitializedEvent": () => (/* binding */ VideoInitializedEvent), /* harmony export */ "WebRtcAutoConnectEvent": () => (/* binding */ WebRtcAutoConnectEvent), /* harmony export */ "WebRtcConnectedEvent": () => (/* binding */ WebRtcConnectedEvent), /* harmony export */ "WebRtcConnectingEvent": () => (/* binding */ WebRtcConnectingEvent), /* harmony export */ "WebRtcDisconnectedEvent": () => (/* binding */ WebRtcDisconnectedEvent), /* harmony export */ "WebRtcFailedEvent": () => (/* binding */ WebRtcFailedEvent), /* harmony export */ "WebRtcSdpEvent": () => (/* binding */ WebRtcSdpEvent), /* harmony export */ "XrFrameEvent": () => (/* binding */ XrFrameEvent), /* harmony export */ "XrSessionEndedEvent": () => (/* binding */ XrSessionEndedEvent), /* harmony export */ "XrSessionStartedEvent": () => (/* binding */ XrSessionStartedEvent) /* harmony export */ }); /** * An event that is emitted when AFK disconnect is about to happen. * Can be cancelled by calling the callback function provided as part of the event. */ class AfkWarningActivateEvent extends Event { constructor(data) { super('afkWarningActivate'); this.data = data; } } /** * An event that is emitted when the AFK disconnect countdown is updated. */ class AfkWarningUpdateEvent extends Event { constructor(data) { super('afkWarningUpdate'); this.data = data; } } /** * An event that is emitted when AFK warning is deactivated. */ class AfkWarningDeactivateEvent extends Event { constructor() { super('afkWarningDeactivate'); } } /** * An event that is emitted when AFK countdown reaches 0 and the user is disconnected. */ class AfkTimedOutEvent extends Event { constructor() { super('afkTimedOut'); } } /** * An event that is emitted when we receive new video quality value. */ class VideoEncoderAvgQPEvent extends Event { constructor(data) { super('videoEncoderAvgQP'); this.data = data; } } /** * An event that is emitted after a WebRtc connection has been negotiated. */ class WebRtcSdpEvent extends Event { constructor() { super('webRtcSdp'); } } /** * An event that is emitted when auto connecting. */ class WebRtcAutoConnectEvent extends Event { constructor() { super('webRtcAutoConnect'); } } /** * An event that is emitted when sending a WebRtc offer. */ class WebRtcConnectingEvent extends Event { constructor() { super('webRtcConnecting'); } } /** * An event that is emitted when WebRtc connection has been established. */ class WebRtcConnectedEvent extends Event { constructor() { super('webRtcConnected'); } } /** * An event that is emitted if WebRtc connection has failed. */ class WebRtcFailedEvent extends Event { constructor() { super('webRtcFailed'); } } /** * An event that is emitted if WebRtc connection is disconnected. */ class WebRtcDisconnectedEvent extends Event { constructor(data) { super('webRtcDisconnected'); this.data = data; } } /** * An event that is emitted when RTCDataChannel is opened. */ class DataChannelOpenEvent extends Event { constructor(data) { super('dataChannelOpen'); this.data = data; } } /** * An event that is emitted when RTCDataChannel is closed. */ class DataChannelCloseEvent extends Event { constructor(data) { super('dataChannelClose'); this.data = data; } } /** * An event that is emitted on RTCDataChannel errors. */ class DataChannelErrorEvent extends Event { constructor(data) { super('dataChannelError'); this.data = data; } } /** * An event that is emitted when the video stream has been initialized. */ class VideoInitializedEvent extends Event { constructor() { super('videoInitialized'); } } /** * An event that is emitted when video stream loading starts. */ class StreamLoadingEvent extends Event { constructor() { super('streamLoading'); } } /** * An event that is emitted when video stream loading has finished. */ class StreamPreConnectEvent extends Event { constructor() { super('streamConnect'); } } /** * An event that is emitted when video stream has stopped. */ class StreamPreDisconnectEvent extends Event { constructor() { super('streamDisconnect'); } } /** * An event that is emitted when video stream is reconnecting. */ class StreamReconnectEvent extends Event { constructor() { super('streamReconnect'); } } /** * An event that is emitted if there are errors loading the video stream. */ class PlayStreamErrorEvent extends Event { constructor(data) { super('playStreamError'); this.data = data; } } /** * An event that is emitted before trying to start video playback. */ class PlayStreamEvent extends Event { constructor() { super('playStream'); } } /** * An event that is emitted if the browser rejects video playback. Can happen for example if * video auto-play without user interaction is refused by the browser. */ class PlayStreamRejectedEvent extends Event { constructor(data) { super('playStreamRejected'); this.data = data; } } /** * An event that is emitted when receiving a full FreezeFrame image from UE. */ class LoadFreezeFrameEvent extends Event { constructor(data) { super('loadFreezeFrame'); this.data = data; } } /** * An event that is emitted when receiving UnfreezeFrame message from UE and video playback is about to be resumed. */ class HideFreezeFrameEvent extends Event { constructor() { super('hideFreezeFrame'); } } /** * An event that is emitted when receiving WebRTC statistics. */ class StatsReceivedEvent extends Event { constructor(data) { super('statsReceived'); this.data = data; } } /** * An event that is emitted when streamer list changes. */ class StreamerListMessageEvent extends Event { constructor(data) { super('streamerListMessage'); this.data = data; } } /** * An event that is emitted when receiving latency test results. */ class LatencyTestResultEvent extends Event { constructor(data) { super('latencyTestResult'); this.data = data; } } /** * An event that is emitted when receiving data channel latency test response from server. * This event is handled by DataChannelLatencyTestController */ class DataChannelLatencyTestResponseEvent extends Event { constructor(data) { super('dataChannelLatencyTestResponse'); this.data = data; } } /** * An event that is emitted when data channel latency test results are ready. */ class DataChannelLatencyTestResultEvent extends Event { constructor(data) { super('dataChannelLatencyTestResult'); this.data = data; } } /** * An event that is emitted when receiving initial settings from UE. */ class InitialSettingsEvent extends Event { constructor(data) { super('initialSettings'); this.data = data; } } /** * An event that is emitted when PixelStreaming settings change. */ class SettingsChangedEvent extends Event { constructor(data) { super('settingsChanged'); this.data = data; } } /** * Event emitted when an XR Session starts */ class XrSessionStartedEvent extends Event { constructor() { super('xrSessionStarted'); } } /** * Event emitted when an XR Session ends */ class XrSessionEndedEvent extends Event { constructor() { super('xrSessionEnded'); } } /** * Event emitted when an XR Frame is complete */ class XrFrameEvent extends Event { constructor(data) { super('xrFrame'); this.data = data; } } /** * An event that is emitted when receiving a player count from the signalling server */ class PlayerCountEvent extends Event { constructor(data) { super('playerCount'); this.data = data; } } class EventEmitter extends EventTarget { /** * Dispatch a new event. * @param e event * @returns */ dispatchEvent(e) { return super.dispatchEvent(e); } /** * Register an event handler. * @param type event name * @param listener event handler function */ addEventListener(type, listener) { super.addEventListener(type, listener); } /** * Remove an event handler. * @param type event name * @param listener event handler function */ removeEventListener(type, listener) { super.removeEventListener(type, listener); } } /***/ }), /***/ "./src/Util/EventListenerTracker.ts": /*!******************************************!*\ !*** ./src/Util/EventListenerTracker.ts ***! \******************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "EventListenerTracker": () => (/* binding */ EventListenerTracker) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. class EventListenerTracker { constructor() { this.unregisterCallbacks = []; } /** * Add a new callback that is executed when unregisterAll is called. * @param callback */ addUnregisterCallback(callback) { this.unregisterCallbacks.push(callback); } /** * Execute all callbacks and clear the list. */ unregisterAll() { for (const callback of this.unregisterCallbacks) { callback(); } this.unregisterCallbacks = []; } } /***/ }), /***/ "./src/Util/FileUtil.ts": /*!******************************!*\ !*** ./src/Util/FileUtil.ts ***! \******************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "FileTemplate": () => (/* binding */ FileTemplate), /* harmony export */ "FileUtil": () => (/* binding */ FileUtil) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Utility function for populate file information from byte buffers. */ class FileUtil { /** * Processes a files extension when received over data channel * @param view - the file extension data */ static setExtensionFromBytes(view, file) { // Reset file if we got a file message and we are not "receiving" it yet if (!file.receiving) { file.mimetype = ''; file.extension = ''; file.receiving = true; file.valid = false; file.size = 0; file.data = []; file.timestampStart = new Date().getTime(); _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Received first chunk of file', 6); } const extensionAsString = new TextDecoder('utf-16').decode(view.slice(1)); _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), extensionAsString, 6); file.extension = extensionAsString; } /** * Processes a files mime type when received over data channel * @param view - the file mime type data */ static setMimeTypeFromBytes(view, file) { // Reset file if we got a file message and we are not "receiving" it yet if (!file.receiving) { file.mimetype = ''; file.extension = ''; file.receiving = true; file.valid = false; file.size = 0; file.data = []; file.timestampStart = new Date().getTime(); _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Received first chunk of file', 6); } const mimeAsString = new TextDecoder('utf-16').decode(view.slice(1)); _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), mimeAsString, 6); file.mimetype = mimeAsString; } /** * Processes a files contents when received over data channel * @param view - the file contents data */ static setContentsFromBytes(view, file) { // If we haven't received the initial setup instructions, return if (!file.receiving) return; // Extract the total size of the file (across all chunks) file.size = Math.ceil(new DataView(view.slice(1, 5).buffer).getInt32(0, true) / 16379 /* The maximum number of payload bits per message*/); // Get the file part of the payload const fileBytes = view.slice(1 + 4); // Append to existing data that holds the file file.data.push(fileBytes); // Uncomment for debug _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Received file chunk: ${file.data.length}/${file.size}`, 6); if (file.data.length === file.size) { file.receiving = false; file.valid = true; _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Received complete file', 6); const transferDuration = new Date().getTime() - file.timestampStart; const transferBitrate = Math.round((file.size * 16 * 1024) / transferDuration); _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Average transfer bitrate: ${transferBitrate}kb/s over ${transferDuration / 1000} seconds`, 6); // File reconstruction /** * Example code to reconstruct the file * * This code reconstructs the received data into the original file based on the mime type and extension provided and then downloads the reconstructed file */ const received = new Blob(file.data, { type: file.mimetype }); const a = document.createElement('a'); a.setAttribute('href', URL.createObjectURL(received)); a.setAttribute('download', `transfer.${file.extension}`); document.body.append(a); // if you are so inclined to make it auto-download, do something like: a.click(); a.remove(); } else if (file.data.length > file.size) { file.receiving = false; _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Received bigger file than advertised: ${file.data.length}/${file.size}`); } } } /** * A class that represents a template for a downloaded file */ class FileTemplate { constructor() { this.mimetype = ''; this.extension = ''; this.receiving = false; this.size = 0; this.data = []; this.valid = false; } } /***/ }), /***/ "./src/Util/RTCUtils.ts": /*!******************************!*\ !*** ./src/Util/RTCUtils.ts ***! \******************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "RTCUtils": () => (/* binding */ RTCUtils) /* harmony export */ }); class RTCUtils { static isVideoTransciever(transceiver) { return this.canTransceiverReceiveVideo(transceiver) || this.canTransceiverSendVideo(transceiver); } static canTransceiverReceiveVideo(transceiver) { return !!transceiver && (transceiver.direction === 'sendrecv' || transceiver.direction === 'recvonly') && transceiver.receiver && transceiver.receiver.track && transceiver.receiver.track.kind === 'video'; } static canTransceiverSendVideo(transceiver) { return !!transceiver && (transceiver.direction === 'sendrecv' || transceiver.direction === 'sendonly') && transceiver.sender && transceiver.sender.track && transceiver.sender.track.kind === 'video'; } static isAudioTransciever(transceiver) { return this.canTransceiverReceiveAudio(transceiver) || this.canTransceiverSendAudio(transceiver); } static canTransceiverReceiveAudio(transceiver) { return !!transceiver && (transceiver.direction === 'sendrecv' || transceiver.direction === 'recvonly') && transceiver.receiver && transceiver.receiver.track && transceiver.receiver.track.kind === 'audio'; } static canTransceiverSendAudio(transceiver) { return !!transceiver && (transceiver.direction === 'sendrecv' || transceiver.direction === 'sendonly') && transceiver.sender && transceiver.sender.track && transceiver.sender.track.kind === 'audio'; } } /***/ }), /***/ "./src/Util/WebGLUtils.ts": /*!********************************!*\ !*** ./src/Util/WebGLUtils.ts ***! \********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "WebGLUtils": () => (/* binding */ WebGLUtils) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. class WebGLUtils { static vertexShader() { return ` attribute vec2 a_position; attribute vec2 a_texCoord; // input uniform vec2 u_resolution; uniform vec4 u_offset; // varying vec2 v_texCoord; void main() { // convert the rectangle from pixels to 0.0 to 1.0 vec2 zeroToOne = a_position / u_resolution; // convert from 0->1 to 0->2 vec2 zeroToTwo = zeroToOne * 2.0; // convert from 0->2 to -1->+1 (clipspace) vec2 clipSpace = zeroToTwo - 1.0; gl_Position = vec4(clipSpace * vec2(1, -1), 0, 1); // pass the texCoord to the fragment shader // The GPU will interpolate this value between points. v_texCoord = (a_texCoord * u_offset.xy) + u_offset.zw; } `; } static fragmentShader() { return ` precision mediump float; // our texture uniform sampler2D u_image; // the texCoords passed in from the vertex shader. varying vec2 v_texCoord; void main() { gl_FragColor = texture2D(u_image, v_texCoord); } `; } } /***/ }), /***/ "./src/Util/WebXRUtils.ts": /*!********************************!*\ !*** ./src/Util/WebXRUtils.ts ***! \********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "WebXRUtils": () => (/* binding */ WebXRUtils) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. class WebXRUtils { /** * Deep copies a gamepad's values by first converting it to a JSON object and then back to a gamepad * * @param gamepad the original gamepad * @returns a new gamepad object, populated with the original gamepads values */ static deepCopyGamepad(gamepad) { return JSON.parse(JSON.stringify({ buttons: gamepad.buttons.map((b) => JSON.parse(JSON.stringify({ pressed: b.pressed, touched: b.touched }))), axes: gamepad.axes })); } } /***/ }), /***/ "./src/VideoPlayer/StreamController.ts": /*!*********************************************!*\ !*** ./src/VideoPlayer/StreamController.ts ***! \*********************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "StreamController": () => (/* binding */ StreamController) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Video Player Controller handles the creation of the video HTML element and all handlers */ class StreamController { /** * @param videoElementProvider Video Player instance */ constructor(videoElementProvider) { this.videoElementProvider = videoElementProvider; this.audioElement = document.createElement('Audio'); this.videoElementProvider.setAudioElement(this.audioElement); } /** * Handles when the Peer connection has a track event * @param rtcTrackEvent - RTC Track Event */ handleOnTrack(rtcTrackEvent) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'handleOnTrack ' + JSON.stringify(rtcTrackEvent.streams), 6); const videoElement = this.videoElementProvider.getVideoElement(); if (rtcTrackEvent.track) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Got track - ' + rtcTrackEvent.track.kind + ' id=' + rtcTrackEvent.track.id + ' readyState=' + rtcTrackEvent.track.readyState, 6); } if (rtcTrackEvent.track.kind == 'audio') { this.CreateAudioTrack(rtcTrackEvent.streams[0]); return; } else if (rtcTrackEvent.track.kind == 'video' && videoElement.srcObject !== rtcTrackEvent.streams[0]) { videoElement.srcObject = rtcTrackEvent.streams[0]; _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Set video source from video track ontrack.'); return; } } /** * Creates the audio device when receiving an RTCTrackEvent with the kind of "audio" * @param audioMediaStream - Audio Media stream track */ CreateAudioTrack(audioMediaStream) { const videoElement = this.videoElementProvider.getVideoElement(); // do nothing the video has the same media stream as the audio track we have here (they are linked) if (videoElement.srcObject == audioMediaStream) { return; } // video element has some other media stream that is not associated with this audio track else if (videoElement.srcObject && videoElement.srcObject !== audioMediaStream) { // create a new audio element this.audioElement.srcObject = audioMediaStream; _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Created new audio element to play separate audio stream.'); } } } /***/ }), /***/ "./src/VideoPlayer/VideoPlayer.ts": /*!****************************************!*\ !*** ./src/VideoPlayer/VideoPlayer.ts ***! \****************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "VideoPlayer": () => (/* binding */ VideoPlayer) /* harmony export */ }); /* harmony import */ var _Config_Config__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Config/Config */ "./src/Config/Config.ts"); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * The video player html element */ class VideoPlayer { /** * @param videoElementParent the html div the the video player will be injected into * @param config the applications configuration. We're interested in the startVideoMuted flag */ constructor(videoElementParent, config) { this.lastTimeResized = new Date().getTime(); this.videoElement = document.createElement('video'); this.config = config; this.videoElement.id = 'streamingVideo'; this.videoElement.disablePictureInPicture = true; this.videoElement.playsInline = true; this.videoElement.style.width = '100%'; this.videoElement.style.height = '100%'; this.videoElement.style.position = 'absolute'; this.videoElement.style.pointerEvents = 'all'; videoElementParent.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?'); }; // set play for video (and audio) this.videoElement.onclick = () => { if (this.audioElement != undefined && this.audioElement.paused) { this.audioElement.play(); } if (this.videoElement.paused) { this.videoElement.play(); } }; this.videoElement.onloadedmetadata = () => { this.onVideoInitialized(); }; // set resize events to the windows if it is resized or its orientation is changed window.addEventListener('resize', () => this.resizePlayerStyle(), true); window.addEventListener('orientationchange', () => this.onOrientationChange()); } setAudioElement(audioElement) { this.audioElement = audioElement; } /** * Sets up the video element with any application config and plays the video element. * @returns A promise for if playing the video was successful or not. */ play() { this.videoElement.muted = this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_0__.Flags.StartVideoMuted); this.videoElement.autoplay = this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_0__.Flags.AutoPlayVideo); return this.videoElement.play(); } /** * @returns True if the video element is paused. */ isPaused() { return this.videoElement.paused; } /** * @returns - whether the video element is playing. */ isVideoReady() { return (this.videoElement.readyState !== undefined && this.videoElement.readyState > 0); } /** * @returns True if the video element has a valid video source (srcObject). */ hasVideoSource() { return (this.videoElement.srcObject !== undefined && this.videoElement.srcObject !== null); } /** * Get the current context of the html video element * @returns - the current context of the video element */ getVideoElement() { return this.videoElement; } /** * Get the current context of the html video elements parent * @returns - the current context of the video elements parent */ getVideoParentElement() { return this.videoElement.parentElement; } /** * Set the Video Elements src object tracks to enable * @param enabled - Enable Tracks on the Src Object */ setVideoEnabled(enabled) { // this is a temporary hack until type scripts video element is updated to reflect the need for tracks on a html video element const videoElement = this.videoElement; videoElement.srcObject .getTracks() .forEach((track) => (track.enabled = enabled)); } /** * An override for when the video has been initialized with a srcObject */ onVideoInitialized() { // Default Functionality: Do Nothing } /** * On the orientation change of a window clear the timeout */ onOrientationChange() { clearTimeout(this.orientationChangeTimeout); this.orientationChangeTimeout = window.setTimeout(() => { this.resizePlayerStyle(); }, 500); } /** * Resizes the player style based on the window height and width * @returns - nil if requirements are satisfied */ resizePlayerStyle() { const videoElementParent = this.getVideoParentElement(); if (!videoElementParent) { return; } this.updateVideoStreamSize(); if (videoElementParent.classList.contains('fixed-size')) { this.onResizePlayerCallback(); return; } // controls for resizing the player this.resizePlayerStyleToFillParentElement(); this.onResizePlayerCallback(); } /** * Resizes the player element to fill the parent element */ resizePlayerStyleToFillParentElement() { const videoElementParent = this.getVideoParentElement(); //Video is not initialized yet so set videoElementParent to size of parent element const styleWidth = '100%'; const styleHeight = '100%'; const styleTop = 0; const styleLeft = 0; videoElementParent.setAttribute('style', 'top: ' + styleTop + 'px; left: ' + styleLeft + 'px; width: ' + styleWidth + '; height: ' + styleHeight + '; cursor: default;'); } updateVideoStreamSize() { if (!this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_0__.Flags.MatchViewportResolution)) { return; } const now = new Date().getTime(); if (now - this.lastTimeResized > 300) { const videoElementParent = this.getVideoParentElement(); if (!videoElementParent) { return; } this.onMatchViewportResolutionCallback(videoElementParent.clientWidth, videoElementParent.clientHeight); this.lastTimeResized = new Date().getTime(); } else { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'Resizing too often - skipping', 6); clearTimeout(this.resizeTimeoutHandle); this.resizeTimeoutHandle = window.setTimeout(() => this.updateVideoStreamSize(), 100); } } } /***/ }), /***/ "./src/WebRtcPlayer/WebRtcPlayerController.ts": /*!****************************************************!*\ !*** ./src/WebRtcPlayer/WebRtcPlayerController.ts ***! \****************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "WebRtcPlayerController": () => (/* binding */ WebRtcPlayerController) /* harmony export */ }); /* harmony import */ var _WebSockets_WebSocketController__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! ../WebSockets/WebSocketController */ "./src/WebSockets/WebSocketController.ts"); /* harmony import */ var _VideoPlayer_StreamController__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../VideoPlayer/StreamController */ "./src/VideoPlayer/StreamController.ts"); /* harmony import */ var _FreezeFrame_FreezeFrameController__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../FreezeFrame/FreezeFrameController */ "./src/FreezeFrame/FreezeFrameController.ts"); /* harmony import */ var _AFK_AFKController__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../AFK/AFKController */ "./src/AFK/AFKController.ts"); /* harmony import */ var _DataChannel_DataChannelController__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../DataChannel/DataChannelController */ "./src/DataChannel/DataChannelController.ts"); /* harmony import */ var _PeerConnectionController_PeerConnectionController__WEBPACK_IMPORTED_MODULE_17__ = __webpack_require__(/*! ../PeerConnectionController/PeerConnectionController */ "./src/PeerConnectionController/PeerConnectionController.ts"); /* harmony import */ var _Config_Config__WEBPACK_IMPORTED_MODULE_11__ = __webpack_require__(/*! ../Config/Config */ "./src/Config/Config.ts"); /* harmony import */ var _DataChannel_InitialSettings__WEBPACK_IMPORTED_MODULE_19__ = __webpack_require__(/*! ../DataChannel/InitialSettings */ "./src/DataChannel/InitialSettings.ts"); /* harmony import */ var _DataChannel_LatencyTestResults__WEBPACK_IMPORTED_MODULE_18__ = __webpack_require__(/*! ../DataChannel/LatencyTestResults */ "./src/DataChannel/LatencyTestResults.ts"); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _Util_FileUtil__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Util/FileUtil */ "./src/Util/FileUtil.ts"); /* harmony import */ var _Inputs_InputClassesFactory__WEBPACK_IMPORTED_MODULE_14__ = __webpack_require__(/*! ../Inputs/InputClassesFactory */ "./src/Inputs/InputClassesFactory.ts"); /* harmony import */ var _VideoPlayer_VideoPlayer__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../VideoPlayer/VideoPlayer */ "./src/VideoPlayer/VideoPlayer.ts"); /* harmony import */ var _UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! ../UeInstanceMessage/StreamMessageController */ "./src/UeInstanceMessage/StreamMessageController.ts"); /* harmony import */ var _UeInstanceMessage_ResponseController__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../UeInstanceMessage/ResponseController */ "./src/UeInstanceMessage/ResponseController.ts"); /* harmony import */ var _UeInstanceMessage_SendMessageController__WEBPACK_IMPORTED_MODULE_12__ = __webpack_require__(/*! ../UeInstanceMessage/SendMessageController */ "./src/UeInstanceMessage/SendMessageController.ts"); /* harmony import */ var _UeInstanceMessage_ToStreamerMessagesController__WEBPACK_IMPORTED_MODULE_13__ = __webpack_require__(/*! ../UeInstanceMessage/ToStreamerMessagesController */ "./src/UeInstanceMessage/ToStreamerMessagesController.ts"); /* harmony import */ var _DataChannel_DataChannelSender__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ../DataChannel/DataChannelSender */ "./src/DataChannel/DataChannelSender.ts"); /* harmony import */ var _Util_CoordinateConverter__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ../Util/CoordinateConverter */ "./src/Util/CoordinateConverter.ts"); /* harmony import */ var _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_16__ = __webpack_require__(/*! ../Util/EventEmitter */ "./src/Util/EventEmitter.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Entry point for the WebRTC Player */ class WebRtcPlayerController { /** * * @param config - the frontend config object * @param pixelStreaming - the PixelStreaming object */ constructor(config, pixelStreaming) { this.shouldShowPlayOverlay = true; this.autoJoinTimer = undefined; this.config = config; this.pixelStreaming = pixelStreaming; this.responseController = new _UeInstanceMessage_ResponseController__WEBPACK_IMPORTED_MODULE_0__.ResponseController(); this.file = new _Util_FileUtil__WEBPACK_IMPORTED_MODULE_1__.FileTemplate(); this.sdpConstraints = { offerToReceiveAudio: true, offerToReceiveVideo: true }; // set up the afk logic class and connect up its method for closing the signaling server this.afkController = new _AFK_AFKController__WEBPACK_IMPORTED_MODULE_2__.AFKController(this.config, this.pixelStreaming, this.onAfkTriggered.bind(this)); this.afkController.onAFKTimedOutCallback = () => { this.closeSignalingServer('You have been disconnected due to inactivity'); }; this.freezeFrameController = new _FreezeFrame_FreezeFrameController__WEBPACK_IMPORTED_MODULE_3__.FreezeFrameController(this.pixelStreaming.videoElementParent); this.videoPlayer = new _VideoPlayer_VideoPlayer__WEBPACK_IMPORTED_MODULE_4__.VideoPlayer(this.pixelStreaming.videoElementParent, this.config); this.videoPlayer.onVideoInitialized = () => this.handleVideoInitialized(); // When in match viewport resolution mode, when the browser viewport is resized we send a resize command back to UE. this.videoPlayer.onMatchViewportResolutionCallback = (width, height) => { const descriptor = { 'Resolution.Width': width, 'Resolution.Height': height }; this.streamMessageController.toStreamerHandlers.get('Command')([JSON.stringify(descriptor)]); }; // Every time video player is resized in browser we need to reinitialize the mouse coordinate conversion and freeze frame sizing logic. this.videoPlayer.onResizePlayerCallback = () => { this.setUpMouseAndFreezeFrame(); }; this.streamController = new _VideoPlayer_StreamController__WEBPACK_IMPORTED_MODULE_5__.StreamController(this.videoPlayer); this.coordinateConverter = new _Util_CoordinateConverter__WEBPACK_IMPORTED_MODULE_6__.CoordinateConverter(this.videoPlayer); this.sendrecvDataChannelController = new _DataChannel_DataChannelController__WEBPACK_IMPORTED_MODULE_7__.DataChannelController(); this.recvDataChannelController = new _DataChannel_DataChannelController__WEBPACK_IMPORTED_MODULE_7__.DataChannelController(); this.registerDataChannelEventEmitters(this.sendrecvDataChannelController); this.registerDataChannelEventEmitters(this.recvDataChannelController); this.dataChannelSender = new _DataChannel_DataChannelSender__WEBPACK_IMPORTED_MODULE_8__.DataChannelSender(this.sendrecvDataChannelController); this.dataChannelSender.resetAfkWarningTimerOnDataSend = () => this.afkController.resetAfkWarningTimer(); this.streamMessageController = new _UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.StreamMessageController(); // set up websocket methods this.webSocketController = new _WebSockets_WebSocketController__WEBPACK_IMPORTED_MODULE_10__.WebSocketController(); this.webSocketController.onConfig = (messageConfig) => this.handleOnConfigMessage(messageConfig); this.webSocketController.onStreamerList = (messageList) => this.handleStreamerListMessage(messageList); this.webSocketController.onPlayerCount = (playerCount) => { this.pixelStreaming._onPlayerCount(playerCount.count); }; this.webSocketController.onOpen.addEventListener('open', () => { const BrowserSendsOffer = this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.BrowserSendOffer); if (!BrowserSendsOffer) { this.webSocketController.requestStreamerList(); } }); this.webSocketController.onClose.addEventListener('close', (event) => { // when we refresh the page during a stream we get the going away code. // in that case we don't want to reconnect since we're navigating away. // https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent/code // lists all the codes. const CODE_GOING_AWAY = 1001; const willTryReconnect = this.shouldReconnect && event.detail.code != CODE_GOING_AWAY && this.config.getNumericSettingValue(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.NumericParameters.MaxReconnectAttempts) > 0; const disconnectMessage = this.disconnectMessage ? this.disconnectMessage : event.detail.reason; this.pixelStreaming._onDisconnect(disconnectMessage, !willTryReconnect && !this.isReconnecting); this.afkController.stopAfkWarningTimer(); // stop sending stats on interval if we have closed our connection if (this.statsTimerHandle && this.statsTimerHandle !== undefined) { window.clearInterval(this.statsTimerHandle); } // reset the stream quality icon. this.setVideoEncoderAvgQP(0); // unregister all input device event handlers on disconnect this.setTouchInputEnabled(false); this.setMouseInputEnabled(false); this.setKeyboardInputEnabled(false); this.setGamePadInputEnabled(false); if (willTryReconnect) { // need a small delay here to prevent reconnect spamming setTimeout(() => { this.isReconnecting = true; this.reconnectAttempt++; this.tryReconnect(event.detail.reason); }, 2000); } }); // set up the final webRtc player controller methods from within our application so a connection can be activated this.sendMessageController = new _UeInstanceMessage_SendMessageController__WEBPACK_IMPORTED_MODULE_12__.SendMessageController(this.dataChannelSender, this.streamMessageController); this.toStreamerMessagesController = new _UeInstanceMessage_ToStreamerMessagesController__WEBPACK_IMPORTED_MODULE_13__.ToStreamerMessagesController(this.sendMessageController); this.registerMessageHandlers(); this.streamMessageController.populateDefaultProtocol(); this.inputClassesFactory = new _Inputs_InputClassesFactory__WEBPACK_IMPORTED_MODULE_14__.InputClassesFactory(this.streamMessageController, this.videoPlayer, this.coordinateConverter); this.isUsingSFU = false; this.isQualityController = false; this.preferredCodec = ''; this.shouldReconnect = true; this.isReconnecting = false; this.reconnectAttempt = 0; this.config._addOnOptionSettingChangedListener(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.OptionParameters.StreamerId, (streamerid) => { if (streamerid === "") { return; } // close the current peer connection and create a new one this.peerConnectionController.peerConnection.close(); this.peerConnectionController.createPeerConnection(this.peerConfig, this.preferredCodec); this.subscribedStream = streamerid; this.webSocketController.sendSubscribe(streamerid); }); this.setVideoEncoderAvgQP(-1); this.signallingUrlBuilder = () => { let signallingServerUrl = this.config.getTextSettingValue(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.TextParameters.SignallingServerUrl); // If we are connecting to the SFU add a special url parameter to the url if (this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.BrowserSendOffer)) { signallingServerUrl += '?' + _Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.BrowserSendOffer + '=true'; } // This code is no longer needed, but is a good example for how subsequent config flags can be appended // if (this.config.isFlagEnabled(Flags.BrowserSendOffer)) { // signallingServerUrl += (signallingServerUrl.includes('?') ? '&' : '?') + Flags.BrowserSendOffer + '=true'; // } return signallingServerUrl; }; } /** * Make a request to UnquantizedAndDenormalizeUnsigned coordinates * @param x x axis coordinate * @param y y axis coordinate */ requestUnquantizedAndDenormalizeUnsigned(x, y) { return this.coordinateConverter.unquantizeAndDenormalizeUnsigned(x, y); } /** * Handles when a message is received * @param event - Message Event */ handleOnMessage(event) { const message = new Uint8Array(event.data); _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'Message incoming:' + message, 6); //try { const messageType = this.streamMessageController.fromStreamerMessages.get(message[0]); this.streamMessageController.fromStreamerHandlers.get(messageType)(event.data); //} catch (e) { //Logger.Error(Logger.GetStackTrace(), `Custom data channel message with message type that is unknown to the Pixel Streaming protocol. Does your PixelStreamingProtocol need updating? The message type was: ${message[0]}`); //} } /** * Register message all handlers */ registerMessageHandlers() { // From Streamer // Message events from the streamer have a data type of ArrayBuffer as we force this type in the DatachannelController this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'QualityControlOwnership', (data) => this.onQualityControlOwnership(data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'Response', (data) => this.responseController.onResponse(data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'Command', (data) => { this.onCommand(data); }); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'FreezeFrame', (data) => this.onFreezeFrameMessage(data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'UnfreezeFrame', () => this.invalidateFreezeFrameAndEnableVideo()); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'VideoEncoderAvgQP', (data) => this.handleVideoEncoderAvgQP(data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'LatencyTest', (data) => this.handleLatencyTestResult(data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'DataChannelLatencyTest', (data) => this.handleDataChannelLatencyTestResponse(data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'InitialSettings', (data) => this.handleInitialSettings(data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'FileExtension', (data) => this.onFileExtension(data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'FileMimeType', (data) => this.onFileMimeType(data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'FileContents', (data) => this.onFileContents(data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'TestEcho', () => { /* Do nothing */ }); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'InputControlOwnership', (data) => this.onInputControlOwnership(data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'GamepadResponse', (data) => this.onGamepadResponse(data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer, 'Protocol', (data) => this.onProtocolMessage(data)); // To Streamer this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'IFrameRequest', () => this.sendMessageController.sendMessageToStreamer('IFrameRequest')); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'RequestQualityControl', () => this.sendMessageController.sendMessageToStreamer('RequestQualityControl')); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'FpsRequest', () => this.sendMessageController.sendMessageToStreamer('FpsRequest')); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'AverageBitrateRequest', () => this.sendMessageController.sendMessageToStreamer('AverageBitrateRequest')); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'StartStreaming', () => this.sendMessageController.sendMessageToStreamer('StartStreaming')); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'StopStreaming', () => this.sendMessageController.sendMessageToStreamer('StopStreaming')); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'LatencyTest', (data) => this.sendMessageController.sendMessageToStreamer('LatencyTest', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'RequestInitialSettings', () => this.sendMessageController.sendMessageToStreamer('RequestInitialSettings')); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'TestEcho', () => { /* Do nothing */ }); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'UIInteraction', (data) => this.sendMessageController.sendMessageToStreamer('UIInteraction', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'Command', (data) => this.sendMessageController.sendMessageToStreamer('Command', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'TextboxEntry', (data) => this.sendMessageController.sendMessageToStreamer('TextboxEntry', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'KeyDown', (data) => this.sendMessageController.sendMessageToStreamer('KeyDown', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'KeyUp', (data) => this.sendMessageController.sendMessageToStreamer('KeyUp', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'KeyPress', (data) => this.sendMessageController.sendMessageToStreamer('KeyPress', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'MouseEnter', (data) => this.sendMessageController.sendMessageToStreamer('MouseEnter', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'MouseLeave', (data) => this.sendMessageController.sendMessageToStreamer('MouseLeave', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'MouseDown', (data) => this.sendMessageController.sendMessageToStreamer('MouseDown', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'MouseUp', (data) => this.sendMessageController.sendMessageToStreamer('MouseUp', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'MouseMove', (data) => this.sendMessageController.sendMessageToStreamer('MouseMove', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'MouseWheel', (data) => this.sendMessageController.sendMessageToStreamer('MouseWheel', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'MouseDouble', (data) => this.sendMessageController.sendMessageToStreamer('MouseDouble', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'TouchStart', (data) => this.sendMessageController.sendMessageToStreamer('TouchStart', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'TouchEnd', (data) => this.sendMessageController.sendMessageToStreamer('TouchEnd', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'TouchMove', (data) => this.sendMessageController.sendMessageToStreamer('TouchMove', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'GamepadConnected', () => this.sendMessageController.sendMessageToStreamer('GamepadConnected')); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'GamepadButtonPressed', (data) => this.sendMessageController.sendMessageToStreamer('GamepadButtonPressed', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'GamepadButtonReleased', (data) => this.sendMessageController.sendMessageToStreamer('GamepadButtonReleased', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'GamepadAnalog', (data) => this.sendMessageController.sendMessageToStreamer('GamepadAnalog', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'GamepadDisconnected', (data) => this.sendMessageController.sendMessageToStreamer('GamepadDisconnected', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'XRHMDTransform', (data) => this.sendMessageController.sendMessageToStreamer('XRHMDTransform', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'XRControllerTransform', (data) => this.sendMessageController.sendMessageToStreamer('XRControllerTransform', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'XRSystem', (data) => this.sendMessageController.sendMessageToStreamer('XRSystem', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'XRButtonTouched', (data) => this.sendMessageController.sendMessageToStreamer('XRButtonTouched', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'XRButtonPressed', (data) => this.sendMessageController.sendMessageToStreamer('XRButtonPressed', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'XRButtonReleased', (data) => this.sendMessageController.sendMessageToStreamer('XRButtonReleased', data)); this.streamMessageController.registerMessageHandler(_UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer, 'XRAnalog', (data) => this.sendMessageController.sendMessageToStreamer('XRAnalog', data)); } /** * Activate the logic associated with a command from UE * @param message */ onCommand(message) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'DataChannelReceiveMessageType.Command', 6); const commandAsString = new TextDecoder('utf-16').decode(message.slice(1)); _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'Data Channel Command: ' + commandAsString, 6); const command = JSON.parse(commandAsString); if (command.command === 'onScreenKeyboard') { this.pixelStreaming._activateOnScreenKeyboard(command); } } /** * Handles a protocol message received from the streamer * @param message the message data from the streamer */ onProtocolMessage(message) { try { const protocolString = new TextDecoder('utf-16').decode(message.slice(1)); const protocolJSON = JSON.parse(protocolString); if (!Object.prototype.hasOwnProperty.call(protocolJSON, 'Direction')) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'Malformed protocol received. Ensure the protocol message contains a direction'); } const direction = protocolJSON.Direction; delete protocolJSON.Direction; _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `Received new ${direction == _UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer ? 'FromStreamer' : 'ToStreamer'} protocol. Updating existing protocol...`); Object.keys(protocolJSON).forEach((messageType) => { const message = protocolJSON[messageType]; switch (direction) { case _UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer: // Check that the message contains all the relevant params if (!Object.prototype.hasOwnProperty.call(message, 'id')) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `ToStreamer->${messageType} protocol definition was malformed as it didn't contain at least an id\n Definition was: ${JSON.stringify(message, null, 2)}`); // return in a forEach is equivalent to a continue in a normal for loop return; } // UE5.1 and UE5.2 don't send a structure for these message types, but they actually do have a structure so ignore updating them if ((messageType === "UIInteraction" || messageType === "Command" || messageType === "LatencyTest")) { return; } if (this.streamMessageController.toStreamerHandlers.get(messageType)) { // If we've registered a handler for this message type we can add it to our supported messages. ie registerMessageHandler(...) this.streamMessageController.toStreamerMessages.set(messageType, message); } else { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `There was no registered handler for "${messageType}" - try adding one using registerMessageHandler(MessageDirection.ToStreamer, "${messageType}", myHandler)`); } break; case _UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer: // Check that the message contains all the relevant params if (!Object.prototype.hasOwnProperty.call(message, 'id')) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `FromStreamer->${messageType} protocol definition was malformed as it didn't contain at least an id\n Definition was: ${JSON.stringify(message, null, 2)}`); // return in a forEach is equivalent to a continue in a normal for loop return; } if (this.streamMessageController.fromStreamerHandlers.get(messageType)) { // If we've registered a handler for this message type. ie registerMessageHandler(...) this.streamMessageController.fromStreamerMessages.set(message.id, messageType); } else { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `There was no registered handler for "${message}" - try adding one using registerMessageHandler(MessageDirection.FromStreamer, "${messageType}", myHandler)`); } break; default: _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `Unknown direction: ${direction}`); } }); // Once the protocol has been received, we can send our control messages this.toStreamerMessagesController.SendRequestInitialSettings(); this.toStreamerMessagesController.SendRequestQualityControl(); } catch (e) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), e); } } /** * Handles an input control message when it is received from the streamer * @param message The input control message */ onInputControlOwnership(message) { const view = new Uint8Array(message); _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'DataChannelReceiveMessageType.InputControlOwnership', 6); const inputControlOwnership = new Boolean(view[1]).valueOf(); _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `Received input controller message - will your input control the stream: ${inputControlOwnership}`); this.pixelStreaming._onInputControlOwnership(inputControlOwnership); } /** * * @param message */ onGamepadResponse(message) { const responseString = new TextDecoder('utf-16').decode(message.slice(1)); const responseJSON = JSON.parse(responseString); this.gamePadController.onGamepadResponseReceived(responseJSON.controllerId); } onAfkTriggered() { this.afkController.onAfkClick(); // if the stream is paused play it, if we can if (this.videoPlayer.isPaused() && this.videoPlayer.hasVideoSource()) { this.playStream(); } } /** * Set whether we should timeout when afk. * @param afkEnabled If true we timeout when idle for some given amount of time. */ setAfkEnabled(afkEnabled) { if (afkEnabled) { this.onAfkTriggered(); } else { this.afkController.stopAfkWarningTimer(); } } /** * Attempt a reconnection to the signalling server */ tryReconnect(message) { // if there is no webSocketController return immediately or this will not work if (!this.webSocketController) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'The Web Socket Controller does not exist so this will not work right now.'); return; } // if the connection is open, first close it. wait some time and try again. this.isReconnecting = true; if (this.webSocketController.webSocket && this.webSocketController.webSocket.readyState != WebSocket.CLOSED) { this.closeSignalingServer(`${message} Restarting stream...`); setTimeout(() => { this.tryReconnect(message); }, 3000); } else { this.pixelStreaming._onWebRtcAutoConnect(); this.connectToSignallingServer(); } } /** * Loads a freeze frame if it is required otherwise shows the play overlay */ loadFreezeFrameOrShowPlayOverlay() { this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_16__.LoadFreezeFrameEvent({ shouldShowPlayOverlay: this.shouldShowPlayOverlay, isValid: this.freezeFrameController.valid, jpegData: this.freezeFrameController.jpeg })); if (this.shouldShowPlayOverlay === true) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'showing play overlay'); this.resizePlayerStyle(); } else { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'showing freeze frame'); this.freezeFrameController.showFreezeFrame(); } setTimeout(() => { this.videoPlayer.setVideoEnabled(false); }, this.freezeFrameController.freezeFrameDelay); } /** * Process the freeze frame and load it * @param message The freeze frame data in bytes */ onFreezeFrameMessage(message) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'DataChannelReceiveMessageType.FreezeFrame', 6); const view = new Uint8Array(message); this.freezeFrameController.processFreezeFrameMessage(view, () => this.loadFreezeFrameOrShowPlayOverlay()); } /** * Enable the video after hiding a freeze frame */ invalidateFreezeFrameAndEnableVideo() { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'DataChannelReceiveMessageType.FreezeFrame', 6); setTimeout(() => { this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_16__.HideFreezeFrameEvent()); this.freezeFrameController.hideFreezeFrame(); }, this.freezeFrameController.freezeFrameDelay); if (this.videoPlayer.getVideoElement()) { this.videoPlayer.setVideoEnabled(true); } } /** * Prep datachannel data for processing file extension * @param data the file extension data */ onFileExtension(data) { const view = new Uint8Array(data); _Util_FileUtil__WEBPACK_IMPORTED_MODULE_1__.FileUtil.setExtensionFromBytes(view, this.file); } /** * Prep datachannel data for processing the file mime type * @param data the file mime type data */ onFileMimeType(data) { const view = new Uint8Array(data); _Util_FileUtil__WEBPACK_IMPORTED_MODULE_1__.FileUtil.setMimeTypeFromBytes(view, this.file); } /** * Prep datachannel data for processing the file contents * @param data the file contents data */ onFileContents(data) { const view = new Uint8Array(data); _Util_FileUtil__WEBPACK_IMPORTED_MODULE_1__.FileUtil.setContentsFromBytes(view, this.file); } /** * Plays the stream audio and video source and sets up other pieces while the stream starts */ playStream() { if (!this.videoPlayer.getVideoElement()) { const message = 'Could not play video stream because the video player was not initialized correctly.'; this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_16__.PlayStreamErrorEvent({ message })); _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), message); // close the connection this.closeSignalingServer('Stream not initialized correctly'); return; } if (!this.videoPlayer.hasVideoSource()) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Warning(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'Cannot play stream, the video element has no srcObject to play.'); return; } this.setTouchInputEnabled(this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.TouchInput)); this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_16__.PlayStreamEvent()); if (this.streamController.audioElement.srcObject) { const startMuted = this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.StartVideoMuted); this.streamController.audioElement.muted = startMuted; if (startMuted) { this.playVideo(); } else { this.streamController.audioElement .play() .then(() => { this.playVideo(); }) .catch((onRejectedReason) => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), onRejectedReason); _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.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 _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_16__.PlayStreamRejectedEvent({ reason: onRejectedReason })); }); } } else { this.playVideo(); } this.shouldShowPlayOverlay = false; this.freezeFrameController.showFreezeFrame(); } /** * Plays the video stream */ playVideo() { // handle play() with promise as it is an asynchronous call this.videoPlayer.play().catch((onRejectedReason) => { if (this.streamController.audioElement.srcObject) { this.streamController.audioElement.pause(); } _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), onRejectedReason); _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.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 _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_16__.PlayStreamRejectedEvent({ reason: onRejectedReason })); }); } /** * Enable the video to play automatically if enableAutoplay is true */ autoPlayVideoOrSetUpPlayOverlay() { if (this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.AutoPlayVideo)) { // attempt to play the video this.playStream(); } this.resizePlayerStyle(); } /** * Connect to the Signaling server */ connectToSignallingServer() { this.locallyClosed = false; this.shouldReconnect = true; this.disconnectMessage = null; const signallingUrl = this.signallingUrlBuilder(); this.webSocketController.connect(signallingUrl); } /** * This will start the handshake to the signalling server * @param peerConfig - RTC Configuration Options from the Signaling server * @remark RTC Peer Connection on Ice Candidate event have it handled by handle Send Ice Candidate */ startSession(peerConfig) { this.peerConfig = peerConfig; // check for forcing turn if (this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.ForceTURN)) { // check for a turn server const hasTurnServer = this.checkTurnServerAvailability(peerConfig); // close and error if turn is forced and there is no turn server if (!hasTurnServer) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Info(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'No turn server was found in the Peer Connection Options. TURN cannot be forced, closing connection. Please use STUN instead'); this.closeSignalingServer('TURN cannot be forced, closing connection. Please use STUN instead.'); return; } } // set up the peer connection controller this.peerConnectionController = new _PeerConnectionController_PeerConnectionController__WEBPACK_IMPORTED_MODULE_17__.PeerConnectionController(this.peerConfig, this.config, this.preferredCodec); // set up peer connection controller video stats this.peerConnectionController.onVideoStats = (event) => this.handleVideoStats(event); /* When the Peer Connection wants to send an offer have it handled */ this.peerConnectionController.onSendWebRTCOffer = (offer) => this.handleSendWebRTCOffer(offer); /* When the Peer Connection wants to send an answer have it handled */ this.peerConnectionController.onSendWebRTCAnswer = (offer) => this.handleSendWebRTCAnswer(offer); /* When the Peer Connection ice candidate is added have it handled */ this.peerConnectionController.onPeerIceCandidate = (peerConnectionIceEvent) => this.handleSendIceCandidate(peerConnectionIceEvent); /* When the Peer Connection has a data channel created for it by the browser, handle it */ this.peerConnectionController.onDataChannel = (datachannelEvent) => this.handleDataChannel(datachannelEvent); // set up webRtc text overlays this.peerConnectionController.showTextOverlayConnecting = () => this.pixelStreaming._onWebRtcConnecting(); this.peerConnectionController.showTextOverlaySetupFailure = () => this.pixelStreaming._onWebRtcFailed(); let webRtcConnectedSent = false; this.peerConnectionController.onIceConnectionStateChange = () => { // Browsers emit "connected" when getting first connection and "completed" when finishing // candidate checking. However, sometimes browsers can skip "connected" and only emit "completed". // Therefore need to check both cases and emit onWebRtcConnected only once on the first hit. if (!webRtcConnectedSent && ["connected", "completed"].includes(this.peerConnectionController.peerConnection.iceConnectionState)) { this.pixelStreaming._onWebRtcConnected(); webRtcConnectedSent = true; } }; /* RTC Peer Connection on Track event -> handle on track */ this.peerConnectionController.onTrack = (trackEvent) => this.streamController.handleOnTrack(trackEvent); /* Start the Hand shake process by creating an Offer */ const BrowserSendsOffer = this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.BrowserSendOffer); if (BrowserSendsOffer) { // If browser is sending the offer, create an offer and send it to the streamer this.sendrecvDataChannelController.createDataChannel(this.peerConnectionController.peerConnection, 'cirrus', this.datachannelOptions); this.sendrecvDataChannelController.handleOnMessage = (ev) => this.handleOnMessage(ev); this.peerConnectionController.createOffer(this.sdpConstraints, this.config); } } /** * Checks the peer connection options for a turn server and returns true or false */ checkTurnServerAvailability(options) { // if iceServers is empty return false this should not be the general use case but is here incase if (!options.iceServers) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Info(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'A turn sever was not found'); return false; } // loop through the ice servers to check for a turn url for (const iceServer of options.iceServers) { for (const url of iceServer.urls) { if (url.includes('turn')) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `A turn sever was found at ${url}`); return true; } } } _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Info(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'A turn sever was not found'); return false; } /** * Handles when a Config Message is received contains the Peer Connection Options required (STUN and TURN Server Info) * @param messageConfig - Config Message received from the signaling server */ handleOnConfigMessage(messageConfig) { this.resizePlayerStyle(); // Tell the WebRtcController to start a session with the peer options sent from the signaling server this.startSession(messageConfig.peerConnectionOptions); // When the signaling server sends a WebRTC Answer over the websocket connection have the WebRtcController handle the message this.webSocketController.onWebRtcAnswer = (messageAnswer) => this.handleWebRtcAnswer(messageAnswer); this.webSocketController.onWebRtcOffer = (messageOffer) => this.handleWebRtcOffer(messageOffer); this.webSocketController.onWebRtcPeerDataChannels = (messageDataChannels) => this.handleWebRtcSFUPeerDatachannels(messageDataChannels); // When the signaling server sends a IceCandidate over the websocket connection have the WebRtcController handle the message this.webSocketController.onIceCandidate = (iceCandidate) => this.handleIceCandidate(iceCandidate); } /** * Handles when the signalling server gives us the list of streamer ids. */ handleStreamerListMessage(messageStreamerList) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `Got streamer list ${messageStreamerList.ids}`, 6); // add the streamers to the UI const settingOptions = [...messageStreamerList.ids]; // copy the original messageStreamerList.ids settingOptions.unshift(''); // add an empty option at the top this.config.setOptionSettingOptions(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.OptionParameters.StreamerId, settingOptions); let wantedStreamerId = null; let autoSelectedStreamerId = null; const waitForStreamer = this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.WaitForStreamer); const reconnectLimit = this.config.getNumericSettingValue(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.NumericParameters.MaxReconnectAttempts); const reconnectDelay = this.config.getNumericSettingValue(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.NumericParameters.StreamerAutoJoinInterval); // first we figure out a wanted streamer id through various means const urlParams = new URLSearchParams(window.location.search); if (urlParams.has(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.OptionParameters.StreamerId)) { // if we've set the streamer id on the url we only want that streamer id wantedStreamerId = urlParams.get(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.OptionParameters.StreamerId); } else if (this.subscribedStream) { // we were previously subscribed to a streamer, we want that wantedStreamerId = this.subscribedStream; } // now lets see if we can pick it. if (wantedStreamerId && messageStreamerList.ids.includes(wantedStreamerId)) { // if the wanted stream is in the list. we pick that autoSelectedStreamerId = wantedStreamerId; } else if ((!wantedStreamerId || !waitForStreamer) && messageStreamerList.ids.length == 1) { // otherwise, if we're not waiting for the wanted streamer and there's only one streamer, connect to it autoSelectedStreamerId = messageStreamerList.ids[0]; } // if we found a streamer id to auto select, select it if (autoSelectedStreamerId) { this.isReconnecting = false; this.reconnectAttempt = 0; this.config.setOptionSettingValue(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.OptionParameters.StreamerId, autoSelectedStreamerId); } else { // no auto selected streamer. // if we're waiting for a streamer then try reconnecting if (waitForStreamer) { if (this.reconnectAttempt < reconnectLimit) { // still reconnects available this.isReconnecting = true; this.reconnectAttempt++; setTimeout(() => { this.webSocketController.requestStreamerList(); }, reconnectDelay); } else { // We've exhausted our reconnect attempts, return to main screen this.reconnectAttempt = 0; this.isReconnecting = false; this.shouldReconnect = false; } } } // dispatch this event finally this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_16__.StreamerListMessageEvent({ messageStreamerList, autoSelectedStreamerId, wantedStreamerId })); } /** * Handle the RTC Answer from the signaling server * @param Answer - Answer SDP from the peer. */ handleWebRtcAnswer(Answer) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `Got answer sdp ${Answer.sdp}`, 6); const sdpAnswer = { sdp: Answer.sdp, type: 'answer' }; this.peerConnectionController.receiveAnswer(sdpAnswer); this.handlePostWebrtcNegotiation(); } /** * Handle the RTC offer from a WebRTC peer (received through the signalling server). * @param Offer - Offer SDP from the peer. */ handleWebRtcOffer(Offer) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `Got offer sdp ${Offer.sdp}`, 6); this.isUsingSFU = Offer.sfu ? Offer.sfu : false; if (this.isUsingSFU) { // Disable negotiating with the sfu as the sfu only supports one codec at a time this.peerConnectionController.preferredCodec = ''; } const sdpOffer = { sdp: Offer.sdp, type: 'offer' }; this.peerConnectionController.receiveOffer(sdpOffer, this.config); this.handlePostWebrtcNegotiation(); } /** * Handle when the SFU provides the peer with its data channels * @param DataChannels - The message from the SFU containing the data channels ids */ handleWebRtcSFUPeerDatachannels(DataChannels) { const SendOptions = { ordered: true, negotiated: true, id: DataChannels.sendStreamId }; const unidirectional = DataChannels.sendStreamId != DataChannels.recvStreamId; this.sendrecvDataChannelController.createDataChannel(this.peerConnectionController.peerConnection, unidirectional ? 'send-datachannel' : 'datachannel', SendOptions); if (unidirectional) { const RecvOptions = { ordered: true, negotiated: true, id: DataChannels.recvStreamId }; this.recvDataChannelController.createDataChannel(this.peerConnectionController.peerConnection, 'recv-datachannel', RecvOptions); this.recvDataChannelController.handleOnOpen = () => this.webSocketController.sendSFURecvDataChannelReady(); // If we're uni-directional, only the recv data channel should handle incoming messages this.recvDataChannelController.handleOnMessage = (ev) => this.handleOnMessage(ev); } else { // else our primary datachannel is send/recv so it can handle incoming messages this.sendrecvDataChannelController.handleOnMessage = (ev) => this.handleOnMessage(ev); } } handlePostWebrtcNegotiation() { // start the afk warning timer as PS is now running this.afkController.startAfkWarningTimer(); // show the overlay that we have negotiated a connection this.pixelStreaming._onWebRtcSdp(); if (this.statsTimerHandle && this.statsTimerHandle !== undefined) { window.clearInterval(this.statsTimerHandle); } this.statsTimerHandle = window.setInterval(() => this.getStats(), 1000); /* */ this.setMouseInputEnabled(this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.MouseInput)); this.setKeyboardInputEnabled(this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.KeyboardInput)); this.setGamePadInputEnabled(this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.GamepadInput)); } /** * When an ice Candidate is received from the Signaling server add it to the Peer Connection Client * @param iceCandidate - Ice Candidate from Server */ handleIceCandidate(iceCandidate) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'Web RTC Controller: onWebRtcIce', 6); const candidate = new RTCIceCandidate(iceCandidate); this.peerConnectionController.handleOnIce(candidate); } /** * Send the ice Candidate to the signaling server via websocket * @param iceEvent - RTC Peer ConnectionIceEvent) { */ handleSendIceCandidate(iceEvent) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'OnIceCandidate', 6); if (iceEvent.candidate && iceEvent.candidate.candidate) { this.webSocketController.sendIceCandidate(iceEvent.candidate); } } /** * Send the ice Candidate to the signaling server via websocket * @param iceEvent - RTC Peer ConnectionIceEvent) { */ handleDataChannel(datachannelEvent) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'Data channel created for us by browser as we are a receiving peer.', 6); this.sendrecvDataChannelController.dataChannel = datachannelEvent.channel; // Data channel was created for us, so we just need to setup its callbacks and array type this.sendrecvDataChannelController.setupDataChannel(); this.sendrecvDataChannelController.handleOnMessage = (ev) => this.handleOnMessage(ev); } /** * Send the RTC Offer Session to the Signaling server via websocket * @param offer - RTC Session Description */ handleSendWebRTCOffer(offer) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'Sending the offer to the Server', 6); this.webSocketController.sendWebRtcOffer(offer); } /** * Send the RTC Offer Session to the Signaling server via websocket * @param answer - RTC Session Description */ handleSendWebRTCAnswer(answer) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'Sending the answer to the Server', 6); this.webSocketController.sendWebRtcAnswer(answer); if (this.isUsingSFU) { this.webSocketController.sendWebRtcDatachannelRequest(); } } /** * Set the freeze frame overlay to the player div */ setUpMouseAndFreezeFrame() { // Calculating and normalizing positions depends on the width and height of the player. this.videoElementParentClientRect = this.videoPlayer .getVideoParentElement() .getBoundingClientRect(); this.coordinateConverter.setupNormalizeAndQuantize(); this.freezeFrameController.freezeFrame.resize(); } /** * Close the Connection to the signaling server */ closeSignalingServer(message) { var _a; // We explicitly called close, therefore we don't want to trigger auto reconnect this.locallyClosed = true; this.shouldReconnect = false; this.disconnectMessage = message; (_a = this.webSocketController) === null || _a === void 0 ? void 0 : _a.close(); } /** * Close the peer connection */ closePeerConnection() { var _a; (_a = this.peerConnectionController) === null || _a === void 0 ? void 0 : _a.close(); } /** * Close all connections */ close() { this.closeSignalingServer(''); this.closePeerConnection(); } /** * Fires a Video Stats Event in the RTC Peer Connection */ getStats() { this.peerConnectionController.generateStats(); } /** * Send a Latency Test Request to the UE Instance */ sendLatencyTest() { this.latencyStartTime = Date.now(); this.streamMessageController.toStreamerHandlers.get('LatencyTest')([JSON.stringify({ StartTime: this.latencyStartTime })]); } /** * Send a Data Channel Latency Test Request to the UE Instance */ sendDataChannelLatencyTest(descriptor) { this.streamMessageController.toStreamerHandlers.get('DataChannelLatencyTest')([JSON.stringify(descriptor)]); } /** * Send the MinQP encoder setting to the UE Instance. * @param minQP - The lower bound for QP when encoding * valid values are (1-51) where: * 1 = Best quality but highest bitrate. * 51 = Worst quality but lowest bitrate. * By default the minQP is 1 meaning the encoder is free * to aim for the best quality it can on the given network link. */ sendEncoderMinQP(minQP) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `MinQP=${minQP}\n`, 6); if (minQP != null) { this.streamMessageController.toStreamerHandlers.get('Command')([JSON.stringify({ 'Encoder.MinQP': minQP })]); } } /** * Send the MaxQP encoder setting to the UE Instance. * @param maxQP - The upper bound for QP when encoding * valid values are (1-51) where: * 1 = Best quality but highest bitrate. * 51 = Worst quality but lowest bitrate. * By default the maxQP is 51 meaning the encoder is free * to drop quality as low as needed on the given network link. */ sendEncoderMaxQP(maxQP) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `MaxQP=${maxQP}\n`, 6); if (maxQP != null) { this.streamMessageController.toStreamerHandlers.get('Command')([JSON.stringify({ 'Encoder.MaxQP': maxQP })]); } } /** * Send the { WebRTC.MinBitrate: SomeNumber }} command to UE to set * the minimum bitrate that we allow WebRTC to use * (note setting this too high in poor networks can be problematic). * @param minBitrate - The minimum bitrate we would like WebRTC to not fall below. */ sendWebRTCMinBitrate(minBitrate) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `WebRTC Min Bitrate=${minBitrate}`, 6); if (minBitrate != null) { this.streamMessageController.toStreamerHandlers.get('Command')([JSON.stringify({ 'WebRTC.MinBitrate': minBitrate })]); } } /** * Send the { WebRTC.MaxBitrate: SomeNumber }} command to UE to set * the minimum bitrate that we allow WebRTC to use * (note setting this too low could result in blocky video). * @param minBitrate - The minimum bitrate we would like WebRTC to not fall below. */ sendWebRTCMaxBitrate(maxBitrate) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `WebRTC Max Bitrate=${maxBitrate}`, 6); if (maxBitrate != null) { this.streamMessageController.toStreamerHandlers.get('Command')([JSON.stringify({ 'WebRTC.MaxBitrate': maxBitrate })]); } } /** * Send the { WebRTC.Fps: SomeNumber }} UE 5.0+ * and { WebRTC.MaxFps } UE 4.27 command to set * the maximum fps we would like WebRTC to stream at. * @param fps - The maximum stream fps. */ sendWebRTCFps(fps) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `WebRTC FPS=${fps}`, 6); if (fps != null) { this.streamMessageController.toStreamerHandlers.get('Command')([JSON.stringify({ 'WebRTC.Fps': fps })]); /* TODO: Remove when UE 4.27 unsupported. */ this.streamMessageController.toStreamerHandlers.get('Command')([JSON.stringify({ 'WebRTC.MaxFps': fps })]); } } /** * Sends the UI Descriptor `stat fps` to the UE Instance */ sendShowFps() { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), '---- Sending show stat to UE ----', 6); this.streamMessageController.toStreamerHandlers.get('Command')([JSON.stringify({ 'stat.fps': '' })]); } /** * Send an Iframe request to the streamer */ sendIframeRequest() { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), '---- Sending Request for an IFrame ----', 6); this.streamMessageController.toStreamerHandlers.get('IFrameRequest')(); } /** * Send a UIInteraction message */ emitUIInteraction(descriptor) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), '---- Sending custom UIInteraction message ----', 6); this.streamMessageController.toStreamerHandlers.get('UIInteraction')([JSON.stringify(descriptor)]); } /** * Send a Command message */ emitCommand(descriptor) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), '---- Sending custom Command message ----', 6); this.streamMessageController.toStreamerHandlers.get('Command')([JSON.stringify(descriptor)]); } /** * Send a console command message */ emitConsoleCommand(command) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), '---- Sending custom Command:ConsoleCommand message ----', 6); this.streamMessageController.toStreamerHandlers.get('Command')([JSON.stringify({ ConsoleCommand: command, })]); } /** * Sends a request to the UE Instance to have ownership of Quality */ sendRequestQualityControlOwnership() { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), '---- Sending Request to Control Quality ----', 6); this.toStreamerMessagesController.SendRequestQualityControl(); } /** * Handles when a Latency Test Result are received from the UE Instance * @param message - Latency Test Timings */ handleLatencyTestResult(message) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'DataChannelReceiveMessageType.latencyTest', 6); const latencyAsString = new TextDecoder('utf-16').decode(message.slice(1)); const latencyTestResults = new _DataChannel_LatencyTestResults__WEBPACK_IMPORTED_MODULE_18__.LatencyTestResults(); Object.assign(latencyTestResults, JSON.parse(latencyAsString)); latencyTestResults.processFields(); latencyTestResults.testStartTimeMs = this.latencyStartTime; latencyTestResults.browserReceiptTimeMs = Date.now(); latencyTestResults.latencyExcludingDecode = ~~(latencyTestResults.browserReceiptTimeMs - latencyTestResults.testStartTimeMs); latencyTestResults.testDuration = ~~(latencyTestResults.TransmissionTimeMs - latencyTestResults.ReceiptTimeMs); latencyTestResults.networkLatency = ~~(latencyTestResults.latencyExcludingDecode - latencyTestResults.testDuration); if (latencyTestResults.frameDisplayDeltaTimeMs && latencyTestResults.browserReceiptTimeMs) { latencyTestResults.endToEndLatency = ~~(latencyTestResults.frameDisplayDeltaTimeMs + latencyTestResults.networkLatency, +latencyTestResults.CaptureToSendMs); } this.pixelStreaming._onLatencyTestResult(latencyTestResults); } /** * Handles when a Data Channel Latency Test Response is received from the UE Instance * @param message - Data Channel Latency Test Response */ handleDataChannelLatencyTestResponse(message) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'DataChannelReceiveMessageType.dataChannelLatencyResponse', 6); const responseAsString = new TextDecoder('utf-16').decode(message.slice(1)); const latencyTestResponse = JSON.parse(responseAsString); this.pixelStreaming._onDataChannelLatencyTestResponse(latencyTestResponse); } /** * Handles when the Encoder and Web RTC Settings are received from the UE Instance * @param message - Initial Encoder and Web RTC Settings */ handleInitialSettings(message) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'DataChannelReceiveMessageType.InitialSettings', 6); const payloadAsString = new TextDecoder('utf-16').decode(message.slice(1)); const parsedInitialSettings = JSON.parse(payloadAsString); const initialSettings = new _DataChannel_InitialSettings__WEBPACK_IMPORTED_MODULE_19__.InitialSettings(); if (parsedInitialSettings.Encoder) { initialSettings.EncoderSettings = parsedInitialSettings.Encoder; } if (parsedInitialSettings.WebRTC) { initialSettings.WebRTCSettings = parsedInitialSettings.WebRTC; } if (parsedInitialSettings.PixelStreaming) { initialSettings.PixelStreamingSettings = parsedInitialSettings.PixelStreaming; } if (parsedInitialSettings.ConfigOptions && parsedInitialSettings.ConfigOptions.DefaultToHover !== undefined) { this.config.setFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.HoveringMouseMode, !!parsedInitialSettings.ConfigOptions.DefaultToHover); } initialSettings.ueCompatible(); _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), payloadAsString, 6); this.pixelStreaming._onInitialSettings(initialSettings); } /** * Handles when the Quantization Parameter are received from the UE Instance * @param message - Encoders Quantization Parameter */ handleVideoEncoderAvgQP(message) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'DataChannelReceiveMessageType.VideoEncoderAvgQP', 6); const AvgQP = Number(new TextDecoder('utf-16').decode(message.slice(1))); this.setVideoEncoderAvgQP(AvgQP); } /** * Handles when the video element has been loaded with a srcObject */ handleVideoInitialized() { this.pixelStreaming._onVideoInitialized(); // either autoplay the video or set up the play overlay this.autoPlayVideoOrSetUpPlayOverlay(); this.resizePlayerStyle(); this.videoPlayer.updateVideoStreamSize(); } /** * Flag set if the user has Quality Ownership * @param message - Does the current client have Quality Ownership */ onQualityControlOwnership(message) { const view = new Uint8Array(message); _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), 'DataChannelReceiveMessageType.QualityControlOwnership', 6); this.isQualityController = new Boolean(view[1]).valueOf(); _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `Received quality controller message, will control quality: ${this.isQualityController}`); this.pixelStreaming._onQualityControlOwnership(this.isQualityController); } /** * Handles when the Aggregated stats are Collected * @param stats - Aggregated Stats */ handleVideoStats(stats) { this.pixelStreaming._onVideoStats(stats); } /** * To Resize the Video Player element */ resizePlayerStyle() { this.videoPlayer.resizePlayerStyle(); } setPreferredCodec(codec) { this.preferredCodec = codec; if (this.peerConnectionController) { this.peerConnectionController.preferredCodec = codec; this.peerConnectionController.updateCodecSelection = false; } } setVideoEncoderAvgQP(avgQP) { this.videoAvgQp = avgQP; this.pixelStreaming._onVideoEncoderAvgQP(this.videoAvgQp); } /** * enables/disables keyboard event listeners */ setKeyboardInputEnabled(isEnabled) { var _a; (_a = this.keyboardController) === null || _a === void 0 ? void 0 : _a.unregisterKeyBoardEvents(); if (isEnabled) { this.keyboardController = this.inputClassesFactory.registerKeyBoard(this.config); } } /** * enables/disables mouse event listeners */ setMouseInputEnabled(isEnabled) { var _a; (_a = this.mouseController) === null || _a === void 0 ? void 0 : _a.unregisterMouseEvents(); if (isEnabled) { const mouseMode = this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.HoveringMouseMode) ? _Config_Config__WEBPACK_IMPORTED_MODULE_11__.ControlSchemeType.HoveringMouse : _Config_Config__WEBPACK_IMPORTED_MODULE_11__.ControlSchemeType.LockedMouse; this.mouseController = this.inputClassesFactory.registerMouse(mouseMode); } } /** * enables/disables touch event listeners */ setTouchInputEnabled(isEnabled) { var _a; (_a = this.touchController) === null || _a === void 0 ? void 0 : _a.unregisterTouchEvents(); if (isEnabled) { this.touchController = this.inputClassesFactory.registerTouch(this.config.isFlagEnabled(_Config_Config__WEBPACK_IMPORTED_MODULE_11__.Flags.FakeMouseWithTouches), this.videoElementParentClientRect); } } /** * enables/disables game pad event listeners */ setGamePadInputEnabled(isEnabled) { var _a; (_a = this.gamePadController) === null || _a === void 0 ? void 0 : _a.unregisterGamePadEvents(); if (isEnabled) { this.gamePadController = this.inputClassesFactory.registerGamePad(); this.gamePadController.onGamepadConnected = () => { this.streamMessageController.toStreamerHandlers.get('GamepadConnected')(); }; this.gamePadController.onGamepadDisconnected = (controllerIdx) => { this.streamMessageController.toStreamerHandlers.get('GamepadDisconnected')([controllerIdx]); }; } } registerDataChannelEventEmitters(dataChannel) { dataChannel.onOpen = (label, event) => this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_16__.DataChannelOpenEvent({ label, event })); dataChannel.onClose = (label, event) => this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_16__.DataChannelCloseEvent({ label, event })); dataChannel.onError = (label, event) => this.pixelStreaming.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_16__.DataChannelErrorEvent({ label, event })); } registerMessageHandler(name, direction, handler) { if (direction === _UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.FromStreamer && typeof handler === 'undefined') { _Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.Warning(_Logger_Logger__WEBPACK_IMPORTED_MODULE_15__.Logger.GetStackTrace(), `Unable to register handler for ${name} as no handler was passed`); } this.streamMessageController.registerMessageHandler(direction, name, (data) => (typeof handler === 'undefined' && direction === _UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_9__.MessageDirection.ToStreamer) ? this.sendMessageController.sendMessageToStreamer(name, data) : handler(data)); } } /***/ }), /***/ "./src/WebSockets/MessageReceive.ts": /*!******************************************!*\ !*** ./src/WebSockets/MessageReceive.ts ***! \******************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "MessageAnswer": () => (/* binding */ MessageAnswer), /* harmony export */ "MessageAuthRequired": () => (/* binding */ MessageAuthRequired), /* harmony export */ "MessageConfig": () => (/* binding */ MessageConfig), /* harmony export */ "MessageIceCandidate": () => (/* binding */ MessageIceCandidate), /* harmony export */ "MessageOffer": () => (/* binding */ MessageOffer), /* harmony export */ "MessageOnScreenKeyboard": () => (/* binding */ MessageOnScreenKeyboard), /* harmony export */ "MessagePeerDataChannels": () => (/* binding */ MessagePeerDataChannels), /* harmony export */ "MessagePlayerCount": () => (/* binding */ MessagePlayerCount), /* harmony export */ "MessageRecv": () => (/* binding */ MessageRecv), /* harmony export */ "MessageRecvTypes": () => (/* binding */ MessageRecvTypes), /* harmony export */ "MessageStreamerList": () => (/* binding */ MessageStreamerList) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * The Types of Messages that will be received */ var MessageRecvTypes; (function (MessageRecvTypes) { MessageRecvTypes["CONFIG"] = "config"; MessageRecvTypes["STREAMER_LIST"] = "streamerList"; MessageRecvTypes["PLAYER_COUNT"] = "playerCount"; MessageRecvTypes["OFFER"] = "offer"; MessageRecvTypes["ANSWER"] = "answer"; MessageRecvTypes["ICE_CANDIDATE"] = "iceCandidate"; MessageRecvTypes["PEER_DATA_CHANNELS"] = "peerDataChannels"; MessageRecvTypes["PING"] = "ping"; MessageRecvTypes["WARNING"] = "warning"; })(MessageRecvTypes || (MessageRecvTypes = {})); /** * Concrete Received Message wrapper */ class MessageRecv { } /** * Authentication Required Message wrapper */ class MessageAuthRequired extends MessageRecv { } /** * Config Message Wrapper */ class MessageConfig extends MessageRecv { } /** * Streamer List Message Wrapper */ class MessageStreamerList extends MessageRecv { } /** * Player Count Message wrapper */ class MessagePlayerCount extends MessageRecv { } /** * Web RTC offer Answer Message wrapper */ class MessageAnswer extends MessageRecv { } /** * WebRTC sdp offer Message wrapper. */ class MessageOffer extends MessageRecv { } /** * Ice Candidate Message wrapper */ class MessageIceCandidate extends MessageRecv { } /** * Peer Data Channels Message wrapper */ class MessagePeerDataChannels extends MessageRecv { } class MessageOnScreenKeyboard { } /***/ }), /***/ "./src/WebSockets/MessageSend.ts": /*!***************************************!*\ !*** ./src/WebSockets/MessageSend.ts ***! \***************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "MessageIceCandidate": () => (/* binding */ MessageIceCandidate), /* harmony export */ "MessageListStreamers": () => (/* binding */ MessageListStreamers), /* harmony export */ "MessagePong": () => (/* binding */ MessagePong), /* harmony export */ "MessageSFURecvDataChannelReady": () => (/* binding */ MessageSFURecvDataChannelReady), /* harmony export */ "MessageSend": () => (/* binding */ MessageSend), /* harmony export */ "MessageSendTypes": () => (/* binding */ MessageSendTypes), /* harmony export */ "MessageSubscribe": () => (/* binding */ MessageSubscribe), /* harmony export */ "MessageUnsubscribe": () => (/* binding */ MessageUnsubscribe), /* harmony export */ "MessageWebRTCAnswer": () => (/* binding */ MessageWebRTCAnswer), /* harmony export */ "MessageWebRTCDatachannelRequest": () => (/* binding */ MessageWebRTCDatachannelRequest), /* harmony export */ "MessageWebRTCOffer": () => (/* binding */ MessageWebRTCOffer) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * The Send Types that are pushed from the signaling server */ var MessageSendTypes; (function (MessageSendTypes) { MessageSendTypes["LIST_STREAMERS"] = "listStreamers"; MessageSendTypes["SUBSCRIBE"] = "subscribe"; MessageSendTypes["UNSUBSCRIBE"] = "unsubscribe"; MessageSendTypes["ICE_CANDIDATE"] = "iceCandidate"; MessageSendTypes["OFFER"] = "offer"; MessageSendTypes["ANSWER"] = "answer"; MessageSendTypes["DATACHANNELREQUEST"] = "dataChannelRequest"; MessageSendTypes["SFURECVDATACHANNELREADY"] = "peerDataChannelsReady"; MessageSendTypes["PONG"] = "pong"; })(MessageSendTypes || (MessageSendTypes = {})); /** * A Wrapper for the message to send to the signaling server */ class MessageSend { /** * Turns the wrapper into a JSON String * @returns - JSON String of the Message to send */ payload() { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Sending => \n' + JSON.stringify(this, undefined, 4), 6); return JSON.stringify(this); } } class MessageListStreamers extends MessageSend { constructor() { super(); this.type = MessageSendTypes.LIST_STREAMERS; } } class MessageSubscribe extends MessageSend { constructor(streamerid) { super(); this.type = MessageSendTypes.SUBSCRIBE; this.streamerId = streamerid; } } class MessageUnsubscribe extends MessageSend { constructor() { super(); this.type = MessageSendTypes.UNSUBSCRIBE; } } /** * Instance Request Message Wrapper */ class MessagePong extends MessageSend { constructor(time) { super(); this.type = MessageSendTypes.PONG; this.time = time; } } /** * Web RTC Offer message wrapper */ class MessageWebRTCOffer extends MessageSend { /** * @param offer - Generated Web RTC Offer */ constructor(offer) { super(); this.type = MessageSendTypes.OFFER; if (offer) { this.type = offer.type; this.sdp = offer.sdp; } } } /** * Web RTC Answer message wrapper */ class MessageWebRTCAnswer extends MessageSend { /** * @param answer - Generated Web RTC Offer */ constructor(answer) { super(); this.type = MessageSendTypes.ANSWER; if (answer) { this.type = answer.type; this.sdp = answer.sdp; } } } /** * Web RTC Data channel request message wrapper */ class MessageWebRTCDatachannelRequest extends MessageSend { constructor() { super(); this.type = MessageSendTypes.DATACHANNELREQUEST; } } /** * Web RTC SFU Data channel ready message wrapper */ class MessageSFURecvDataChannelReady extends MessageSend { constructor() { super(); this.type = MessageSendTypes.SFURECVDATACHANNELREADY; } } /** * RTC Ice Candidate Wrapper */ class MessageIceCandidate { /** * @param candidate - RTC Ice Candidate */ constructor(candidate) { this.type = MessageSendTypes.ICE_CANDIDATE; this.candidate = candidate; } /** * Turns the wrapper into a JSON String * @returns - JSON String of the Message to send */ payload() { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Sending => \n' + JSON.stringify(this, undefined, 4), 6); return JSON.stringify(this); } } /***/ }), /***/ "./src/WebSockets/SignallingProtocol.ts": /*!**********************************************!*\ !*** ./src/WebSockets/SignallingProtocol.ts ***! \**********************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SignallingProtocol": () => (/* binding */ SignallingProtocol) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _MessageReceive__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./MessageReceive */ "./src/WebSockets/MessageReceive.ts"); /* harmony import */ var _MessageSend__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./MessageSend */ "./src/WebSockets/MessageSend.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Signalling protocol for handling messages from the signalling server. */ class SignallingProtocol { constructor() { this.FromUEMessageHandlers = new Map(); } addMessageHandler(messageId, messageHandler) { this.FromUEMessageHandlers.set(messageId, messageHandler); } handleMessage(messageId, messageData) { if (this.FromUEMessageHandlers.has(messageId)) { this.FromUEMessageHandlers.get(messageId)(messageData); } else { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Message type of ${messageId} does not have a message handler registered on the frontend - ignoring message.`); } } /** * Setup any default signalling message handling, these can be overridden or additional handlers added with `addMessageHandler`. * @param websocketController The controller to setup these handlers on. */ static setupDefaultHandlers(websocketController) { // PING websocketController.signallingProtocol.addMessageHandler(_MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.PING, (pingPayload) => { // send our pong payload back to the signalling server const pongPayload = new _MessageSend__WEBPACK_IMPORTED_MODULE_2__.MessagePong(new Date().getTime()).payload(); _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), _MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.PING + ': ' + pingPayload, 6); websocketController.webSocket.send(pongPayload); }); // CONFIG websocketController.signallingProtocol.addMessageHandler(_MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.CONFIG, (configPayload) => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), _MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.CONFIG, 6); const config = JSON.parse(configPayload); websocketController.onConfig(config); }); // STREAMER_LIST websocketController.signallingProtocol.addMessageHandler(_MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.STREAMER_LIST, (listPayload) => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), _MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.STREAMER_LIST, 6); const streamerList = JSON.parse(listPayload); websocketController.onStreamerList(streamerList); }); // PLAYER_COUNT websocketController.signallingProtocol.addMessageHandler(_MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.PLAYER_COUNT, (playerCountPayload) => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), _MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.PLAYER_COUNT, 6); const playerCount = JSON.parse(playerCountPayload); _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Player Count: ' + playerCount.count, 6); websocketController.onPlayerCount(playerCount); }); // ANSWER websocketController.signallingProtocol.addMessageHandler(_MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.ANSWER, (answerPayload) => { // send our pong payload back to the signalling server _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), _MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.ANSWER, 6); const answer = JSON.parse(answerPayload); websocketController.onWebRtcAnswer(answer); }); // OFFER websocketController.signallingProtocol.addMessageHandler(_MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.OFFER, (offerPayload) => { // send our pong payload back to the signalling server _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), _MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.OFFER, 6); const offer = JSON.parse(offerPayload); websocketController.onWebRtcOffer(offer); }); // ICE CANDIDATE websocketController.signallingProtocol.addMessageHandler(_MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.ICE_CANDIDATE, (iceCandidatePayload) => { // send our pong payload back to the signalling server _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), _MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.ICE_CANDIDATE, 6); const iceCandidate = JSON.parse(iceCandidatePayload); websocketController.onIceCandidate(iceCandidate.candidate); }); // WARNING websocketController.signallingProtocol.addMessageHandler(_MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.WARNING, (warningPayload) => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Warning(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Warning received: ${warningPayload}`); }); // PEER DATA CHANNELS websocketController.signallingProtocol.addMessageHandler(_MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.PEER_DATA_CHANNELS, (peerDataChannelsPayload) => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), _MessageReceive__WEBPACK_IMPORTED_MODULE_1__.MessageRecvTypes.PEER_DATA_CHANNELS, 6); const peerDataChannels = JSON.parse(peerDataChannelsPayload); websocketController.onWebRtcPeerDataChannels(peerDataChannels); }); } } /***/ }), /***/ "./src/WebSockets/WebSocketController.ts": /*!***********************************************!*\ !*** ./src/WebSockets/WebSocketController.ts ***! \***********************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "WebSocketController": () => (/* binding */ WebSocketController) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _MessageSend__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./MessageSend */ "./src/WebSockets/MessageSend.ts"); /* harmony import */ var _SignallingProtocol__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./SignallingProtocol */ "./src/WebSockets/SignallingProtocol.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * The controller for the WebSocket and all associated methods */ class WebSocketController { constructor() { this.WS_OPEN_STATE = 1; this.onOpen = new EventTarget(); this.onClose = new EventTarget(); this.signallingProtocol = new _SignallingProtocol__WEBPACK_IMPORTED_MODULE_0__.SignallingProtocol(); _SignallingProtocol__WEBPACK_IMPORTED_MODULE_0__.SignallingProtocol.setupDefaultHandlers(this); } /** * Connect to the signaling server * @param connectionURL - The Address of the signaling server * @returns - If there is a connection */ connect(connectionURL) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), connectionURL, 6); try { this.webSocket = new WebSocket(connectionURL); this.webSocket.onopen = (event) => this.handleOnOpen(event); this.webSocket.onerror = () => this.handleOnError(); this.webSocket.onclose = (event) => this.handleOnClose(event); this.webSocket.onmessage = (event) => this.handleOnMessage(event); this.webSocket.onmessagebinary = (event) => this.handleOnMessageBinary(event); return true; } catch (error) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Error(error, error); return false; } } /** * Handles what happens when a message is received in binary form * @param event - Message Received */ handleOnMessageBinary(event) { // if the event is empty return if (!event || !event.data) { return; } // handle the binary and then handle the message event.data .text() .then((messageString) => { // build a new message const constructedMessage = new MessageEvent('messageFromBinary', { data: messageString }); // send the new stringified event back into `onmessage` this.handleOnMessage(constructedMessage); }) .catch((error) => { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), `Failed to parse binary blob from websocket, reason: ${error}`); }); } /** * Handles what happens when a message is received * @param event - Message Received */ handleOnMessage(event) { // Check if websocket message is binary, if so, stringify it. if (event.data && event.data instanceof Blob) { this.handleOnMessageBinary(event); return; } const message = JSON.parse(event.data); _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'received => \n' + JSON.stringify(JSON.parse(event.data), undefined, 4), 6); // Send to our signalling protocol to handle the incoming message this.signallingProtocol.handleMessage(message.type, event.data); } /** * Handles when the Websocket is opened * @param event - Not Used */ // eslint-disable-next-line @typescript-eslint/no-unused-vars handleOnOpen(event) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'Connected to the signalling server via WebSocket', 6); this.onOpen.dispatchEvent(new Event('open')); } /** * Handles when there is an error on the websocket * @param event - Error Payload */ handleOnError() { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Error(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'WebSocket error'); } /** * Handles when the Websocket is closed * @param event - Close Event */ handleOnClose(event) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'Disconnected to the signalling server via WebSocket: ' + JSON.stringify(event.code) + ' - ' + event.reason); this.onClose.dispatchEvent(new CustomEvent('close', { 'detail': event })); } requestStreamerList() { const payload = new _MessageSend__WEBPACK_IMPORTED_MODULE_2__.MessageListStreamers(); this.webSocket.send(payload.payload()); } sendSubscribe(streamerid) { const payload = new _MessageSend__WEBPACK_IMPORTED_MODULE_2__.MessageSubscribe(streamerid); this.webSocket.send(payload.payload()); } sendUnsubscribe() { const payload = new _MessageSend__WEBPACK_IMPORTED_MODULE_2__.MessageUnsubscribe(); this.webSocket.send(payload.payload()); } sendWebRtcOffer(offer) { const payload = new _MessageSend__WEBPACK_IMPORTED_MODULE_2__.MessageWebRTCOffer(offer); this.webSocket.send(payload.payload()); } sendWebRtcAnswer(answer) { const payload = new _MessageSend__WEBPACK_IMPORTED_MODULE_2__.MessageWebRTCAnswer(answer); this.webSocket.send(payload.payload()); } sendWebRtcDatachannelRequest() { const payload = new _MessageSend__WEBPACK_IMPORTED_MODULE_2__.MessageWebRTCDatachannelRequest(); this.webSocket.send(payload.payload()); } sendSFURecvDataChannelReady() { const payload = new _MessageSend__WEBPACK_IMPORTED_MODULE_2__.MessageSFURecvDataChannelReady(); this.webSocket.send(payload.payload()); } /** * Sends an RTC Ice Candidate to the Server * @param candidate - RTC Ice Candidate */ sendIceCandidate(candidate) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'Sending Ice Candidate'); if (this.webSocket && this.webSocket.readyState === this.WS_OPEN_STATE) { //ws.send(JSON.stringify({ type: 'iceCandidate', candidate: candidate })); const IceCandidate = new _MessageSend__WEBPACK_IMPORTED_MODULE_2__.MessageIceCandidate(candidate); this.webSocket.send(IceCandidate.payload()); } } /** * Closes the Websocket connection */ close() { var _a; (_a = this.webSocket) === null || _a === void 0 ? void 0 : _a.close(); } /** * The Message Contains the payload of the peer connection options used for the RTC Peer hand shake * @param messageConfig - Config Message received from he signaling server */ // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function onConfig(messageConfig) { } /** * The Message Contains the payload of the peer connection options used for the RTC Peer hand shake * @param messageConfig - Config Message received from he signaling server */ // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function onStreamerList(messageStreamerList) { } /** * @param iceCandidate - Ice Candidate sent from the Signaling server server's RTC hand shake */ // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function onIceCandidate(iceCandidate) { } /** * Event is fired when the websocket receives the answer for the RTC peer Connection * @param messageAnswer - The RTC Answer payload from the signaling server */ // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function onWebRtcAnswer(messageAnswer) { } /** * Event is fired when the websocket receives the offer for the RTC peer Connection * @param messageOffer - The sdp offer */ // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function onWebRtcOffer(messageOffer) { } /** * Event is fired when the websocket receives the data channels for the RTC peer Connection from the SFU * @param messageDataChannels - The data channels details */ // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function onWebRtcPeerDataChannels(messageDataChannels) { } /** * Event is fired when the websocket receives the an updated player count from cirrus * @param MessagePlayerCount - The new player count */ // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function onPlayerCount(playerCount) { } } /***/ }), /***/ "./src/WebXR/WebXRController.ts": /*!**************************************!*\ !*** ./src/WebXR/WebXRController.ts ***! \**************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "WebXRController": () => (/* binding */ WebXRController) /* harmony export */ }); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _Util_WebGLUtils__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../Util/WebGLUtils */ "./src/Util/WebGLUtils.ts"); /* harmony import */ var _Inputs_XRGamepadController__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../Inputs/XRGamepadController */ "./src/Inputs/XRGamepadController.ts"); /* harmony import */ var _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../Util/EventEmitter */ "./src/Util/EventEmitter.ts"); /* harmony import */ var _pixelstreamingfrontend__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../pixelstreamingfrontend */ "./src/Config/Config.ts"); // Copyright Epic Games, Inc. All Rights Reserved. class WebXRController { constructor(webRtcPlayerController) { this.xrSession = null; this.webRtcController = webRtcPlayerController; this.xrControllers = []; this.xrGamepadController = new _Inputs_XRGamepadController__WEBPACK_IMPORTED_MODULE_0__.XRGamepadController(this.webRtcController.streamMessageController); this.onSessionEnded = new EventTarget(); this.onSessionStarted = new EventTarget(); this.onFrame = new EventTarget(); } xrClicked() { if (!this.xrSession) { navigator.xr .requestSession('immersive-vr') .then((session) => { this.onXrSessionStarted(session); }); } else { this.xrSession.end(); } } onXrSessionEnded() { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'XR Session ended'); this.xrSession = null; this.onSessionEnded.dispatchEvent(new Event('xrSessionEnded')); } onXrSessionStarted(session) { _Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.Log(_Logger_Logger__WEBPACK_IMPORTED_MODULE_1__.Logger.GetStackTrace(), 'XR Session started'); this.xrSession = session; this.xrSession.addEventListener('end', () => { this.onXrSessionEnded(); }); const canvas = document.createElement('canvas'); this.gl = canvas.getContext('webgl2', { xrCompatible: true }); this.xrSession.updateRenderState({ baseLayer: new XRWebGLLayer(this.xrSession, this.gl) }); // setup vertex shader const vertexShader = this.gl.createShader(this.gl.VERTEX_SHADER); this.gl.shaderSource(vertexShader, _Util_WebGLUtils__WEBPACK_IMPORTED_MODULE_2__.WebGLUtils.vertexShader()); this.gl.compileShader(vertexShader); // setup fragment shader const fragmentShader = this.gl.createShader(this.gl.FRAGMENT_SHADER); this.gl.shaderSource(fragmentShader, _Util_WebGLUtils__WEBPACK_IMPORTED_MODULE_2__.WebGLUtils.fragmentShader()); this.gl.compileShader(fragmentShader); // setup GLSL program const shaderProgram = this.gl.createProgram(); this.gl.attachShader(shaderProgram, vertexShader); this.gl.attachShader(shaderProgram, fragmentShader); this.gl.linkProgram(shaderProgram); this.gl.useProgram(shaderProgram); // look up where vertex data needs to go this.positionLocation = this.gl.getAttribLocation(shaderProgram, 'a_position'); this.texcoordLocation = this.gl.getAttribLocation(shaderProgram, 'a_texCoord'); // Create a buffer to put three 2d clip space points in this.positionBuffer = this.gl.createBuffer(); // Bind it to ARRAY_BUFFER (think of it as ARRAY_BUFFER = positionBuffer) this.gl.bindBuffer(this.gl.ARRAY_BUFFER, this.positionBuffer); // Turn on the position attribute this.gl.enableVertexAttribArray(this.positionLocation); // Create a texture. const texture = this.gl.createTexture(); this.gl.bindTexture(this.gl.TEXTURE_2D, texture); // Set the parameters so we can render any size image. 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(); // lookup uniforms this.resolutionLocation = this.gl.getUniformLocation(shaderProgram, 'u_resolution'); this.offsetLocation = this.gl.getUniformLocation(shaderProgram, 'u_offset'); session.requestReferenceSpace('local').then((refSpace) => { this.xrRefSpace = refSpace; this.xrSession.requestAnimationFrame((time, frame) => this.onXrFrame(time, frame)); }); this.onSessionStarted.dispatchEvent(new Event('xrSessionStarted')); } onXrFrame(time, frame) { const pose = frame.getViewerPose(this.xrRefSpace); if (pose) { const matrix = pose.transform.matrix; const mat = []; for (let i = 0; i < 16; i++) { mat[i] = new Float32Array([matrix[i]])[0]; } // prettier-ignore this.webRtcController.streamMessageController.toStreamerHandlers.get('XRHMDTransform')([ mat[0], mat[4], mat[8], mat[12], mat[1], mat[5], mat[9], mat[13], mat[2], mat[6], mat[10], mat[14], mat[3], mat[7], mat[11], mat[15] ]); const glLayer = this.xrSession.renderState.baseLayer; // If we do have a valid pose, bind the WebGL layer's framebuffer, // which is where any content to be displayed on the XRDevice must be // rendered. this.gl.bindFramebuffer(this.gl.FRAMEBUFFER, glLayer.framebuffer); // Upload the image into the texture. WebGL knows how to extract the current frame from the video element 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()); } if (this.webRtcController.config.isFlagEnabled(_pixelstreamingfrontend__WEBPACK_IMPORTED_MODULE_3__.Flags.XRControllerInput)) { this.xrSession.inputSources.forEach((source, index, array) => { this.xrGamepadController.updateStatus(source, frame, this.xrRefSpace); }, this); } this.xrSession.requestAnimationFrame((time, frame) => this.onXrFrame(time, frame)); this.onFrame.dispatchEvent(new _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_4__.XrFrameEvent({ time, frame })); } render(videoElement) { if (!this.gl) { return; } const glLayer = this.xrSession.renderState.baseLayer; this.gl.viewport(0, 0, glLayer.framebufferWidth, glLayer.framebufferHeight); this.gl.uniform4f(this.offsetLocation, 1.0, 1.0, 0.0, 0.0); // Set rectangle // prettier-ignore this.gl.bufferData(this.gl.ARRAY_BUFFER, new Float32Array([ 0, 0, videoElement.videoWidth, 0, 0, videoElement.videoHeight, 0, videoElement.videoHeight, videoElement.videoWidth, 0, videoElement.videoWidth, videoElement.videoHeight ]), this.gl.STATIC_DRAW); // Provide texture coordinates for the rectangle this.gl.bindBuffer(this.gl.ARRAY_BUFFER, this.texcoordBuffer); this.gl.bufferData(this.gl.ARRAY_BUFFER, new Float32Array([ 0.0, 0.0, 1.0, 0.0, 0.0, 1.0, 0.0, 1.0, 1.0, 0.0, 1.0, 1.0 ]), this.gl.STATIC_DRAW); let size; // components per iteration let type; // the data type let normalize; // normalize the data let stride; // 0 = move forward size * sizeof(type) each iteration to get the next position let offset; // start position of the buffer // Bind the position buffer. this.gl.bindBuffer(this.gl.ARRAY_BUFFER, this.positionBuffer); // Tell the position attribute how to get data out of positionBuffer (ARRAY_BUFFER) size = 2; // 2 components per iteration type = this.gl.FLOAT; // the data is 32bit floats normalize = false; // don't normalize the data stride = 0; // 0 = move forward size * sizeof(type) each iteration to get the next position offset = 0; // start at the beginning of the buffer this.gl.vertexAttribPointer(this.positionLocation, size, type, normalize, stride, offset); // Turn on the texcoord attribute this.gl.enableVertexAttribArray(this.texcoordLocation); // bind the texcoord buffer. this.gl.bindBuffer(this.gl.ARRAY_BUFFER, this.texcoordBuffer); // Tell the texcoord attribute how to get data out of texcoordBuffer (ARRAY_BUFFER) size = 2; // 2 components per iteration type = this.gl.FLOAT; // the data is 32bit floats normalize = false; // don't normalize the data stride = 0; // 0 = move forward size * sizeof(type) each iteration to get the next position offset = 0; // start at the beginning of the buffer this.gl.vertexAttribPointer(this.texcoordLocation, size, type, normalize, stride, offset); // set the resolution this.gl.uniform2f(this.resolutionLocation, videoElement.videoWidth, videoElement.videoHeight); // draw the rectangle. const primitiveType = this.gl.TRIANGLES; const count = 6; offset = 0; this.gl.drawArrays(primitiveType, offset, count); } static isSessionSupported(mode) { if (navigator.xr) { return navigator.xr.isSessionSupported(mode); } else { return new Promise(() => { return false; }); } } } /***/ }), /***/ "sdp": /*!**********************!*\ !*** external "sdp" ***! \**********************/ /***/ ((module) => { module.exports = __WEBPACK_EXTERNAL_MODULE_sdp__; /***/ }) /******/ }); /************************************************************************/ /******/ // The module cache /******/ var __webpack_module_cache__ = {}; /******/ /******/ // The require function /******/ function __webpack_require__(moduleId) { /******/ // Check if module is in cache /******/ var cachedModule = __webpack_module_cache__[moduleId]; /******/ if (cachedModule !== undefined) { /******/ return cachedModule.exports; /******/ } /******/ // Create a new module (and put it into the cache) /******/ var module = __webpack_module_cache__[moduleId] = { /******/ // no module.id needed /******/ // no module.loaded needed /******/ exports: {} /******/ }; /******/ /******/ // Execute the module function /******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__); /******/ /******/ // Return the exports of the module /******/ return module.exports; /******/ } /******/ /************************************************************************/ /******/ /* webpack/runtime/compat get default export */ /******/ (() => { /******/ // getDefaultExport function for compatibility with non-harmony modules /******/ __webpack_require__.n = (module) => { /******/ var getter = module && module.__esModule ? /******/ () => (module['default']) : /******/ () => (module); /******/ __webpack_require__.d(getter, { a: getter }); /******/ return getter; /******/ }; /******/ })(); /******/ /******/ /* webpack/runtime/define property getters */ /******/ (() => { /******/ // define getter functions for harmony exports /******/ __webpack_require__.d = (exports, definition) => { /******/ for(var key in definition) { /******/ if(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) { /******/ Object.defineProperty(exports, key, { enumerable: true, get: definition[key] }); /******/ } /******/ } /******/ }; /******/ })(); /******/ /******/ /* webpack/runtime/hasOwnProperty shorthand */ /******/ (() => { /******/ __webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop)) /******/ })(); /******/ /******/ /* webpack/runtime/make namespace object */ /******/ (() => { /******/ // define __esModule on exports /******/ __webpack_require__.r = (exports) => { /******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) { /******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' }); /******/ } /******/ Object.defineProperty(exports, '__esModule', { value: true }); /******/ }; /******/ })(); /******/ /************************************************************************/ var __webpack_exports__ = {}; // This entry need to be wrapped in an IIFE because it need to be isolated against other modules in the chunk. (() => { /*!***************************************!*\ !*** ./src/pixelstreamingfrontend.ts ***! \***************************************/ __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "AfkLogic": () => (/* reexport safe */ _AFK_AFKController__WEBPACK_IMPORTED_MODULE_9__.AFKController), /* harmony export */ "AfkTimedOutEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.AfkTimedOutEvent), /* harmony export */ "AfkWarningActivateEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.AfkWarningActivateEvent), /* harmony export */ "AfkWarningDeactivateEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.AfkWarningDeactivateEvent), /* harmony export */ "AfkWarningUpdateEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.AfkWarningUpdateEvent), /* harmony export */ "AggregatedStats": () => (/* reexport safe */ _PeerConnectionController_AggregatedStats__WEBPACK_IMPORTED_MODULE_12__.AggregatedStats), /* harmony export */ "CandidatePairStats": () => (/* reexport safe */ _PeerConnectionController_CandidatePairStats__WEBPACK_IMPORTED_MODULE_20__.CandidatePairStats), /* harmony export */ "CandidateStat": () => (/* reexport safe */ _PeerConnectionController_CandidateStat__WEBPACK_IMPORTED_MODULE_21__.CandidateStat), /* harmony export */ "Config": () => (/* reexport safe */ _Config_Config__WEBPACK_IMPORTED_MODULE_2__.Config), /* harmony export */ "ControlSchemeType": () => (/* reexport safe */ _Config_Config__WEBPACK_IMPORTED_MODULE_2__.ControlSchemeType), /* harmony export */ "DataChannelCloseEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.DataChannelCloseEvent), /* harmony export */ "DataChannelErrorEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.DataChannelErrorEvent), /* harmony export */ "DataChannelLatencyTestResponseEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.DataChannelLatencyTestResponseEvent), /* harmony export */ "DataChannelLatencyTestResultEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.DataChannelLatencyTestResultEvent), /* harmony export */ "DataChannelOpenEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.DataChannelOpenEvent), /* harmony export */ "DataChannelStats": () => (/* reexport safe */ _PeerConnectionController_DataChannelStats__WEBPACK_IMPORTED_MODULE_22__.DataChannelStats), /* harmony export */ "EncoderSettings": () => (/* reexport safe */ _DataChannel_InitialSettings__WEBPACK_IMPORTED_MODULE_11__.EncoderSettings), /* harmony export */ "EventEmitter": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.EventEmitter), /* harmony export */ "Flags": () => (/* reexport safe */ _Config_Config__WEBPACK_IMPORTED_MODULE_2__.Flags), /* harmony export */ "HideFreezeFrameEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.HideFreezeFrameEvent), /* harmony export */ "InboundAudioStats": () => (/* reexport safe */ _PeerConnectionController_InboundRTPStats__WEBPACK_IMPORTED_MODULE_23__.InboundAudioStats), /* harmony export */ "InboundVideoStats": () => (/* reexport safe */ _PeerConnectionController_InboundRTPStats__WEBPACK_IMPORTED_MODULE_23__.InboundVideoStats), /* harmony export */ "InitialSettings": () => (/* reexport safe */ _DataChannel_InitialSettings__WEBPACK_IMPORTED_MODULE_11__.InitialSettings), /* harmony export */ "InitialSettingsEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.InitialSettingsEvent), /* harmony export */ "LatencyTestResultEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.LatencyTestResultEvent), /* harmony export */ "LatencyTestResults": () => (/* reexport safe */ _DataChannel_LatencyTestResults__WEBPACK_IMPORTED_MODULE_10__.LatencyTestResults), /* harmony export */ "LoadFreezeFrameEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.LoadFreezeFrameEvent), /* harmony export */ "Logger": () => (/* reexport safe */ _Logger_Logger__WEBPACK_IMPORTED_MODULE_13__.Logger), /* harmony export */ "MessageDirection": () => (/* reexport safe */ _UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_15__.MessageDirection), /* harmony export */ "MessageRecv": () => (/* reexport safe */ _WebSockets_MessageReceive__WEBPACK_IMPORTED_MODULE_17__.MessageRecv), /* harmony export */ "MessageSend": () => (/* reexport safe */ _WebSockets_MessageSend__WEBPACK_IMPORTED_MODULE_16__.MessageSend), /* harmony export */ "MessageStreamerList": () => (/* reexport safe */ _WebSockets_MessageReceive__WEBPACK_IMPORTED_MODULE_17__.MessageStreamerList), /* harmony export */ "NumericParameters": () => (/* reexport safe */ _Config_Config__WEBPACK_IMPORTED_MODULE_2__.NumericParameters), /* harmony export */ "OptionParameters": () => (/* reexport safe */ _Config_Config__WEBPACK_IMPORTED_MODULE_2__.OptionParameters), /* harmony export */ "OutBoundVideoStats": () => (/* reexport safe */ _PeerConnectionController_OutBoundRTPStats__WEBPACK_IMPORTED_MODULE_24__.OutBoundVideoStats), /* harmony export */ "PixelStreaming": () => (/* reexport safe */ _PixelStreaming_PixelStreaming__WEBPACK_IMPORTED_MODULE_8__.PixelStreaming), /* harmony export */ "PlayStreamErrorEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.PlayStreamErrorEvent), /* harmony export */ "PlayStreamEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.PlayStreamEvent), /* harmony export */ "PlayStreamRejectedEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.PlayStreamRejectedEvent), /* harmony export */ "PlayerCountEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.PlayerCountEvent), /* harmony export */ "SettingBase": () => (/* reexport safe */ _Config_SettingBase__WEBPACK_IMPORTED_MODULE_3__.SettingBase), /* harmony export */ "SettingFlag": () => (/* reexport safe */ _Config_SettingFlag__WEBPACK_IMPORTED_MODULE_4__.SettingFlag), /* harmony export */ "SettingNumber": () => (/* reexport safe */ _Config_SettingNumber__WEBPACK_IMPORTED_MODULE_5__.SettingNumber), /* harmony export */ "SettingOption": () => (/* reexport safe */ _Config_SettingOption__WEBPACK_IMPORTED_MODULE_6__.SettingOption), /* harmony export */ "SettingText": () => (/* reexport safe */ _Config_SettingText__WEBPACK_IMPORTED_MODULE_7__.SettingText), /* harmony export */ "SettingsChangedEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.SettingsChangedEvent), /* harmony export */ "SignallingProtocol": () => (/* reexport safe */ _WebSockets_SignallingProtocol__WEBPACK_IMPORTED_MODULE_19__.SignallingProtocol), /* harmony export */ "StatsReceivedEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.StatsReceivedEvent), /* harmony export */ "StreamLoadingEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.StreamLoadingEvent), /* harmony export */ "StreamPreConnectEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.StreamPreConnectEvent), /* harmony export */ "StreamPreDisconnectEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.StreamPreDisconnectEvent), /* harmony export */ "StreamReconnectEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.StreamReconnectEvent), /* harmony export */ "StreamerListMessageEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.StreamerListMessageEvent), /* harmony export */ "TextParameters": () => (/* reexport safe */ _Config_Config__WEBPACK_IMPORTED_MODULE_2__.TextParameters), /* harmony export */ "UnquantizedAndDenormalizeUnsigned": () => (/* reexport safe */ _Util_CoordinateConverter__WEBPACK_IMPORTED_MODULE_14__.UnquantizedDenormalizedUnsignedCoord), /* harmony export */ "VideoEncoderAvgQPEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.VideoEncoderAvgQPEvent), /* harmony export */ "VideoInitializedEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.VideoInitializedEvent), /* harmony export */ "WebRTCSettings": () => (/* reexport safe */ _DataChannel_InitialSettings__WEBPACK_IMPORTED_MODULE_11__.WebRTCSettings), /* harmony export */ "WebRtcAutoConnectEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.WebRtcAutoConnectEvent), /* harmony export */ "WebRtcConnectedEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.WebRtcConnectedEvent), /* harmony export */ "WebRtcConnectingEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.WebRtcConnectingEvent), /* harmony export */ "WebRtcDisconnectedEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.WebRtcDisconnectedEvent), /* harmony export */ "WebRtcFailedEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.WebRtcFailedEvent), /* harmony export */ "WebRtcPlayerController": () => (/* reexport safe */ _WebRtcPlayer_WebRtcPlayerController__WEBPACK_IMPORTED_MODULE_0__.WebRtcPlayerController), /* harmony export */ "WebRtcSdpEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.WebRtcSdpEvent), /* harmony export */ "WebSocketController": () => (/* reexport safe */ _WebSockets_WebSocketController__WEBPACK_IMPORTED_MODULE_18__.WebSocketController), /* harmony export */ "WebXRController": () => (/* reexport safe */ _WebXR_WebXRController__WEBPACK_IMPORTED_MODULE_1__.WebXRController), /* harmony export */ "XrFrameEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.XrFrameEvent), /* harmony export */ "XrSessionEndedEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.XrSessionEndedEvent), /* harmony export */ "XrSessionStartedEvent": () => (/* reexport safe */ _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__.XrSessionStartedEvent) /* harmony export */ }); /* harmony import */ var _WebRtcPlayer_WebRtcPlayerController__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./WebRtcPlayer/WebRtcPlayerController */ "./src/WebRtcPlayer/WebRtcPlayerController.ts"); /* harmony import */ var _WebXR_WebXRController__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./WebXR/WebXRController */ "./src/WebXR/WebXRController.ts"); /* harmony import */ var _Config_Config__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./Config/Config */ "./src/Config/Config.ts"); /* harmony import */ var _Config_SettingBase__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./Config/SettingBase */ "./src/Config/SettingBase.ts"); /* harmony import */ var _Config_SettingFlag__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./Config/SettingFlag */ "./src/Config/SettingFlag.ts"); /* harmony import */ var _Config_SettingNumber__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./Config/SettingNumber */ "./src/Config/SettingNumber.ts"); /* harmony import */ var _Config_SettingOption__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./Config/SettingOption */ "./src/Config/SettingOption.ts"); /* harmony import */ var _Config_SettingText__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ./Config/SettingText */ "./src/Config/SettingText.ts"); /* harmony import */ var _PixelStreaming_PixelStreaming__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ./PixelStreaming/PixelStreaming */ "./src/PixelStreaming/PixelStreaming.ts"); /* harmony import */ var _AFK_AFKController__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! ./AFK/AFKController */ "./src/AFK/AFKController.ts"); /* harmony import */ var _DataChannel_LatencyTestResults__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! ./DataChannel/LatencyTestResults */ "./src/DataChannel/LatencyTestResults.ts"); /* harmony import */ var _DataChannel_InitialSettings__WEBPACK_IMPORTED_MODULE_11__ = __webpack_require__(/*! ./DataChannel/InitialSettings */ "./src/DataChannel/InitialSettings.ts"); /* harmony import */ var _PeerConnectionController_AggregatedStats__WEBPACK_IMPORTED_MODULE_12__ = __webpack_require__(/*! ./PeerConnectionController/AggregatedStats */ "./src/PeerConnectionController/AggregatedStats.ts"); /* harmony import */ var _Logger_Logger__WEBPACK_IMPORTED_MODULE_13__ = __webpack_require__(/*! ./Logger/Logger */ "./src/Logger/Logger.ts"); /* harmony import */ var _Util_CoordinateConverter__WEBPACK_IMPORTED_MODULE_14__ = __webpack_require__(/*! ./Util/CoordinateConverter */ "./src/Util/CoordinateConverter.ts"); /* harmony import */ var _UeInstanceMessage_StreamMessageController__WEBPACK_IMPORTED_MODULE_15__ = __webpack_require__(/*! ./UeInstanceMessage/StreamMessageController */ "./src/UeInstanceMessage/StreamMessageController.ts"); /* harmony import */ var _WebSockets_MessageSend__WEBPACK_IMPORTED_MODULE_16__ = __webpack_require__(/*! ./WebSockets/MessageSend */ "./src/WebSockets/MessageSend.ts"); /* harmony import */ var _WebSockets_MessageReceive__WEBPACK_IMPORTED_MODULE_17__ = __webpack_require__(/*! ./WebSockets/MessageReceive */ "./src/WebSockets/MessageReceive.ts"); /* harmony import */ var _WebSockets_WebSocketController__WEBPACK_IMPORTED_MODULE_18__ = __webpack_require__(/*! ./WebSockets/WebSocketController */ "./src/WebSockets/WebSocketController.ts"); /* harmony import */ var _WebSockets_SignallingProtocol__WEBPACK_IMPORTED_MODULE_19__ = __webpack_require__(/*! ./WebSockets/SignallingProtocol */ "./src/WebSockets/SignallingProtocol.ts"); /* harmony import */ var _PeerConnectionController_CandidatePairStats__WEBPACK_IMPORTED_MODULE_20__ = __webpack_require__(/*! ./PeerConnectionController/CandidatePairStats */ "./src/PeerConnectionController/CandidatePairStats.ts"); /* harmony import */ var _PeerConnectionController_CandidateStat__WEBPACK_IMPORTED_MODULE_21__ = __webpack_require__(/*! ./PeerConnectionController/CandidateStat */ "./src/PeerConnectionController/CandidateStat.ts"); /* harmony import */ var _PeerConnectionController_DataChannelStats__WEBPACK_IMPORTED_MODULE_22__ = __webpack_require__(/*! ./PeerConnectionController/DataChannelStats */ "./src/PeerConnectionController/DataChannelStats.ts"); /* harmony import */ var _PeerConnectionController_InboundRTPStats__WEBPACK_IMPORTED_MODULE_23__ = __webpack_require__(/*! ./PeerConnectionController/InboundRTPStats */ "./src/PeerConnectionController/InboundRTPStats.ts"); /* harmony import */ var _PeerConnectionController_OutBoundRTPStats__WEBPACK_IMPORTED_MODULE_24__ = __webpack_require__(/*! ./PeerConnectionController/OutBoundRTPStats */ "./src/PeerConnectionController/OutBoundRTPStats.ts"); /* harmony import */ var _Util_EventEmitter__WEBPACK_IMPORTED_MODULE_25__ = __webpack_require__(/*! ./Util/EventEmitter */ "./src/Util/EventEmitter.ts"); // Copyright Epic Games, Inc. All Rights Reserved. })(); /******/ return __webpack_exports__; /******/ })() ; }); //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"lib-pixelstreamingfrontend.js","mappings":"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","sources":["webpack://lib-pixelstreamingfrontend/webpack/universalModuleDefinition","webpack://lib-pixelstreamingfrontend/./src/AFK/AFKController.ts","webpack://lib-pixelstreamingfrontend/./src/Config/Config.ts","webpack://lib-pixelstreamingfrontend/./src/Config/SettingBase.ts","webpack://lib-pixelstreamingfrontend/./src/Config/SettingFlag.ts","webpack://lib-pixelstreamingfrontend/./src/Config/SettingNumber.ts","webpack://lib-pixelstreamingfrontend/./src/Config/SettingOption.ts","webpack://lib-pixelstreamingfrontend/./src/Config/SettingText.ts","webpack://lib-pixelstreamingfrontend/./src/DataChannel/DataChannelController.ts","webpack://lib-pixelstreamingfrontend/./src/DataChannel/DataChannelLatencyTestController.ts","webpack://lib-pixelstreamingfrontend/./src/DataChannel/DataChannelLatencyTestResults.ts","webpack://lib-pixelstreamingfrontend/./src/DataChannel/DataChannelSender.ts","webpack://lib-pixelstreamingfrontend/./src/DataChannel/InitialSettings.ts","webpack://lib-pixelstreamingfrontend/./src/DataChannel/LatencyTestResults.ts","webpack://lib-pixelstreamingfrontend/./src/FreezeFrame/FreezeFrame.ts","webpack://lib-pixelstreamingfrontend/./src/FreezeFrame/FreezeFrameController.ts","webpack://lib-pixelstreamingfrontend/./src/Inputs/FakeTouchController.ts","webpack://lib-pixelstreamingfrontend/./src/Inputs/GamepadController.ts","webpack://lib-pixelstreamingfrontend/./src/Inputs/HoveringMouseEvents.ts","webpack://lib-pixelstreamingfrontend/./src/Inputs/InputClassesFactory.ts","webpack://lib-pixelstreamingfrontend/./src/Inputs/KeyboardController.ts","webpack://lib-pixelstreamingfrontend/./src/Inputs/LockedMouseEvents.ts","webpack://lib-pixelstreamingfrontend/./src/Inputs/MouseButtons.ts","webpack://lib-pixelstreamingfrontend/./src/Inputs/MouseController.ts","webpack://lib-pixelstreamingfrontend/./src/Inputs/SpecialKeyCodes.ts","webpack://lib-pixelstreamingfrontend/./src/Inputs/TouchController.ts","webpack://lib-pixelstreamingfrontend/./src/Inputs/XRGamepadController.ts","webpack://lib-pixelstreamingfrontend/./src/Logger/Logger.ts","webpack://lib-pixelstreamingfrontend/./src/PeerConnectionController/AggregatedStats.ts","webpack://lib-pixelstreamingfrontend/./src/PeerConnectionController/CandidatePairStats.ts","webpack://lib-pixelstreamingfrontend/./src/PeerConnectionController/CandidateStat.ts","webpack://lib-pixelstreamingfrontend/./src/PeerConnectionController/DataChannelStats.ts","webpack://lib-pixelstreamingfrontend/./src/PeerConnectionController/InboundRTPStats.ts","webpack://lib-pixelstreamingfrontend/./src/PeerConnectionController/OutBoundRTPStats.ts","webpack://lib-pixelstreamingfrontend/./src/PeerConnectionController/PeerConnectionController.ts","webpack://lib-pixelstreamingfrontend/./src/PeerConnectionController/SessionStats.ts","webpack://lib-pixelstreamingfrontend/./src/PeerConnectionController/StreamStats.ts","webpack://lib-pixelstreamingfrontend/./src/PixelStreaming/PixelStreaming.ts","webpack://lib-pixelstreamingfrontend/./src/UI/OnScreenKeyboard.ts","webpack://lib-pixelstreamingfrontend/./src/UeInstanceMessage/ResponseController.ts","webpack://lib-pixelstreamingfrontend/./src/UeInstanceMessage/SendMessageController.ts","webpack://lib-pixelstreamingfrontend/./src/UeInstanceMessage/StreamMessageController.ts","webpack://lib-pixelstreamingfrontend/./src/UeInstanceMessage/ToStreamerMessagesController.ts","webpack://lib-pixelstreamingfrontend/./src/Util/CoordinateConverter.ts","webpack://lib-pixelstreamingfrontend/./src/Util/EventEmitter.ts","webpack://lib-pixelstreamingfrontend/./src/Util/EventListenerTracker.ts","webpack://lib-pixelstreamingfrontend/./src/Util/FileUtil.ts","webpack://lib-pixelstreamingfrontend/./src/Util/RTCUtils.ts","webpack://lib-pixelstreamingfrontend/./src/Util/WebGLUtils.ts","webpack://lib-pixelstreamingfrontend/./src/Util/WebXRUtils.ts","webpack://lib-pixelstreamingfrontend/./src/VideoPlayer/StreamController.ts","webpack://lib-pixelstreamingfrontend/./src/VideoPlayer/VideoPlayer.ts","webpack://lib-pixelstreamingfrontend/./src/WebRtcPlayer/WebRtcPlayerController.ts","webpack://lib-pixelstreamingfrontend/./src/WebSockets/MessageReceive.ts","webpack://lib-pixelstreamingfrontend/./src/WebSockets/MessageSend.ts","webpack://lib-pixelstreamingfrontend/./src/WebSockets/SignallingProtocol.ts","webpack://lib-pixelstreamingfrontend/./src/WebSockets/WebSocketController.ts","webpack://lib-pixelstreamingfrontend/./src/WebXR/WebXRController.ts","webpack://lib-pixelstreamingfrontend/external umd \"sdp\"","webpack://lib-pixelstreamingfrontend/webpack/bootstrap","webpack://lib-pixelstreamingfrontend/webpack/runtime/compat get default export","webpack://lib-pixelstreamingfrontend/webpack/runtime/define property getters","webpack://lib-pixelstreamingfrontend/webpack/runtime/hasOwnProperty shorthand","webpack://lib-pixelstreamingfrontend/webpack/runtime/make namespace object","webpack://lib-pixelstreamingfrontend/./src/pixelstreamingfrontend.ts"],"sourcesContent":["(function webpackUniversalModuleDefinition(root, factory) {\n\tif(typeof exports === 'object' && typeof module === 'object')\n\t\tmodule.exports = factory(require(\"sdp\"));\n\telse if(typeof define === 'function' && define.amd)\n\t\tdefine([\"sdp\"], factory);\n\telse if(typeof exports === 'object')\n\t\texports[\"lib-pixelstreamingfrontend\"] = factory(require(\"sdp\"));\n\telse\n\t\troot[\"lib-pixelstreamingfrontend\"] = factory(root[\"sdp\"]);\n})(this, (__WEBPACK_EXTERNAL_MODULE_sdp__) => {\nreturn ","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Config, Flags, NumericParameters } from '../Config/Config';\nimport { Logger } from '../Logger/Logger';\nimport { PixelStreaming } from '../PixelStreaming/PixelStreaming';\nimport {\n    AfkTimedOutEvent,\n    AfkWarningActivateEvent,\n    AfkWarningDeactivateEvent,\n    AfkWarningUpdateEvent\n} from '../Util/EventEmitter';\n\nexport class AFKController {\n    // time out logic details\n    closeTimeout = 10;\n    active = false;\n    countdownActive = false;\n    warnTimer: ReturnType<typeof setTimeout> = undefined;\n    countDown = 0;\n    countDownTimer: ReturnType<typeof setInterval> = undefined;\n    config: Config;\n    pixelStreaming: PixelStreaming;\n    onDismissAfk: () => void;\n\n    onAFKTimedOutCallback: () => void;\n\n    constructor(\n        config: Config,\n        pixelStreaming: PixelStreaming,\n        onDismissAfk: () => void\n    ) {\n        this.config = config;\n        this.pixelStreaming = pixelStreaming;\n        this.onDismissAfk = onDismissAfk;\n        this.onAFKTimedOutCallback = () => {\n            console.log(\n                'AFK timed out, did you want to override this callback?'\n            );\n        };\n    }\n\n    /**\n     * The methods that occur when an afk event listener is clicked\n     */\n    onAfkClick() {\n        clearInterval(this.countDownTimer);\n\n        if (this.active || this.countdownActive) {\n            this.startAfkWarningTimer();\n            this.pixelStreaming.dispatchEvent(\n                new AfkWarningDeactivateEvent()\n            );\n        }\n    }\n\n    /**\n     * Start the warning timer if a timeout is set greater that 0 seconds\n     */\n    startAfkWarningTimer() {\n        if (\n            this.config.getNumericSettingValue(\n                NumericParameters.AFKTimeoutSecs\n            ) > 0 &&\n            this.config.isFlagEnabled(Flags.AFKDetection)\n        ) {\n            this.active = true;\n        } else {\n            this.active = false;\n        }\n        this.resetAfkWarningTimer();\n    }\n\n    /**\n     * Stop the afk warning timer\n     */\n    stopAfkWarningTimer() {\n        this.active = false;\n        this.countdownActive = false;\n        clearTimeout(this.warnTimer);\n        clearInterval(this.countDownTimer);\n    }\n\n    /**\n     * Pause the timer which when elapsed will warn the user they are inactive.\n     */\n    pauseAfkWarningTimer() {\n        this.active = false;\n    }\n\n    /**\n     * If the user interacts then reset the warning timer.\n     */\n    resetAfkWarningTimer() {\n        if (this.active && this.config.isFlagEnabled(Flags.AFKDetection)) {\n            clearTimeout(this.warnTimer);\n            this.warnTimer = setTimeout(\n                () => this.activateAfkEvent(),\n                this.config.getNumericSettingValue(\n                    NumericParameters.AFKTimeoutSecs\n                ) * 1000\n            );\n        }\n    }\n\n    /**\n     * Show the AFK overlay and begin the countDown\n     */\n    activateAfkEvent() {\n        // Pause the timer while the user is looking at the inactivity warning overlay\n        this.pauseAfkWarningTimer();\n\n        // instantiate a new overlay\n        this.pixelStreaming.dispatchEvent(\n            new AfkWarningActivateEvent({\n                countDown: this.countDown,\n                dismissAfk: this.onDismissAfk\n            })\n        );\n\n        // update our countDown timer and overlay contents\n        this.countDown = this.closeTimeout;\n        this.countdownActive = true;\n        this.pixelStreaming.dispatchEvent(\n            new AfkWarningUpdateEvent({ countDown: this.countDown })\n        );\n\n        // if we are in locked mouse exit pointerlock\n        if (!this.config.isFlagEnabled(Flags.HoveringMouseMode)) {\n            // minor hack to alleviate ios not supporting pointerlock\n            if (document.exitPointerLock) {\n                document.exitPointerLock();\n            }\n        }\n\n        // reset our countDown interval accordingly\n        this.countDownTimer = setInterval(() => {\n            this.countDown--;\n            if (this.countDown == 0) {\n                // The user failed to click so hide the overlay and disconnect them.\n                this.pixelStreaming.dispatchEvent(\n                    new AfkTimedOutEvent()\n                );\n                this.onAFKTimedOutCallback();\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    'You have been disconnected due to inactivity'\n                );\n\n                // switch off the afk feature as stream has closed\n                this.stopAfkWarningTimer();\n            } else {\n                this.pixelStreaming.dispatchEvent(\n                    new AfkWarningUpdateEvent({ countDown: this.countDown })\n                );\n            }\n        }, 1000);\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\nimport { SettingFlag } from './SettingFlag';\nimport { SettingNumber } from './SettingNumber';\nimport { SettingText } from './SettingText';\nimport { SettingOption } from './SettingOption';\nimport { EventEmitter, SettingsChangedEvent } from '../Util/EventEmitter';\nimport { SettingBase } from './SettingBase';\n\n/**\n * A collection of flags that can be toggled and are core to all Pixel Streaming experiences.\n * These are used in the `Config.Flags` map.\n */\nexport class Flags {\n    static AutoConnect = 'AutoConnect' as const;\n    static AutoPlayVideo = 'AutoPlayVideo' as const;\n    static AFKDetection = 'TimeoutIfIdle' as const;\n    static BrowserSendOffer = 'OfferToReceive' as const;\n    static HoveringMouseMode = 'HoveringMouse' as const;\n    static ForceMonoAudio = 'ForceMonoAudio' as const;\n    static ForceTURN = 'ForceTURN' as const;\n    static FakeMouseWithTouches = 'FakeMouseWithTouches' as const;\n    static IsQualityController = 'ControlsQuality' as const;\n    static MatchViewportResolution = 'MatchViewportRes' as const;\n    static StartVideoMuted = 'StartVideoMuted' as const;\n    static SuppressBrowserKeys = 'SuppressBrowserKeys' as const;\n    static UseMic = 'UseMic' as const;\n    static KeyboardInput = 'KeyboardInput' as const;\n    static MouseInput = 'MouseInput' as const;\n    static TouchInput = 'TouchInput' as const;\n    static GamepadInput = 'GamepadInput' as const;\n    static XRControllerInput = 'XRControllerInput' as const;\n    static WaitForStreamer = \"WaitForStreamer\" as const;\n}\n\nexport type FlagsKeys = Exclude<keyof typeof Flags, 'prototype'>;\nexport type FlagsIds = typeof Flags[FlagsKeys];\n\nconst isFlagId = (id: string): id is FlagsIds =>\n    Object.getOwnPropertyNames(Flags).some(\n        (name: FlagsKeys) => Flags[name] === id\n    );\n\n/**\n * A collection of numeric parameters that are core to all Pixel Streaming experiences.\n *\n */\nexport class NumericParameters {\n    static AFKTimeoutSecs = 'AFKTimeout' as const;\n    static MinQP = 'MinQP' as const;\n    static MaxQP = 'MaxQP' as const;\n    static WebRTCFPS = 'WebRTCFPS' as const;\n    static WebRTCMinBitrate = 'WebRTCMinBitrate' as const;\n    static WebRTCMaxBitrate = 'WebRTCMaxBitrate' as const;\n    static MaxReconnectAttempts = 'MaxReconnectAttempts' as const;\n    static StreamerAutoJoinInterval = 'StreamerAutoJoinInterval' as const;\n}\n\nexport type NumericParametersKeys = Exclude<\n    keyof typeof NumericParameters,\n    'prototype'\n>;\nexport type NumericParametersIds =\n    typeof NumericParameters[NumericParametersKeys];\n\nconst isNumericId = (id: string): id is NumericParametersIds =>\n    Object.getOwnPropertyNames(NumericParameters).some(\n        (name: NumericParametersKeys) => NumericParameters[name] === id\n    );\n\n/**\n * A collection of textual parameters that are core to all Pixel Streaming experiences.\n *\n */\nexport class TextParameters {\n    static SignallingServerUrl = 'ss' as const;\n}\n\nexport type TextParametersKeys = Exclude<\n    keyof typeof TextParameters,\n    'prototype'\n>;\nexport type TextParametersIds = typeof TextParameters[TextParametersKeys];\n\nconst isTextId = (id: string): id is TextParametersIds =>\n    Object.getOwnPropertyNames(TextParameters).some(\n        (name: TextParametersKeys) => TextParameters[name] === id\n    );\n\n/**\n * A collection of enum based parameters that are core to all Pixel Streaming experiences.\n *\n */\nexport class OptionParameters {\n    static PreferredCodec = 'PreferredCodec' as const;\n    static StreamerId = 'StreamerId' as const;\n}\n\nexport type OptionParametersKeys = Exclude<\n    keyof typeof OptionParameters,\n    'prototype'\n>;\nexport type OptionParametersIds = typeof OptionParameters[OptionParametersKeys];\n\nconst isOptionId = (id: string): id is OptionParametersIds =>\n    Object.getOwnPropertyNames(OptionParameters).some(\n        (name: OptionParametersKeys) => OptionParameters[name] === id\n    );\n\n/**\n * Utility types for inferring data type based on setting ID\n */\nexport type OptionIds =\n    | FlagsIds\n    | NumericParametersIds\n    | TextParametersIds\n    | OptionParametersIds;\nexport type OptionKeys<T> = T extends FlagsIds\n    ? boolean\n    : T extends NumericParametersIds\n    ? number\n    : T extends TextParametersIds\n    ? string\n    : T extends OptionParametersIds\n    ? string\n    : never;\n\nexport type AllSettings = {\n    [K in OptionIds]: OptionKeys<K>;\n};\n\nexport interface ConfigParams {\n    /** Initial Pixel Streaming settings */\n    initialSettings?: Partial<AllSettings>;\n    /** If useUrlParams is set true, will read initial values from URL parameters and persist changed settings into URL */\n    useUrlParams?: boolean;\n}\nexport class Config {\n    /* A map of flags that can be toggled - options that can be set in the application - e.g. Use Mic? */\n    private flags = new Map<FlagsIds, SettingFlag>();\n\n    /* A map of numerical settings - options that can be in the application - e.g. MinBitrate */\n    private numericParameters = new Map<NumericParametersIds, SettingNumber>();\n\n    /* A map of text settings - e.g. signalling server url */\n    private textParameters = new Map<TextParametersIds, SettingText>();\n\n    /* A map of enum based settings - e.g. preferred codec */\n    private optionParameters = new Map<OptionParametersIds, SettingOption>();\n\n    private _useUrlParams: boolean;\n\n    // ------------ Settings -----------------\n\n    constructor(config: ConfigParams = {}) {\n        const { initialSettings, useUrlParams } = config;\n        this._useUrlParams = !!useUrlParams;\n        this.populateDefaultSettings(this._useUrlParams);\n        if (initialSettings) {\n            this.setSettings(initialSettings);\n        }\n    }\n\n    /**\n     * True if reading configuration initial values from URL parameters, and\n     * persisting changes in URL when changed.\n     */\n    public get useUrlParams() {\n        return this._useUrlParams;\n    }\n\n    /**\n     * Populate the default settings for a Pixel Streaming application\n     */\n    private populateDefaultSettings(useUrlParams: boolean): void {\n        /**\n         * Text Parameters\n         */\n\n        this.textParameters.set(\n            TextParameters.SignallingServerUrl,\n            new SettingText(\n                TextParameters.SignallingServerUrl,\n                'Signalling url',\n                'Url of the signalling server',\n                (location.protocol === 'https:' ? 'wss://' : 'ws://') +\n                    window.location.hostname +\n                    // for readability, we omit the port if it's 80\n                    (window.location.port === '80' ||\n                    window.location.port === ''\n                        ? ''\n                        : `:${window.location.port}`),\n                useUrlParams\n            )\n        );\n\n        this.optionParameters.set(\n            OptionParameters.StreamerId,\n            new SettingOption(\n                OptionParameters.StreamerId,\n                'Streamer ID',\n                'The ID of the streamer to stream.',\n                '',\n                [],\n                useUrlParams\n            )\n        );\n\n        /**\n         * Enum Parameters\n         */\n        this.optionParameters.set(\n            OptionParameters.PreferredCodec,\n            new SettingOption(\n                OptionParameters.PreferredCodec,\n                'Preferred Codec',\n                'The preferred codec to be used during codec negotiation',\n                'H264 level-asymmetry-allowed=1;packetization-mode=1;profile-level-id=42e01f',\n                (function (): Array<string> {\n                    const browserSupportedCodecs: Array<string> = [];\n                    // Try get the info needed from the RTCRtpReceiver. This is only available on chrome\n                    if (!RTCRtpReceiver.getCapabilities) {\n                        browserSupportedCodecs.push('Only available on Chrome');\n                        return browserSupportedCodecs;\n                    }\n\n                    const matcher = /(VP\\d|H26\\d|AV1).*/;\n                    const codecs =\n                        RTCRtpReceiver.getCapabilities('video').codecs;\n                    codecs.forEach((codec) => {\n                        const str =\n                            codec.mimeType.split('/')[1] +\n                            ' ' +\n                            (codec.sdpFmtpLine || '');\n                        const match = matcher.exec(str);\n                        if (match !== null) {\n                            browserSupportedCodecs.push(str);\n                        }\n                    });\n                    return browserSupportedCodecs;\n                })(),\n                useUrlParams\n            )\n        );\t\n\n        /**\n         * Boolean parameters\n         */\n\n        this.flags.set(\n            Flags.AutoConnect,\n            new SettingFlag(\n                Flags.AutoConnect,\n                'Auto connect to stream',\n                'Whether we should attempt to auto connect to the signalling server or show a click to start prompt.',\n                false,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.AutoPlayVideo,\n            new SettingFlag(\n                Flags.AutoPlayVideo,\n                'Auto play video',\n                'When video is ready automatically start playing it as opposed to showing a play button.',\n                true,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.BrowserSendOffer,\n            new SettingFlag(\n                Flags.BrowserSendOffer,\n                'Browser send offer',\n                'Browser will initiate the WebRTC handshake by sending the offer to the streamer',\n                false,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.UseMic,\n            new SettingFlag(\n                Flags.UseMic,\n                'Use microphone',\n                'Make browser request microphone access and open an input audio track.',\n                false,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.StartVideoMuted,\n            new SettingFlag(\n                Flags.StartVideoMuted,\n                'Start video muted',\n                'Video will start muted if true.',\n                false,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.SuppressBrowserKeys,\n            new SettingFlag(\n                Flags.SuppressBrowserKeys,\n                'Suppress browser keys',\n                'Suppress certain browser keys that we use in UE, for example F5 to show shader complexity instead of refresh the page.',\n                true,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.IsQualityController,\n            new SettingFlag(\n                Flags.IsQualityController,\n                'Is quality controller?',\n                'True if this peer controls stream quality',\n                true,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.ForceMonoAudio,\n            new SettingFlag(\n                Flags.ForceMonoAudio,\n                'Force mono audio',\n                'Force browser to request mono audio in the SDP',\n                false,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.ForceTURN,\n            new SettingFlag(\n                Flags.ForceTURN,\n                'Force TURN',\n                'Only generate TURN/Relayed ICE candidates.',\n                false,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.AFKDetection,\n            new SettingFlag(\n                Flags.AFKDetection,\n                'AFK if idle',\n                'Timeout the experience if user is AFK for a period.',\n                false,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.MatchViewportResolution,\n            new SettingFlag(\n                Flags.MatchViewportResolution,\n                'Match viewport resolution',\n                'Pixel Streaming will be instructed to dynamically resize the video stream to match the size of the video element.',\n                false,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.HoveringMouseMode,\n            new SettingFlag(\n                Flags.HoveringMouseMode,\n                'Control Scheme: Locked Mouse',\n                'Either locked mouse, where the pointer is consumed by the video and locked to it, or hovering mouse, where the mouse is not consumed.',\n                false,\n                useUrlParams,\n                (isHoveringMouse: boolean, setting: SettingBase) => {\n                    setting.label = `Control Scheme: ${isHoveringMouse ? 'Hovering' : 'Locked'} Mouse`;\n                }\n            )\n        );\n\n        this.flags.set(\n            Flags.FakeMouseWithTouches,\n            new SettingFlag(\n                Flags.FakeMouseWithTouches,\n                'Fake mouse with touches',\n                'A single finger touch is converted into a mouse event. This allows a non-touch application to be controlled partially via a touch device.',\n                false,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.KeyboardInput,\n            new SettingFlag(\n                Flags.KeyboardInput,\n                'Keyboard input',\n                'If enabled, send keyboard events to streamer',\n                true,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.MouseInput,\n            new SettingFlag(\n                Flags.MouseInput,\n                'Mouse input',\n                'If enabled, send mouse events to streamer',\n                true,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.TouchInput,\n            new SettingFlag(\n                Flags.TouchInput,\n                'Touch input',\n                'If enabled, send touch events to streamer',\n                true,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.GamepadInput,\n            new SettingFlag(\n                Flags.GamepadInput,\n                'Gamepad input',\n                'If enabled, send gamepad events to streamer',\n                true,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.XRControllerInput,\n            new SettingFlag(\n                Flags.XRControllerInput,\n                'XR controller input',\n                'If enabled, send XR controller events to streamer',\n                true,\n                useUrlParams\n            )\n        );\n\n        this.flags.set(\n            Flags.WaitForStreamer,\n            new SettingFlag(\n                Flags.WaitForStreamer,\n                'Wait for streamer',\n                'Will continue trying to connect to the first streamer available.',\n                true,\n                useUrlParams\n            )\n        );\n\n        /**\n         * Numeric parameters\n         */\n\n        this.numericParameters.set(\n            NumericParameters.AFKTimeoutSecs,\n            new SettingNumber(\n                NumericParameters.AFKTimeoutSecs,\n                'AFK timeout',\n                'The time (in seconds) it takes for the application to time out if AFK timeout is enabled.',\n                0 /*min*/,\n                600 /*max*/,\n                120 /*value*/,\n                useUrlParams\n            )\n        );\n\n        this.numericParameters.set(\n            NumericParameters.MaxReconnectAttempts,\n            new SettingNumber(\n                NumericParameters.MaxReconnectAttempts,\n                'Max Reconnects',\n                'Maximum number of reconnects the application will attempt when a streamer disconnects.',\n                0 /*min*/,\n                999 /*max*/,\n                3 /*value*/,\n                useUrlParams\n            )\n        );\n\n        this.numericParameters.set(\n            NumericParameters.MinQP,\n            new SettingNumber(\n                NumericParameters.MinQP,\n                'Min QP',\n                'The lower bound for the quantization parameter (QP) of the encoder. 0 = Best quality, 51 = worst quality.',\n                0 /*min*/,\n                51 /*max*/,\n                0 /*value*/,\n                useUrlParams\n            )\n        );\n\n        this.numericParameters.set(\n            NumericParameters.MaxQP,\n            new SettingNumber(\n                NumericParameters.MaxQP,\n                'Max QP',\n                'The upper bound for the quantization parameter (QP) of the encoder. 0 = Best quality, 51 = worst quality.',\n                0 /*min*/,\n                51 /*max*/,\n                51 /*value*/,\n                useUrlParams\n            )\n        );\n\n        this.numericParameters.set(\n            NumericParameters.WebRTCFPS,\n            new SettingNumber(\n                NumericParameters.WebRTCFPS,\n                'Max FPS',\n                'The maximum FPS that WebRTC will try to transmit frames at.',\n                1 /*min*/,\n                999 /*max*/,\n                60 /*value*/,\n                useUrlParams\n            )\n        );\n\n        this.numericParameters.set(\n            NumericParameters.WebRTCMinBitrate,\n            new SettingNumber(\n                NumericParameters.WebRTCMinBitrate,\n                'Min Bitrate (kbps)',\n                'The minimum bitrate that WebRTC should use.',\n                0 /*min*/,\n                500000 /*max*/,\n                0 /*value*/,\n                useUrlParams\n            )\n        );\n\n        this.numericParameters.set(\n            NumericParameters.WebRTCMaxBitrate,\n            new SettingNumber(\n                NumericParameters.WebRTCMaxBitrate,\n                'Max Bitrate (kbps)',\n                'The maximum bitrate that WebRTC should use.',\n                0 /*min*/,\n                500000 /*max*/,\n                0 /*value*/,\n                useUrlParams\n            )\n        );\n\n        this.numericParameters.set(\n            NumericParameters.StreamerAutoJoinInterval,\n            new SettingNumber(\n                NumericParameters.StreamerAutoJoinInterval,\n                'Streamer Auto Join Interval (ms)',\n                'Delay between retries when waiting for an available streamer.',\n                500 /*min*/,\n                900000 /*max*/,\n                3000 /*value*/,\n                useUrlParams\n            )\n        );\n    }\n\n    /**\n     * Add a callback to fire when the numeric setting is toggled.\n     * @param id The id of the flag.\n     * @param onChangedListener The callback to fire when the numeric value changes.\n     */\n    _addOnNumericSettingChangedListener(\n        id: NumericParametersIds,\n        onChangedListener: (newValue: number) => void\n    ): void {\n        if (this.numericParameters.has(id)) {\n            this.numericParameters\n                .get(id)\n                .addOnChangedListener(onChangedListener);\n        }\n    }\n\n    _addOnOptionSettingChangedListener(\n        id: OptionParametersIds,\n        onChangedListener: (newValue: string) => void\n    ): void {\n        if (this.optionParameters.has(id)) {\n            this.optionParameters\n                .get(id)\n                .addOnChangedListener(onChangedListener);\n        }\n    }\n\n    /**\n     * @param id The id of the numeric setting we are interested in getting a value for.\n     * @returns The numeric value stored in the parameter with the passed id.\n     */\n    getNumericSettingValue(id: NumericParametersIds): number {\n        if (this.numericParameters.has(id)) {\n            return this.numericParameters.get(id).number;\n        } else {\n            throw new Error(`There is no numeric setting with the id of ${id}`);\n        }\n    }\n\n    /**\n     * @param id The id of the text setting we are interested in getting a value for.\n     * @returns The text value stored in the parameter with the passed id.\n     */\n    getTextSettingValue(id: TextParametersIds): string {\n        if (this.textParameters.has(id)) {\n            return this.textParameters.get(id).value as string;\n        } else {\n            throw new Error(`There is no numeric setting with the id of ${id}`);\n        }\n    }\n\n    /**\n     * Set number in the setting.\n     * @param id The id of the numeric setting we are interested in.\n     * @param value The numeric value to set.\n     */\n    setNumericSetting(id: NumericParametersIds, value: number): void {\n        if (this.numericParameters.has(id)) {\n            this.numericParameters.get(id).number = value;\n        } else {\n            throw new Error(`There is no numeric setting with the id of ${id}`);\n        }\n    }\n\n    /**\n     * Add a callback to fire when the flag is toggled.\n     * @param id The id of the flag.\n     * @param onChangeListener The callback to fire when the value changes.\n     */\n    _addOnSettingChangedListener(\n        id: FlagsIds,\n        onChangeListener: (newFlagValue: boolean) => void\n    ): void {\n        if (this.flags.has(id)) {\n            this.flags.get(id).onChange = onChangeListener;\n        }\n    }\n\n    /**\n     * Add a callback to fire when the text is changed.\n     * @param id The id of the flag.\n     * @param onChangeListener The callback to fire when the value changes.\n     */\n    _addOnTextSettingChangedListener(\n        id: TextParametersIds,\n        onChangeListener: (newTextValue: string) => void\n    ): void {\n        if (this.textParameters.has(id)) {\n            this.textParameters.get(id).onChange = onChangeListener;\n        }\n    }\n\n    /**\n     * Get the option which has the given id.\n     * @param id The id of the option.\n     * @returns The SettingOption object matching id\n     */\n    getSettingOption(id: OptionParametersIds): SettingOption {\n        return this.optionParameters.get(id);\n    }\n\n    /**\n     * Get the value of the configuration flag which has the given id.\n     * @param id The unique id for the flag.\n     * @returns True if the flag is enabled.\n     */\n    isFlagEnabled(id: FlagsIds): boolean {\n        return this.flags.get(id).flag as boolean;\n    }\n\n    /**\n     * Set flag to be enabled/disabled.\n     * @param id The id of the flag to toggle.\n     * @param flagEnabled True if the flag should be enabled.\n     */\n    setFlagEnabled(id: FlagsIds, flagEnabled: boolean) {\n        if (!this.flags.has(id)) {\n            Logger.Warning(\n                Logger.GetStackTrace(),\n                `Cannot toggle flag called ${id} - it does not exist in the Config.flags map.`\n            );\n        } else {\n            this.flags.get(id).flag = flagEnabled;\n        }\n    }\n\n    /**\n     * Set the text setting.\n     * @param id The id of the setting\n     * @param settingValue The value to set in the setting.\n     */\n    setTextSetting(id: TextParametersIds, settingValue: string) {\n        if (!this.textParameters.has(id)) {\n            Logger.Warning(\n                Logger.GetStackTrace(),\n                `Cannot set text setting called ${id} - it does not exist in the Config.textParameters map.`\n            );\n        } else {\n            this.textParameters.get(id).text = settingValue;\n        }\n    }\n\n    /**\n     * Set the option setting list of options.\n     * @param id The id of the setting\n     * @param settingOptions The values the setting could take\n     */\n    setOptionSettingOptions(\n        id: OptionParametersIds,\n        settingOptions: Array<string>\n    ) {\n        if (!this.optionParameters.has(id)) {\n            Logger.Warning(\n                Logger.GetStackTrace(),\n                `Cannot set text setting called ${id} - it does not exist in the Config.optionParameters map.`\n            );\n        } else {\n            this.optionParameters.get(id).options = settingOptions;\n        }\n    }\n\n    /**\n     * Set option enum settings selected option.\n     * @param id The id of the setting\n     * @param settingOptions The value to select out of all the options\n     */\n    setOptionSettingValue(id: OptionParametersIds, settingValue: string) {\n        if (!this.optionParameters.has(id)) {\n            Logger.Warning(\n                Logger.GetStackTrace(),\n                `Cannot set text setting called ${id} - it does not exist in the Config.enumParameters map.`\n            );\n        } else {\n            this.optionParameters.get(id).selected = settingValue;\n        }\n    }\n\n    /**\n     * Set the label for the flag.\n     * @param id The id of the flag.\n     * @param label The new label to use for the flag.\n     */\n    setFlagLabel(id: FlagsIds, label: string) {\n        if (!this.flags.has(id)) {\n            Logger.Warning(\n                Logger.GetStackTrace(),\n                `Cannot set label for flag called ${id} - it does not exist in the Config.flags map.`\n            );\n        } else {\n            this.flags.get(id).label = label;\n        }\n    }\n\n    /**\n     * Set a subset of all settings in one function call.\n     *\n     * @param settings A (partial) list of settings to set\n     */\n    setSettings(settings: Partial<AllSettings>) {\n        for (const key of Object.keys(settings)) {\n            if (isFlagId(key)) {\n                this.setFlagEnabled(key, settings[key]);\n            } else if (isNumericId(key)) {\n                this.setNumericSetting(key, settings[key]);\n            } else if (isTextId(key)) {\n                this.setTextSetting(key, settings[key]);\n            } else if (isOptionId(key)) {\n                this.setOptionSettingValue(key, settings[key]);\n            }\n        }\n    }\n\n    /**\n     * Get all settings\n     * @returns All setting values as an object with setting ids as keys\n     */\n    getSettings(): Partial<AllSettings> {\n        const settings: Partial<AllSettings> = {};\n        for (const [key, value] of this.flags.entries()) {\n            settings[key] = value.flag;\n        }\n        for (const [key, value] of this.numericParameters.entries()) {\n            settings[key] = value.number;\n        }\n        for (const [key, value] of this.textParameters.entries()) {\n            settings[key] = value.text;\n        }\n        for (const [key, value] of this.optionParameters.entries()) {\n            settings[key] = value.selected;\n        }\n        return settings;\n    }\n\n    /**\n     * Get all Flag settings as an array.\n     * @returns All SettingFlag objects\n     */\n    getFlags(): Array<SettingFlag> {\n        return Array.from(this.flags.values());\n    }\n\n    /**\n     * Get all Text settings as an array.\n     * @returns All SettingText objects\n     */\n    getTextSettings(): Array<SettingText> {\n        return Array.from(this.textParameters.values());\n    }\n\n    /**\n     * Get all Number settings as an array.\n     * @returns All SettingNumber objects\n     */\n    getNumericSettings(): Array<SettingNumber> {\n        return Array.from(this.numericParameters.values());\n    }\n\n    /**\n     * Get all Option settings as an array.\n     * @returns All SettingOption objects\n     */\n    getOptionSettings(): Array<SettingOption> {\n        return Array.from(this.optionParameters.values());\n    }\n\n    /**\n     * Emit events when settings change.\n     * @param eventEmitter\n     */\n    _registerOnChangeEvents(eventEmitter: EventEmitter) {\n        for (const key of this.flags.keys()) {\n            const flag = this.flags.get(key);\n            if (flag) {\n                flag.onChangeEmit = (newValue: boolean) =>\n                    eventEmitter.dispatchEvent(\n                        new SettingsChangedEvent({\n                            id: flag.id,\n                            type: 'flag',\n                            value: newValue,\n                            target: flag\n                        })\n                    );\n            }\n        }\n        for (const key of this.numericParameters.keys()) {\n            const number = this.numericParameters.get(key);\n            if (number) {\n                number.onChangeEmit = (newValue: number) =>\n                    eventEmitter.dispatchEvent(\n                        new SettingsChangedEvent({\n                            id: number.id,\n                            type: 'number',\n                            value: newValue,\n                            target: number\n                        })\n                    );\n            }\n        }\n        for (const key of this.textParameters.keys()) {\n            const text = this.textParameters.get(key);\n            if (text) {\n                text.onChangeEmit = (newValue: string) =>\n                    eventEmitter.dispatchEvent(\n                        new SettingsChangedEvent({\n                            id: text.id,\n                            type: 'text',\n                            value: newValue,\n                            target: text\n                        })\n                    );\n            }\n        }\n        for (const key of this.optionParameters.keys()) {\n            const option = this.optionParameters.get(key);\n            if (option) {\n                option.onChangeEmit = (newValue: string) =>\n                    eventEmitter.dispatchEvent(\n                        new SettingsChangedEvent({\n                            id: option.id,\n                            type: 'option',\n                            value: newValue,\n                            target: option\n                        })\n                    );\n            }\n        }\n    }\n}\n\n/**\n * The enum associated with the mouse being locked or hovering\n */\nexport enum ControlSchemeType {\n    LockedMouse = 0,\n    HoveringMouse = 1\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Base class for a setting that has a text label and an arbitrary setting value it stores.\n */\nexport class SettingBase {\n    id: string;\n    description: string;\n    _label: string;\n    _value: unknown;\n    onChange: (changedValue: unknown, setting: SettingBase) => void;\n    onChangeEmit: (changedValue: unknown) => void;\n\n    constructor(\n        id: string,\n        label: string,\n        description: string,\n        defaultSettingValue: unknown,\n\t\t// eslint-disable-next-line @typescript-eslint/no-empty-function\n\t\tdefaultOnChangeListener: (changedValue: unknown, setting: SettingBase) => void = () => { /* Do nothing, to be overridden. */ }\n    ) {\n        this.onChange = defaultOnChangeListener;\n\n        this.onChangeEmit = () => {\n            /* Do nothing, to be overridden. */\n        };\n        this.id = id;\n        this.description = description;\n        this.label = label;\n        this.value = defaultSettingValue;\n    }\n\n    /**\n     * Set the label text for the setting.\n     * @param label setting label.\n     */\n    public set label(inLabel: string) {\n        this._label = inLabel;\n        this.onChangeEmit(this._value);\n    }\n\n    /**\n     * @returns The label text for the setting.\n     */\n    public get label(): string {\n        return this._label;\n    }\n\n    /**\n     * @return The setting's value.\n     */\n    public get value(): unknown {\n        return this._value;\n    }\n\n    /**\n     * Update the setting's stored value.\n     * @param inValue The new value for the setting.\n     */\n    public set value(inValue: unknown) {\n        this._value = inValue;\n        this.onChange(this._value, this);\n        this.onChangeEmit(this._value);\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport type { FlagsIds } from './Config';\nimport { SettingBase } from './SettingBase';\n\n/**\n * A boolean flag setting object with a text label.\n */\nexport class SettingFlag<\n    CustomIds extends string = FlagsIds\n> extends SettingBase {\n    id: FlagsIds | CustomIds;\n    onChangeEmit: (changedValue: boolean) => void;\n    useUrlParams: boolean;\n\n    constructor(\n        id: FlagsIds | CustomIds,\n        label: string,\n        description: string,\n        defaultFlagValue: boolean,\n        useUrlParams: boolean,\n\t\t// eslint-disable-next-line @typescript-eslint/no-empty-function\n\t\tdefaultOnChangeListener: (changedValue: unknown, setting: SettingBase) => void = () => { /* Do nothing, to be overridden. */ }\n    ) {\n        super(id, label, description, defaultFlagValue, defaultOnChangeListener);\n\n        const urlParams = new URLSearchParams(window.location.search);\n        if (!useUrlParams || !urlParams.has(this.id)) {\n            this.flag = defaultFlagValue;\n        } else {\n            // parse flag from url parameters\n            const urlParamFlag = this.getUrlParamFlag();\n            this.flag = urlParamFlag;\n        }\n        this.useUrlParams = useUrlParams;\n    }\n\n    /**\n     * Parse the flag value from the url parameters.\n     * @returns True if the url parameters contains /?id, but False if /?id=false\n     */\n    getUrlParamFlag(): boolean {\n        const urlParams = new URLSearchParams(window.location.search);\n        if (urlParams.has(this.id)) {\n            if (\n                urlParams.get(this.id) === 'false' ||\n                urlParams.get(this.id) === 'False'\n            ) {\n                return false;\n            }\n            return true;\n        }\n        return false;\n    }\n\n    /**\n     * Persist the setting value in URL.\n     */\n    public updateURLParams() {\n        if (this.useUrlParams) {\n            // set url params\n            const urlParams = new URLSearchParams(window.location.search);\n            if (this.flag === true) {\n                urlParams.set(this.id, 'true');\n            } else {\n                urlParams.set(this.id, 'false');\n            }\n            window.history.replaceState(\n                {},\n                '',\n                urlParams.toString() !== ''\n                    ? `${location.pathname}?${urlParams}`\n                    : `${location.pathname}`\n            );\n        }\n    }\n\n    /**\n     * Enables this flag.\n     */\n    public enable(): void {\n        this.flag = true;\n    }\n\n    /**\n     * @return The setting's value.\n     */\n    public get flag(): boolean {\n        return !!this.value;\n    }\n\n    /**\n     * Update the setting's stored value.\n     * @param inValue The new value for the setting.\n     */\n    public set flag(inValue: boolean) {\n        this.value = inValue;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport type { NumericParametersIds } from './Config';\nimport { SettingBase } from './SettingBase';\n\n/**\n * A number setting object with a text label. Min and max limit the range of allowed values.\n */\nexport class SettingNumber<\n    CustomIds extends string = NumericParametersIds\n> extends SettingBase {\n    _min: number;\n    _max: number;\n\n    id: NumericParametersIds | CustomIds;\n    onChangeEmit: (changedValue: number) => void;\n    useUrlParams: boolean;\n\n    constructor(\n        id: NumericParametersIds | CustomIds,\n        label: string,\n        description: string,\n        min: number,\n        max: number,\n        defaultNumber: number,\n        useUrlParams: boolean,\n\t\t// eslint-disable-next-line @typescript-eslint/no-empty-function\n\t\tdefaultOnChangeListener: (changedValue: unknown, setting: SettingBase) => void = () => { /* Do nothing, to be overridden. */ }\n    ) {\n        super(id, label, description, defaultNumber, defaultOnChangeListener);\n\n        this._min = min;\n        this._max = max;\n\n        // attempt to read the number from the url params\n        const urlParams = new URLSearchParams(window.location.search);\n        if (!useUrlParams || !urlParams.has(this.id)) {\n            this.number = defaultNumber;\n        } else {\n            const parsedValue = Number.parseInt(urlParams.get(this.id));\n            this.number = Number.isNaN(parsedValue)\n                ? defaultNumber\n                : parsedValue;\n        }\n        this.useUrlParams = useUrlParams;\n    }\n\n    /**\n     * Persist the setting value in URL.\n     */\n    public updateURLParams(): void {\n        if (this.useUrlParams) {\n            // set url params like ?id=number\n            const urlParams = new URLSearchParams(window.location.search);\n            urlParams.set(this.id, this.number.toString());\n            window.history.replaceState(\n                {},\n                '',\n                urlParams.toString() !== ''\n                    ? `${location.pathname}?${urlParams}`\n                    : `${location.pathname}`\n            );\n        }\n    }\n\n    /**\n     * Set the number value (will be clamped within range).\n     */\n    public set number(newNumber: number) {\n        this.value = this.clamp(newNumber);\n    }\n\n    /**\n     * @returns The number stored.\n     */\n    public get number(): number {\n        return this.value as number;\n    }\n\n    /**\n     * Clamps a number between the min and max values (inclusive).\n     * @param inNumber The number to clamp.\n     * @returns The clamped number.\n     */\n    public clamp(inNumber: number): number {\n        return Math.max(Math.min(this._max, inNumber), this._min);\n    }\n\n    /**\n     * Returns the minimum value\n     * @returns The minimum value\n     */\n    public get min(): number {\n        return this._min;\n    }\n\n    /**\n     * Returns the maximum value\n     * @returns The maximum value\n     */\n    public get max(): number {\n        return this._max;\n    }\n\n    /**\n     * Add a change listener to the number object.\n     */\n    public addOnChangedListener(onChangedFunc: (newNumber: number) => void) {\n        this.onChange = onChangedFunc;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport type { OptionParametersIds } from './Config';\nimport { SettingBase } from './SettingBase';\n\n/**\n * An Option setting object with a text label. Allows you to specify an array of options and select one of them.\n */\nexport class SettingOption<\n    CustomIds extends string = OptionParametersIds\n> extends SettingBase {\n    id: OptionParametersIds | CustomIds;\n    onChangeEmit: (changedValue: string) => void;\n    _options: Array<string>;\n    useUrlParams: boolean;\n\n    constructor(\n        id: OptionParametersIds | CustomIds,\n        label: string,\n        description: string,\n        defaultTextValue: string,\n        options: Array<string>,\n        useUrlParams: boolean,\n\t\t// eslint-disable-next-line @typescript-eslint/no-empty-function\n\t\tdefaultOnChangeListener: (changedValue: unknown, setting: SettingBase) => void = () => { /* Do nothing, to be overridden. */ }\n    ) {\n        super(id, label, description, [defaultTextValue, defaultTextValue], defaultOnChangeListener);\n\n        this.options = options;\n        const urlParams = new URLSearchParams(window.location.search);\n        const stringToMatch: string =\n            useUrlParams && urlParams.has(this.id)\n                ? this.getUrlParamText()\n                : defaultTextValue;\n        this.selected = stringToMatch;\n        this.useUrlParams = useUrlParams;\n    }\n\n    /**\n     * Parse the text value from the url parameters.\n     * @returns The text value parsed from the url if the url parameters contains /?id=value, but empty string if just /?id or no url param found.\n     */\n    getUrlParamText(): string {\n        const urlParams = new URLSearchParams(window.location.search);\n        if (urlParams.has(this.id)) {\n            return urlParams.get(this.id) ?? '';\n        }\n        return '';\n    }\n\n    /**\n     * Persist the setting value in URL.\n     */\n    public updateURLParams() {\n        if (this.useUrlParams) {\n            // set url params\n            const urlParams = new URLSearchParams(window.location.search);\n            urlParams.set(this.id, this.selected);\n            window.history.replaceState(\n                {},\n                '',\n                urlParams.toString() !== ''\n                    ? `${location.pathname}?${urlParams}`\n                    : `${location.pathname}`\n            );\n        }\n    }\n\n    /**\n     * Add a change listener to the select element.\n     */\n    public addOnChangedListener(onChangedFunc: (newValue: string) => void) {\n        this.onChange = onChangedFunc;\n    }\n\n    /**\n     * @returns All available options as an array\n     */\n    public get options(): Array<string> {\n        return this._options;\n    }\n\n    /**\n     * Set options\n     * @param values Array of options\n     */\n    public set options(values: Array<string>) {\n        this._options = values;\n        this.onChangeEmit(this.selected);\n    }\n\n    /**\n     * @returns Selected option as a string\n     */\n    public get selected(): string {\n        return this.value as string;\n    }\n\n    /**\n     * Set selected option if it matches one of the available options\n     * @param value Selected option\n     */\n    public set selected(value: string) {\n        // A user may not specify the full possible value so we instead use the closest match.\n        // eg ?xxx=H264 would select 'H264 level-asymmetry-allowed=1;packetization-mode=1;profile-level-id=42001f'\n        let filteredList = this.options.filter(\n            (option: string) => option.indexOf(value) !== -1\n        );\n        if (filteredList.length) {\n            this.value = filteredList[0];\n            return;\n        } \n\n        // A user has specified a codec with a fmtp string but this codec + fmtp line isn't available.\n        // in that case, just use the codec\n        filteredList = this.options.filter(\n            (option: string) => option.indexOf(value.split(' ')[0]) !== -1\n        );\n        if (filteredList.length) {\n            this.value = filteredList[0];\n            return;\n        }\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport type { TextParametersIds } from './Config';\nimport { SettingBase } from './SettingBase';\n\n/**\n * A text setting object with a text label.\n */\nexport class SettingText<\n    CustomIds extends string = TextParametersIds\n> extends SettingBase {\n    id: TextParametersIds | CustomIds;\n    onChangeEmit: (changedValue: string) => void;\n    useUrlParams: boolean;\n\n    constructor(\n        id: TextParametersIds | CustomIds,\n        label: string,\n        description: string,\n        defaultTextValue: string,\n        useUrlParams: boolean,\n\t\t// eslint-disable-next-line @typescript-eslint/no-empty-function\n\t\tdefaultOnChangeListener: (changedValue: unknown, setting: SettingBase) => void = () => { /* Do nothing, to be overridden. */ }\n    ) {\n        super(id, label, description, defaultTextValue, defaultOnChangeListener);\n\n        const urlParams = new URLSearchParams(window.location.search);\n        if (!useUrlParams || !urlParams.has(this.id)) {\n            this.text = defaultTextValue;\n        } else {\n            // parse flag from url parameters\n            const urlParamFlag = this.getUrlParamText();\n            this.text = urlParamFlag;\n        }\n        this.useUrlParams = useUrlParams;\n    }\n\n    /**\n     * Parse the text value from the url parameters.\n     * @returns The text value parsed from the url if the url parameters contains /?id=value, but empty string if just /?id or no url param found.\n     */\n    getUrlParamText(): string {\n        const urlParams = new URLSearchParams(window.location.search);\n        if (urlParams.has(this.id)) {\n            return urlParams.get(this.id) ?? '';\n        }\n        return '';\n    }\n\n    /**\n     * Persist the setting value in URL.\n     */\n    public updateURLParams() {\n        if (this.useUrlParams) {\n            // set url params\n            const urlParams = new URLSearchParams(window.location.search);\n            urlParams.set(this.id, this.text);\n            window.history.replaceState(\n                {},\n                '',\n                urlParams.toString() !== ''\n                    ? `${location.pathname}?${urlParams}`\n                    : `${location.pathname}`\n            );\n        }\n    }\n\n    /**\n     * @return The setting's value.\n     */\n    public get text(): string {\n        return this.value as string;\n    }\n\n    /**\n     * Update the setting's stored value.\n     * @param inValue The new value for the setting.\n     */\n    public set text(inValue: string) {\n        this.value = inValue;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\n\n/**\n * Handles the Sending and Receiving of messages to the UE Instance via the Data Channel\n */\nexport class DataChannelController {\n    dataChannel: RTCDataChannel;\n    peerConnection: RTCPeerConnection;\n    datachannelOptions: RTCDataChannelInit;\n    label: string;\n    isReceivingFreezeFrame = false;\n\n    /**\n     * return the current state of a datachannel controller instance\n     * @returns the current DataChannelController instance\n     */\n    getDataChannelInstance(): DataChannelController {\n        return this;\n    }\n\n    /**\n     * To Create and Set up a Data Channel\n     * @param peerConnection - The RTC Peer Connection\n     * @param label - Label of the Data Channel\n     * @param datachannelOptions - Optional RTC DataChannel options\n     */\n    createDataChannel(\n        peerConnection: RTCPeerConnection,\n        label: string,\n        datachannelOptions?: RTCDataChannelInit\n    ) {\n        this.peerConnection = peerConnection;\n        this.label = label;\n        this.datachannelOptions = datachannelOptions;\n        if (datachannelOptions == null) {\n            this.datachannelOptions = {} as RTCDataChannelInit;\n            this.datachannelOptions.ordered = true;\n        }\n\n        this.dataChannel = this.peerConnection.createDataChannel(\n            this.label,\n            this.datachannelOptions\n        );\n        this.setupDataChannel();\n    }\n\n    setupDataChannel() {\n        //We Want an Array Buffer not a blob\n        this.dataChannel.binaryType = 'arraybuffer';\n        this.dataChannel.onopen = (ev: Event) => this.handleOnOpen(ev);\n        this.dataChannel.onclose = (ev: Event) => this.handleOnClose(ev);\n        this.dataChannel.onmessage = (ev: MessageEvent) =>\n            this.handleOnMessage(ev);\n        this.dataChannel.onerror = (ev: MessageEvent) => this.handleOnError(ev);\n    }\n\n    /**\n     * Handles when the Data Channel is opened\n     */\n    handleOnOpen(ev: Event) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            `Data Channel (${this.label}) opened.`,\n            7\n        );\n        this.onOpen(this.dataChannel?.label, ev);\n    }\n\n    /**\n     * Handles when the Data Channel is closed\n     */\n    handleOnClose(ev: Event) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            `Data Channel (${this.label}) closed.`,\n            7\n        );\n        this.onClose(this.dataChannel?.label, ev);\n    }\n\n    /**\n     * Handles when a message is received\n     * @param event - Message Event\n     */\n    handleOnMessage(event: MessageEvent) {\n        // Higher log level to prevent log spam with messages received\n        Logger.Log(\n            Logger.GetStackTrace(),\n            `Data Channel (${this.label}) message: ${event}`,\n            8\n        );\n    }\n\n    /**\n     * Handles when an error is thrown\n     * @param event - Error Event\n     */\n    handleOnError(event: MessageEvent) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            `Data Channel (${this.label}) error: ${event}`,\n            7\n        );\n        this.onError(this.dataChannel?.label, event);\n    }\n\n    /**\n     * Override to register onOpen handler\n     * @param label Data channel label (\"datachannel\", \"send-datachannel\", \"recv-datachannel\")\n     * @param ev event\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    onOpen(label: string, ev: Event) {\n        // empty default implementation\n    }\n\n    /**\n     * Override to register onClose handler\n     * @param label Data channel label (\"datachannel\", \"send-datachannel\", \"recv-datachannel\")\n     * @param ev event\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    onClose(label: string, ev: Event) {\n        // empty default implementation\n    }\n\n    /**\n     * Override to register onError handler\n     * @param label Data channel label (\"datachannel\", \"send-datachannel\", \"recv-datachannel\")\n     * @param ev event\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    onError(label: string, ev: Event) {\n        // empty default implementation\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\nimport {\n    DataChannelLatencyTestRecord,\n    DataChannelLatencyTestRequest,\n    DataChannelLatencyTestResponse,\n    DataChannelLatencyTestResult,\n    DataChannelLatencyTestSeq,\n    DataChannelLatencyTestTimestamp\n} from \"./DataChannelLatencyTestResults\";\n\nexport type DataChannelLatencyTestConfig = {\n    // test duration in milliseconds\n    duration: number;\n    //requests per second\n    rps: number;\n    //request filler size\n    requestSize: number;\n    //response filler size\n    responseSize: number;\n}\n\nexport type DataChannelLatencyTestSink = (request: DataChannelLatencyTestRequest) => void;\nexport type DataChannelLatencyTestResultCallback = (result: DataChannelLatencyTestResult) => void;\n\nexport class DataChannelLatencyTestController {\n    startTime: DataChannelLatencyTestTimestamp;\n    sink: DataChannelLatencyTestSink;\n    callback: DataChannelLatencyTestResultCallback;\n    records: Map<DataChannelLatencyTestSeq, DataChannelLatencyTestRecord>;\n    seq: DataChannelLatencyTestSeq;\n    interval: NodeJS.Timer;\n\n    constructor(sink: DataChannelLatencyTestSink, callback: DataChannelLatencyTestResultCallback) {\n        this.sink = sink;\n        this.callback = callback;\n        this.records = new Map();\n        this.seq = 0;\n    }\n\n    start(config: DataChannelLatencyTestConfig) {\n        if (this.isRunning()) {\n            return false;\n        }\n        this.startTime = Date.now();\n        this.records.clear();\n        this.interval = setInterval((() => {\n            if (Date.now() - this.startTime >= config.duration) {\n                this.stop();\n            } else {\n                this.sendRequest(config.requestSize, config.responseSize);\n            }\n        }).bind(this), Math.floor(1000/config.rps));\n        return true;\n    }\n\n    stop() {\n        if (this.interval) {\n            clearInterval(this.interval);\n            this.interval = undefined;\n            this.callback(this.produceResult());\n        }\n    }\n\n    produceResult(): DataChannelLatencyTestResult {\n        const resultRecords = new Map(this.records);\n        return {\n            records: resultRecords,\n            dataChannelRtt: Math.ceil(Array.from(this.records.values()).reduce((acc, next) => {\n                return acc + (next.playerReceivedTimestamp - next.playerSentTimestamp);\n            }, 0) / this.records.size),\n            playerToStreamerTime: Math.ceil(Array.from(this.records.values()).reduce((acc, next) => {\n                return acc + (next.streamerReceivedTimestamp - next.playerSentTimestamp);\n            }, 0) / this.records.size),\n            streamerToPlayerTime: Math.ceil(Array.from(this.records.values()).reduce((acc, next) => {\n                return acc + (next.playerReceivedTimestamp - next.streamerSentTimestamp);\n            }, 0) / this.records.size),\n            exportLatencyAsCSV: () => {\n                let csv = \"Timestamp;RTT;PlayerToStreamer;StreamerToPlayer;\\n\";\n                resultRecords.forEach((record) => {\n                    csv += record.playerSentTimestamp + \";\";\n                    csv += (record.playerReceivedTimestamp - record.playerSentTimestamp) + \";\";\n                    csv += (record.streamerReceivedTimestamp - record.playerSentTimestamp) + \";\";\n                    csv += (record.playerReceivedTimestamp - record.streamerSentTimestamp) + \";\";\n                    csv += \"\\n\";\n                })\n                return csv;\n            }\n        }\n    }\n\n    isRunning() {\n        return !!this.interval;\n    }\n\n    receive(response: DataChannelLatencyTestResponse) {\n        if (!this.isRunning()) {\n            return;\n        }\n        if (!response) {\n            Logger.Error(\n                Logger.GetStackTrace(),\n                \"Undefined response from server\"\n            );\n            return;\n        }\n        let record = this.records.get(response.Seq);\n        if (record) {\n            record.update(response);\n        }\n    }\n\n    sendRequest(requestSize: number, responseSize: number) {\n        let request = this.createRequest(requestSize, responseSize);\n        let record = new DataChannelLatencyTestRecord(request);\n        this.records.set(record.seq, record);\n        this.sink(request);\n    }\n\n    createRequest(requestSize: number, responseSize: number): DataChannelLatencyTestRequest {\n        return {\n            Seq: this.seq++,\n            FillResponseSize: responseSize,\n            Filler: requestSize ? \"A\".repeat(requestSize) : \"\"\n        }\n    }\n\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Data Channel Latency Test types\n */\n\n\n/**\n * Unix epoch\n */\nexport type DataChannelLatencyTestTimestamp = number;\n\n/**\n * Sequence number represented by unsigned int\n */\nexport type DataChannelLatencyTestSeq = number;\n\n/**\n * Request sent to Streamer\n */\nexport type DataChannelLatencyTestRequest = {\n    Seq: DataChannelLatencyTestSeq;\n    FillResponseSize: number;\n    Filler: string;\n}\n\n/**\n * Response from the Streamer\n */\nexport type DataChannelLatencyTestResponse = {\n    Seq: DataChannelLatencyTestSeq;\n    Filler: string;\n    ReceivedTimestamp: DataChannelLatencyTestTimestamp;\n    SentTimestamp: DataChannelLatencyTestTimestamp;\n}\n\nexport type DataChannelLatencyTestResult = {\n    records: Map<DataChannelLatencyTestSeq, DataChannelLatencyTestRecord>\n    dataChannelRtt: number,\n    playerToStreamerTime: number,\n    streamerToPlayerTime: number,\n    exportLatencyAsCSV: () => string\n}\n\nexport class DataChannelLatencyTestRecord {\n    seq: DataChannelLatencyTestSeq;\n    playerSentTimestamp: DataChannelLatencyTestTimestamp;\n    playerReceivedTimestamp: DataChannelLatencyTestTimestamp;\n    streamerReceivedTimestamp: DataChannelLatencyTestTimestamp;\n    streamerSentTimestamp: DataChannelLatencyTestTimestamp;\n    requestFillerSize: number;\n    responseFillerSize: number;\n\n    constructor(request: DataChannelLatencyTestRequest) {\n        this.seq = request.Seq;\n        this.playerSentTimestamp = Date.now();\n        this.requestFillerSize = request.Filler ? request.Filler.length : 0;\n    }\n\n    update(response: DataChannelLatencyTestResponse) {\n        this.playerReceivedTimestamp = Date.now();\n        this.streamerReceivedTimestamp = response.ReceivedTimestamp;\n        this.streamerSentTimestamp = response.SentTimestamp;\n        this.responseFillerSize = response.Filler ? response.Filler.length : 0;\n    }\n\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\nimport { DataChannelController } from './DataChannelController';\n\n/**\n * A class for sending data channel messages\n */\nexport class DataChannelSender {\n    dataChannelProvider: DataChannelController;\n\n    /**\n     * @param dataChannelProvider - Data channel object type\n     */\n    constructor(dataChannelProvider: DataChannelController) {\n        this.dataChannelProvider = dataChannelProvider;\n    }\n\n    canSend(): boolean {\n        return (\n            this.dataChannelProvider.getDataChannelInstance().dataChannel !==\n                undefined &&\n            this.dataChannelProvider.getDataChannelInstance().dataChannel\n                .readyState == 'open'\n        );\n    }\n\n    /**\n     * Send Data over the Data channel to the UE Instance\n     * @param data - Message Data Array Buffer\n     */\n    sendData(data: ArrayBuffer) {\n        // reset the afk inactivity\n        const dataChannelInstance =\n            this.dataChannelProvider.getDataChannelInstance();\n\n        if (dataChannelInstance.dataChannel.readyState == 'open') {\n            dataChannelInstance.dataChannel.send(data);\n            Logger.Log(\n                Logger.GetStackTrace(),\n                `Message Sent: ${new Uint8Array(data)}`,\n                6\n            );\n            this.resetAfkWarningTimerOnDataSend();\n        } else {\n            Logger.Error(\n                Logger.GetStackTrace(),\n                `Message Failed: ${new Uint8Array(data)}`\n            );\n        }\n    }\n\n    /**\n     * An override method for resetting the Afk warning timer when data is sent over the data channel\n     */\n    resetAfkWarningTimerOnDataSend() {\n        // Base Functionality: Do Nothing\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Latency Test Results Data\n */\nexport class InitialSettings {\n    PixelStreamingSettings: PixelStreamingSettings;\n    EncoderSettings: EncoderSettings;\n    WebRTCSettings: WebRTCSettings;\n\n    constructor() {\n        this.PixelStreamingSettings = new PixelStreamingSettings();\n        this.EncoderSettings = new EncoderSettings();\n        this.WebRTCSettings = new WebRTCSettings();\n    }\n\n    /**\n     * Checks for compatibility with the FPS and MaxFPS stats between 4.27 and 5\n     */\n    ueCompatible() {\n        if (this.WebRTCSettings.MaxFPS != null) {\n            this.WebRTCSettings.FPS = this.WebRTCSettings.MaxFPS;\n        }\n    }\n}\n\n/**\n * A class for handling Pixel Streaming details\n */\nexport class PixelStreamingSettings {\n    AllowPixelStreamingCommands?: boolean;\n    DisableLatencyTest?: boolean;\n}\n\n/**\n * A class for handling encoder stats\n */\nexport class EncoderSettings {\n    TargetBitrate?: number;\n    MaxBitrate?: number;\n    MinQP?: number;\n    MaxQP?: number;\n    RateControl?: 'CBR' | 'VBR' | 'ConstQP';\n    FillerData?: boolean;\n    MultiPass?: 'DISABLED' | 'QUARTER' | 'FULL';\n}\n\n/**\n * A class for handling web rtc stats\n */\nexport class WebRTCSettings {\n    DegradationPref?: 'BALANCED' | 'MAINTAIN_FRAMERATE' | 'MAINTAIN_RESOLUTION';\n    MinBitrate?: number;\n    MaxBitrate?: number;\n    LowQP?: number;\n    HighQP?: number;\n    // UE4.27 compatible\n    MaxFPS?: number;\n    // UE5 compatible\n    FPS?: number;\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\n/**\n * Latency Test Results Data\n */\nexport class LatencyTestResults {\n    //Fields Set from the latency payload regardless of version\n    ReceiptTimeMs: number = null;\n    TransmissionTimeMs: number = null;\n\n    //Fields Set from the latency payload from 4.27.2\n    PreCaptureTimeMs: number = null;\n    PostCaptureTimeMs: number = null;\n    PreEncodeTimeMs: number = null;\n    PostEncodeTimeMs: number = null;\n\n    //Fields Set from the latency payload from 5.0\n    EncodeMs: number = null;\n    CaptureToSendMs: number = null;\n\n    //Fields Set when processed\n    testStartTimeMs = 0;\n    browserReceiptTimeMs = 0;\n\n    //Fields set from calculations\n    latencyExcludingDecode = 0;\n    testDuration = 0;\n    //ueLatency: number = 0;\n    networkLatency = 0;\n    browserSendLatency = 0;\n    frameDisplayDeltaTimeMs = 0;\n    endToEndLatency = 0;\n    //uePixelStreamLatency: number = 0;\n    encodeLatency = 0;\n\n    /**\n     * Sets the Delta Time Milliseconds\n     * @param DeltaTimeMs - Delta Time Milliseconds\n     */\n    setFrameDisplayDeltaTime(DeltaTimeMs: number) {\n        if (this.frameDisplayDeltaTimeMs == 0) {\n            this.frameDisplayDeltaTimeMs = Math.round(DeltaTimeMs);\n        }\n    }\n\n    /**\n     * Process the encoder times and set them\n     */\n    processFields() {\n        if (\n            this.EncodeMs == null &&\n            (this.PreEncodeTimeMs != null || this.PostEncodeTimeMs != null)\n        ) {\n            Logger.Log(\n                Logger.GetStackTrace(),\n                `Setting Encode Ms \\n ${this.PostEncodeTimeMs} \\n ${this.PreEncodeTimeMs}`,\n                6\n            );\n            this.EncodeMs = this.PostEncodeTimeMs - this.PreEncodeTimeMs;\n        }\n\n        if (\n            this.CaptureToSendMs == null &&\n            (this.PreCaptureTimeMs != null || this.PostCaptureTimeMs != null)\n        ) {\n            Logger.Log(\n                Logger.GetStackTrace(),\n                `Setting CaptureToSendMs Ms \\n ${this.PostCaptureTimeMs} \\n ${this.PreCaptureTimeMs}`,\n                6\n            );\n            this.CaptureToSendMs =\n                this.PostCaptureTimeMs - this.PreCaptureTimeMs;\n        }\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * A class for managing the freeze frame object\n */\nexport class FreezeFrame {\n    protected rootDiv: HTMLElement;\n    protected rootElement: HTMLDivElement;\n    imageElement: HTMLImageElement;\n    freezeFrameHeight = 0;\n    freezeFrameWidth = 0;\n\n    /**\n     * Construct a freeze frame\n     * @param rootDiv the div that a freeze frame element will be injected into\n     */\n    constructor(rootDiv: HTMLElement) {\n        this.rootDiv = rootDiv;\n\n        // create the overlay\n        this.rootElement = document.createElement('div');\n        this.rootElement.id = 'freezeFrame';\n        this.rootElement.style.display = 'none';\n        this.rootElement.style.pointerEvents = 'none';\n        this.rootElement.style.position = 'absolute';\n        this.rootElement.style.zIndex = '20';\n\n        // create the image place holder\n        this.imageElement = document.createElement('img');\n        this.imageElement.style.position = 'absolute';\n\n        // append the image into the root element and append the element to the root div\n        this.rootElement.appendChild(this.imageElement);\n        this.rootDiv.appendChild(this.rootElement);\n    }\n\n    /**\n     * Set the freeze frame element for showing\n     */\n    setElementForShow() {\n        this.rootElement.style.display = 'block';\n    }\n\n    /**\n     * Set the freeze frame element for hiding\n     */\n    setElementForHide() {\n        this.rootElement.style.display = 'none';\n    }\n\n    /**\n     * Update the freeze frames image source\n     * @param jpeg - the freeze frame image as a byte array data\n     */\n    updateImageElementSource(jpeg: Uint8Array) {\n        const base64 = btoa(\n            jpeg.reduce((data, byte) => data + String.fromCharCode(byte), '')\n        );\n        this.imageElement.src = 'data:image/jpeg;base64,' + base64;\n    }\n\n    /**\n     * Set the dimensions for the freeze frame from the element and resize it\n     */\n    setDimensionsFromElementAndResize() {\n        this.freezeFrameHeight = this.imageElement.naturalHeight;\n        this.freezeFrameWidth = this.imageElement.naturalWidth;\n        this.resize();\n    }\n\n    /**\n     * Resize a freeze frame element\n     */\n    resize() {\n        if (this.freezeFrameWidth !== 0 && this.freezeFrameHeight !== 0) {\n            let displayWidth = 0;\n            let displayHeight = 0;\n            let displayTop = 0;\n            let displayLeft = 0;\n            const parentAspectRatio =\n                this.rootDiv.clientWidth / this.rootDiv.clientHeight;\n            const videoAspectRatio =\n                this.freezeFrameWidth / this.freezeFrameHeight;\n            if (parentAspectRatio < videoAspectRatio) {\n                displayWidth = this.rootDiv.clientWidth;\n                displayHeight = Math.floor(\n                    this.rootDiv.clientWidth / videoAspectRatio\n                );\n                displayTop = Math.floor(\n                    (this.rootDiv.clientHeight - displayHeight) * 0.5\n                );\n                displayLeft = 0;\n            } else {\n                displayWidth = Math.floor(\n                    this.rootDiv.clientHeight * videoAspectRatio\n                );\n                displayHeight = this.rootDiv.clientHeight;\n                displayTop = 0;\n                displayLeft = Math.floor(\n                    (this.rootDiv.clientWidth - displayWidth) * 0.5\n                );\n            }\n            this.rootElement.style.width = this.rootDiv.offsetWidth + 'px';\n            this.rootElement.style.height = this.rootDiv.offsetHeight + 'px';\n            this.rootElement.style.left = 0 + 'px';\n            this.rootElement.style.top = 0 + 'px';\n\n            this.imageElement.style.width = displayWidth + 'px';\n            this.imageElement.style.height = displayHeight + 'px';\n            this.imageElement.style.left = displayLeft + 'px';\n            this.imageElement.style.top = displayTop + 'px';\n        }\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\nimport { FreezeFrame } from './FreezeFrame';\n\n/**\n * A class for controlling freeze frame functionality\n */\nexport class FreezeFrameController {\n    freezeFrame: FreezeFrame;\n    receiving = false;\n    size = 0;\n    jpeg: Uint8Array = undefined;\n    valid = false;\n    freezeFrameDelay = 50;\n\n    /**\n     * Construct a freeze frame controller\n     * @param rootDiv - the div that a freeze frame element will be injected into\n     */\n    constructor(rootDiv: HTMLElement) {\n        this.freezeFrame = new FreezeFrame(rootDiv);\n    }\n\n    /**\n     * Show the freeze frame if it is valid\n     */\n    showFreezeFrame() {\n        if (this.valid) {\n            this.freezeFrame.setElementForShow();\n        }\n    }\n\n    /**\n     * Hide the freeze frame and set the validity to false\n     */\n    hideFreezeFrame() {\n        this.valid = false;\n        this.freezeFrame.setElementForHide();\n    }\n\n    /**\n     * Update the freeze frames image source and load it\n     * @param jpeg - the freeze frame image as a byte array data\n     * @param onLoadCallBack - a call back for managing if the play overlay needs to be shown or not\n     */\n    updateFreezeFrameAndShow(jpeg: Uint8Array, onLoadCallBack: () => void) {\n        this.freezeFrame.updateImageElementSource(jpeg);\n        this.freezeFrame.imageElement.onload = () => {\n            this.freezeFrame.setDimensionsFromElementAndResize();\n            onLoadCallBack();\n        };\n    }\n\n    /**\n     * Process the new freeze frame image and update it\n     * @param view - the freeze frame image as a byte array data\n     * @param onLoadCallBack - a call back for managing if the play overlay needs to be shown or not\n     */\n    processFreezeFrameMessage(view: Uint8Array, onLoadCallBack: () => void) {\n        // Reset freeze frame if we got a freeze frame message and we are not \"receiving\" yet.\n        if (!this.receiving) {\n            this.receiving = true;\n            this.valid = false;\n            this.size = 0;\n            this.jpeg = undefined;\n        }\n\n        // Extract total size of freeze frame (across all chunks)\n        this.size = new DataView(view.slice(1, 5).buffer).getInt32(0, true);\n\n        // Get the jpeg part of the payload\n        const jpegBytes = view.slice(1 + 4);\n\n        // Append to existing jpeg that holds the freeze frame\n        if (this.jpeg) {\n            const jpeg = new Uint8Array(this.jpeg.length + jpegBytes.length);\n            jpeg.set(this.jpeg, 0);\n            jpeg.set(jpegBytes, this.jpeg.length);\n            this.jpeg = jpeg;\n        }\n        // No existing freeze frame jpeg, make one\n        else {\n            this.jpeg = jpegBytes;\n            this.receiving = true;\n            Logger.Log(\n                Logger.GetStackTrace(),\n                `received first chunk of freeze frame: ${this.jpeg.length}/${this.size}`,\n                6\n            );\n        }\n\n        // Finished receiving freeze frame, we can show it now\n        if (this.jpeg.length === this.size) {\n            this.receiving = false;\n            this.valid = true;\n            Logger.Log(\n                Logger.GetStackTrace(),\n                `received complete freeze frame ${this.size}`,\n                6\n            );\n            this.updateFreezeFrameAndShow(this.jpeg, onLoadCallBack);\n        }\n        // We received more data than the freeze frame payload message indicate (this is an error)\n        else if (this.jpeg.length > this.size) {\n            Logger.Error(\n                Logger.GetStackTrace(),\n                `received bigger freeze frame than advertised: ${this.jpeg.length}/${this.size}`\n            );\n            this.jpeg = undefined;\n            this.receiving = false;\n        }\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { CoordinateConverter } from '../Util/CoordinateConverter';\nimport { StreamMessageController } from '../UeInstanceMessage/StreamMessageController';\nimport { VideoPlayer } from '../VideoPlayer/VideoPlayer';\nimport { ITouchController } from './ITouchController';\nimport { MouseButton } from './MouseButtons';\nimport { EventListenerTracker } from '../Util/EventListenerTracker';\n\n/**\n * Allows for the usage of fake touch events and implements ITouchController\n * @param dataChannelController - The controller for the Data channel\n * @param videoElementParent - The video player DOM element\n */\nexport class FakeTouchController implements ITouchController {\n    fakeTouchFinger: FakeTouchFinger;\n    toStreamerMessagesProvider: StreamMessageController;\n    videoElementProvider: VideoPlayer;\n    coordinateConverter: CoordinateConverter;\n    videoElementParentClientRect: DOMRect;\n\n    // Utility for keeping track of event handlers and unregistering them\n    private touchEventListenerTracker = new EventListenerTracker();\n\n    /**\n     * @param toStreamerMessagesProvider - Stream message instance\n     * @param videoElementProvider - Video element instance\n     * @param coordinateConverter - A coordinate converter instance\n     */\n    constructor(\n        toStreamerMessagesProvider: StreamMessageController,\n        videoElementProvider: VideoPlayer,\n        coordinateConverter: CoordinateConverter\n    ) {\n        this.toStreamerMessagesProvider = toStreamerMessagesProvider;\n        this.videoElementProvider = videoElementProvider;\n        this.coordinateConverter = coordinateConverter;\n        const ontouchstart = (ev: TouchEvent) => this.onTouchStart(ev);\n        const ontouchend = (ev: TouchEvent) => this.onTouchEnd(ev);\n        const ontouchmove = (ev: TouchEvent) => this.onTouchMove(ev);\n        document.addEventListener('touchstart', ontouchstart, { passive: false });\n        document.addEventListener('touchend', ontouchend, { passive: false });\n        document.addEventListener('touchmove', ontouchmove, { passive: false });\n        this.touchEventListenerTracker.addUnregisterCallback(\n            () => document.removeEventListener('touchstart', ontouchstart)\n        );\n        this.touchEventListenerTracker.addUnregisterCallback(\n            () => document.removeEventListener('touchend', ontouchend)\n        );\n        this.touchEventListenerTracker.addUnregisterCallback(\n            () => document.removeEventListener('touchmove', ontouchmove)\n        );\n    }\n\n    /**\n     * Unregister all touch events\n     */\n    unregisterTouchEvents() {\n        this.touchEventListenerTracker.unregisterAll();\n    }\n\n    /**\n     * Sets the video Element Parent Client Rect numbers for this class\n     * @param videoElementParentClientRect - a html ElementParentClientRect object\n     */\n    setVideoElementParentClientRect(videoElementParentClientRect: DOMRect) {\n        this.videoElementParentClientRect = videoElementParentClientRect;\n    }\n\n    /**\n     * When a touch event begins\n     * @param touch - the activating touch event\n     */\n    onTouchStart(touch: TouchEvent): void {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        if (this.fakeTouchFinger == null) {\n            const first_touch = touch.changedTouches[0];\n            this.fakeTouchFinger = new FakeTouchFinger(\n                first_touch.identifier,\n                first_touch.clientX - this.videoElementParentClientRect.left,\n                first_touch.clientY - this.videoElementParentClientRect.top\n            );\n\n            const videoElementParent =\n                this.videoElementProvider.getVideoParentElement() as HTMLDivElement;\n            const mouseEvent = new MouseEvent('mouseenter', first_touch);\n            videoElementParent.dispatchEvent(mouseEvent);\n\n            const coord = this.coordinateConverter.normalizeAndQuantizeUnsigned(\n                this.fakeTouchFinger.x,\n                this.fakeTouchFinger.y\n            );\n            const toStreamerHandlers =\n                this.toStreamerMessagesProvider.toStreamerHandlers;\n            toStreamerHandlers.get('MouseDown')([\n                MouseButton.mainButton,\n                coord.x,\n                coord.y\n            ]);\n        }\n        touch.preventDefault();\n    }\n\n    /**\n     * When a touch event ends\n     * @param touchEvent - the activating touch event\n     */\n    onTouchEnd(touchEvent: TouchEvent): void {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        const videoElementParent =\n            this.videoElementProvider.getVideoParentElement();\n        const toStreamerHandlers =\n            this.toStreamerMessagesProvider.toStreamerHandlers;\n\n        for (let t = 0; t < touchEvent.changedTouches.length; t++) {\n            const touch = touchEvent.changedTouches[t];\n            if (touch.identifier === this.fakeTouchFinger.id) {\n                const x =\n                    touch.clientX - this.videoElementParentClientRect.left;\n                const y = touch.clientY - this.videoElementParentClientRect.top;\n                const coord =\n                    this.coordinateConverter.normalizeAndQuantizeUnsigned(x, y);\n                toStreamerHandlers.get('MouseUp')([\n                    MouseButton.mainButton,\n                    coord.x,\n                    coord.y\n                ]);\n\n                const mouseEvent = new MouseEvent('mouseleave', touch);\n                videoElementParent.dispatchEvent(mouseEvent);\n                this.fakeTouchFinger = null;\n                break;\n            }\n        }\n        touchEvent.preventDefault();\n    }\n\n    /**\n     * On a Move touch event\n     * @param touchEvent - the activating touch event\n     */\n    onTouchMove(touchEvent: TouchEvent): void {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        const toStreamerHandlers =\n            this.toStreamerMessagesProvider.toStreamerHandlers;\n\n        for (let t = 0; t < touchEvent.touches.length; t++) {\n            const touch = touchEvent.touches[t];\n            if (touch.identifier === this.fakeTouchFinger.id) {\n                const x =\n                    touch.clientX - this.videoElementParentClientRect.left;\n                const y = touch.clientY - this.videoElementParentClientRect.top;\n                const coord =\n                    this.coordinateConverter.normalizeAndQuantizeUnsigned(x, y);\n                const delta =\n                    this.coordinateConverter.normalizeAndQuantizeSigned(\n                        x - this.fakeTouchFinger.x,\n                        y - this.fakeTouchFinger.y\n                    );\n                toStreamerHandlers.get('MouseMove')([\n                    coord.x,\n                    coord.y,\n                    delta.x,\n                    delta.y\n                ]);\n                this.fakeTouchFinger.x = x;\n                this.fakeTouchFinger.y = y;\n                break;\n            }\n        }\n        touchEvent.preventDefault();\n    }\n}\n\n/**\n * The interface for finger position mapping\n */\nexport class FakeTouchFinger {\n    id: number;\n    x: number;\n    y: number;\n\n    /**\n     * @param id - the button id\n     * @param x - the x axis value\n     * @param y - the y axis value\n     */\n    constructor(id: number, x: number, y: number) {\n        this.id = id;\n        this.x = x;\n        this.y = y;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\nimport { StreamMessageController } from '../UeInstanceMessage/StreamMessageController';\nimport { EventListenerTracker } from '../Util/EventListenerTracker';\nimport { Controller } from './GamepadTypes';\n\n/**\n * The class that handles the functionality of gamepads and controllers\n */\nexport class GamePadController {\n    controllers: Array<Controller>;\n    requestAnimationFrame: (callback: FrameRequestCallback) => number;\n    toStreamerMessagesProvider: StreamMessageController;\n\n    // Utility for keeping track of event handlers and unregistering them\n    private gamePadEventListenerTracker = new EventListenerTracker();\n\n    /**\n     * @param toStreamerMessagesProvider - Stream message instance\n     */\n    constructor(toStreamerMessagesProvider: StreamMessageController) {\n        this.toStreamerMessagesProvider = toStreamerMessagesProvider;\n\n        this.requestAnimationFrame = (\n            window.mozRequestAnimationFrame ||\n            window.webkitRequestAnimationFrame ||\n            window.requestAnimationFrame\n        ).bind(window);\n        const browserWindow = window as Window;\n        if ('GamepadEvent' in browserWindow) {\n            const onGamePadConnected = (ev: GamepadEvent) =>\n                this.gamePadConnectHandler(ev);\n            const onGamePadDisconnected = (ev: GamepadEvent) =>\n                this.gamePadDisconnectHandler(ev);\n            window.addEventListener('gamepadconnected', onGamePadConnected);\n            window.addEventListener('gamepaddisconnected', onGamePadDisconnected);\n            this.gamePadEventListenerTracker.addUnregisterCallback(\n                () => window.removeEventListener('gamepadconnected', onGamePadConnected)\n            );\n            this.gamePadEventListenerTracker.addUnregisterCallback(\n                () => window.removeEventListener('gamepaddisconnected', onGamePadDisconnected)\n            );\n        } else if ('WebKitGamepadEvent' in browserWindow) {\n            const onWebkitGamePadConnected = (ev: GamepadEvent) => this.gamePadConnectHandler(ev);\n            const onWebkitGamePadDisconnected = (ev: GamepadEvent) => this.gamePadDisconnectHandler(ev);\n            window.addEventListener('webkitgamepadconnected', onWebkitGamePadConnected);\n            window.addEventListener('webkitgamepaddisconnected', onWebkitGamePadDisconnected);\n            this.gamePadEventListenerTracker.addUnregisterCallback(\n                () => window.removeEventListener('webkitgamepadconnected', onWebkitGamePadConnected)\n            );\n            this.gamePadEventListenerTracker.addUnregisterCallback(\n                () => window.removeEventListener('webkitgamepaddisconnected', onWebkitGamePadDisconnected)\n            );\n        }\n        this.controllers = [];\n        if (navigator.getGamepads) {\n            for (const gamepad of navigator.getGamepads()) {\n                if (gamepad) {\n                    this.gamePadConnectHandler(new GamepadEvent('gamepadconnected', { gamepad }));\n                }\n            }\n        }\n    }\n\n    /**\n     * Unregisters all event handlers\n     */\n    unregisterGamePadEvents() {\n        this.gamePadEventListenerTracker.unregisterAll();\n        for(const controller of this.controllers) {\n            if(controller.id !== undefined) {\n                this.onGamepadDisconnected(controller.id);\n            }\n        }\n        this.controllers = [];\n        this.onGamepadConnected = () => { /* */ };\n        this.onGamepadDisconnected = () => { /* */ };\n    }\n\n    /**\n     * Connects the gamepad handler\n     * @param gamePadEvent - the activating gamepad event\n     */\n    gamePadConnectHandler(gamePadEvent: GamepadEvent) {\n        Logger.Log(Logger.GetStackTrace(), 'Gamepad connect handler', 6);\n        const gamepad = gamePadEvent.gamepad;\n\n        const temp: Controller = {\n            currentState: gamepad,\n            prevState: gamepad,\n            id: undefined\n        };\n\n        this.controllers.push(temp);\n        this.controllers[gamepad.index].currentState = gamepad;\n        this.controllers[gamepad.index].prevState = gamepad;\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'gamepad: ' + gamepad.id + ' connected',\n            6\n        );\n        window.requestAnimationFrame(() => this.updateStatus());\n        this.onGamepadConnected();\n    }\n\n    /**\n     * Disconnects the gamepad handler\n     * @param gamePadEvent - the activating gamepad event\n     */\n    gamePadDisconnectHandler(gamePadEvent: GamepadEvent) {\n        Logger.Log(Logger.GetStackTrace(), 'Gamepad disconnect handler', 6);\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'gamepad: ' + gamePadEvent.gamepad.id + ' disconnected',\n            6\n        );\n        const deletedController = this.controllers[gamePadEvent.gamepad.index];\n        delete this.controllers[gamePadEvent.gamepad.index];\n        this.controllers = this.controllers.filter(\n            (controller) => controller !== undefined\n        );\n        this.onGamepadDisconnected(deletedController.id);\n    }\n\n    /**\n     * Scan for connected gamepads\n     */\n    scanGamePads() {\n        const gamepads = navigator.getGamepads\n            ? navigator.getGamepads()\n            : navigator.webkitGetGamepads\n            ? navigator.webkitGetGamepads()\n            : [];\n        for (let i = 0; i < gamepads.length; i++) {\n            if (gamepads[i] && gamepads[i].index in this.controllers) {\n                this.controllers[gamepads[i].index].currentState = gamepads[i];\n            }\n        }\n    }\n\n    /**\n     * Updates the status of the gamepad and sends the inputs\n     */\n    updateStatus() {\n        this.scanGamePads();\n        const toStreamerHandlers =\n            this.toStreamerMessagesProvider.toStreamerHandlers;\n\n        // Iterate over multiple controllers in the case the multiple gamepads are connected\n        for (const controller of this.controllers) {\n            // If we haven't received an id (possible if using an older version of UE), return to original functionality\n            const controllerIndex = (controller.id === undefined) ? this.controllers.indexOf(controller) : controller.id;\n            const currentState = controller.currentState;\n            for (let i = 0; i < controller.currentState.buttons.length; i++) {\n                const currentButton = controller.currentState.buttons[i];\n                const previousButton = controller.prevState.buttons[i];\n                if (currentButton.pressed) {\n                    // press\n                    if (i == gamepadLayout.LeftTrigger) {\n                        //                       UEs left analog has a button index of 5\n                        toStreamerHandlers.get('GamepadAnalog')([\n                            controllerIndex,\n                            5,\n                            currentButton.value\n                        ]);\n                    } else if (i == gamepadLayout.RightTrigger) {\n                        //                       UEs right analog has a button index of 6\n                        toStreamerHandlers.get('GamepadAnalog')([\n                            controllerIndex,\n                            6,\n                            currentButton.value\n                        ]);\n                    } else {\n                        toStreamerHandlers.get('GamepadButtonPressed')([\n                            controllerIndex,\n                            i,\n                            previousButton.pressed ? 1 : 0\n                        ]);\n                    }\n                } else if (!currentButton.pressed && previousButton.pressed) {\n                    // release\n                    if (i == gamepadLayout.LeftTrigger) {\n                        //                       UEs left analog has a button index of 5\n                        toStreamerHandlers.get('GamepadAnalog')([\n                            controllerIndex,\n                            5,\n                            0\n                        ]);\n                    } else if (i == gamepadLayout.RightTrigger) {\n                        //                       UEs right analog has a button index of 6\n                        toStreamerHandlers.get('GamepadAnalog')([\n                            controllerIndex,\n                            6,\n                            0\n                        ]);\n                    } else {\n                        toStreamerHandlers.get('GamepadButtonReleased')([\n                            controllerIndex,\n                            i\n                        ]);\n                    }\n                }\n            }\n            // Iterate over gamepad axes (we will increment in lots of 2 as there is 2 axes per stick)\n            for (let i = 0; i < currentState.axes.length; i += 2) {\n                // Horizontal axes are even numbered\n                const x = parseFloat(currentState.axes[i].toFixed(4));\n\n                // Vertical axes are odd numbered\n                // https://w3c.github.io/gamepad/#remapping Gamepad browser side standard mapping has positive down, negative up. This is downright disgusting. So we fix it.\n                const y = -parseFloat(currentState.axes[i + 1].toFixed(4));\n\n                // UE's analog axes follow the same order as the browsers, but start at index 1 so we will offset as such\n                toStreamerHandlers.get('GamepadAnalog')([\n                    controllerIndex,\n                    i + 1,\n                    x\n                ]); // Horizontal axes, only offset by 1\n                toStreamerHandlers.get('GamepadAnalog')([\n                    controllerIndex,\n                    i + 2,\n                    y\n                ]); // Vertical axes, offset by two (1 to match UEs axes convention and then another 1 for the vertical axes)\n            }\n            this.controllers[controllerIndex].prevState = currentState;\n        }\n        if (this.controllers.length > 0) {\n            this.requestAnimationFrame(() => this.updateStatus());\n        }\n    }\n\n    onGamepadResponseReceived(gamepadId: number) {\n        for(const controller of this.controllers) {\n            if(controller.id === undefined) {\n                controller.id = gamepadId;\n                break;\n            }\n        }\n    }\n\n    /**\n     * Event to send the gamepadconnected message to the application\n     */\n    onGamepadConnected() {\n        // Default Functionality: Do Nothing\n    }\n\n    /**\n     * Event to send the gamepaddisconnected message to the application\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    onGamepadDisconnected(controllerIdx: number) {\n        // Default Functionality: Do Nothing\n    }\n}\n\n\n\n/**\n * Additional types for Window and Navigator\n */\ndeclare global {\n    interface Window {\n        mozRequestAnimationFrame(callback: FrameRequestCallback): number;\n        webkitRequestAnimationFrame(callback: FrameRequestCallback): number;\n    }\n\n    interface Navigator {\n        webkitGetGamepads(): Gamepad[];\n    }\n}\n\n/**\n * Gamepad layout codes enum\n */\nexport enum gamepadLayout {\n    RightClusterBottomButton = 0,\n    RightClusterRightButton = 1,\n    RightClusterLeftButton = 2,\n    RightClusterTopButton = 3,\n    LeftShoulder = 4,\n    RightShoulder = 5,\n    LeftTrigger = 6,\n    RightTrigger = 7,\n    SelectOrBack = 8,\n    StartOrForward = 9,\n    LeftAnalogPress = 10,\n    RightAnalogPress = 11,\n    LeftClusterTopButton = 12,\n    LeftClusterBottomButton = 13,\n    LeftClusterLeftButton = 14,\n    LeftClusterRightButton = 15,\n    CentreButton = 16,\n    // Axes\n    LeftStickHorizontal = 0,\n    LeftStickVertical = 1,\n    RightStickHorizontal = 2,\n    RightStickVertical = 3\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { MouseController } from './MouseController';\nimport { Logger } from '../Logger/Logger';\nimport { IMouseEvents } from './IMouseEvents';\n\n/**\n * Video Player mouse Hover handler\n */\nexport class HoveringMouseEvents implements IMouseEvents {\n    mouseController: MouseController;\n\n    /**\n     * @param mouseController - Mouse Controller instance\n     */\n    constructor(mouseController: MouseController) {\n        this.mouseController = mouseController;\n    }\n\n    /**\n     * Unregister event handlers\n     */\n    unregisterMouseEvents(): void {\n        // empty for HoveringMouseEvents implementation\n    }\n\n    /**\n     * Handle the mouse move event, sends the mouse data to the UE Instance\n     * @param mouseEvent - Mouse Event\n     */\n    updateMouseMovePosition(mouseEvent: MouseEvent) {\n        if (!this.mouseController.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        Logger.Log(Logger.GetStackTrace(), 'MouseMove', 6);\n        const coord =\n            this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(\n                mouseEvent.offsetX,\n                mouseEvent.offsetY\n            );\n        const delta =\n            this.mouseController.coordinateConverter.normalizeAndQuantizeSigned(\n                mouseEvent.movementX,\n                mouseEvent.movementY\n            );\n        const toStreamerHandlers =\n            this.mouseController.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseMove')([\n            coord.x,\n            coord.y,\n            delta.x,\n            delta.y\n        ]);\n        mouseEvent.preventDefault();\n    }\n\n    /**\n     * Handle the mouse Down event, sends the mouse data to the UE Instance\n     * @param mouseEvent - Mouse Event\n     */\n    handleMouseDown(mouseEvent: MouseEvent) {\n        if (!this.mouseController.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        Logger.Log(Logger.GetStackTrace(), 'onMouse Down', 6);\n        const coord =\n            this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(\n                mouseEvent.offsetX,\n                mouseEvent.offsetY\n            );\n        const toStreamerHandlers =\n            this.mouseController.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseDown')([\n            mouseEvent.button,\n            coord.x,\n            coord.y\n        ]);\n        mouseEvent.preventDefault();\n    }\n\n    /**\n     * Handle the mouse Up event, sends the mouse data to the UE Instance\n     * @param mouseEvent - Mouse Event\n     */\n    handleMouseUp(mouseEvent: MouseEvent) {\n        if (!this.mouseController.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        Logger.Log(Logger.GetStackTrace(), 'onMouse Up', 6);\n        const coord =\n            this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(\n                mouseEvent.offsetX,\n                mouseEvent.offsetY\n            );\n        const toStreamerHandlers =\n            this.mouseController.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseUp')([\n            mouseEvent.button,\n            coord.x,\n            coord.y\n        ]);\n        mouseEvent.preventDefault();\n    }\n\n    /**\n     * Consumes the mouse context event. The UE instance has no equivalent and doesn't need to be informed.\n     * @param mouseEvent - Mouse Event\n     */\n    handleContextMenu(mouseEvent: MouseEvent) {\n        if (!this.mouseController.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        mouseEvent.preventDefault();\n    }\n\n    /**\n     * Handle the mouse wheel event, sends the mouse wheel data to the UE Instance\n     * @param wheelEvent - Mouse Event\n     */\n    handleMouseWheel(wheelEvent: WheelEvent) {\n        if (!this.mouseController.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        const coord =\n            this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(\n                wheelEvent.offsetX,\n                wheelEvent.offsetY\n            );\n        const toStreamerHandlers =\n            this.mouseController.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseWheel')([\n            wheelEvent.wheelDelta,\n            coord.x,\n            coord.y\n        ]);\n        wheelEvent.preventDefault();\n    }\n\n    /**\n     * Handle the mouse double click event, sends the mouse data to the UE Instance\n     * @param mouseEvent - Mouse Event\n     */\n    handleMouseDouble(mouseEvent: MouseEvent) {\n        if (!this.mouseController.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        const coord =\n            this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(\n                mouseEvent.offsetX,\n                mouseEvent.offsetY\n            );\n        const toStreamerHandlers =\n            this.mouseController.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseDouble')([\n            mouseEvent.button,\n            coord.x,\n            coord.y\n        ]);\n    }\n\n    /**\n     * Handle the press mouse buttons event, sends the mouse data to the UE Instance\n     * @param mouseEvent - Mouse Event\n     */\n    handlePressMouseButtons(mouseEvent: MouseEvent) {\n        if (!this.mouseController.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        Logger.Log(Logger.GetStackTrace(), 'onMouse press', 6);\n        this.mouseController.pressMouseButtons(\n            mouseEvent.buttons,\n            mouseEvent.offsetX,\n            mouseEvent.offsetY\n        );\n    }\n\n    /**\n     * Handle the release mouse buttons event, sends the mouse data to the UE Instance\n     * @param mouseEvent - Mouse Event\n     */\n    handleReleaseMouseButtons(mouseEvent: MouseEvent) {\n        if (!this.mouseController.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        Logger.Log(Logger.GetStackTrace(), 'onMouse release', 6);\n        this.mouseController.releaseMouseButtons(\n            mouseEvent.buttons,\n            mouseEvent.offsetX,\n            mouseEvent.offsetY\n        );\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { FakeTouchController } from './FakeTouchController';\nimport { KeyboardController } from './KeyboardController';\nimport { MouseController } from './MouseController';\nimport { TouchController } from './TouchController';\nimport { GamePadController } from './GamepadController';\nimport { Config, ControlSchemeType } from '../Config/Config';\nimport { Logger } from '../Logger/Logger';\nimport { CoordinateConverter } from '../Util/CoordinateConverter';\nimport { StreamMessageController } from '../UeInstanceMessage/StreamMessageController';\nimport { VideoPlayer } from '../VideoPlayer/VideoPlayer';\n\n/**\n * Class for making and setting up input class types\n */\nexport class InputClassesFactory {\n    toStreamerMessagesProvider: StreamMessageController;\n    videoElementProvider: VideoPlayer;\n    coordinateConverter: CoordinateConverter;\n    activeKeys: ActiveKeys = new ActiveKeys();\n\n    /**\n     * @param toStreamerMessagesProvider - Stream message instance\n     * @param videoElementProvider - Video Player instance\n     * @param coordinateConverter - A coordinateConverter instance\n     */\n    constructor(\n        toStreamerMessagesProvider: StreamMessageController,\n        videoElementProvider: VideoPlayer,\n        coordinateConverter: CoordinateConverter\n    ) {\n        this.toStreamerMessagesProvider = toStreamerMessagesProvider;\n        this.videoElementProvider = videoElementProvider;\n        this.coordinateConverter = coordinateConverter;\n    }\n\n    /**\n     * Registers browser key events.\n     */\n    registerKeyBoard(config: Config) {\n        Logger.Log(Logger.GetStackTrace(), 'Register Keyboard Events', 7);\n        const keyboardController = new KeyboardController(\n            this.toStreamerMessagesProvider,\n            config,\n            this.activeKeys\n        );\n        keyboardController.registerKeyBoardEvents();\n        return keyboardController;\n    }\n\n    /**\n     * register mouse events based on a control type\n     * @param controlScheme - if the mouse is either hovering or locked\n     */\n    registerMouse(controlScheme: ControlSchemeType) {\n        Logger.Log(Logger.GetStackTrace(), 'Register Mouse Events', 7);\n        const mouseController = new MouseController(\n            this.toStreamerMessagesProvider,\n            this.videoElementProvider,\n            this.coordinateConverter,\n            this.activeKeys\n        );\n\n        switch (controlScheme) {\n            case ControlSchemeType.LockedMouse:\n                mouseController.registerLockedMouseEvents(mouseController);\n                break;\n            case ControlSchemeType.HoveringMouse:\n                mouseController.registerHoveringMouseEvents(mouseController);\n                break;\n            default:\n                Logger.Info(\n                    Logger.GetStackTrace(),\n                    'unknown Control Scheme Type Defaulting to Locked Mouse Events'\n                );\n                mouseController.registerLockedMouseEvents(mouseController);\n                break;\n        }\n\n        return mouseController;\n    }\n\n    /**\n     * register touch events\n     * @param fakeMouseTouch - the faked mouse touch event\n     */\n    registerTouch(\n        fakeMouseTouch: boolean,\n        videoElementParentClientRect: DOMRect\n    ) {\n        Logger.Log(Logger.GetStackTrace(), 'Registering Touch', 6);\n        if (fakeMouseTouch) {\n            const fakeTouchController = new FakeTouchController(\n                this.toStreamerMessagesProvider,\n                this.videoElementProvider,\n                this.coordinateConverter\n            );\n            fakeTouchController.setVideoElementParentClientRect(\n                videoElementParentClientRect\n            );\n            return fakeTouchController;\n        } else {\n            return new TouchController(\n                this.toStreamerMessagesProvider,\n                this.videoElementProvider,\n                this.coordinateConverter\n            );\n        }\n    }\n\n    /**\n     * registers a gamepad\n     */\n    registerGamePad() {\n        Logger.Log(Logger.GetStackTrace(), 'Register Game Pad', 7);\n        const gamePadController = new GamePadController(\n            this.toStreamerMessagesProvider\n        );\n        return gamePadController;\n    }\n}\n\n/**\n * A class that keeps track of current active keys\n */\nexport class ActiveKeys {\n    activeKeys: Array<number> = [];\n    constructor() {\n        this.activeKeys = [];\n    }\n\n    /**\n     * Get the current array of active keys\n     * @returns - an array of active keys\n     */\n    getActiveKeys(): number[] {\n        return this.activeKeys;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { SpecialKeyCodes } from './SpecialKeyCodes';\nimport { Logger } from '../Logger/Logger';\nimport { ActiveKeys } from './InputClassesFactory';\nimport { StreamMessageController } from '../UeInstanceMessage/StreamMessageController';\nimport { Config, Flags } from '../Config/Config';\nimport { EventListenerTracker } from '../Util/EventListenerTracker';\n\ninterface ICodeToKeyCode {\n    [key: string]: number;\n}\n\n/**\n * Handles the Keyboard Inputs for the document\n */\nexport class KeyboardController {\n    toStreamerMessagesProvider: StreamMessageController;\n    activeKeysProvider: ActiveKeys;\n    config: Config;\n\n    // Utility for keeping track of event handlers and unregistering them\n    private keyboardEventListenerTracker = new EventListenerTracker();\n\n    /*\n     * New browser APIs have moved away from KeyboardEvent.keyCode to KeyboardEvent.Code.\n     * For details see: https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/keyCode#constants_for_keycode_value\n     * We still use old KeyboardEvent.keyCode integers in the UE C++ side, so we need a way to map the new\n     * string-based KeyboardEvent.Code to the old integers.\n     */\n    CodeToKeyCode: ICodeToKeyCode = {\n        Escape: 27,\n        Digit0: 48,\n        Digit1: 49,\n        Digit2: 50,\n        Digit3: 51,\n        Digit4: 52,\n        Digit5: 53,\n        Digit6: 54,\n        Digit7: 55,\n        Digit8: 56,\n        Digit9: 57,\n        Minus: 173,\n        Equal: 187,\n        Backspace: 8,\n        Tab: 9,\n        KeyQ: 81,\n        KeyW: 87,\n        KeyE: 69,\n        KeyR: 82,\n        KeyT: 84,\n        KeyY: 89,\n        KeyU: 85,\n        KeyI: 73,\n        KeyO: 79,\n        KeyP: 80,\n        BracketLeft: 219,\n        BracketRight: 221,\n        Enter: 13,\n        ControlLeft: 17,\n        KeyA: 65,\n        KeyS: 83,\n        KeyD: 68,\n        KeyF: 70,\n        KeyG: 71,\n        KeyH: 72,\n        KeyJ: 74,\n        KeyK: 75,\n        KeyL: 76,\n        Semicolon: 186,\n        Quote: 222,\n        Backquote: 192,\n        ShiftLeft: 16,\n        Backslash: 220,\n        KeyZ: 90,\n        KeyX: 88,\n        KeyC: 67,\n        KeyV: 86,\n        KeyB: 66,\n        KeyN: 78,\n        KeyM: 77,\n        Comma: 188,\n        Period: 190,\n        Slash: 191,\n        ShiftRight: 253,\n        AltLeft: 18,\n        Space: 32,\n        CapsLock: 20,\n        F1: 112,\n        F2: 113,\n        F3: 114,\n        F4: 115,\n        F5: 116,\n        F6: 117,\n        F7: 118,\n        F8: 119,\n        F9: 120,\n        F10: 121,\n        F11: 122,\n        F12: 123,\n        Pause: 19,\n        ScrollLock: 145,\n        NumpadDivide: 111,\n        NumpadMultiply: 106,\n        NumpadSubtract: 109,\n        NumpadAdd: 107,\n        NumpadDecimal: 110,\n        Numpad9: 105,\n        Numpad8: 104,\n        Numpad7: 103,\n        Numpad6: 102,\n        Numpad5: 101,\n        Numpad4: 100,\n        Numpad3: 99,\n        Numpad2: 98,\n        Numpad1: 97,\n        Numpad0: 96,\n        NumLock: 144,\n        ControlRight: 254,\n        AltRight: 255,\n        Home: 36,\n        End: 35,\n        ArrowUp: 38,\n        ArrowLeft: 37,\n        ArrowRight: 39,\n        ArrowDown: 40,\n        PageUp: 33,\n        PageDown: 34,\n        Insert: 45,\n        Delete: 46,\n        ContextMenu: 93\n    };\n\n    /**\n     * @param toStreamerMessagesProvider Stream message provider class object\n     * @param config The applications configuration. We're interested in the suppress browser keys option\n     * @param activeKeysProvider Active keys provider class object\n     */\n    constructor(\n        toStreamerMessagesProvider: StreamMessageController,\n        config: Config,\n        activeKeysProvider: ActiveKeys\n    ) {\n        this.toStreamerMessagesProvider = toStreamerMessagesProvider;\n        this.config = config;\n        this.activeKeysProvider = activeKeysProvider;\n    }\n\n    /**\n     * Registers document keyboard events with the controller\n     */\n    registerKeyBoardEvents() {\n        const keyDownHandler = (ev: KeyboardEvent) => this.handleOnKeyDown(ev);\n        const keyUpHandler = (ev: KeyboardEvent) => this.handleOnKeyUp(ev);\n        const keyPressHandler = (ev: KeyboardEvent) => this.handleOnKeyPress(ev);\n\n        document.addEventListener(\"keydown\", keyDownHandler);\n        document.addEventListener(\"keyup\", keyUpHandler);\n\n        //This has been deprecated as at Jun 13 2021\n        document.addEventListener(\"keypress\", keyPressHandler);\n\n        this.keyboardEventListenerTracker.addUnregisterCallback(\n            () => document.removeEventListener(\"keydown\", keyDownHandler)\n        );\n        this.keyboardEventListenerTracker.addUnregisterCallback(\n            () => document.removeEventListener(\"keyup\", keyUpHandler)\n        );\n        this.keyboardEventListenerTracker.addUnregisterCallback(\n            () => document.removeEventListener(\"keypress\", keyPressHandler)\n        );\n    }\n\n    /**\n     * Unregisters document keyboard events\n     */\n    unregisterKeyBoardEvents() {\n        this.keyboardEventListenerTracker.unregisterAll();\n    }\n\n    /**\n     * Handles When a key is down\n     * @param keyboardEvent - Keyboard event\n     */\n    handleOnKeyDown(keyboardEvent: KeyboardEvent) {\n        const keyCode = this.getKeycode(keyboardEvent);\n        if (!keyCode) {\n            return;\n        }\n\n        Logger.Log(\n            Logger.GetStackTrace(),\n            `key down ${keyCode}, repeat = ${keyboardEvent.repeat}`,\n            6\n        );\n        const toStreamerHandlers =\n            this.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('KeyDown')([\n            this.getKeycode(keyboardEvent),\n            keyboardEvent.repeat ? 1 : 0\n        ]);\n        const activeKeys = this.activeKeysProvider.getActiveKeys();\n        activeKeys.push(keyCode);\n        // Backspace is not considered a keypress in JavaScript but we need it\n        // to be so characters may be deleted in a UE text entry field.\n        if (keyCode === SpecialKeyCodes.backSpace) {\n            document.dispatchEvent(\n                new KeyboardEvent('keypress', {\n                    charCode: SpecialKeyCodes.backSpace\n                })\n            );\n        }\n\n        if (\n            this.config.isFlagEnabled(Flags.SuppressBrowserKeys) &&\n            this.isKeyCodeBrowserKey(keyCode)\n        ) {\n            keyboardEvent.preventDefault();\n        }\n    }\n\n    /**\n     * handles when a key is up\n     * @param keyboardEvent - Keyboard event\n     */\n    handleOnKeyUp(keyboardEvent: KeyboardEvent) {\n        const keyCode = this.getKeycode(keyboardEvent);\n        if (!keyCode) {\n            return;\n        }\n\n        Logger.Log(Logger.GetStackTrace(), `key up ${keyCode}`, 6);\n        const toStreamerHandlers =\n            this.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('KeyUp')([ keyCode ]);\n\n        if (\n            this.config.isFlagEnabled(Flags.SuppressBrowserKeys) &&\n            this.isKeyCodeBrowserKey(keyCode)\n        ) {\n            keyboardEvent.preventDefault();\n        }\n    }\n\n    /**\n     * Handles when a key is press\n     * @param keyboard - Keyboard Event\n     */\n    handleOnKeyPress(keyboard: KeyboardEvent) {\n        if (!('charCode' in keyboard)) {\n            Logger.Warning(\n                Logger.GetStackTrace(),\n                'KeyboardEvent.charCode is deprecated in this browser, cannot send key press.'\n            );\n            return;\n        }\n\n        const charCode = keyboard.charCode;\n        Logger.Log(Logger.GetStackTrace(), `key press ${charCode}`, 6);\n\n        const toStreamerHandlers =\n            this.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('KeyPress')([charCode]);\n    }\n\n    /**\n     * Gets the Keycode of the Key pressed\n     * @param keyboardEvent - Key board Event\n     * @returns - the key code of the Key\n     */\n    getKeycode(keyboardEvent: KeyboardEvent) {\n        // If we don't have keyCode property because browser API is deprecated then use KeyboardEvent.code instead.\n        // See: https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/keyCode#constants_for_keycode_value\n        if (!('keyCode' in keyboardEvent)) {\n            // Convert KeyboardEvent.code string into integer-based key code for backwards compatibility reasons.\n            const event = keyboardEvent as KeyboardEvent;\n            if (event.code in this.CodeToKeyCode) {\n                return this.CodeToKeyCode[event.code];\n            } else {\n                Logger.Warning(\n                    Logger.GetStackTrace(),\n                    `Keyboard code of ${event.code} is not supported in our mapping, ignoring this key.`\n                );\n                return null;\n            }\n        }\n\n        // If we made it here KeyboardEvent.keyCode is still supported so we can safely use it.\n\n        if (\n            keyboardEvent.keyCode === SpecialKeyCodes.shift &&\n            keyboardEvent.code === 'ShiftRight'\n        ) {\n            return SpecialKeyCodes.rightShift;\n        } else if (\n            keyboardEvent.keyCode === SpecialKeyCodes.control &&\n            keyboardEvent.code === 'ControlRight'\n        ) {\n            return SpecialKeyCodes.rightControl;\n        } else if (\n            keyboardEvent.keyCode === SpecialKeyCodes.alt &&\n            keyboardEvent.code === 'AltRight'\n        ) {\n            return SpecialKeyCodes.rightAlt;\n        } else {\n            return keyboardEvent.keyCode;\n        }\n    }\n\n    /**\n     * Browser keys do not have a charCode so we only need to test keyCode.\n     * @param keyCode - the browser keycode number\n     */\n    isKeyCodeBrowserKey(keyCode: number) {\n        // Function keys or tab key are considered \"browser keys\" that we may wish to suppress by preventing them being process by browser.\n        return (keyCode >= 112 && keyCode <= 123) || keyCode === 9;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { MouseController } from './MouseController';\nimport { Logger } from '../Logger/Logger';\nimport { IMouseEvents } from './IMouseEvents';\nimport { NormalizedQuantizedUnsignedCoord } from '../Util/CoordinateConverter';\nimport { ActiveKeys } from './InputClassesFactory';\nimport { VideoPlayer } from '../VideoPlayer/VideoPlayer';\nimport { EventListenerTracker } from '../Util/EventListenerTracker';\n\n/**\n * Handle the mouse locked events\n */\nexport class LockedMouseEvents implements IMouseEvents {\n    x = 0;\n    y = 0;\n    coord: NormalizedQuantizedUnsignedCoord;\n    videoElementProvider: VideoPlayer;\n    mouseController: MouseController;\n    activeKeysProvider: ActiveKeys;\n    updateMouseMovePositionEvent = (mouseEvent: MouseEvent) => {\n        this.updateMouseMovePosition(mouseEvent);\n    };\n\n    // Utility for keeping track of event handlers and unregistering them\n    private mouseEventListenerTracker = new EventListenerTracker();\n\n    /**\n     * @param videoElementProvider - Video Player instance\n     * @param mouseController - Mouse controller instance\n     * @param activeKeysProvider - Active keys provider instance\n     * @param playerStyleAttributesProvider - Player style attributes instance\n     */\n    constructor(\n        videoElementProvider: VideoPlayer,\n        mouseController: MouseController,\n        activeKeysProvider: ActiveKeys\n    ) {\n        this.videoElementProvider = videoElementProvider;\n        this.mouseController = mouseController;\n        this.activeKeysProvider = activeKeysProvider;\n        const videoElementParent =\n            this.videoElementProvider.getVideoParentElement();\n        this.x = videoElementParent.getBoundingClientRect().width / 2;\n        this.y = videoElementParent.getBoundingClientRect().height / 2;\n        this.coord =\n            this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(\n                this.x,\n                this.y\n            );\n    }\n\n    /**\n     * Unregisters all event handlers\n     */\n    unregisterMouseEvents() {\n        this.mouseEventListenerTracker.unregisterAll();\n    }\n\n    /**\n     * Handle when the locked state Changed\n     */\n    lockStateChange() {\n        const videoElementParent =\n            this.videoElementProvider.getVideoParentElement();\n        const toStreamerHandlers =\n            this.mouseController.toStreamerMessagesProvider.toStreamerHandlers;\n\n        if (\n            document.pointerLockElement === videoElementParent ||\n            document.mozPointerLockElement === videoElementParent\n        ) {\n            Logger.Log(Logger.GetStackTrace(), 'Pointer locked', 6);\n            document.addEventListener(\n                'mousemove',\n                this.updateMouseMovePositionEvent,\n                false\n            );\n            this.mouseEventListenerTracker.addUnregisterCallback(\n                () => document.removeEventListener(\n                    'mousemove',\n                    this.updateMouseMovePositionEvent,\n                    false\n                )\n            );\n        } else {\n            Logger.Log(\n                Logger.GetStackTrace(),\n                'The pointer lock status is now unlocked',\n                6\n            );\n            // !a new arrow function must not be used here as it will be counted as a new function that cannot be removed\n            document.removeEventListener(\n                'mousemove',\n                this.updateMouseMovePositionEvent,\n                false\n            );\n\n            // If mouse loses focus, send a key up for all of the currently held-down keys\n            // This is necessary as when the mouse loses focus, the windows stops listening for events and as such\n            // the keyup listener won't get fired\n            let activeKeys = this.activeKeysProvider.getActiveKeys();\n            const setKeys = new Set(activeKeys);\n            const newKeysIterable: Array<number> = [];\n\n            setKeys.forEach((setKey: number) => {\n                newKeysIterable[setKey];\n            });\n\n            newKeysIterable.forEach((uniqueKeycode) => {\n                toStreamerHandlers.get('KeyUp')([uniqueKeycode]);\n            });\n            // Reset the active keys back to nothing\n            activeKeys = [];\n        }\n    }\n\n    /**\n     * Handle the mouse move event, sends the mouse data to the UE Instance\n     * @param mouseEvent - Mouse Event\n     */\n    updateMouseMovePosition(mouseEvent: MouseEvent) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        const toStreamerHandlers =\n            this.mouseController.toStreamerMessagesProvider.toStreamerHandlers;\n        const styleWidth =\n            this.videoElementProvider.getVideoParentElement().clientWidth;\n        const styleHeight =\n            this.videoElementProvider.getVideoParentElement().clientHeight;\n\n        this.x += mouseEvent.movementX;\n        this.y += mouseEvent.movementY;\n\n        if (this.x > styleWidth) {\n            this.x -= styleWidth;\n        }\n        if (this.y > styleHeight) {\n            this.y -= styleHeight;\n        }\n        if (this.x < 0) {\n            this.x = styleWidth + this.x;\n        }\n        if (this.y < 0) {\n            this.y = styleHeight - this.y;\n        }\n\n        this.coord =\n            this.mouseController.coordinateConverter.normalizeAndQuantizeUnsigned(\n                this.x,\n                this.y\n            );\n        const delta =\n            this.mouseController.coordinateConverter.normalizeAndQuantizeSigned(\n                mouseEvent.movementX,\n                mouseEvent.movementY\n            );\n        toStreamerHandlers.get('MouseMove')([\n            this.coord.x,\n            this.coord.y,\n            delta.x,\n            delta.y\n        ]);\n    }\n\n    /**\n     * Handle the mouse Down event, sends the mouse data to the UE Instance\n     * @param mouseEvent - Mouse Event\n     */\n    handleMouseDown(mouseEvent: MouseEvent) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n\n        const toStreamerHandlers =\n            this.mouseController.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseDown')([\n            mouseEvent.button,\n            // We use the store value of this.coord as opposed to the mouseEvent.x/y as the mouseEvent location\n            // uses the system cursor location which hasn't moved\n            this.coord.x,\n            this.coord.y\n        ]);\n    }\n\n    /**\n     * Handle the mouse Up event, sends the mouse data to the UE Instance\n     * @param mouseEvent - Mouse Event\n     */\n    handleMouseUp(mouseEvent: MouseEvent) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        const toStreamerHandlers =\n            this.mouseController.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseUp')([\n            mouseEvent.button,\n            // We use the store value of this.coord as opposed to the mouseEvent.x/y as the mouseEvent location\n            // uses the system cursor location which hasn't moved\n            this.coord.x,\n            this.coord.y\n        ]);\n    }\n\n    /**\n     * Handle the mouse wheel event, sends the mouse wheel data to the UE Instance\n     * @param wheelEvent - Mouse Event\n     */\n    handleMouseWheel(wheelEvent: WheelEvent) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        const toStreamerHandlers =\n            this.mouseController.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseWheel')([\n            wheelEvent.wheelDelta,\n            // We use the store value of this.coord as opposed to the mouseEvent.x/y as the mouseEvent location\n            // uses the system cursor location which hasn't moved\n            this.coord.x,\n            this.coord.y\n        ]);\n    }\n\n    /**\n     * Handle the mouse double click event, sends the mouse data to the UE Instance\n     * @param mouseEvent - Mouse Event\n     */\n    handleMouseDouble(mouseEvent: MouseEvent) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        const toStreamerHandlers =\n            this.mouseController.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseDouble')([\n            mouseEvent.button,\n            // We use the store value of this.coord as opposed to the mouseEvent.x/y as the mouseEvent location\n            // uses the system cursor location which hasn't moved\n            this.coord.x,\n            this.coord.y\n        ]);\n    }\n\n    /**\n     * Handle the press mouse buttons event, sends the mouse data to the UE Instance\n     * @param mouseEvent - Mouse Event\n     */\n    handlePressMouseButtons(mouseEvent: MouseEvent) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        this.mouseController.pressMouseButtons(\n            mouseEvent.buttons,\n            this.x,\n            this.y\n        );\n    }\n\n    /**\n     * Handle the release mouse buttons event, sends the mouse data to the UE Instance\n     * @param mouseEvent - Mouse Event\n     */\n    handleReleaseMouseButtons(mouseEvent: MouseEvent) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        this.mouseController.releaseMouseButtons(\n            mouseEvent.buttons,\n            this.x,\n            this.y\n        );\n    }\n}\n\n/**\n * Extra types for Document and WheelEvent\n */\ndeclare global {\n    interface Document {\n        mozPointerLockElement: unknown;\n        mozExitPointerLock?(): void;\n    }\n\n    interface WheelEvent {\n        wheelDelta: number;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Mouse Button Data\n * {@link https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/button}\n */\nexport class MouseButton {\n    static mainButton = 0; // Left button.\n    static auxiliaryButton = 1; // Wheel button.\n    static secondaryButton = 2; // Right button.\n    static fourthButton = 3; // Browser Back button.\n    static fifthButton = 4; // Browser Forward button.\n}\n\n/**\n * Mouse Button Mask Data\n * {@link https://developer.mozilla.org/en-US/docs/Web/API/MouseEvent/buttons}\n */\nexport class MouseButtonsMask {\n    static primaryButton = 1; // Left button.\n    static secondaryButton = 2; // Right button.\n    static auxiliaryButton = 4; // Wheel button.\n    static fourthButton = 8; // Browser Back button.\n    static fifthButton = 16; // Browser Forward button.\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { MouseButtonsMask, MouseButton } from './MouseButtons';\nimport { Logger } from '../Logger/Logger';\nimport { StreamMessageController } from '../UeInstanceMessage/StreamMessageController';\nimport { CoordinateConverter } from '../Util/CoordinateConverter';\nimport { VideoPlayer } from '../VideoPlayer/VideoPlayer';\nimport { IMouseEvents } from './IMouseEvents';\nimport { LockedMouseEvents } from './LockedMouseEvents';\nimport { HoveringMouseEvents } from './HoveringMouseEvents';\nimport type { ActiveKeys } from './InputClassesFactory';\nimport { EventListenerTracker } from '../Util/EventListenerTracker';\n\n/**\n * Handles the Mouse Inputs for the document\n */\nexport class MouseController {\n    videoElementProvider: VideoPlayer;\n    toStreamerMessagesProvider: StreamMessageController;\n    coordinateConverter: CoordinateConverter;\n    activeKeysProvider: ActiveKeys;\n\n    // Utility for keeping track of event handlers and unregistering them\n    private mouseEventListenerTracker = new EventListenerTracker();\n\n    /**\n     * @param toStreamerMessagesProvider - Stream message instance\n     * @param videoElementProvider - Video Player instance\n     * @param normalizeAndQuantize - A normalize and quantize instance\n     */\n    constructor(\n        toStreamerMessagesProvider: StreamMessageController,\n        videoElementProvider: VideoPlayer,\n        coordinateConverter: CoordinateConverter,\n        activeKeysProvider: ActiveKeys\n    ) {\n        this.toStreamerMessagesProvider = toStreamerMessagesProvider;\n        this.coordinateConverter = coordinateConverter;\n        this.videoElementProvider = videoElementProvider;\n        this.activeKeysProvider = activeKeysProvider;\n        this.registerMouseEnterAndLeaveEvents();\n    }\n\n    /**\n     * Clears all the click events on the current video element parent div\n     */\n    unregisterMouseEvents() {\n        this.mouseEventListenerTracker.unregisterAll();\n    }\n\n    /**\n     * Register a locked mouse class\n     * @param mouseController - a mouse controller instance\n     * @param playerStyleAttributesProvider - a player style attributes instance\n     */\n    registerLockedMouseEvents(mouseController: MouseController) {\n        const videoElementParent =\n            this.videoElementProvider.getVideoParentElement() as HTMLDivElement;\n        const lockedMouseEvents: IMouseEvents = new LockedMouseEvents(\n            this.videoElementProvider,\n            mouseController,\n            this.activeKeysProvider\n        );\n\n        videoElementParent.requestPointerLock =\n            videoElementParent.requestPointerLock ||\n            videoElementParent.mozRequestPointerLock;\n        document.exitPointerLock =\n            document.exitPointerLock || document.mozExitPointerLock;\n\n        // minor hack to alleviate ios not supporting pointerlock\n        if (videoElementParent.requestPointerLock) {\n            const onclick = () => {\n                videoElementParent.requestPointerLock();\n            };\n            videoElementParent.addEventListener('click', onclick);\n            this.mouseEventListenerTracker.addUnregisterCallback(\n                () => videoElementParent.removeEventListener('click', onclick)\n            );\n        }\n\n        const lockStateChangeListener = () =>\n            lockedMouseEvents.lockStateChange();\n        document.addEventListener(\n            'pointerlockchange',\n            lockStateChangeListener,\n            false\n        );\n        document.addEventListener(\n            'mozpointerlockchange',\n            lockStateChangeListener,\n            false\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => document.removeEventListener(\n                'pointerlockchange',\n                lockStateChangeListener,\n                false\n            )\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => document.removeEventListener(\n                'mozpointerlockchange',\n                lockStateChangeListener,\n                false\n            )\n        );\n\n        const onmousedown = (mouseEvent: MouseEvent) =>\n            lockedMouseEvents.handleMouseDown(mouseEvent);\n        const onmouseup = (mouseEvent: MouseEvent) =>\n            lockedMouseEvents.handleMouseUp(mouseEvent);\n        const onwheel = (wheelEvent: WheelEvent) =>\n            lockedMouseEvents.handleMouseWheel(wheelEvent);\n        const ondblclick = (mouseEvent: MouseEvent) =>\n            lockedMouseEvents.handleMouseDouble(mouseEvent);\n        videoElementParent.addEventListener('mousedown', onmousedown);\n        videoElementParent.addEventListener('mouseup', onmouseup);\n        videoElementParent.addEventListener('wheel', onwheel);\n        videoElementParent.addEventListener('dblclick', ondblclick);\n\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => videoElementParent.removeEventListener('mousedown', onmousedown)\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => videoElementParent.removeEventListener('mouseup', onmouseup)\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => videoElementParent.removeEventListener('wheel', onwheel)\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => videoElementParent.removeEventListener('dblclick', ondblclick)\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => lockedMouseEvents.unregisterMouseEvents()\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(() => {\n            if (\n                document.exitPointerLock &&\n                (document.pointerLockElement === videoElementParent ||\n                    document.mozPointerLockElement === videoElementParent)\n            ) {\n                document.exitPointerLock();\n            }\n        });\n    }\n\n    /**\n     * Register a hovering mouse class\n     * @param mouseController - A mouse controller object\n     */\n    registerHoveringMouseEvents(mouseController: MouseController) {\n        const videoElementParent =\n            this.videoElementProvider.getVideoParentElement() as HTMLDivElement;\n        const hoveringMouseEvents = new HoveringMouseEvents(mouseController);\n\n        const onmousemove = (mouseEvent: MouseEvent) =>\n            hoveringMouseEvents.updateMouseMovePosition(mouseEvent);\n        const onmousedown = (mouseEvent: MouseEvent) =>\n            hoveringMouseEvents.handleMouseDown(mouseEvent);\n        const onmouseup = (mouseEvent: MouseEvent) =>\n            hoveringMouseEvents.handleMouseUp(mouseEvent);\n        const oncontextmenu = (mouseEvent: MouseEvent) =>\n            hoveringMouseEvents.handleContextMenu(mouseEvent);\n        const onwheel = (wheelEvent: WheelEvent) =>\n            hoveringMouseEvents.handleMouseWheel(wheelEvent);\n        const ondblclick = (mouseEvent: MouseEvent) =>\n            hoveringMouseEvents.handleMouseDouble(mouseEvent);\n        videoElementParent.addEventListener('mousemove', onmousemove);\n        videoElementParent.addEventListener('mousedown', onmousedown);\n        videoElementParent.addEventListener('mouseup', onmouseup);\n        videoElementParent.addEventListener('contextmenu', oncontextmenu);\n        videoElementParent.addEventListener('wheel', onwheel);\n        videoElementParent.addEventListener('dblclick', ondblclick);\n\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => videoElementParent.removeEventListener('mousemove', onmousemove)\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => videoElementParent.removeEventListener('mousedown', onmousedown)\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => videoElementParent.removeEventListener('mouseup', onmouseup)\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => videoElementParent.removeEventListener('contextmenu', oncontextmenu)\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => videoElementParent.removeEventListener('wheel', onwheel)\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => videoElementParent.removeEventListener('dblclick', ondblclick)\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => hoveringMouseEvents.unregisterMouseEvents()\n        );\n    }\n\n    /**\n     * Set the mouse enter and mouse leave events\n     */\n    registerMouseEnterAndLeaveEvents() {\n        const videoElementParent =\n            this.videoElementProvider.getVideoParentElement() as HTMLDivElement;\n\n        // Handle when the Mouse has entered the element\n        const onmouseenter = (event: MouseEvent) => {\n            if (!this.videoElementProvider.isVideoReady()) {\n                return;\n            }\n            Logger.Log(Logger.GetStackTrace(), 'Mouse Entered', 6);\n            this.sendMouseEnter();\n            this.pressMouseButtons(event.buttons, event.x, event.y);\n        };\n\n        // Handles when the mouse has left the element\n        const onmouseleave = (event: MouseEvent) => {\n            if (!this.videoElementProvider.isVideoReady()) {\n                return;\n            }\n            Logger.Log(Logger.GetStackTrace(), 'Mouse Left', 6);\n            this.sendMouseLeave();\n            this.releaseMouseButtons(event.buttons, event.x, event.y);\n        };\n        videoElementParent.addEventListener('mouseenter', onmouseenter);\n        videoElementParent.addEventListener('mouseleave', onmouseleave);\n\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => videoElementParent.removeEventListener('mouseenter', onmouseenter)\n        );\n        this.mouseEventListenerTracker.addUnregisterCallback(\n            () => videoElementParent.removeEventListener('mouseleave', onmouseleave)\n        );\n    }\n\n    /**\n     * Handle when a mouse button is released\n     * @param buttons - Mouse Button\n     * @param X - Mouse pointer X coordinate\n     * @param Y - Mouse pointer Y coordinate\n     */\n    releaseMouseButtons(buttons: number, X: number, Y: number) {\n        const coord = this.coordinateConverter.normalizeAndQuantizeUnsigned(\n            X,\n            Y\n        );\n        if (buttons & MouseButtonsMask.primaryButton) {\n            this.sendMouseUp(MouseButton.mainButton, coord.x, coord.y);\n        }\n        if (buttons & MouseButtonsMask.secondaryButton) {\n            this.sendMouseUp(MouseButton.secondaryButton, coord.x, coord.y);\n        }\n        if (buttons & MouseButtonsMask.auxiliaryButton) {\n            this.sendMouseUp(MouseButton.auxiliaryButton, coord.x, coord.y);\n        }\n        if (buttons & MouseButtonsMask.fourthButton) {\n            this.sendMouseUp(MouseButton.fourthButton, coord.x, coord.y);\n        }\n        if (buttons & MouseButtonsMask.fifthButton) {\n            this.sendMouseUp(MouseButton.fifthButton, coord.x, coord.y);\n        }\n    }\n\n    /**\n     * Handle when a mouse button is pressed\n     * @param buttons - Mouse Button\n     * @param X - Mouse pointer X coordinate\n     * @param Y - Mouse pointer Y coordinate\n     */\n    pressMouseButtons(buttons: number, X: number, Y: number) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        const coord = this.coordinateConverter.normalizeAndQuantizeUnsigned(\n            X,\n            Y\n        );\n        if (buttons & MouseButtonsMask.primaryButton) {\n            this.sendMouseDown(MouseButton.mainButton, coord.x, coord.y);\n        }\n        if (buttons & MouseButtonsMask.secondaryButton) {\n            this.sendMouseDown(MouseButton.secondaryButton, coord.x, coord.y);\n        }\n        if (buttons & MouseButtonsMask.auxiliaryButton) {\n            this.sendMouseDown(MouseButton.auxiliaryButton, coord.x, coord.y);\n        }\n        if (buttons & MouseButtonsMask.fourthButton) {\n            this.sendMouseDown(MouseButton.fourthButton, coord.x, coord.y);\n        }\n        if (buttons & MouseButtonsMask.fifthButton) {\n            this.sendMouseDown(MouseButton.fifthButton, coord.x, coord.y);\n        }\n    }\n\n    /**\n     * Handles mouse enter\n     */\n    sendMouseEnter() {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        const toStreamerHandlers =\n            this.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseEnter')();\n    }\n\n    /**\n     * Handles mouse Leave\n     */\n    sendMouseLeave() {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        const toStreamerHandlers =\n            this.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseLeave')();\n    }\n\n    /**\n     * Handles when a mouse button is pressed down\n     * @param button - Mouse Button Pressed\n     * @param X  - Mouse X Coordinate\n     * @param Y  - Mouse Y Coordinate\n     */\n    sendMouseDown(button: number, X: number, Y: number) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        Logger.Log(\n            Logger.GetStackTrace(),\n            `mouse button ${button} down at (${X}, ${Y})`,\n            6\n        );\n        const toStreamerHandlers =\n            this.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseDown')([button, X, Y]);\n    }\n\n    /**\n     * Handles when a mouse button is pressed up\n     * @param button - Mouse Button Pressed\n     * @param X  - Mouse X Coordinate\n     * @param Y  - Mouse Y Coordinate\n     */\n    sendMouseUp(button: number, X: number, Y: number) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        Logger.Log(\n            Logger.GetStackTrace(),\n            `mouse button ${button} up at (${X}, ${Y})`,\n            6\n        );\n        const coord = this.coordinateConverter.normalizeAndQuantizeUnsigned(\n            X,\n            Y\n        );\n        const toStreamerHandlers =\n            this.toStreamerMessagesProvider.toStreamerHandlers;\n        toStreamerHandlers.get('MouseUp')([button, coord.x, coord.y]);\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Registers the Special Key codes\n *  Must be kept in sync with JavaScriptKeyCodeToFKey C++ array.\n * The index of the entry in the array is the special key code given below.\n */\nexport class SpecialKeyCodes {\n    static backSpace = 8;\n    static shift = 16;\n    static control = 17;\n    static alt = 18;\n    static rightShift = 253;\n    static rightControl = 254;\n    static rightAlt = 255;\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\nimport { CoordinateConverter } from '../Util/CoordinateConverter';\nimport { StreamMessageController } from '../UeInstanceMessage/StreamMessageController';\nimport { VideoPlayer } from '../VideoPlayer/VideoPlayer';\nimport { ITouchController } from './ITouchController';\nimport { EventListenerTracker } from '../Util/EventListenerTracker';\n/**\n * Handles the Touch input Events\n */\nexport class TouchController implements ITouchController {\n    toStreamerMessagesProvider: StreamMessageController;\n    videoElementProvider: VideoPlayer;\n    coordinateConverter: CoordinateConverter;\n    videoElementParent: HTMLVideoElement;\n    fingers = [9, 8, 7, 6, 5, 4, 3, 2, 1, 0];\n    fingerIds = new Map();\n    maxByteValue = 255;\n\n    // Utility for keeping track of event handlers and unregistering them\n    private touchEventListenerTracker = new EventListenerTracker();\n\n    /**\n     * @param toStreamerMessagesProvider - Stream message instance\n     * @param videoElementProvider - Video Player instance\n     * @param coordinateConverter - A coordinate converter instance\n     */\n    constructor(\n        toStreamerMessagesProvider: StreamMessageController,\n        videoElementProvider: VideoPlayer,\n        coordinateConverter: CoordinateConverter\n    ) {\n        this.toStreamerMessagesProvider = toStreamerMessagesProvider;\n        this.videoElementProvider = videoElementProvider;\n        this.coordinateConverter = coordinateConverter;\n        this.videoElementParent = videoElementProvider.getVideoElement();\n        const ontouchstart = (ev: TouchEvent) =>\n            this.onTouchStart(ev);\n        const ontouchend = (ev: TouchEvent) =>\n            this.onTouchEnd(ev);\n        const ontouchmove = (ev: TouchEvent) =>\n            this.onTouchMove(ev);\n        this.videoElementParent.addEventListener('touchstart', ontouchstart);\n        this.videoElementParent.addEventListener('touchend', ontouchend);\n        this.videoElementParent.addEventListener('touchmove', ontouchmove);\n        this.touchEventListenerTracker.addUnregisterCallback(\n            () => this.videoElementParent.removeEventListener('touchstart', ontouchstart)\n        );\n        this.touchEventListenerTracker.addUnregisterCallback(\n            () => this.videoElementParent.removeEventListener('touchend', ontouchend)\n        );\n        this.touchEventListenerTracker.addUnregisterCallback(\n            () => this.videoElementParent.removeEventListener('touchmove', ontouchmove)\n        );\n        Logger.Log(Logger.GetStackTrace(), 'Touch Events Registered', 6);\n\n        // is this strictly necessary?\n        const preventOnTouchMove = (event: TouchEvent) => {\n            event.preventDefault();\n        };\n        document.addEventListener('touchmove', preventOnTouchMove);\n        this.touchEventListenerTracker.addUnregisterCallback(\n            () => document.removeEventListener('touchmove', preventOnTouchMove)\n        );\n    }\n\n    /**\n     * Unregister all touch events\n     */\n    unregisterTouchEvents() {\n        this.touchEventListenerTracker.unregisterAll();\n    }\n\n    /**\n     * Remember a touch command\n     * @param touch - the touch command\n     */\n    rememberTouch(touch: Touch) {\n        const finger = this.fingers.pop();\n        if (finger === undefined) {\n            Logger.Log(\n                Logger.GetStackTrace(),\n                'exhausted touch identifiers',\n                6\n            );\n        }\n        this.fingerIds.set(touch.identifier, finger);\n    }\n\n    /**\n     * Forgets a touch command\n     * @param touch - the touch command\n     */\n    forgetTouch(touch: Touch) {\n        this.fingers.push(this.fingerIds.get(touch.identifier));\n        // Sort array back into descending order. This means if finger '1' were to lift after finger '0', we would ensure that 0 will be the first index to pop\n        this.fingers.sort(function (a, b) {\n            return b - a;\n        });\n        this.fingerIds.delete(touch.identifier);\n    }\n\n    /**\n     * When a touch event starts\n     * @param touchEvent - the touch event being intercepted\n     */\n    onTouchStart(touchEvent: TouchEvent) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        for (let t = 0; t < touchEvent.changedTouches.length; t++) {\n            this.rememberTouch(touchEvent.changedTouches[t]);\n        }\n        Logger.Log(Logger.GetStackTrace(), 'touch start', 6);\n\n        this.emitTouchData('TouchStart', touchEvent.changedTouches);\n        touchEvent.preventDefault();\n    }\n\n    /**\n     * When a touch event ends\n     * @param touchEvent - the touch event being intercepted\n     */\n    onTouchEnd(touchEvent: TouchEvent) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        Logger.Log(Logger.GetStackTrace(), 'touch end', 6);\n        this.emitTouchData('TouchEnd', touchEvent.changedTouches);\n        // Re-cycle unique identifiers previously assigned to each touch.\n        for (let t = 0; t < touchEvent.changedTouches.length; t++) {\n            this.forgetTouch(touchEvent.changedTouches[t]);\n        }\n        touchEvent.preventDefault();\n    }\n\n    /**\n     * when a moving touch event occurs\n     * @param touchEvent - the touch event being intercepted\n     */\n    onTouchMove(touchEvent: TouchEvent) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        Logger.Log(Logger.GetStackTrace(), 'touch move', 6);\n        this.emitTouchData('TouchMove', touchEvent.touches);\n        touchEvent.preventDefault();\n    }\n\n    emitTouchData(type: string, touches: TouchList) {\n        if (!this.videoElementProvider.isVideoReady()) {\n            return;\n        }\n        const offset = this.videoElementProvider.getVideoParentElement().getBoundingClientRect();\n        const toStreamerHandlers =\n            this.toStreamerMessagesProvider.toStreamerHandlers;\n\n        for (let t = 0; t < touches.length; t++) {\n            const numTouches = 1; // the number of touches to be sent this message\n            const touch = touches[t];\n            const x = touch.clientX - offset.left;\n            const y = touch.clientY - offset.top;\n            Logger.Log(\n                Logger.GetStackTrace(),\n                `F${this.fingerIds.get(touch.identifier)}=(${x}, ${y})`,\n                6\n            );\n\n            const coord = this.coordinateConverter.normalizeAndQuantizeUnsigned(\n                x,\n                y\n            );\n            switch (type) {\n                case 'TouchStart':\n                    toStreamerHandlers.get('TouchStart')([\n                        numTouches,\n                        coord.x,\n                        coord.y,\n                        this.fingerIds.get(touch.identifier),\n                        this.maxByteValue * touch.force,\n                        coord.inRange ? 1 : 0\n                    ]);\n                    break;\n                case 'TouchEnd':\n                    toStreamerHandlers.get('TouchEnd')([\n                        numTouches,\n                        coord.x,\n                        coord.y,\n                        this.fingerIds.get(touch.identifier),\n                        this.maxByteValue * touch.force,\n                        coord.inRange ? 1 : 0\n                    ]);\n                    break;\n                case 'TouchMove':\n                    toStreamerHandlers.get('TouchMove')([\n                        numTouches,\n                        coord.x,\n                        coord.y,\n                        this.fingerIds.get(touch.identifier),\n                        this.maxByteValue * touch.force,\n                        coord.inRange ? 1 : 0\n                    ]);\n                    break;\n            }\n        }\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { StreamMessageController } from '../UeInstanceMessage/StreamMessageController';\nimport { Controller } from './GamepadTypes';\nimport { WebXRUtils } from '../Util/WebXRUtils';\n\n/**\n * The class that handles the functionality of xrgamepads and controllers\n */\nexport class XRGamepadController {\n    controllers: Array<Controller>;\n    toStreamerMessagesProvider: StreamMessageController;\n\n    /**\n     * @param toStreamerMessagesProvider - Stream message instance\n     */\n    constructor(toStreamerMessagesProvider: StreamMessageController) {\n        this.toStreamerMessagesProvider = toStreamerMessagesProvider;\n        this.controllers = [];\n    }\n\n    updateStatus(\n        source: XRInputSource,\n        frame: XRFrame,\n        refSpace: XRReferenceSpace\n    ) {\n        if (source.gamepad) {\n            const gamepadPose = frame.getPose(source.gripSpace, refSpace);\n            if (!gamepadPose) {\n                return;\n            }\n\n            let system = 0;\n            if (source.profiles.includes('htc-vive')) {\n                system = 1;\n            } else if (source.profiles.includes('oculus-touch')) {\n                system = 2;\n            }\n            // TODO (william.belcher): Add other profiles (Quest, Microsoft Mixed Reality, etc)\n            this.toStreamerMessagesProvider.toStreamerHandlers.get('XRSystem')([\n                system\n            ]);\n\n            // Default: AnyHand (2)\n            let handedness = 2;\n            switch (source.handedness) {\n                case 'left':\n                    handedness = 0;\n                    break;\n                case 'right':\n                    handedness = 1;\n                    break;\n            }\n\n            // Send controller transform\n            const matrix = gamepadPose.transform.matrix;\n            const mat = [];\n            for (let i = 0; i < 16; i++) {\n                mat[i] = new Float32Array([matrix[i]])[0];\n            }\n\n            // prettier-ignore\n            this.toStreamerMessagesProvider.toStreamerHandlers.get('XRControllerTransform')([\n                mat[0], mat[4], mat[8], mat[12],\n                mat[1], mat[5], mat[9], mat[13],\n                mat[2], mat[6], mat[10], mat[14],\n                mat[3], mat[7], mat[11], mat[15],\n                handedness\n            ]);\n\n            // Handle controller buttons and axes\n            if (this.controllers[handedness] === undefined) {\n                this.controllers[handedness] = {\n                    prevState: undefined,\n                    currentState: undefined,\n\t\t\t\t\tid: undefined\n                };\n                this.controllers[handedness].prevState =\n                    WebXRUtils.deepCopyGamepad(source.gamepad);\n            }\n\n            this.controllers[handedness].currentState =\n                WebXRUtils.deepCopyGamepad(source.gamepad);\n\n            const controller = this.controllers[handedness];\n            const currState = controller.currentState;\n            const prevState = controller.prevState;\n            // Iterate over buttons\n            for (let i = 0; i < currState.buttons.length; i++) {\n                const currButton = currState.buttons[i];\n                const prevButton = prevState.buttons[i];\n\n                if (currButton.pressed) {\n                    // press\n                    this.toStreamerMessagesProvider.toStreamerHandlers.get(\n                        'XRButtonPressed'\n                    )([handedness, i, prevButton.pressed ? 1 : 0]);\n                } else if (!currButton.pressed && prevButton.pressed) {\n                    this.toStreamerMessagesProvider.toStreamerHandlers.get(\n                        'XRButtonReleased'\n                    )([handedness, i, 0]);\n                }\n\n                if (currButton.touched && !currButton.pressed) {\n                    // press\n                    this.toStreamerMessagesProvider.toStreamerHandlers.get(\n                        'XRButtonPressed'\n                    )([handedness, 3, prevButton.touched ? 1 : 0]);\n                } else if (!currButton.touched && prevButton.touched) {\n                    this.toStreamerMessagesProvider.toStreamerHandlers.get(\n                        'XRButtonReleased'\n                    )([handedness, 3, 0]);\n                }\n            }\n\n            // Iterate over gamepad axes\n            for (let i = 0; i < currState.axes.length; i++) {\n                this.toStreamerMessagesProvider.toStreamerHandlers.get(\n                    'XRAnalog'\n                )([handedness, i, currState.axes[i]]);\n            }\n\n            this.controllers[handedness].prevState = currState;\n        }\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nexport class Logger {\n    static verboseLogLevel = 5;\n\n    /**\n     * Captures the stack and returns it\n     * @returns the current stack\n     */\n    static GetStackTrace() {\n        const error = new Error();\n        let formattedStack = 'No Stack Available for this browser';\n\n        // format the error\n        if (error.stack) {\n            formattedStack = error.stack.toString().replace(/Error/g, '');\n        }\n\n        return formattedStack;\n    }\n\n    /**\n     * Set the log verbosity level\n     */\n    static SetLoggerVerbosity(verboseLogLevel: number) {\n        if (this.verboseLogLevel != null) {\n            this.verboseLogLevel = verboseLogLevel;\n        }\n    }\n\n    /**\n     * The standard logging output\n     * @param stack - the stack trace\n     * @param message - the message to be logged\n     * @param verbosity - the verbosity level\n     */\n    static Log(stack: string, message: string, verbosity?: number) {\n        if (verbosity > this.verboseLogLevel) {\n            return;\n        }\n\n        const returnString = `Level: Log\\nMsg: ${message}\\nCaller: ${stack}`;\n        console.log(returnString);\n    }\n\n    /**\n     * The standard logging output\n     * @param stack - the stack trace\n     * @param message - the message to be logged\n     * @param verbosity - the verbosity level\n     */\n    static Info(stack: string, message: string, verbosity?: number) {\n        if (verbosity > this.verboseLogLevel) {\n            return;\n        }\n\n        const returnString = `Level: Info\\nMsg: ${message}`;\n        console.info(returnString);\n    }\n\n    /**\n     * The standard logging output\n     * @param stack - the stack trace\n     * @param message - the message to be logged\n     */\n    static Error(stack: string, message: string) {\n        const returnString = `Level: Error\\nMsg: ${message}\\nCaller: ${stack}`;\n        console.error(returnString);\n    }\n\n    /**\n     * The standard logging output\n     * @param stack - the stack trace\n     * @param message - the message to be logged\n     */\n    static Warning(stack: string, message: string) {\n        const returnString = `Level: Warning\\nCaller: ${stack}\\nMsg: ${message}`;\n        console.warn(returnString);\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport {\n    InboundRTPStats,\n    InboundVideoStats,\n    InboundAudioStats\n} from './InboundRTPStats';\nimport { InboundTrackStats } from './InboundTrackStats';\nimport { DataChannelStats } from './DataChannelStats';\nimport { CandidateStat } from './CandidateStat';\nimport { CandidatePairStats } from './CandidatePairStats';\nimport { OutBoundRTPStats, OutBoundVideoStats } from './OutBoundRTPStats';\nimport { SessionStats } from './SessionStats';\nimport { StreamStats } from './StreamStats';\nimport { CodecStats } from './CodecStats';\nimport { Logger } from '../Logger/Logger';\n\n/**\n * The Aggregated Stats that is generated from the RTC Stats Report\n */\n\ntype RTCStatsTypePS = RTCStatsType | 'stream' | 'media-playout';\nexport class AggregatedStats {\n    inboundVideoStats: InboundVideoStats;\n    inboundAudioStats: InboundAudioStats;\n    lastVideoStats: InboundVideoStats;\n    lastAudioStats: InboundAudioStats;\n    candidatePair: CandidatePairStats;\n    DataChannelStats: DataChannelStats;\n    localCandidates: Array<CandidateStat>;\n    remoteCandidates: Array<CandidateStat>;\n    outBoundVideoStats: OutBoundVideoStats;\n    sessionStats: SessionStats;\n    streamStats: StreamStats;\n    codecs: Map<string, string>;\n\n    constructor() {\n        this.inboundVideoStats = new InboundVideoStats();\n        this.inboundAudioStats = new InboundAudioStats();\n        this.candidatePair = new CandidatePairStats();\n        this.DataChannelStats = new DataChannelStats();\n        this.outBoundVideoStats = new OutBoundVideoStats();\n        this.sessionStats = new SessionStats();\n        this.streamStats = new StreamStats();\n        this.codecs = new Map<string, string>();\n    }\n\n    /**\n     * Gather all the information from the RTC Peer Connection Report\n     * @param rtcStatsReport - RTC Stats Report\n     */\n    processStats(rtcStatsReport: RTCStatsReport) {\n        this.localCandidates = new Array<CandidateStat>();\n        this.remoteCandidates = new Array<CandidateStat>();\n\n        rtcStatsReport.forEach((stat) => {\n            const type: RTCStatsTypePS = stat.type;\n\n            switch (type) {\n                case 'candidate-pair':\n                    this.handleCandidatePair(stat);\n                    break;\n                case 'certificate':\n                    break;\n                case 'codec':\n                    this.handleCodec(stat);\n                    break;\n                case 'data-channel':\n                    this.handleDataChannel(stat);\n                    break;\n                case 'inbound-rtp':\n                    this.handleInBoundRTP(stat);\n                    break;\n                case 'local-candidate':\n                    this.handleLocalCandidate(stat);\n                    break;\n                case 'media-source':\n                    break;\n                case 'media-playout':\n                    break;\n                case 'outbound-rtp':\n                    break;\n                case 'peer-connection':\n                    break;\n                case 'remote-candidate':\n                    this.handleRemoteCandidate(stat);\n                    break;\n                case 'remote-inbound-rtp':\n                    break;\n                case 'remote-outbound-rtp':\n                    this.handleRemoteOutBound(stat);\n                    break;\n                case 'track':\n                    this.handleTrack(stat);\n                    break;\n                case 'transport':\n                    break;\n                case 'stream':\n                    this.handleStream(stat);\n                    break;\n                default:\n                    Logger.Error(Logger.GetStackTrace(), 'unhandled Stat Type');\n                    Logger.Log(Logger.GetStackTrace(), stat);\n                    break;\n            }\n        });\n    }\n\n    /**\n     * Process stream stats data from webrtc\n     *\n     * @param stat - the stats coming in from webrtc\n     */\n    handleStream(stat: StreamStats) {\n        this.streamStats = stat;\n    }\n\n    /**\n     * Process the Ice Candidate Pair Data\n     * @param stat - the stats coming in from ice candidates\n     */\n    handleCandidatePair(stat: CandidatePairStats) {\n        this.candidatePair.bytesReceived = stat.bytesReceived;\n        this.candidatePair.bytesSent = stat.bytesSent;\n        this.candidatePair.localCandidateId = stat.localCandidateId;\n        this.candidatePair.remoteCandidateId = stat.remoteCandidateId;\n        this.candidatePair.nominated = stat.nominated;\n        this.candidatePair.readable = stat.readable;\n        this.candidatePair.selected = stat.selected;\n        this.candidatePair.writable = stat.writable;\n        this.candidatePair.state = stat.state;\n        this.candidatePair.currentRoundTripTime = stat.currentRoundTripTime;\n    }\n\n    /**\n     * Process the Data Channel Data\n     * @param stat - the stats coming in from the data channel\n     */\n    handleDataChannel(stat: DataChannelStats) {\n        this.DataChannelStats.bytesReceived = stat.bytesReceived;\n        this.DataChannelStats.bytesSent = stat.bytesSent;\n        this.DataChannelStats.dataChannelIdentifier =\n            stat.dataChannelIdentifier;\n        this.DataChannelStats.id = stat.id;\n        this.DataChannelStats.label = stat.label;\n        this.DataChannelStats.messagesReceived = stat.messagesReceived;\n        this.DataChannelStats.messagesSent = stat.messagesSent;\n        this.DataChannelStats.protocol = stat.protocol;\n        this.DataChannelStats.state = stat.state;\n        this.DataChannelStats.timestamp = stat.timestamp;\n    }\n\n    /**\n     * Process the Local Ice Candidate Data\n     * @param stat - local stats\n     */\n    handleLocalCandidate(stat: CandidateStat) {\n        const localCandidate = new CandidateStat();\n        localCandidate.label = 'local-candidate';\n        localCandidate.address = stat.address;\n        localCandidate.port = stat.port;\n        localCandidate.protocol = stat.protocol;\n        localCandidate.candidateType = stat.candidateType;\n        localCandidate.id = stat.id;\n        this.localCandidates.push(localCandidate);\n    }\n\n    /**\n     * Process the Remote Ice Candidate Data\n     * @param stat - ice candidate stats\n     */\n    handleRemoteCandidate(stat: CandidateStat) {\n        const RemoteCandidate = new CandidateStat();\n        RemoteCandidate.label = 'local-candidate';\n        RemoteCandidate.address = stat.address;\n        RemoteCandidate.port = stat.port;\n        RemoteCandidate.protocol = stat.protocol;\n        RemoteCandidate.id = stat.id;\n        RemoteCandidate.candidateType = stat.candidateType;\n        this.remoteCandidates.push(RemoteCandidate);\n    }\n\n    /**\n     * Process the Inbound RTP Audio and Video Data\n     * @param stat - inbound rtp stats\n     */\n    handleInBoundRTP(stat: InboundRTPStats) {\n        switch (stat.kind) {\n            case 'video':\n                // Need to convert to unknown first to remove an error around\n                // InboundVideoStats having the bitrate member which isn't found on\n                // the InboundRTPStats\n                this.inboundVideoStats = stat as unknown as InboundVideoStats;\n\n                if (this.lastVideoStats != undefined) {\n                    this.inboundVideoStats.bitrate =\n                        (8 *\n                            (this.inboundVideoStats.bytesReceived -\n                                this.lastVideoStats.bytesReceived)) /\n                        (this.inboundVideoStats.timestamp -\n                            this.lastVideoStats.timestamp);\n                    this.inboundVideoStats.bitrate = Math.floor(\n                        this.inboundVideoStats.bitrate\n                    );\n                }\n                this.lastVideoStats = { ...this.inboundVideoStats };\n                break;\n            case 'audio':\n                // Need to convert to unknown first to remove an error around\n                // InboundAudioStats having the bitrate member which isn't found on\n                // the InboundRTPStats\n                this.inboundAudioStats = stat as unknown as InboundAudioStats;\n\n                if (this.lastAudioStats != undefined) {\n                    this.inboundAudioStats.bitrate =\n                        (8 *\n                            (this.inboundAudioStats.bytesReceived -\n                                this.lastAudioStats.bytesReceived)) /\n                        (this.inboundAudioStats.timestamp -\n                            this.lastAudioStats.timestamp);\n                    this.inboundAudioStats.bitrate = Math.floor(\n                        this.inboundAudioStats.bitrate\n                    );\n                }\n                this.lastAudioStats = { ...this.inboundAudioStats };\n                break;\n            default:\n                Logger.Log(Logger.GetStackTrace(), 'Kind is not handled');\n                break;\n        }\n    }\n\n    /**\n     * Process the outbound RTP Audio and Video Data\n     * @param stat - remote outbound stats\n     */\n    handleRemoteOutBound(stat: OutBoundRTPStats) {\n        switch (stat.kind) {\n            case 'video':\n                this.outBoundVideoStats.bytesSent = stat.bytesSent;\n                this.outBoundVideoStats.id = stat.id;\n                this.outBoundVideoStats.localId = stat.localId;\n                this.outBoundVideoStats.packetsSent = stat.packetsSent;\n                this.outBoundVideoStats.remoteTimestamp = stat.remoteTimestamp;\n                this.outBoundVideoStats.timestamp = stat.timestamp;\n                break;\n            case 'audio':\n                break;\n\n            default:\n                break;\n        }\n    }\n\n    /**\n     * Process the Inbound Video Track Data\n     * @param stat - video track stats\n     */\n    handleTrack(stat: InboundTrackStats) {\n        // we only want to extract stats from the video track\n        if (\n            stat.type === 'track' &&\n            (stat.trackIdentifier === 'video_label' || stat.kind === 'video')\n        ) {\n            this.inboundVideoStats.framesDropped = stat.framesDropped;\n            this.inboundVideoStats.framesReceived = stat.framesReceived;\n            this.inboundVideoStats.frameHeight = stat.frameHeight;\n            this.inboundVideoStats.frameWidth = stat.frameWidth;\n        }\n    }\n\n    handleCodec(stat: CodecStats) {\n        const codecId = stat.id;\n        const codecType = `${stat.mimeType\n            .replace('video/', '')\n            .replace('audio/', '')}${\n            stat.sdpFmtpLine ? ` ${stat.sdpFmtpLine}` : ''\n        }`;\n        this.codecs.set(codecId, codecType);\n    }\n\n    handleSessionStatistics(\n        videoStartTime: number,\n        inputController: boolean | null,\n        videoEncoderAvgQP: number\n    ) {\n        const deltaTime = Date.now() - videoStartTime;\n        this.sessionStats.runTime = new Date(deltaTime)\n            .toISOString()\n            .substr(11, 8)\n            .toString();\n\n        const controlsStreamInput =\n            inputController === null\n                ? 'Not sent yet'\n                : inputController\n                ? 'true'\n                : 'false';\n        this.sessionStats.controlsStreamInput = controlsStreamInput;\n\n        this.sessionStats.videoEncoderAvgQP = videoEncoderAvgQP;\n    }\n\n    /**\n     * Check if a value coming in from our stats is actually a number\n     * @param value - the number to be checked\n     */\n    isNumber(value: unknown): boolean {\n        return typeof value === 'number' && isFinite(value);\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * ICE Candidate Pair Stats collected from the RTC Stats Report\n */\nexport class CandidatePairStats {\n    bytesReceived: number;\n    bytesSent: number;\n    localCandidateId: string;\n    remoteCandidateId: string;\n    nominated: boolean;\n    readable: boolean;\n    writable: boolean;\n    selected: boolean;\n    state: string;\n    currentRoundTripTime: number;\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * ICE Candidate Stat collected from the RTC Stats Report\n */\nexport class CandidateStat {\n    label: string;\n    id: string;\n    address: string;\n    candidateType: string;\n    port: number;\n    protocol: 'tcp' | 'udp';\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Data Channel Stats collected from the RTC Stats Report\n */\nexport class DataChannelStats {\n    bytesReceived: number;\n    bytesSent: number;\n    dataChannelIdentifier: number;\n    id: string;\n    label: string;\n    messagesReceived: number;\n    messagesSent: number;\n    protocol: string;\n    state: string;\n    timestamp: number;\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Inbound Audio Stats collected from the RTC Stats Report\n */\nexport class InboundAudioStats {\n    audioLevel: number;\n    bytesReceived: number;\n    codecId: string;\n    concealedSamples: number;\n    concealmentEvents: number;\n    fecPacketsDiscarded: number;\n    fecPacketsReceived: number;\n    headerBytesReceived: number;\n    id: string;\n    insertedSamplesForDeceleration: number;\n    jitter: number;\n    jitterBufferDelay: number;\n    jitterBufferEmittedCount: number;\n    jitterBufferMinimumDelay: number;\n    jitterBufferTargetDelay: number;\n    kind: string;\n    lastPacketReceivedTimestamp: number;\n    mediaType: string;\n    mid: string;\n    packetsDiscarded: number;\n    packetsLost: number;\n    packetsReceived: number;\n    removedSamplesForAcceleration: number;\n    silentConcealedSamples: number;\n    ssrc: number;\n    timestamp: number;\n    totalAudioEnergy: number;\n    totalSamplesDuration: number;\n    totalSamplesReceived: number;\n    trackIdentifier: string;\n    transportId: string;\n    type: string;\n\n    /* additional, custom stats */\n    bitrate: number;\n}\n\n/**\n * Inbound Video Stats collected from the RTC Stats Report\n */\nexport class InboundVideoStats {\n    bytesReceived: number;\n    codecId: string;\n    firCount: number;\n    frameHeight: number;\n    frameWidth: number;\n    framesAssembledFromMultiplePackets: number;\n    framesDecoded: number;\n    framesDropped: number;\n    framesPerSecond: number;\n    framesReceived: number;\n    freezeCount: number;\n    googTimingFrameInfo: string;\n    headerBytesReceived: number;\n    id: string;\n    jitter: number;\n    jitterBufferDelay: number;\n    jitterBufferEmittedCount: number;\n    keyFramesDecoded: number;\n    kind: string;\n    lastPacketReceivedTimestamp: number;\n    mediaType: string;\n    mid: string;\n    nackCount: number;\n    packetsLost: number;\n    packetsReceived: number;\n    pauseCount: number;\n    pliCount: number;\n    ssrc: number;\n    timestamp: number;\n    totalAssemblyTime: number;\n    totalDecodeTime: number;\n    totalFreezesDuration: number;\n    totalInterFrameDelay: number;\n    totalPausesDuration: number;\n    totalProcessingDelay: number;\n    totalSquaredInterFrameDelay: number;\n    trackIdentifier: string;\n    transportId: string;\n    type: string;\n\n    /* additional, custom stats */\n    bitrate: number;\n}\n\n/**\n * Inbound Stats collected from the RTC Stats Report\n */\nexport class InboundRTPStats {\n    /* common stats */\n    bytesReceived: number;\n    codecId: string;\n    headerBytesReceived: number;\n    id: string;\n    jitter: number;\n    jitterBufferDelay: number;\n    jitterBufferEmittedCount: number;\n    kind: string;\n    lastPacketReceivedTimestamp: number;\n    mediaType: string;\n    mid: string;\n    packetsLost: number;\n    packetsReceived: number;\n    ssrc: number;\n    timestamp: number;\n    trackIdentifier: string;\n    transportId: string;\n    type: string;\n\n    /* audio specific stats */\n    audioLevel: number;\n    concealedSamples: number;\n    concealmentEvents: number;\n    fecPacketsDiscarded: number;\n    fecPacketsReceived: number;\n    insertedSamplesForDeceleration: number;\n    jitterBufferMinimumDelay: number;\n    jitterBufferTargetDelay: number;\n    packetsDiscarded: number;\n    removedSamplesForAcceleration: number;\n    silentConcealedSamples: number;\n    totalAudioEnergy: number;\n    totalSamplesDuration: number;\n    totalSamplesReceived: number;\n\n    /* video specific stats */\n    firCount: number;\n    frameHeight: number;\n    frameWidth: number;\n    framesAssembledFromMultiplePackets: number;\n    framesDecoded: number;\n    framesDropped: number;\n    framesPerSecond: number;\n    framesReceived: number;\n    freezeCount: number;\n    googTimingFrameInfo: string;\n    keyFramesDecoded: number;\n    nackCount: number;\n    pauseCount: number;\n    pliCount: number;\n    totalAssemblyTime: number;\n    totalDecodeTime: number;\n    totalFreezesDuration: number;\n    totalInterFrameDelay: number;\n    totalPausesDuration: number;\n    totalProcessingDelay: number;\n    totalSquaredInterFrameDelay: number;\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Outbound Video Stats collected from the RTC Stats Report\n */\nexport class OutBoundVideoStats {\n    bytesSent: number;\n    id: string;\n    localId: string;\n    packetsSent: number;\n    remoteTimestamp: number;\n    timestamp: number;\n}\n\n/**\n * Outbound Stats collected from the RTC Stats Report\n */\nexport class OutBoundRTPStats {\n    kind: string;\n    bytesSent: number;\n    id: string;\n    localId: string;\n    packetsSent: number;\n    remoteTimestamp: number;\n    timestamp: number;\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\nimport { Config, OptionParameters, Flags } from '../Config/Config';\nimport { AggregatedStats } from './AggregatedStats';\nimport { parseRtpParameters, splitSections } from 'sdp';\nimport { RTCUtils } from '../Util/RTCUtils';\n\n/**\n * Handles the Peer Connection\n */\nexport class PeerConnectionController {\n    peerConnection: RTCPeerConnection;\n    aggregatedStats: AggregatedStats;\n    config: Config;\n    preferredCodec: string;\n    updateCodecSelection: boolean;\n\n    /**\n     * Create a new RTC Peer Connection client\n     * @param options - Peer connection Options\n     * @param config - The config for our PS experience.\n     */\n    constructor(\n        options: RTCConfiguration,\n        config: Config,\n        preferredCodec: string\n    ) {\n        this.config = config;\n        this.createPeerConnection(options, preferredCodec);\n    }\n\n    createPeerConnection(options: RTCConfiguration, preferredCodec: string) {\n        // Set the ICE transport to relay if TURN enabled\n        if (this.config.isFlagEnabled(Flags.ForceTURN)) {\n            options.iceTransportPolicy = 'relay';\n            Logger.Log(\n                Logger.GetStackTrace(),\n                'Forcing TURN usage by setting ICE Transport Policy in peer connection config.'\n            );\n        }\n\n        // build a new peer connection with the options\n        this.peerConnection = new RTCPeerConnection(options);\n        this.peerConnection.onsignalingstatechange = (ev: Event) =>\n            this.handleSignalStateChange(ev);\n        this.peerConnection.oniceconnectionstatechange = (ev: Event) =>\n            this.handleIceConnectionStateChange(ev);\n        this.peerConnection.onicegatheringstatechange = (ev: Event) =>\n            this.handleIceGatheringStateChange(ev);\n        this.peerConnection.ontrack = (ev: RTCTrackEvent) =>\n            this.handleOnTrack(ev);\n        this.peerConnection.onicecandidate = (ev: RTCPeerConnectionIceEvent) =>\n            this.handleIceCandidate(ev);\n        this.peerConnection.ondatachannel = (ev: RTCDataChannelEvent) =>\n            this.handleDataChannel(ev);\n        this.aggregatedStats = new AggregatedStats();\n        this.preferredCodec = preferredCodec;\n        this.updateCodecSelection = true;\n    }\n\n    /**\n     * Create an offer for the Web RTC handshake and send the offer to the signaling server via websocket\n     * @param offerOptions - RTC Offer Options\n     */\n    async createOffer(offerOptions: RTCOfferOptions, config: Config) {\n        Logger.Log(Logger.GetStackTrace(), 'Create Offer', 6);\n\n        const isLocalhostConnection =\n            location.hostname === 'localhost' ||\n            location.hostname === '127.0.0.1';\n        const isHttpsConnection = location.protocol === 'https:';\n        let useMic = config.isFlagEnabled(Flags.UseMic);\n        if (useMic && !(isLocalhostConnection || isHttpsConnection)) {\n            useMic = false;\n            Logger.Error(\n                Logger.GetStackTrace(),\n                'Microphone access in the browser will not work if you are not on HTTPS or localhost. Disabling mic access.'\n            );\n            Logger.Error(\n                Logger.GetStackTrace(),\n                \"For testing you can enable HTTP microphone access Chrome by visiting chrome://flags/ and enabling 'unsafely-treat-insecure-origin-as-secure'\"\n            );\n        }\n\n        this.setupTransceiversAsync(useMic).finally(() => {\n            this.peerConnection\n                ?.createOffer(offerOptions)\n                .then((offer: RTCSessionDescriptionInit) => {\n                    this.showTextOverlayConnecting();\n                    offer.sdp = this.mungeSDP(offer.sdp, useMic);\n                    this.peerConnection?.setLocalDescription(offer);\n                    this.onSendWebRTCOffer(offer);\n                })\n                .catch(() => {\n                    this.showTextOverlaySetupFailure();\n                });\n        });\n    }\n\n    /**\n     *\n     */\n    async receiveOffer(offer: RTCSessionDescriptionInit, config: Config) {\n        Logger.Log(Logger.GetStackTrace(), 'Receive Offer', 6);\n\n        this.peerConnection?.setRemoteDescription(offer).then(() => {\n            const isLocalhostConnection =\n                location.hostname === 'localhost' ||\n                location.hostname === '127.0.0.1';\n            const isHttpsConnection = location.protocol === 'https:';\n            let useMic = config.isFlagEnabled(Flags.UseMic);\n            if (useMic && !(isLocalhostConnection || isHttpsConnection)) {\n                useMic = false;\n                Logger.Error(\n                    Logger.GetStackTrace(),\n                    'Microphone access in the browser will not work if you are not on HTTPS or localhost. Disabling mic access.'\n                );\n                Logger.Error(\n                    Logger.GetStackTrace(),\n                    \"For testing you can enable HTTP microphone access Chrome by visiting chrome://flags/ and enabling 'unsafely-treat-insecure-origin-as-secure'\"\n                );\n            }\n\n            this.setupTransceiversAsync(useMic).finally(() => {\n                this.peerConnection\n                    ?.createAnswer()\n                    .then((Answer: RTCSessionDescriptionInit) => {\n                        Answer.sdp = this.mungeSDP(Answer.sdp, useMic);\n                        return this.peerConnection?.setLocalDescription(Answer);\n                    })\n                    .then(() => {\n                        this.onSendWebRTCAnswer(\n                            this.peerConnection?.currentLocalDescription\n                        );\n                    })\n                    .catch(() => {\n                        Logger.Error(\n                            Logger.GetStackTrace(),\n                            'createAnswer() failed'\n                        );\n                    });\n            });\n        });\n\n        // Ugly syntax, but this achieves the intersection of the browser supported list and the UE supported list\n        this.config.setOptionSettingOptions(\n            OptionParameters.PreferredCodec,\n            this.parseAvailableCodecs(offer).filter((value) =>\n                this.config\n                    .getSettingOption(OptionParameters.PreferredCodec)\n                    .options.includes(value)\n            )\n        );\n    }\n\n    /**\n     * Set the Remote Descriptor from the signaling server to the RTC Peer Connection\n     * @param answer - RTC Session Descriptor from the Signaling Server\n     */\n    receiveAnswer(answer: RTCSessionDescriptionInit) {\n        this.peerConnection?.setRemoteDescription(answer);\n        // Ugly syntax, but this achieves the intersection of the browser supported list and the UE supported list\n        this.config.setOptionSettingOptions(\n            OptionParameters.PreferredCodec,\n            this.parseAvailableCodecs(answer).filter((value) =>\n                this.config\n                    .getSettingOption(OptionParameters.PreferredCodec)\n                    .options.includes(value)\n            )\n        );\n    }\n\n    /**\n     * Generate Aggregated Stats and then fire a onVideo Stats event\n     */\n    generateStats() {\n        this.peerConnection?.getStats(null).then((StatsData: RTCStatsReport) => {\n            this.aggregatedStats.processStats(StatsData);\n            this.onVideoStats(this.aggregatedStats);\n\n            // Update the preferred codec selection based on what was actually negotiated\n            if (this.updateCodecSelection) {\n                this.config.setOptionSettingValue(\n                    OptionParameters.PreferredCodec,\n                    this.aggregatedStats.codecs.get(\n                        this.aggregatedStats.inboundVideoStats.codecId\n                    )\n                );\n            }\n        });\n    }\n\n    /**\n     * Close The Peer Connection\n     */\n    close() {\n        if (this.peerConnection) {\n            this.peerConnection.close();\n            this.peerConnection = null;\n        }\n    }\n\n    /**\n     * Modify the Session Descriptor\n     * @param sdp - Session Descriptor as a string\n     * @param useMic - Is the microphone in use\n     * @returns A modified Session Descriptor\n     */\n    mungeSDP(sdp: string, useMic: boolean) {\n        let mungedSDP = sdp.replace(\n            /(a=fmtp:\\d+ .*level-asymmetry-allowed=.*)\\r\\n/gm,\n            '$1;x-google-start-bitrate=10000;x-google-max-bitrate=100000\\r\\n'\n        );\n\n        // set max bitrate to highest bitrate Opus supports\n        let audioSDP = 'maxaveragebitrate=510000;';\n\n        if (useMic) {\n            // set the max capture rate to 48khz (so we can send high quality audio from mic)\n            audioSDP += 'sprop-maxcapturerate=48000;';\n        }\n\n        // Force mono or stereo based on whether ?forceMono was passed or not\n        audioSDP += this.config.isFlagEnabled(Flags.ForceMonoAudio)\n            ? 'stereo=0;'\n            : 'stereo=1;';\n\n        // enable in-band forward error correction for opus audio\n        audioSDP += 'useinbandfec=1';\n\n        // We use the line 'useinbandfec=1' (which Opus uses) to set our Opus specific audio parameters.\n        mungedSDP = mungedSDP.replace('useinbandfec=1', audioSDP);\n\n        return mungedSDP;\n    }\n\n    /**\n     * When a Ice Candidate is received add to the RTC Peer Connection\n     * @param iceCandidate - RTC Ice Candidate from the Signaling Server\n     */\n    handleOnIce(iceCandidate: RTCIceCandidate) {\n        Logger.Log(Logger.GetStackTrace(), 'peerconnection handleOnIce', 6);\n\n        // // if forcing TURN, reject any candidates not relay\n        if (this.config.isFlagEnabled(Flags.ForceTURN)) {\n            // check if no relay address is found, if so, we are assuming it means no TURN server\n            if (iceCandidate.candidate.indexOf('relay') < 0) {\n                Logger.Info(\n                    Logger.GetStackTrace(),\n                    `Dropping candidate because it was not TURN relay. | Type= ${iceCandidate.type} | Protocol= ${iceCandidate.protocol} | Address=${iceCandidate.address} | Port=${iceCandidate.port} |`,\n                    6\n                );\n                return;\n            }\n        }\n\n        this.peerConnection?.addIceCandidate(iceCandidate);\n    }\n\n    /**\n     * When the RTC Peer Connection Signaling server state Changes\n     * @param state - Signaling Server State Change Event\n     */\n    handleSignalStateChange(state: Event) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'signaling state change: ' + state,\n            6\n        );\n    }\n\n    /**\n     * Handle when the Ice Connection State Changes\n     * @param state - Ice Connection State\n     */\n    handleIceConnectionStateChange(state: Event) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'ice connection state change: ' + state,\n            6\n        );\n        this.onIceConnectionStateChange(state);\n    }\n\n    /**\n     * Handle when the Ice Gathering State Changes\n     * @param state - Ice Gathering State Change\n     */\n    handleIceGatheringStateChange(state: Event) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'ice gathering state change: ' + JSON.stringify(state),\n            6\n        );\n    }\n\n    /**\n     * Activates the onTrack method\n     * @param event - The webRtc track event\n     */\n    handleOnTrack(event: RTCTrackEvent) {\n        this.onTrack(event);\n    }\n\n    /**\n     * Activates the onPeerIceCandidate\n     * @param event - The peer ice candidate\n     */\n    handleIceCandidate(event: RTCPeerConnectionIceEvent) {\n        this.onPeerIceCandidate(event);\n    }\n\n    /**\n     * Activates the onDataChannel\n     * @param event - The peer's data channel\n     */\n    handleDataChannel(event: RTCDataChannelEvent) {\n        this.onDataChannel(event);\n    }\n\n    /**\n     * An override method for onTrack for use outside of the PeerConnectionController\n     * @param trackEvent - The webRtc track event\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    onTrack(trackEvent: RTCTrackEvent) {\n        // Default Functionality: Do Nothing\n    }\n\n    /**\n     * An override method for onIceConnectionStateChange for use outside of the PeerConnectionController\n     * @param event - The webRtc iceconnectionstatechange event\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    onIceConnectionStateChange(event: Event) {\n        // Default Functionality: Do Nothing\n    }\n\n    /**\n     * An override method for onPeerIceCandidate for use outside of the PeerConnectionController\n     * @param peerConnectionIceEvent - The peer ice candidate\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    onPeerIceCandidate(peerConnectionIceEvent: RTCPeerConnectionIceEvent) {\n        // Default Functionality: Do Nothing\n    }\n\n    /**\n     * An override method for onDataChannel for use outside of the PeerConnectionController\n     * @param datachannelEvent - The peer's data channel\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    onDataChannel(datachannelEvent: RTCDataChannelEvent) {\n        // Default Functionality: Do Nothing\n    }\n\n    /**\n     * Setup tracks on the RTC Peer Connection\n     * @param useMic - is mic in use\n     */\n    async setupTransceiversAsync(useMic: boolean) {\n        const hasTransceivers =\n            this.peerConnection?.getTransceivers().length > 0;\n\n        // Setup a transceiver for getting UE video\n        this.peerConnection?.addTransceiver('video', { direction: 'recvonly' });\n\n        // We can only set preferrec codec on Chrome\n        if (RTCRtpReceiver.getCapabilities && this.preferredCodec != '') {\n            for (const transceiver of this.peerConnection?.getTransceivers() ?? []) {\n                if (\n                    transceiver &&  \n                    transceiver.receiver &&\n                    transceiver.receiver.track &&\n                    transceiver.receiver.track.kind === 'video' &&\n                    // As of 06/2023, FireFox has added RTCRtpReceiver.getCapabilities, but hasn't added the ability to set codec preferences\n                    transceiver.setCodecPreferences\n                ) {\n                    const preferredRTPCodec = this.preferredCodec.split(' ');\n                    const codecs = [\n                        {\n                            mimeType:\n                                'video/' + preferredRTPCodec[0] /* Name */,\n                            clockRate: 90000,\n                            sdpFmtpLine: preferredRTPCodec[1] /* sdpFmtpLine */\n                                ? preferredRTPCodec[1]\n                                : ''\n                        }\n                    ];\n\n                    this.config\n                        .getSettingOption(OptionParameters.PreferredCodec)\n                        .options.filter((option) => {\n                            // Remove the preferred codec from the list of possible codecs as we've set it already\n                            return option != this.preferredCodec;\n                        })\n                        .forEach((option) => {\n                            // Ammend the rest of the browsers supported codecs\n                            const altCodec = option.split(' ');\n                            codecs.push({\n                                mimeType: 'video/' + altCodec[0] /* Name */,\n                                clockRate: 90000,\n                                sdpFmtpLine: altCodec[1] /* sdpFmtpLine */\n                                    ? altCodec[1]\n                                    : ''\n                            });\n                        });\n\n                    for (const codec of codecs) {\n                        if (codec.sdpFmtpLine === '') {\n                            // We can't dynamically add members to the codec, so instead remove the field if it's empty\n                            delete codec.sdpFmtpLine;\n                        }\n                    }\n\n                    transceiver.setCodecPreferences(codecs);\n                }\n            }\n        }\n\n        // Setup a transceiver for sending mic audio to UE and receiving audio from UE\n        if (!useMic) {\n            this.peerConnection?.addTransceiver('audio', {\n                direction: 'recvonly'\n            });\n        } else {\n            // set the audio options based on mic usage\n            const audioOptions = {\n                autoGainControl: false,\n                channelCount: 1,\n                echoCancellation: false,\n                latency: 0,\n                noiseSuppression: false,\n                sampleRate: 48000,\n                sampleSize: 16,\n                volume: 1.0\n            }\n\n            // set the media send options\n            const mediaSendOptions: MediaStreamConstraints = {\n                video: false,\n                audio: audioOptions\n            };\n\n            // Note using mic on android chrome requires SSL or chrome://flags/ \"unsafely-treat-insecure-origin-as-secure\"\n            const stream = await navigator.mediaDevices.getUserMedia(\n                mediaSendOptions\n            );\n            if (stream) {\n                if (hasTransceivers) {\n                    for (const transceiver of this.peerConnection?.getTransceivers() ?? []) {\n                        if (RTCUtils.canTransceiverReceiveAudio(transceiver)) {\n                            for (const track of stream.getTracks()) {\n                                if (track.kind && track.kind == 'audio') {\n                                    transceiver.sender.replaceTrack(track);\n                                    transceiver.direction = 'sendrecv';\n                                }\n                            }\n                        }\n                    }\n                } else {\n                    for (const track of stream.getTracks()) {\n                        if (track.kind && track.kind == 'audio') {\n                            this.peerConnection?.addTransceiver(track, {\n                                direction: 'sendrecv'\n                            });\n                        }\n                    }\n                }\n            } else {\n                this.peerConnection?.addTransceiver('audio', {\n                    direction: 'recvonly'\n                });\n            }\n        }\n    }\n\n    /**\n     * And override event for when the video stats are fired\n     * @param event - Aggregated Stats\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    onVideoStats(event: AggregatedStats) {\n        // Default Functionality: Do Nothing\n    }\n\n    /**\n     * Event to send the RTC offer to the Signaling server\n     * @param offer - RTC Offer\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    onSendWebRTCOffer(offer: RTCSessionDescriptionInit) {\n        // Default Functionality: Do Nothing\n    }\n\n    /**\n     * Event to send the RTC Answer to the Signaling server\n     * @param answer - RTC Answer\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    onSendWebRTCAnswer(answer: RTCSessionDescriptionInit) {\n        // Default Functionality: Do Nothing\n    }\n\n    /**\n     * An override for showing the Peer connection connecting Overlay\n     */\n    showTextOverlayConnecting() {\n        // Default Functionality: Do Nothing\n    }\n\n    /**\n     * An override for showing the Peer connection Failed overlay\n     */\n    showTextOverlaySetupFailure() {\n        // Default Functionality: Do Nothing\n    }\n\n    parseAvailableCodecs(\n        rtcSessionDescription: RTCSessionDescriptionInit\n    ): Array<string> {\n        // No point in updating the available codecs if on FF\n        if (!RTCRtpReceiver.getCapabilities)\n            return ['Only available on Chrome'];\n\n        const ueSupportedCodecs: Array<string> = [];\n        const sections = splitSections(rtcSessionDescription.sdp);\n        // discard the session information as we only want media related info\n        sections.shift();\n        sections.forEach((mediaSection) => {\n            const { codecs } = parseRtpParameters(mediaSection);\n            // Filter only for VPX / H26X / AV1\n            const matcher = /(VP\\d|H26\\d|AV1).*/;\n            codecs.forEach((c) => {\n                const str =\n                    c.name +\n                    ' ' +\n                    Object.keys(c.parameters || {})\n                        .map((p) => p + '=' + c.parameters[p])\n                        .join(';');\n                const match = matcher.exec(str);\n                if (match !== null) {\n                    if (c.name == 'VP9') {\n                        // UE answers don't specify profile but we know we want profile 0\n                        c.parameters = {\n                            'profile-id': '0'\n                        };\n                    }\n                    const codecStr: string =\n                        c.name +\n                        ' ' +\n                        Object.keys(c.parameters || {})\n                            .map((p) => p + '=' + c.parameters[p])\n                            .join(';');\n                    ueSupportedCodecs.push(codecStr);\n                }\n            });\n        });\n\n        return ueSupportedCodecs;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Session statistics\n */\nexport class SessionStats {\n    runTime: string;\n    controlsStreamInput: string;\n    videoEncoderAvgQP: number;\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Class to hold the stream stats data coming in from webRtc\n */\nexport class StreamStats {\n    id: string;\n    streamIdentifier: string;\n    timestamp: number;\n    trackIds: string[];\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Config, OptionParameters } from '../Config/Config';\nimport { LatencyTestResults } from '../DataChannel/LatencyTestResults';\nimport { AggregatedStats } from '../PeerConnectionController/AggregatedStats';\nimport { WebRtcPlayerController } from '../WebRtcPlayer/WebRtcPlayerController';\nimport { Flags, NumericParameters } from '../Config/Config';\nimport { Logger } from '../Logger/Logger';\nimport { InitialSettings } from '../DataChannel/InitialSettings';\nimport { OnScreenKeyboard } from '../UI/OnScreenKeyboard';\nimport {\n    EventEmitter,\n    InitialSettingsEvent,\n    LatencyTestResultEvent,\n    PixelStreamingEvent,\n    StatsReceivedEvent,\n    StreamLoadingEvent,\n    StreamPreConnectEvent,\n    StreamReconnectEvent,\n    StreamPreDisconnectEvent,\n    VideoEncoderAvgQPEvent,\n    VideoInitializedEvent,\n    WebRtcAutoConnectEvent,\n    WebRtcConnectedEvent,\n    WebRtcConnectingEvent,\n    WebRtcDisconnectedEvent,\n    WebRtcFailedEvent,\n    WebRtcSdpEvent,\n    DataChannelLatencyTestResponseEvent,\n    DataChannelLatencyTestResultEvent,\n    PlayerCountEvent\n} from '../Util/EventEmitter';\nimport { MessageOnScreenKeyboard } from '../WebSockets/MessageReceive';\nimport { WebXRController } from '../WebXR/WebXRController';\nimport { MessageDirection } from '../UeInstanceMessage/StreamMessageController';\nimport {\n    DataChannelLatencyTestConfig,\n    DataChannelLatencyTestController\n} from \"../DataChannel/DataChannelLatencyTestController\";\nimport {\n    DataChannelLatencyTestResponse,\n    DataChannelLatencyTestResult\n} from \"../DataChannel/DataChannelLatencyTestResults\";\nimport { RTCUtils } from '../Util/RTCUtils';\n\n\nexport interface PixelStreamingOverrides {\n    /** The DOM elment where Pixel Streaming video and user input event handlers are attached to.\n     * You can give an existing DOM element here. If not given, the library will create a new div element\n     * that is not attached anywhere. In this case you can later get access to this new element and\n     * attach it to your web page. */\n    videoElementParent?: HTMLElement;\n}\n\n/**\n * The key class for the browser side of a Pixel Streaming application, it includes:\n * WebRTC handling, XR support, input handling, and emitters for lifetime and state change events.\n * Users are encouraged to use this class as is, through composition, or extend it. In any case, \n * this will likely be the core of your Pixel Streaming experience in terms of functionality.\n */\nexport class PixelStreaming {\n    protected _webRtcController: WebRtcPlayerController;\n    protected _webXrController: WebXRController;\n    protected _dataChannelLatencyTestController: DataChannelLatencyTestController;\n    /**\n     * Configuration object. You can read or modify config through this object. Whenever\n     * the configuration is changed, the library will emit a `settingsChanged` event.\n     */\n    public config: Config;\n\n    private _videoElementParent: HTMLElement;\n\n    private allowConsoleCommands = false;\n\n    private onScreenKeyboardHelper: OnScreenKeyboard;\n\n    private _videoStartTime: number;\n    private _inputController: boolean;\n\n    private _eventEmitter: EventEmitter;\n\n    /**\n     * @param config - A newly instantiated config object\n     * @param overrides - Parameters to override default behaviour\n     * returns the base Pixel streaming object\n     */\n    constructor(config: Config, overrides?: PixelStreamingOverrides) {\n        this.config = config;\n\n        if (overrides?.videoElementParent) {\n            this._videoElementParent = overrides.videoElementParent;\n        }\n\n        this._eventEmitter = new EventEmitter();\n\n        this.configureSettings();\n\n        // setup WebRTC\n        this.setWebRtcPlayerController(\n            new WebRtcPlayerController(this.config, this)\n        );\n\n        // Onscreen keyboard\n        this.onScreenKeyboardHelper = new OnScreenKeyboard(\n            this.videoElementParent\n        );\n        this.onScreenKeyboardHelper.unquantizeAndDenormalizeUnsigned = (\n            x: number,\n            y: number\n        ) =>\n            this._webRtcController.requestUnquantizedAndDenormalizeUnsigned(\n                x,\n                y\n            );\n        this._activateOnScreenKeyboard = (command: MessageOnScreenKeyboard) =>\n            this.onScreenKeyboardHelper.showOnScreenKeyboard(command);\n\n        this._webXrController = new WebXRController(this._webRtcController);\n    }\n\n    /**\n     * Gets the element that contains the video stream element.\n     */\n    public get videoElementParent(): HTMLElement {\n        if (!this._videoElementParent) {\n            this._videoElementParent = document.createElement('div');\n            this._videoElementParent.id = 'videoElementParent';\n        }\n        return this._videoElementParent;\n    }\n\n    /**\n     * Configure the settings with on change listeners and any additional per experience settings.\n     */\n    private configureSettings(): void {\n        this.config._addOnSettingChangedListener(\n            Flags.IsQualityController,\n            (wantsQualityController: boolean) => {\n                // If the setting has been set to true (either programatically or the user has flicked the toggle)\n                // and we aren't currently quality controller, send the request\n                if (\n                    wantsQualityController === true &&\n                    !this._webRtcController.isQualityController\n                ) {\n                    this._webRtcController.sendRequestQualityControlOwnership();\n                }\n            }\n        );\n\n        this.config._addOnSettingChangedListener(\n            Flags.AFKDetection,\n            (isAFKEnabled: boolean) => {\n                this._webRtcController.setAfkEnabled(isAFKEnabled);\n            }\n        );\n\n        this.config._addOnSettingChangedListener(\n            Flags.MatchViewportResolution,\n            () => {\n                this._webRtcController.videoPlayer.updateVideoStreamSize();\n            }\n        );\n\n        this.config._addOnSettingChangedListener(\n            Flags.HoveringMouseMode,\n            (isHoveringMouse: boolean) => {\n                this.config.setFlagLabel(\n                    Flags.HoveringMouseMode,\n                    `Control Scheme: ${\n                        isHoveringMouse ? 'Hovering' : 'Locked'\n                    } Mouse`\n                );\n                this._webRtcController.setMouseInputEnabled(this.config.isFlagEnabled(Flags.MouseInput));\n            }\n        );\n\n        // user input\n        this.config._addOnSettingChangedListener(\n            Flags.KeyboardInput,\n            (isEnabled: boolean) => {\n                this._webRtcController.setKeyboardInputEnabled(isEnabled);\n            }\n        );\n\n        this.config._addOnSettingChangedListener(\n            Flags.MouseInput,\n            (isEnabled: boolean) => {\n                this._webRtcController.setMouseInputEnabled(isEnabled);\n            }\n        );\n\n        this.config._addOnSettingChangedListener(\n            Flags.TouchInput,\n            (isEnabled: boolean) => {\n                this._webRtcController.setTouchInputEnabled(isEnabled);\n            }\n        );\n\n        this.config._addOnSettingChangedListener(\n            Flags.GamepadInput,\n            (isEnabled: boolean) => {\n                this._webRtcController.setGamePadInputEnabled(isEnabled);\n            }\n        );\n\n        // encoder settings\n        this.config._addOnNumericSettingChangedListener(\n            NumericParameters.MinQP,\n            (newValue: number) => {\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    '--------  Sending MinQP  --------',\n                    7\n                );\n                this._webRtcController.sendEncoderMinQP(newValue);\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    '-------------------------------------------',\n                    7\n                );\n            }\n        );\n\n        this.config._addOnNumericSettingChangedListener(\n            NumericParameters.MaxQP,\n            (newValue: number) => {\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    '--------  Sending encoder settings  --------',\n                    7\n                );\n                this._webRtcController.sendEncoderMaxQP(newValue);\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    '-------------------------------------------',\n                    7\n                );\n            }\n        );\n\n        // WebRTC settings\n        this.config._addOnNumericSettingChangedListener(\n            NumericParameters.WebRTCMinBitrate,\n            (newValue: number) => {\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    '--------  Sending web rtc settings  --------',\n                    7\n                );\n                this._webRtcController.sendWebRTCMinBitrate(newValue * 1000 /* kbps to bps */);\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    '-------------------------------------------',\n                    7\n                );\n            }\n        );\n\n        this.config._addOnNumericSettingChangedListener(\n            NumericParameters.WebRTCMaxBitrate,\n            (newValue: number) => {\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    '--------  Sending web rtc settings  --------',\n                    7\n                );\n                this._webRtcController.sendWebRTCMaxBitrate(newValue * 1000 /* kbps to bps */);\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    '-------------------------------------------',\n                    7\n                );\n            }\n        );\n\n        this.config._addOnNumericSettingChangedListener(\n            NumericParameters.WebRTCFPS,\n            (newValue: number) => {\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    '--------  Sending web rtc settings  --------',\n                    7\n                );\n                this._webRtcController.sendWebRTCFps(newValue);\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    '-------------------------------------------',\n                    7\n                );\n            }\n        );\n\n        this.config._addOnOptionSettingChangedListener(\n            OptionParameters.PreferredCodec,\n            (newValue: string) => {\n                if (this._webRtcController) {\n                    this._webRtcController.setPreferredCodec(newValue);\n                }\n            }\n        );\n\n        this.config._registerOnChangeEvents(this._eventEmitter);\n    }\n\n    /**\n     * Activate the on screen keyboard when receiving the command from the streamer\n     * @param command - the keyboard command\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    _activateOnScreenKeyboard(command: MessageOnScreenKeyboard): void {\n        throw new Error('Method not implemented.');\n    }\n\n    /**\n     * Set the input control ownership\n     * @param inputControlOwnership - does the user have input control ownership\n     */\n    _onInputControlOwnership(inputControlOwnership: boolean): void {\n        this._inputController = inputControlOwnership;\n    }\n\n    /**\n     * Instantiate the WebRTCPlayerController interface to provide WebRTCPlayerController functionality within this class and set up anything that requires it\n     * @param webRtcPlayerController - a WebRtcPlayerController controller instance\n     */\n    private setWebRtcPlayerController(\n        webRtcPlayerController: WebRtcPlayerController\n    ) {\n        this._webRtcController = webRtcPlayerController;\n\n        this._webRtcController.setPreferredCodec(\n            this.config.getSettingOption(OptionParameters.PreferredCodec)\n                .selected\n        );\n        this._webRtcController.resizePlayerStyle();\n\n        // connect if auto connect flag is enabled\n        this.checkForAutoConnect();\n    }\n\n    /**\n     * Connect to signaling server.\n     */\n    public connect() {\n        this._eventEmitter.dispatchEvent(new StreamPreConnectEvent());\n        this._webRtcController.connectToSignallingServer();\n    }\n\n    /**\n     * Reconnects to the signaling server. If connection is up, disconnects first\n     * before establishing a new connection\n     */\n    public reconnect() {\n        this._eventEmitter.dispatchEvent(new StreamReconnectEvent());\n        this._webRtcController.tryReconnect(\"Reconnecting...\");\n    }\n\n    /**\n     * Disconnect from the signaling server and close open peer connections.\n     */\n    public disconnect() {\n        this._eventEmitter.dispatchEvent(new StreamPreDisconnectEvent());\n        this._webRtcController.close();\n    }\n\n    /**\n     * Play the stream. Can be called only after a peer connection has been established.\n     */\n    public play() {\n        this._onStreamLoading();\n        this._webRtcController.playStream();\n    }\n\n    /**\n     * Auto connect if AutoConnect flag is enabled\n     */\n    private checkForAutoConnect() {\n        // set up if the auto play will be used or regular click to start\n        if (this.config.isFlagEnabled(Flags.AutoConnect)) {\n            // if autoplaying show an info overlay while while waiting for the connection to begin\n            this._onWebRtcAutoConnect();\n            this._webRtcController.connectToSignallingServer();\n        }\n    }\n\n    /** \n     * Will unmute the microphone track which is sent to Unreal Engine.\n     * By default, will only unmute an existing mic track.\n     * \n     * @param forceEnable Can be used for cases when this object wasn't initialized with a mic track.\n     * If this parameter is true, the connection will be restarted with a microphone.\n     * Warning: this takes some time, as a full renegotiation and reconnection will happen.\n     */\n    public unmuteMicrophone(forceEnable = false) : void {\n        // If there's an existing mic track, we just set muted state\n        if (this.config.isFlagEnabled('UseMic')) {\n            this.setMicrophoneMuted(false);\n            return;\n        }\n        \n        // If there's no pre-existing mic track, and caller is ok with full reset, we enable and reset\n        if (forceEnable) {\n            this.config.setFlagEnabled(\"UseMic\", true);\n            this.reconnect();\n            return;\n        }\n          \n        // If we prefer not to force a reconnection, just warn the user that this operation didn't happen\n        Logger.Warning(\n            Logger.GetStackTrace(),\n            'Trying to unmute mic, but PixelStreaming was initialized with no microphone track. Call with forceEnable == true to re-connect with a mic track.'\n        );\n    }\n\n    public muteMicrophone() : void {\n        if (this.config.isFlagEnabled('UseMic')) {\n            this.setMicrophoneMuted(true);\n            return;\n        }\n\n        // If there wasn't a mic track, just let user know there's nothing to mute\n        Logger.Info(\n            Logger.GetStackTrace(),\n            'Trying to mute mic, but PixelStreaming has no microphone track, so sending sound is already disabled.'\n        );\n    }\n\n    private setMicrophoneMuted(mute: boolean) : void\n    {\n        for (const transceiver of this._webRtcController?.peerConnectionController?.peerConnection?.getTransceivers() ?? []) {\n            if (RTCUtils.canTransceiverSendAudio(transceiver)) {\n                transceiver.sender.track.enabled = !mute;\n            }\n        }\n    }\n\n    /**\n     * Emit an event on auto connecting\n     */\n    _onWebRtcAutoConnect() {\n        this._eventEmitter.dispatchEvent(new WebRtcAutoConnectEvent());\n    }\n\n    /**\n     * Set up functionality to happen when receiving a webRTC answer\n     */\n    _onWebRtcSdp() {\n        this._eventEmitter.dispatchEvent(new WebRtcSdpEvent());\n    }\n\n    /**\n     * Emits a StreamLoading event\n     */\n    _onStreamLoading() {\n        this._eventEmitter.dispatchEvent(new StreamLoadingEvent());\n    }\n\n    /**\n     * Event fired when the video is disconnected - emits given eventString or an override\n     * message from webRtcController if one has been set\n     * @param eventString - a string describing why the connection closed\n     * @param allowClickToReconnect - true if we want to allow the user to retry the connection with a click\n     */\n    _onDisconnect(eventString: string, allowClickToReconnect: boolean) {\n        this._eventEmitter.dispatchEvent(\n            new WebRtcDisconnectedEvent({\n                eventString: eventString,\n                allowClickToReconnect: allowClickToReconnect\n            })\n        );\n    }\n\n    /**\n     * Handles when Web Rtc is connecting\n     */\n    _onWebRtcConnecting() {\n        this._eventEmitter.dispatchEvent(new WebRtcConnectingEvent());\n    }\n\n    /**\n     * Handles when Web Rtc has connected\n     */\n    _onWebRtcConnected() {\n        this._eventEmitter.dispatchEvent(new WebRtcConnectedEvent());\n    }\n\n    /**\n     * Handles when Web Rtc fails to connect\n     */\n    _onWebRtcFailed() {\n        this._eventEmitter.dispatchEvent(new WebRtcFailedEvent());\n    }\n\n    /**\n     * Handle when the Video has been Initialized\n     */\n    _onVideoInitialized() {\n        this._eventEmitter.dispatchEvent(new VideoInitializedEvent());\n        this._videoStartTime = Date.now();\n    }\n\n    /**\n     * Set up functionality to happen when receiving latency test results\n     * @param latency - latency test results object\n     */\n    _onLatencyTestResult(latencyTimings: LatencyTestResults) {\n        this._eventEmitter.dispatchEvent(\n            new LatencyTestResultEvent({ latencyTimings })\n        );\n    }\n\n    _onDataChannelLatencyTestResponse(response: DataChannelLatencyTestResponse) {\n        this._eventEmitter.dispatchEvent(\n            new DataChannelLatencyTestResponseEvent({ response })\n        );\n    }\n\n    /**\n     * Set up functionality to happen when receiving video statistics\n     * @param videoStats - video statistics as a aggregate stats object\n     */\n    _onVideoStats(videoStats: AggregatedStats) {\n        // Duration\n        if (!this._videoStartTime || this._videoStartTime === undefined) {\n            this._videoStartTime = Date.now();\n        }\n        videoStats.handleSessionStatistics(\n            this._videoStartTime,\n            this._inputController,\n            this._webRtcController.videoAvgQp\n        );\n\n        this._eventEmitter.dispatchEvent(\n            new StatsReceivedEvent({ aggregatedStats: videoStats })\n        );\n    }\n\n    /**\n     * Set up functionality to happen when calculating the average video encoder qp\n     * @param QP - the quality number of the stream\n     */\n    _onVideoEncoderAvgQP(QP: number) {\n        this._eventEmitter.dispatchEvent(\n            new VideoEncoderAvgQPEvent({ avgQP: QP })\n        );\n    }\n\n    /**\n     * Set up functionality to happen when receiving and handling initial settings for the UE app\n     * @param settings - initial UE app settings\n     */\n    _onInitialSettings(settings: InitialSettings) {\n        this._eventEmitter.dispatchEvent(\n            new InitialSettingsEvent({ settings })\n        );\n        if (settings.PixelStreamingSettings) {\n            this.allowConsoleCommands =\n                settings.PixelStreamingSettings.AllowPixelStreamingCommands ?? false;\n            if (this.allowConsoleCommands === false) {\n                Logger.Info(\n                    Logger.GetStackTrace(),\n                    '-AllowPixelStreamingCommands=false, sending arbitrary console commands from browser to UE is disabled.'\n                );\n            }\n        }\n\n        const useUrlParams = this.config.useUrlParams;\n        const urlParams = new URLSearchParams(window.location.search);\n        if (settings.EncoderSettings) {\n            this.config.setNumericSetting(\n                NumericParameters.MinQP,\n                // If a setting is set in the URL, make sure we respect that value as opposed to what the application sends us\n                (useUrlParams && urlParams.has(NumericParameters.MinQP)) \n                    ? Number.parseInt(urlParams.get(NumericParameters.MinQP)) \n                    : settings.EncoderSettings.MinQP\n            );\n\n            \n            this.config.setNumericSetting(\n                NumericParameters.MaxQP,\n                (useUrlParams && urlParams.has(NumericParameters.MaxQP)) \n                    ? Number.parseInt(urlParams.get(NumericParameters.MaxQP)) \n                    : settings.EncoderSettings.MaxQP\n            );\n        }\n        if (settings.WebRTCSettings) {\n            this.config.setNumericSetting(\n                NumericParameters.WebRTCMinBitrate,\n                (useUrlParams && urlParams.has(NumericParameters.WebRTCMinBitrate)) \n                    ? Number.parseInt(urlParams.get(NumericParameters.WebRTCMinBitrate))\n                    : settings.WebRTCSettings.MinBitrate / 1000 /* bps to kbps */\n            );\n            this.config.setNumericSetting(\n                NumericParameters.WebRTCMaxBitrate,\n                (useUrlParams && urlParams.has(NumericParameters.WebRTCMaxBitrate)) \n                    ? Number.parseInt(urlParams.get(NumericParameters.WebRTCMaxBitrate))\n                    : settings.WebRTCSettings.MaxBitrate / 1000 /* bps to kbps */\n                \n            );\n            this.config.setNumericSetting(\n                NumericParameters.WebRTCFPS,\n                (useUrlParams && urlParams.has(NumericParameters.WebRTCFPS)) \n                    ? Number.parseInt(urlParams.get(NumericParameters.WebRTCFPS))\n                    : settings.WebRTCSettings.FPS\n            );\n        }\n    }\n\n    /**\n     * Set up functionality to happen when setting quality control ownership of a stream\n     * @param hasQualityOwnership - does this user have quality ownership of the stream true / false\n     */\n    _onQualityControlOwnership(hasQualityOwnership: boolean) {\n        this.config.setFlagEnabled(\n            Flags.IsQualityController,\n            hasQualityOwnership\n        );\n    }\n\n    _onPlayerCount(playerCount: number) {\n        this._eventEmitter.dispatchEvent(\n            new PlayerCountEvent({ count: playerCount })\n        );\n    }\n\n    /**\n     * Request a connection latency test.\n     * NOTE: There are plans to refactor all request* functions. Expect changes if you use this!\n     * @returns\n     */\n    public requestLatencyTest() {\n        if (!this._webRtcController.videoPlayer.isVideoReady()) {\n            return false;\n        }\n        this._webRtcController.sendLatencyTest();\n        return true;\n    }\n\n    /**\n     * Request a data channel latency test.\n     * NOTE: There are plans to refactor all request* functions. Expect changes if you use this!\n     */\n    public requestDataChannelLatencyTest(config: DataChannelLatencyTestConfig) {\n        if (!this._webRtcController.videoPlayer.isVideoReady()) {\n            return false;\n        }\n        if (!this._dataChannelLatencyTestController) {\n            this._dataChannelLatencyTestController = new DataChannelLatencyTestController(\n                this._webRtcController.sendDataChannelLatencyTest.bind(this._webRtcController),\n                (result: DataChannelLatencyTestResult) => {\n                    this._eventEmitter.dispatchEvent(new DataChannelLatencyTestResultEvent( { result }))\n                });\n            this.addEventListener(\n                \"dataChannelLatencyTestResponse\",\n                ({data: {response} }) => {\n                    this._dataChannelLatencyTestController.receive(response);\n                }\n            )\n        }\n        return this._dataChannelLatencyTestController.start(config);\n    }\n\n    /**\n     * Request for the UE application to show FPS counter.\n     * NOTE: There are plans to refactor all request* functions. Expect changes if you use this!\n     * @returns\n     */\n    public requestShowFps() {\n        if (!this._webRtcController.videoPlayer.isVideoReady()) {\n            return false;\n        }\n        this._webRtcController.sendShowFps();\n        return true;\n    }\n\n    /**\n     * Request for a new IFrame from the UE application.\n     * NOTE: There are plans to refactor all request* functions. Expect changes if you use this!\n     * @returns\n     */\n    public requestIframe() {\n        if (!this._webRtcController.videoPlayer.isVideoReady()) {\n            return false;\n        }\n        this._webRtcController.sendIframeRequest();\n        return true;\n    }\n\n    /**\n     * Send data to UE application. The data will be run through JSON.stringify() so e.g. strings\n     * and any serializable plain JSON objects with no recurrence can be sent.\n     * @returns true if succeeded, false if rejected\n     */\n    public emitUIInteraction(descriptor: object | string) {\n        if (!this._webRtcController.videoPlayer.isVideoReady()) {\n            return false;\n        }\n        this._webRtcController.emitUIInteraction(descriptor);\n        return true;\n    }\n\n    /**\n     * Send a command to UE application. Blocks ConsoleCommand descriptors unless UE\n     * has signaled that it allows console commands.\n     * @returns true if succeeded, false if rejected\n     */\n    public emitCommand(descriptor: object) {\n        if (!this._webRtcController.videoPlayer.isVideoReady()) {\n            return false;\n        }\n        if (!this.allowConsoleCommands && 'ConsoleCommand' in descriptor) {\n            return false;\n        }\n        this._webRtcController.emitCommand(descriptor);\n        return true;\n    }\n\n    /**\n     * Send a console command to UE application. Only allowed if UE has signaled that it allows\n     * console commands.\n     * @returns true if succeeded, false if rejected\n     */\n    public emitConsoleCommand(command: string) {\n        if (!this.allowConsoleCommands || !this._webRtcController.videoPlayer.isVideoReady()) {\n            return false;\n        }\n        this._webRtcController.emitConsoleCommand(command);\n        return true;\n    }\n\n    /**\n     * Add a UE -> browser response event listener\n     * @param name - The name of the response handler\n     * @param listener - The method to be activated when a message is received\n     */\n    public addResponseEventListener(\n        name: string,\n        listener: (response: string) => void\n    ) {\n        this._webRtcController.responseController.addResponseEventListener(name, listener);\n    }\n\n    /**\n     * Remove a UE -> browser response event listener\n     * @param name - The name of the response handler\n     */\n    public removeResponseEventListener(name: string) {\n        this._webRtcController.responseController.removeResponseEventListener(name);\n    }\n\n    /**\n     * Dispatch a new event.\n     * @param e event\n     * @returns\n     */\n    public dispatchEvent(e: PixelStreamingEvent): boolean {\n        return this._eventEmitter.dispatchEvent(e);\n    }\n    \n    /**\n     * Register an event handler.\n     * @param type event name\n     * @param listener event handler function\n     */\n    public addEventListener<\n        T extends PixelStreamingEvent['type'],\n        E extends PixelStreamingEvent & { type: T }\n    >(type: T, listener: (e: Event & E) => void) {\n        this._eventEmitter.addEventListener(type, listener);\n    }\n\n    /**\n     * Remove an event handler.\n     * @param type event name\n     * @param listener event handler function\n     */\n    public removeEventListener<\n        T extends PixelStreamingEvent['type'],\n        E extends PixelStreamingEvent & { type: T }\n    >(type: T, listener: (e: Event & E) => void) {\n        this._eventEmitter.removeEventListener(type, listener);\n    }\n\n    /**\n     * Enable/disable XR mode.\n     */\n    public toggleXR() {\n        this.webXrController.xrClicked();\n    }\n\n    /**\n     * Pass in a function to generate a signalling server URL.\n     * This function is useful if you need to programmatically construct your signalling server URL.\n     * @param signallingUrlBuilderFunc A function that generates a signalling server url.\n     */\n    public setSignallingUrlBuilder(signallingUrlBuilderFunc: ()=>string) {\n        this._webRtcController.signallingUrlBuilder = signallingUrlBuilderFunc;\n    }\n\n    /**\n     * Public getter for the websocket controller. Access to this property allows you to send\n     * custom websocket messages.\n     */\n    public get webSocketController() {\n        return this._webRtcController.webSocketController;\n    }\n\n    /**\n     * Public getter for the webXrController controller. Used for all XR features.\n     */\n    public get webXrController() {\n        return this._webXrController;\n    }\n\n    public registerMessageHandler(name: string, direction: MessageDirection, handler?: (data: ArrayBuffer | Array<number | string>) => void) {\n        if(direction === MessageDirection.FromStreamer && typeof handler === 'undefined') {\n            Logger.Warning(Logger.GetStackTrace(), `Unable to register an undefined handler for ${name}`)\n            return;\n        }\n\n        if(direction === MessageDirection.ToStreamer && typeof handler === 'undefined') {\n            this._webRtcController.streamMessageController.registerMessageHandler(\n                direction,\n                name,\n                (data: Array<number | string>) =>\n                this._webRtcController.sendMessageController.sendMessageToStreamer(\n                    name,\n                    data\n                )\n            );\n        } else {\n            this._webRtcController.streamMessageController.registerMessageHandler(\n                direction,\n                name,\n                (data: ArrayBuffer) => handler(data)\n            );\n        }\n    }\n\n    public get toStreamerHandlers() {\n        return this._webRtcController.streamMessageController.toStreamerHandlers;\n    }\n\n    public isReconnecting() {\n        return this._webRtcController.isReconnecting;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { UnquantizedDenormalizedUnsignedCoord } from '../Util/CoordinateConverter';\nimport { MessageOnScreenKeyboard } from '../WebSockets/MessageReceive';\n\n/**\n * Class for handling on screen keyboard usage\n */\nexport class OnScreenKeyboard {\n    // If the user focuses on a UE input widget then we show them a button to open\n    // the on-screen keyboard. JavaScript security means we can only show the\n    // on-screen keyboard in response to a user interaction.\n    editTextButton: HTMLButtonElement;\n\n    // A hidden input text box which is used only for focusing and opening the\n    // on-screen keyboard.\n    hiddenInput: HTMLInputElement;\n\n    /**\n     *\n     * @param videoElementParent The div element the video player is injected into\n     */\n    constructor(videoElementParent: HTMLElement) {\n        this.editTextButton = null;\n        this.hiddenInput = null;\n\n        if ('ontouchstart' in document.documentElement) {\n            this.createOnScreenKeyboardHelpers(videoElementParent);\n        }\n    }\n\n    /**\n     * An override for unquantizeAndDenormalizeUnsigned\n     * @param x the x axis point\n     * @param y the y axis point\n     * @returns unquantizeAndDenormalizeUnsigned object\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    unquantizeAndDenormalizeUnsigned(\n        x: number,\n        y: number\n    ): UnquantizedDenormalizedUnsignedCoord {\n        return null;\n    }\n\n    /**\n     * Creates on screen keyboard helpers\n     * @param videoElementParent The div element the video player i injected into\n     */\n    createOnScreenKeyboardHelpers(videoElementParent: HTMLElement) {\n        if (!this.hiddenInput) {\n            this.hiddenInput = document.createElement('input');\n            this.hiddenInput.id = 'hiddenInput';\n            this.hiddenInput.maxLength = 0;\n            videoElementParent.appendChild(this.hiddenInput);\n        }\n\n        if (!this.editTextButton) {\n            this.editTextButton = document.createElement('button');\n            this.editTextButton.id = 'editTextButton';\n            this.editTextButton.innerHTML = 'edit text';\n            videoElementParent.appendChild(this.editTextButton);\n\n            // Hide the 'edit text' button.\n            this.editTextButton.classList.add('hiddenState');\n\n            this.editTextButton.addEventListener('touchend', (event: Event) => {\n                // Show the on-screen keyboard.\n                this.hiddenInput.focus();\n                event.preventDefault();\n            });\n        }\n    }\n\n    /**\n     * Shows the on screen keyboard\n     * @param command the command received via the data channel containing keyboard positions\n     */\n    showOnScreenKeyboard(command: MessageOnScreenKeyboard) {\n        if (command.showOnScreenKeyboard) {\n            // Show the 'edit text' button.\n            this.editTextButton.classList.remove('hiddenState');\n            // Place the 'edit text' button near the UE input widget.\n            const pos = this.unquantizeAndDenormalizeUnsigned(\n                command.x,\n                command.y\n            );\n            this.editTextButton.style.top = pos.y.toString() + 'px';\n            this.editTextButton.style.left = (pos.x - 40).toString() + 'px';\n        } else {\n            // Hide the 'edit text' button.\n            this.editTextButton.classList.add('hiddenState');\n            // Hide the on-screen keyboard.\n            this.hiddenInput.blur();\n        }\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\n\nexport class ResponseController {\n    responseEventListeners: Map<string, (response: string) => void> = new Map();\n\n    /**\n     * Add a response event listener to the response map\n     * @param name - The name of the response\n     * @param listener - The method to be activated when the response is selected\n     */\n    addResponseEventListener(\n        name: string,\n        listener: (response: string) => void\n    ) {\n        this.responseEventListeners.set(name, listener);\n    }\n\n    /**\n     * Remove a response event listener to the response map\n     * @param name - The name of the response\n     */\n    removeResponseEventListener(name: string) {\n        this.responseEventListeners.delete(name);\n    }\n\n    /**\n     * Handle a response when receiving one form the streamer\n     * @param message - Data received from the data channel with the command in question\n     */\n    onResponse(message: ArrayBuffer) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'DataChannelReceiveMessageType.Response',\n            6\n        );\n        const responses = new TextDecoder('utf-16').decode(message.slice(1));\n\n        Logger.Log(Logger.GetStackTrace(), responses, 6);\n        this.responseEventListeners.forEach(\n            (listener: (response: string) => void) => {\n                listener(responses);\n            }\n        );\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { DataChannelSender } from '../DataChannel/DataChannelSender';\nimport { Logger } from '../Logger/Logger';\nimport { StreamMessageController } from './StreamMessageController';\n\nexport class SendMessageController {\n    toStreamerMessagesMapProvider: StreamMessageController;\n    dataChannelSender: DataChannelSender;\n\n    /**\n     * @param dataChannelSender - Data channel instance\n     * @param toStreamerMessagesMapProvider - Stream Messages instance\n     */\n    constructor(\n        dataChannelSender: DataChannelSender,\n        toStreamerMessagesMapProvider: StreamMessageController\n    ) {\n        this.dataChannelSender = dataChannelSender;\n        this.toStreamerMessagesMapProvider = toStreamerMessagesMapProvider;\n    }\n\n    /**\n     * Send a message to the streamer through the data channel\n     * @param messageType - the type of message we are sending\n     * @param messageData - the message data we are sending over the data channel\n     * @returns - nil\n     */\n    sendMessageToStreamer(messageType: string, messageData?: Array<number | string>) {\n        if (messageData === undefined) {\n            messageData = [];\n        }\n\n        const toStreamerMessages =\n            this.toStreamerMessagesMapProvider.toStreamerMessages;\n        const messageFormat = toStreamerMessages.get(messageType);\n        if (messageFormat === undefined) {\n            Logger.Error(\n                Logger.GetStackTrace(),\n                `Attempted to send a message to the streamer with message type: ${messageType}, but the frontend hasn't been configured to send such a message. Check you've added the message type in your cpp`\n            );\n            return;\n        }\n\n        if(messageFormat.structure && messageData && messageFormat.structure.length !== messageData.length) {\n            Logger.Error(\n                Logger.GetStackTrace(),\n                `Provided message data doesn't match expected layout. Expected [ ${messageFormat.structure.map((element: string) => {\n                    switch (element) {\n                        case 'uint8':\n                        case 'uint16':\n                        case 'int16':\n                        case 'float':\n                        case 'double':\n                            return 'number';\n                        case 'string':\n                            return 'string';\n                    }\n                }).toString() } ] but received [ ${messageData.map((element: number | string) => typeof element).toString()} ]`\n            );\n            return;\n        }\n\n        let byteLength = 0;\n        const textEncoder = new TextEncoder();\n        // One loop to calculate the length in bytes of all of the provided data\n        messageData.forEach((element: number | string, idx: number) => {\n            const type = messageFormat.structure[idx];\n            switch (type) {\n                case 'uint8':\n                    byteLength += 1;\n                    break;\n\n                case 'uint16':\n                    byteLength += 2;\n                    break;\n\n                case 'int16':\n                    byteLength += 2;\n                    break;\n\n                case 'float':\n                    byteLength += 4;\n                    break;\n\n                case 'double':\n                    byteLength += 8;\n                    break;\n\n                case 'string':\n                    // 2 bytes for string length\n                    byteLength += 2;\n                    // 2 bytes per characters\n                    byteLength += 2 * textEncoder.encode(element as string).length;\n                    break;\n            }\n        });\n\n        const data = new DataView(new ArrayBuffer(byteLength + 1));\n        data.setUint8(0, messageFormat.id);\n        let byteOffset = 1;\n\n        messageData.forEach((element: number | string, idx: number) => {\n            const type = messageFormat.structure[idx];\n            switch (type) {\n                case 'uint8':\n                    data.setUint8(byteOffset, element as number);\n                    byteOffset += 1;\n                    break;\n\n                case 'uint16':\n                    data.setUint16(byteOffset, element as number, true);\n                    byteOffset += 2;\n                    break;\n\n                case 'int16':\n                    data.setInt16(byteOffset, element as number, true);\n                    byteOffset += 2;\n                    break;\n\n                case 'float':\n                    data.setFloat32(byteOffset, element as number, true);\n                    byteOffset += 4;\n                    break;\n\n                case 'double':\n                    data.setFloat64(byteOffset, element as number, true);\n                    byteOffset += 8;\n                    break;\n\n                case 'string':\n                    data.setUint16(byteOffset, (element as string).length, true);\n                    byteOffset += 2;\n                    for (let i = 0; i < (element as string).length; i++) {\n                        data.setUint16(byteOffset, (element as string).charCodeAt(i), true);\n                        byteOffset += 2;\n                    }\n                    break;\n            }\n        });\n\n        if (!this.dataChannelSender.canSend()) {\n            Logger.Info(\n                Logger.GetStackTrace(),\n                `Data channel cannot send yet, skipping sending message: ${messageType} - ${new Uint8Array(\n                    data.buffer\n                )}`\n            );\n            return;\n        }\n\n        this.dataChannelSender.sendData(data.buffer);\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\n\nexport class ToStreamerMessage {\n    id: number;\n    structure?: Array<string>;\n}\n\nexport class StreamMessageController {\n    toStreamerHandlers: Map<\n        string,\n        (messageData?: Array<number | string> | undefined) => void\n    >;\n    fromStreamerHandlers: Map<\n        string,\n        (messageType: string, messageData?: ArrayBuffer | undefined) => void\n    >;\n    //                        Type      Format\n    toStreamerMessages: Map<string, ToStreamerMessage>;\n    //                         ID      Type\n    fromStreamerMessages: Map<number, string>;\n\n    constructor() {\n        this.toStreamerHandlers = new Map();\n        this.fromStreamerHandlers = new Map();\n        this.toStreamerMessages = new Map();\n        this.fromStreamerMessages = new Map();\n    }\n\n    /**\n     * Populate the Default message protocol\n     */\n    populateDefaultProtocol() {\n        /*\n         * Control Messages. Range = 0..49.\n         */\n        this.toStreamerMessages.set('IFrameRequest', {\n            id: 0,\n            structure: []\n        });\n        this.toStreamerMessages.set('RequestQualityControl', {\n            id: 1,\n            structure: []\n        });\n        this.toStreamerMessages.set('FpsRequest', {\n            id: 2,\n            structure: []\n        });\n        this.toStreamerMessages.set('AverageBitrateRequest', {\n            id: 3,\n            structure: []\n        });\n        this.toStreamerMessages.set('StartStreaming', {\n            id: 4,\n            structure: []\n        });\n        this.toStreamerMessages.set('StopStreaming', {\n            id: 5,\n            structure: []\n        });\n        this.toStreamerMessages.set('LatencyTest', {\n            id: 6,\n            structure: ['string']\n        });\n        this.toStreamerMessages.set('RequestInitialSettings', {\n            id: 7,\n            structure: []\n        });\n        this.toStreamerMessages.set('TestEcho', {\n            id: 8,\n            structure: []\n        });\n        this.toStreamerMessages.set('DataChannelLatencyTest', {\n            id: 9,\n            structure: []\n        });\n        /*\n         * Input Messages. Range = 50..89.\n         */\n        // Generic Input Messages. Range = 50..59.\n        this.toStreamerMessages.set('UIInteraction', {\n            id: 50,\n            structure: ['string']\n        });\n        this.toStreamerMessages.set('Command', {\n            id: 51,\n            structure: ['string']\n        });\n        // Keyboard Input Message. Range = 60..69.\n        this.toStreamerMessages.set('KeyDown', {\n            id: 60,\n            //            keyCode  isRepeat\n            structure: ['uint8', 'uint8']\n        });\n        this.toStreamerMessages.set('KeyUp', {\n            id: 61,\n            //            keyCode\n            structure: ['uint8']\n        });\n        this.toStreamerMessages.set('KeyPress', {\n            id: 62,\n            //            charcode\n            structure: ['uint16']\n        });\n        // Mouse Input Messages. Range = 70..79.\n        this.toStreamerMessages.set('MouseEnter', {\n            id: 70,\n            structure: []\n        });\n        this.toStreamerMessages.set('MouseLeave', {\n            id: 71,\n            structure: []\n        });\n        this.toStreamerMessages.set('MouseDown', {\n            id: 72,\n            //              button     x         y\n            structure: ['uint8', 'uint16', 'uint16']\n        });\n        this.toStreamerMessages.set('MouseUp', {\n            id: 73,\n            //              button     x         y\n            structure: ['uint8', 'uint16', 'uint16']\n        });\n        this.toStreamerMessages.set('MouseMove', {\n            id: 74,\n            //              x           y      deltaX    deltaY\n            structure: ['uint16', 'uint16', 'int16', 'int16']\n        });\n        this.toStreamerMessages.set('MouseWheel', {\n            id: 75,\n            //              delta       x        y\n            structure: ['int16', 'uint16', 'uint16']\n        });\n        this.toStreamerMessages.set('MouseDouble', {\n            id: 76,\n            //              button     x         y\n            structure: ['uint8', 'uint16', 'uint16']\n        });\n        // Touch Input Messages. Range = 80..89.\n        this.toStreamerMessages.set('TouchStart', {\n            id: 80,\n            //          numtouches(1)   x       y        idx     force     valid\n            structure: ['uint8', 'uint16', 'uint16', 'uint8', 'uint8', 'uint8']\n        });\n        this.toStreamerMessages.set('TouchEnd', {\n            id: 81,\n            //          numtouches(1)   x       y        idx     force     valid\n            structure: ['uint8', 'uint16', 'uint16', 'uint8', 'uint8', 'uint8']\n        });\n        this.toStreamerMessages.set('TouchMove', {\n            id: 82,\n            //          numtouches(1)   x       y       idx      force     valid\n            structure: ['uint8', 'uint16', 'uint16', 'uint8', 'uint8', 'uint8']\n        });\n        // Gamepad Input Messages. Range = 90..99\n        this.toStreamerMessages.set('GamepadConnected', {\n            id: 93,\n            structure: []\n        });\n        this.toStreamerMessages.set('GamepadButtonPressed', {\n            id: 90,\n            //         ctrlerId   button  isRepeat\n            structure: ['uint8', 'uint8', 'uint8']\n        });\n        this.toStreamerMessages.set('GamepadButtonReleased', {\n            id: 91,\n            //         ctrlerId   button  isRepeat(0)\n            structure: ['uint8', 'uint8', 'uint8']\n        });\n        this.toStreamerMessages.set('GamepadAnalog', {\n            id: 92,\n            //         ctrlerId   button  analogValue\n            structure: ['uint8', 'uint8', 'double']\n        });\n        this.toStreamerMessages.set('GamepadDisconnected', {\n            id: 94,\n            //          ctrlerId\n            structure: ['uint8']\n        });\n\n        this.fromStreamerMessages.set(0, 'QualityControlOwnership');\n        this.fromStreamerMessages.set(1, 'Response');\n        this.fromStreamerMessages.set(2, 'Command');\n        this.fromStreamerMessages.set(3, 'FreezeFrame');\n        this.fromStreamerMessages.set(4, 'UnfreezeFrame');\n        this.fromStreamerMessages.set(5, 'VideoEncoderAvgQP');\n        this.fromStreamerMessages.set(6, 'LatencyTest');\n        this.fromStreamerMessages.set(7, 'InitialSettings');\n        this.fromStreamerMessages.set(8, 'FileExtension');\n        this.fromStreamerMessages.set(9, 'FileMimeType');\n        this.fromStreamerMessages.set(10, 'FileContents');\n        this.fromStreamerMessages.set(11, 'TestEcho');\n        this.fromStreamerMessages.set(12, 'InputControlOwnership');\n        this.fromStreamerMessages.set(13, 'GamepadResponse');\n        this.fromStreamerMessages.set(14, 'DataChannelLatencyTest');\n        this.fromStreamerMessages.set(255, 'Protocol');\n    }\n\n    /**\n     * Register a message handler\n     * @param messageDirection - the direction of the message; toStreamer or fromStreamer\n     * @param messageType - the type of the message\n     * @param messageHandler - the function or method to be executed when this handler is called\n     */\n    registerMessageHandler(\n        messageDirection: MessageDirection,\n        messageType: string,\n        messageHandler: (messageData?: unknown | undefined) => void\n    ) {\n        switch (messageDirection) {\n            case MessageDirection.ToStreamer:\n                this.toStreamerHandlers.set(messageType, messageHandler);\n                break;\n            case MessageDirection.FromStreamer:\n                this.fromStreamerHandlers.set(messageType, messageHandler);\n                break;\n            default:\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    `Unknown message direction ${messageDirection}`\n                );\n        }\n    }\n}\n\n/**\n * The enum for message directions\n */\nexport enum MessageDirection {\n    ToStreamer = 0,\n    FromStreamer = 1\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { SendMessageController } from './SendMessageController';\n\nexport class ToStreamerMessagesController {\n    sendMessageController: SendMessageController;\n\n    /**\n     * @param sendMessageController - Stream message controller instance\n     */\n    constructor(sendMessageController: SendMessageController) {\n        this.sendMessageController = sendMessageController;\n    }\n\n    /**\n     * Send Request to Take Quality Control to the UE Instance\n     */\n    SendRequestQualityControl() {\n        this.sendMessageController.sendMessageToStreamer(\n            'RequestQualityControl'\n        );\n    }\n\n    /**\n     * Send Max FPS Request to the UE Instance\n     */\n    SendMaxFpsRequest() {\n        this.sendMessageController.sendMessageToStreamer('FpsRequest');\n    }\n\n    /**\n     * Send Average Bitrate Request to the UE Instance\n     */\n    SendAverageBitrateRequest() {\n        this.sendMessageController.sendMessageToStreamer(\n            'AverageBitrateRequest'\n        );\n    }\n\n    /**\n     * Send a Start Streaming Message to the UE Instance\n     */\n    SendStartStreaming() {\n        this.sendMessageController.sendMessageToStreamer('StartStreaming');\n    }\n\n    /**\n     * Send a Stop Streaming Message to the UE Instance\n     */\n    SendStopStreaming() {\n        this.sendMessageController.sendMessageToStreamer('StopStreaming');\n    }\n\n    /**\n     * Send a Request Initial Settings to the UE Instance\n     */\n    SendRequestInitialSettings() {\n        this.sendMessageController.sendMessageToStreamer(\n            'RequestInitialSettings'\n        );\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\nimport { VideoPlayer } from '../VideoPlayer/VideoPlayer';\n\n/**\n * Converts coordinates from element relative coordinates to values normalized within the value range of a short (and back again)\n */\nexport class CoordinateConverter {\n    videoElementProvider: VideoPlayer;\n    videoElementParent: HTMLElement;\n    videoElement: HTMLVideoElement;\n    ratio: number;\n\n    normalizeAndQuantizeUnsignedFunc: (\n        x: number,\n        y: number\n    ) => NormalizedQuantizedUnsignedCoord;\n    normalizeAndQuantizeSignedFunc: (\n        x: number,\n        y: number\n    ) => NormalizedQuantizedSignedCoord;\n    denormalizeAndUnquantizeUnsignedFunc: (\n        x: number,\n        y: number\n    ) => UnquantizedDenormalizedUnsignedCoord;\n\n    /**\n     * @param videoElementProvider - the div element that the video player will be injected into\n     */\n    constructor(videoElementProvider: VideoPlayer) {\n        this.videoElementProvider = videoElementProvider;\n        this.normalizeAndQuantizeUnsignedFunc = () => {\n            throw new Error(\n                'Normalize and quantize unsigned, method not implemented.'\n            );\n        };\n        this.normalizeAndQuantizeSignedFunc = () => {\n            throw new Error(\n                'Normalize and unquantize signed, method not implemented.'\n            );\n        };\n        this.denormalizeAndUnquantizeUnsignedFunc = () => {\n            throw new Error(\n                'Denormalize and unquantize unsigned, method not implemented.'\n            );\n        };\n    }\n\n    /**\n     * The surface method for setterNormalizeAndQuantizeUnsigned\n     * @param x - x axis point\n     * @param y - y axis point\n     */\n    normalizeAndQuantizeUnsigned(\n        x: number,\n        y: number\n    ): NormalizedQuantizedUnsignedCoord {\n        return this.normalizeAndQuantizeUnsignedFunc(x, y);\n    }\n\n    /**\n     * The surface method for setterUnquantizeAndDenormalizeUnsigned\n     * @param x - x axis point\n     * @param y - y axis point\n     */\n    unquantizeAndDenormalizeUnsigned(\n        x: number,\n        y: number\n    ): UnquantizedDenormalizedUnsignedCoord {\n        return this.denormalizeAndUnquantizeUnsignedFunc(x, y);\n    }\n\n    /**\n     * The surface method for setterNormalizeAndQuantizeSigned\n     * @param x - x axis point\n     * @param y - y axis point\n     */\n    normalizeAndQuantizeSigned(\n        x: number,\n        y: number\n    ): NormalizedQuantizedSignedCoord {\n        return this.normalizeAndQuantizeSignedFunc(x, y);\n    }\n\n    /**\n     * set up the Normalize And Quantize methods based on the aspect ratio and the video player ratio\n     */\n    setupNormalizeAndQuantize() {\n        this.videoElementParent =\n            this.videoElementProvider.getVideoParentElement();\n        this.videoElement = this.videoElementProvider.getVideoElement();\n\n        if (this.videoElementParent && this.videoElement) {\n            const playerAspectRatio =\n                this.videoElementParent.clientHeight /\n                this.videoElementParent.clientWidth;\n            const videoAspectRatio =\n                this.videoElement.videoHeight / this.videoElement.videoWidth;\n            if (playerAspectRatio > videoAspectRatio) {\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    'Setup Normalize and Quantize for playerAspectRatio > videoAspectRatio',\n                    6\n                );\n                this.ratio = playerAspectRatio / videoAspectRatio;\n                this.normalizeAndQuantizeUnsignedFunc = (\n                    x: number,\n                    y: number\n                ) => this.normalizeAndQuantizeUnsignedPlayerBigger(x, y);\n                this.normalizeAndQuantizeSignedFunc = (x: number, y: number) =>\n                    this.normalizeAndQuantizeSignedPlayerBigger(x, y);\n                this.denormalizeAndUnquantizeUnsignedFunc = (\n                    x: number,\n                    y: number\n                ) => this.denormalizeAndUnquantizeUnsignedPlayerBigger(x, y);\n            } else {\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    'Setup Normalize and Quantize for playerAspectRatio <= videoAspectRatio',\n                    6\n                );\n                this.ratio = videoAspectRatio / playerAspectRatio;\n                this.normalizeAndQuantizeUnsignedFunc = (\n                    x: number,\n                    y: number\n                ) => this.normalizeAndQuantizeUnsignedPlayerSmaller(x, y);\n                this.normalizeAndQuantizeSignedFunc = (x: number, y: number) =>\n                    this.normalizeAndQuantizeSignedPlayerSmaller(x, y);\n                this.denormalizeAndUnquantizeUnsignedFunc = (\n                    x: number,\n                    y: number\n                ) => this.denormalizeAndUnquantizeUnsignedPlayerSmaller(x, y);\n            }\n        }\n    }\n\n    /**\n     * normalizeAndQuantizeUnsigned for playerAspectRatio > videoAspectRatio\n     * @param x - x axis point\n     * @param y - y axis point\n     */\n    normalizeAndQuantizeUnsignedPlayerBigger(\n        x: number,\n        y: number\n    ): NormalizedQuantizedUnsignedCoord {\n        const normalizedX = x / this.videoElementParent.clientWidth;\n        const normalizedY =\n            this.ratio * (y / this.videoElementParent.clientHeight - 0.5) + 0.5;\n        if (\n            normalizedX < 0.0 ||\n            normalizedX > 1.0 ||\n            normalizedY < 0.0 ||\n            normalizedY > 1.0\n        ) {\n            return new NormalizedQuantizedUnsignedCoord(false, 65535, 65535);\n        } else {\n            return new NormalizedQuantizedUnsignedCoord(\n                true,\n                normalizedX * 65536,\n                normalizedY * 65536\n            );\n        }\n    }\n\n    /**\n     * unquantizeAndDenormalizeUnsigned for playerAspectRatio > videoAspectRatio\n     * @param x - x axis point\n     * @param y - y axis point\n     */\n    denormalizeAndUnquantizeUnsignedPlayerBigger(x: number, y: number) {\n        const normalizedX = x / 65536;\n        const normalizedY = (y / 65536 - 0.5) / this.ratio + 0.5;\n        return new UnquantizedDenormalizedUnsignedCoord(\n            normalizedX * this.videoElementParent.clientWidth,\n            normalizedY * this.videoElementParent.clientHeight\n        );\n    }\n\n    /**\n     * normalizeAndQuantizeSigned for playerAspectRatio > videoAspectRatio\n     * @param x - x axis point\n     * @param y - y axis point\n     */\n    normalizeAndQuantizeSignedPlayerBigger(x: number, y: number) {\n        const normalizedX = x / (0.5 * this.videoElementParent.clientWidth);\n        const normalizedY =\n            (this.ratio * y) / (0.5 * this.videoElementParent.clientHeight);\n        return new NormalizedQuantizedSignedCoord(\n            normalizedX * 32767,\n            normalizedY * 32767\n        );\n    }\n\n    /**\n     * normalizeAndQuantizeUnsigned for playerAspectRatio <= videoAspectRatio\n     * @param x - x axis point\n     * @param y - y axis point\n     */\n    normalizeAndQuantizeUnsignedPlayerSmaller(x: number, y: number) {\n        const normalizedX =\n            this.ratio * (x / this.videoElementParent.clientWidth - 0.5) + 0.5;\n        const normalizedY = y / this.videoElementParent.clientHeight;\n        if (\n            normalizedX < 0.0 ||\n            normalizedX > 1.0 ||\n            normalizedY < 0.0 ||\n            normalizedY > 1.0\n        ) {\n            return new NormalizedQuantizedUnsignedCoord(false, 65535, 65535);\n        } else {\n            return new NormalizedQuantizedUnsignedCoord(\n                true,\n                normalizedX * 65536,\n                normalizedY * 65536\n            );\n        }\n    }\n\n    /**\n     * unquantizeAndDenormalizeUnsigned for playerAspectRatio <= videoAspectRatio\n     * @param x - x axis point\n     * @param y - y axis point\n     */\n    denormalizeAndUnquantizeUnsignedPlayerSmaller(x: number, y: number) {\n        const normalizedX = (x / 65536 - 0.5) / this.ratio + 0.5;\n        const normalizedY = y / 65536;\n        return new UnquantizedDenormalizedUnsignedCoord(\n            normalizedX * this.videoElementParent.clientWidth,\n            normalizedY * this.videoElementParent.clientHeight\n        );\n    }\n\n    /**\n     * normalizeAndQuantizeSigned for playerAspectRatio <= videoAspectRatio\n     * @param x - x axis point\n     * @param y - y axis point\n     */\n    normalizeAndQuantizeSignedPlayerSmaller(x: number, y: number) {\n        const normalizedX =\n            (this.ratio * x) / (0.5 * this.videoElementParent.clientWidth);\n        const normalizedY = y / (0.5 * this.videoElementParent.clientHeight);\n        return new NormalizedQuantizedSignedCoord(\n            normalizedX * 32767,\n            normalizedY * 32767\n        );\n    }\n}\n\n/**\n * A class for NormalizeAndQuantizeUnsigned objects\n */\nexport class NormalizedQuantizedUnsignedCoord {\n    inRange: boolean;\n    x: number;\n    y: number;\n\n    constructor(inRange: boolean, x: number, y: number) {\n        this.inRange = inRange;\n        this.x = x;\n        this.y = y;\n    }\n}\n\n/**\n * A class for UnquantizedAndDenormalizeUnsigned objects\n */\nexport class UnquantizedDenormalizedUnsignedCoord {\n    x: number;\n    y: number;\n\n    constructor(x: number, y: number) {\n        this.x = x;\n        this.y = y;\n    }\n}\n\n/**\n * A class for NormalizedQuantizedSignedCoord objects\n */\nexport class NormalizedQuantizedSignedCoord {\n    x: number;\n    y: number;\n\n    constructor(x: number, y: number) {\n        this.x = x;\n        this.y = y;\n    }\n}\n","import {\n    FlagsIds,\n    NumericParametersIds,\n    OptionParametersIds,\n    TextParametersIds\n} from '../Config/Config';\nimport { LatencyTestResults } from '../DataChannel/LatencyTestResults';\nimport { AggregatedStats } from '../PeerConnectionController/AggregatedStats';\nimport { InitialSettings } from '../pixelstreamingfrontend';\nimport { MessageStreamerList } from '../WebSockets/MessageReceive';\nimport { SettingFlag } from '../Config/SettingFlag';\nimport { SettingNumber } from '../Config/SettingNumber';\nimport { SettingText } from '../Config/SettingText';\nimport { SettingOption } from '../Config/SettingOption';\nimport {\n    DataChannelLatencyTestResponse,\n    DataChannelLatencyTestResult\n} from \"../DataChannel/DataChannelLatencyTestResults\";\n\n/**\n * An event that is emitted when AFK disconnect is about to happen.\n * Can be cancelled by calling the callback function provided as part of the event.\n */\nexport class AfkWarningActivateEvent extends Event {\n    readonly type: 'afkWarningActivate';\n    readonly data: {\n        /** How many seconds until the session is disconnected */\n        countDown: number;\n        /** Callback function that needs to be called if you wish to cancel the AFK disconnect timeout. */\n        dismissAfk: () => void;\n    };\n    constructor(data: AfkWarningActivateEvent['data']) {\n        super('afkWarningActivate');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when the AFK disconnect countdown is updated.\n */\nexport class AfkWarningUpdateEvent extends Event {\n    readonly type: 'afkWarningUpdate';\n    readonly data: {\n        /** How many seconds until the session is disconnected */\n        countDown: number\n    };\n    constructor(data: AfkWarningUpdateEvent['data']) {\n        super('afkWarningUpdate');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when AFK warning is deactivated.\n */\nexport class AfkWarningDeactivateEvent extends Event {\n    readonly type: 'afkWarningDeactivate';\n    constructor() {\n        super('afkWarningDeactivate');\n    }\n}\n\n/**\n * An event that is emitted when AFK countdown reaches 0 and the user is disconnected.\n */\nexport class AfkTimedOutEvent extends Event {\n    readonly type: 'afkTimedOut';\n    constructor() {\n        super('afkTimedOut');\n    }\n}\n\n/**\n * An event that is emitted when we receive new video quality value.\n */\nexport class VideoEncoderAvgQPEvent extends Event {\n    readonly type: 'videoEncoderAvgQP';\n    readonly data: {\n        /** Average video quality value */\n        avgQP: number\n    };\n    constructor(data: VideoEncoderAvgQPEvent['data']) {\n        super('videoEncoderAvgQP');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted after a WebRtc connection has been negotiated. \n */\nexport class WebRtcSdpEvent extends Event {\n    readonly type: 'webRtcSdp';\n    constructor() {\n        super('webRtcSdp');\n    }\n}\n\n/**\n * An event that is emitted when auto connecting.\n */\nexport class WebRtcAutoConnectEvent extends Event {\n    readonly type: 'webRtcAutoConnect';\n    constructor() {\n        super('webRtcAutoConnect');\n    }\n}\n\n/**\n * An event that is emitted when sending a WebRtc offer.\n */\nexport class WebRtcConnectingEvent extends Event {\n    readonly type: 'webRtcConnecting';\n    constructor() {\n        super('webRtcConnecting');\n    }\n}\n\n/**\n * An event that is emitted when WebRtc connection has been established.\n */\nexport class WebRtcConnectedEvent extends Event {\n    readonly type: 'webRtcConnected';\n    constructor() {\n        super('webRtcConnected');\n    }\n}\n\n/**\n * An event that is emitted if WebRtc connection has failed.\n */\nexport class WebRtcFailedEvent extends Event {\n    readonly type: 'webRtcFailed';\n    constructor() {\n        super('webRtcFailed');\n    }\n}\n\n/**\n * An event that is emitted if WebRtc connection is disconnected.\n */\nexport class WebRtcDisconnectedEvent extends Event {\n    readonly type: 'webRtcDisconnected';\n    readonly data: {\n        /** Message describing the disconnect reason */\n        eventString: string;\n        /** true if the user is able to reconnect, false if disconnected because of unrecoverable reasons like not able to connect to the signaling server */\n        allowClickToReconnect: boolean;\n    };\n    constructor(data: WebRtcDisconnectedEvent['data']) {\n        super('webRtcDisconnected');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when RTCDataChannel is opened.\n */\nexport class DataChannelOpenEvent extends Event {\n    readonly type: 'dataChannelOpen';\n    readonly data: {\n        /** Data channel label. One of 'datachannel', 'send-datachannel', 'recv-datachannel' */\n        label: string;\n        /** RTCDataChannel onOpen event */\n        event: Event\n    };\n    constructor(data: DataChannelOpenEvent['data']) {\n        super('dataChannelOpen');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when RTCDataChannel is closed.\n */\nexport class DataChannelCloseEvent extends Event {\n    readonly type: 'dataChannelClose';\n    readonly data: {\n        /** Data channel label. One of 'datachannel', 'send-datachannel', 'recv-datachannel' */\n        label: string;\n        /** RTCDataChannel onClose event */\n        event: Event\n    };\n    constructor(data: DataChannelCloseEvent['data']) {\n        super('dataChannelClose');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted on RTCDataChannel errors.\n */\nexport class DataChannelErrorEvent extends Event {\n    readonly type: 'dataChannelError';\n    readonly data: {\n        /** Data channel label. One of 'datachannel', 'send-datachannel', 'recv-datachannel' */\n        label: string;\n        /** RTCDataChannel onError event */\n        event: Event\n    };\n    constructor(data: DataChannelErrorEvent['data']) {\n        super('dataChannelError');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when the video stream has been initialized.\n */\nexport class VideoInitializedEvent extends Event {\n    readonly type: 'videoInitialized';\n    constructor() {\n        super('videoInitialized');\n    }\n}\n\n/**\n * An event that is emitted when video stream loading starts.\n */\nexport class StreamLoadingEvent extends Event {\n    readonly type: 'streamLoading';\n    constructor() {\n        super('streamLoading');\n    }\n}\n\n/**\n * An event that is emitted when video stream loading has finished.\n */\nexport class StreamPreConnectEvent extends Event {\n    readonly type: 'streamConnect';\n    constructor() {\n        super('streamConnect');\n    }\n}\n\n/**\n * An event that is emitted when video stream has stopped.\n */\nexport class StreamPreDisconnectEvent extends Event {\n    readonly type: 'streamDisconnect';\n    constructor() {\n        super('streamDisconnect');\n    }\n}\n\n/**\n * An event that is emitted when video stream is reconnecting.\n */\nexport class StreamReconnectEvent extends Event {\n    readonly type: 'streamReconnect';\n    constructor() {\n        super('streamReconnect');\n    }\n}\n\n/**\n * An event that is emitted if there are errors loading the video stream.\n */\nexport class PlayStreamErrorEvent extends Event {\n    readonly type: 'playStreamError';\n    readonly data: {\n        /** Error message */\n        message: string\n    };\n    constructor(data: PlayStreamErrorEvent['data']) {\n        super('playStreamError');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted before trying to start video playback.\n */\nexport class PlayStreamEvent extends Event {\n    readonly type: 'playStream';\n    constructor() {\n        super('playStream');\n    }\n}\n\n/**\n * An event that is emitted if the browser rejects video playback. Can happen for example if\n * video auto-play without user interaction is refused by the browser.\n */\nexport class PlayStreamRejectedEvent extends Event {\n    readonly type: 'playStreamRejected';\n    readonly data: {\n        /** Rejection reason */\n        reason: unknown\n    };\n    constructor(data: PlayStreamRejectedEvent['data']) {\n        super('playStreamRejected');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when receiving a full FreezeFrame image from UE.\n */\nexport class LoadFreezeFrameEvent extends Event {\n    readonly type: 'loadFreezeFrame';\n    readonly data: {\n        /** true if should show click-to-play overlay, not the freeze frame contents */\n        shouldShowPlayOverlay: boolean;\n        /** true if the received image is valid */\n        isValid: boolean;\n        /** Image data. Can be e.g. displayed by encoding as a data url. */\n        jpegData?: Uint8Array;\n    };\n    constructor(data: LoadFreezeFrameEvent['data']) {\n        super('loadFreezeFrame');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when receiving UnfreezeFrame message from UE and video playback is about to be resumed.\n */\nexport class HideFreezeFrameEvent extends Event {\n    readonly type: 'hideFreezeFrame';\n    constructor() {\n        super('hideFreezeFrame');\n    }\n}\n\n/**\n * An event that is emitted when receiving WebRTC statistics.\n */\nexport class StatsReceivedEvent extends Event {\n    readonly type: 'statsReceived';\n    readonly data: {\n        /** Statistics object */\n        aggregatedStats: AggregatedStats\n    };\n    constructor(data: StatsReceivedEvent['data']) {\n        super('statsReceived');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when streamer list changes.\n */\nexport class StreamerListMessageEvent extends Event {\n    readonly type: 'streamerListMessage';\n    readonly data: {\n        /** Streamer list message containing an array of streamer ids */\n        messageStreamerList: MessageStreamerList;\n        /** Auto-selected streamer from the list, or null if unable to auto-select and user should be prompted to select */\n        autoSelectedStreamerId: string;\n        /** Wanted streamer id from various configurations. */\n        wantedStreamerId: string;\n    };\n    constructor(data: StreamerListMessageEvent['data']) {\n        super('streamerListMessage');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when receiving latency test results.\n */\nexport class LatencyTestResultEvent extends Event {\n    readonly type: 'latencyTestResult';\n    readonly data: {\n        /** Latency test result object */\n        latencyTimings: LatencyTestResults\n    };\n    constructor(data: LatencyTestResultEvent['data']) {\n        super('latencyTestResult');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when receiving data channel latency test response from server.\n * This event is handled by DataChannelLatencyTestController\n */\nexport class DataChannelLatencyTestResponseEvent extends Event {\n    readonly type: 'dataChannelLatencyTestResponse';\n    readonly data: {\n        /** Latency test result object */\n        response: DataChannelLatencyTestResponse\n    };\n    constructor(data: DataChannelLatencyTestResponseEvent['data']) {\n        super('dataChannelLatencyTestResponse');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when data channel latency test results are ready.\n */\nexport class DataChannelLatencyTestResultEvent extends Event {\n    readonly type: 'dataChannelLatencyTestResult';\n    readonly data: {\n        /** Latency test result object */\n        result: DataChannelLatencyTestResult\n    };\n    constructor(data: DataChannelLatencyTestResultEvent['data']) {\n        super('dataChannelLatencyTestResult');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when receiving initial settings from UE.\n */\nexport class InitialSettingsEvent extends Event {\n    readonly type: 'initialSettings';\n    readonly data: {\n        /** Initial settings from UE */\n        settings: InitialSettings\n    };\n    constructor(data: InitialSettingsEvent['data']) {\n        super('initialSettings');\n        this.data = data;\n    }\n}\n\nexport type SettingsData =\n    | {\n          /** Flag id */\n          id: FlagsIds;\n          type: 'flag';\n          /** Flag value */\n          value: boolean;\n          /** SettingFlag object */\n          target: SettingFlag;\n      }\n    | {\n          /** Numeric setting id */\n          id: NumericParametersIds;\n          type: 'number';\n          /** Numeric setting value */\n          value: number;\n          /** SettingNumber object */\n          target: SettingNumber;\n      }\n    | {\n          /** Text setting id */\n          id: TextParametersIds;\n          type: 'text';\n          /** Text setting value */\n          value: string;\n          /** SettingText object */\n          target: SettingText;\n      }\n    | {\n          /** Option setting id */\n          id: OptionParametersIds;\n          type: 'option';\n          /** Option setting selected value */\n          value: string;\n          /** SettingOption object */\n          target: SettingOption;\n      };\n\n/**\n * An event that is emitted when PixelStreaming settings change.\n */\nexport class SettingsChangedEvent extends Event {\n    readonly type: 'settingsChanged';\n    readonly data: SettingsData;\n    constructor(data: SettingsChangedEvent['data']) {\n        super('settingsChanged');\n        this.data = data;\n    }\n}\n\n/**\n * Event emitted when an XR Session starts\n */\nexport class XrSessionStartedEvent extends Event {\n    readonly type: 'xrSessionStarted';\n    constructor() {\n        super('xrSessionStarted');\n    }\n}\n\n/**\n * Event emitted when an XR Session ends\n */\nexport class XrSessionEndedEvent extends Event {\n    readonly type: 'xrSessionEnded';\n    constructor() {\n        super('xrSessionEnded');\n    }\n}\n\nexport type XrFrameData = {\n    /** The frame timestamp  */\n    time: DOMHighResTimeStamp;\n    /** The XRFrame */\n    frame: XRFrame;\n};\n\n/**\n * Event emitted when an XR Frame is complete\n */\nexport class XrFrameEvent extends Event {\n    readonly type: 'xrFrame';\n    readonly data: XrFrameData\n    constructor(data: XrFrameEvent['data']) {\n        super('xrFrame');\n        this.data = data;\n    }\n}\n\n/**\n * An event that is emitted when receiving a player count from the signalling server\n */\nexport class PlayerCountEvent extends Event {\n    readonly type: 'playerCount';\n    readonly data: {\n        /** count object */\n        count: number\n    };\n    constructor(data: PlayerCountEvent['data']) {\n        super('playerCount');\n        this.data = data;\n    }\n}\n\nexport type PixelStreamingEvent =\n    | AfkWarningActivateEvent\n    | AfkWarningUpdateEvent\n    | AfkWarningDeactivateEvent\n    | AfkTimedOutEvent\n    | VideoEncoderAvgQPEvent\n    | WebRtcSdpEvent\n    | WebRtcAutoConnectEvent\n    | WebRtcConnectingEvent\n    | WebRtcConnectedEvent\n    | WebRtcFailedEvent\n    | WebRtcDisconnectedEvent\n    | DataChannelOpenEvent\n    | DataChannelCloseEvent\n    | DataChannelErrorEvent\n    | VideoInitializedEvent\n    | StreamLoadingEvent\n    | StreamPreConnectEvent\n    | StreamReconnectEvent\n    | StreamPreDisconnectEvent\n    | PlayStreamErrorEvent\n    | PlayStreamEvent\n    | PlayStreamRejectedEvent\n    | LoadFreezeFrameEvent\n    | HideFreezeFrameEvent\n    | StatsReceivedEvent\n    | StreamerListMessageEvent\n    | LatencyTestResultEvent\n    | DataChannelLatencyTestResponseEvent\n    | DataChannelLatencyTestResultEvent\n    | InitialSettingsEvent\n    | SettingsChangedEvent\n    | XrSessionStartedEvent\n    | XrSessionEndedEvent\n    | XrFrameEvent\n    | PlayerCountEvent;\n\nexport class EventEmitter extends EventTarget {\n    /**\n     * Dispatch a new event.\n     * @param e event\n     * @returns\n     */\n    public dispatchEvent(e: PixelStreamingEvent): boolean {\n        return super.dispatchEvent(e);\n    }\n\n    /**\n     * Register an event handler.\n     * @param type event name\n     * @param listener event handler function\n     */\n    public addEventListener<\n        T extends PixelStreamingEvent['type'],\n        E extends PixelStreamingEvent & { type: T }\n    >(type: T, listener: (e: Event & E) => void) {\n        super.addEventListener(type, listener);\n    }\n\n    /**\n     * Remove an event handler.\n     * @param type event name\n     * @param listener event handler function\n     */\n    public removeEventListener<\n        T extends PixelStreamingEvent['type'],\n        E extends PixelStreamingEvent & { type: T }\n    >(type: T, listener: (e: Event & E) => void) {\n        super.removeEventListener(type, listener);\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nexport type UnregisterFunction = () => void;\n\nexport class EventListenerTracker {\n    private unregisterCallbacks: UnregisterFunction[];\n\n    constructor() {\n        this.unregisterCallbacks = [];\n    }\n\n    /**\n     * Add a new callback that is executed when unregisterAll is called.\n     * @param callback \n     */\n    addUnregisterCallback(callback: UnregisterFunction) {\n        this.unregisterCallbacks.push(callback);\n    }\n\n    /**\n     * Execute all callbacks and clear the list.\n     */\n    unregisterAll() {\n        for (const callback of this.unregisterCallbacks) {\n            callback();\n        }\n        this.unregisterCallbacks = [];\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\n\n/**\n * Utility function for populate file information from byte buffers.\n */\nexport class FileUtil {\n    /**\n     * Processes a files extension when received over data channel\n     * @param view - the file extension data\n     */\n    static setExtensionFromBytes(view: Uint8Array, file: FileTemplate) {\n        // Reset file if we got a file message and we are not \"receiving\" it yet\n        if (!file.receiving) {\n            file.mimetype = '';\n            file.extension = '';\n            file.receiving = true;\n            file.valid = false;\n            file.size = 0;\n            file.data = [];\n            file.timestampStart = new Date().getTime();\n            Logger.Log(\n                Logger.GetStackTrace(),\n                'Received first chunk of file',\n                6\n            );\n        }\n\n        const extensionAsString = new TextDecoder('utf-16').decode(\n            view.slice(1)\n        );\n        Logger.Log(Logger.GetStackTrace(), extensionAsString, 6);\n        file.extension = extensionAsString;\n    }\n\n    /**\n     * Processes a files mime type when received over data channel\n     * @param view - the file mime type data\n     */\n    static setMimeTypeFromBytes(view: Uint8Array, file: FileTemplate) {\n        // Reset file if we got a file message and we are not \"receiving\" it yet\n        if (!file.receiving) {\n            file.mimetype = '';\n            file.extension = '';\n            file.receiving = true;\n            file.valid = false;\n            file.size = 0;\n            file.data = [];\n            file.timestampStart = new Date().getTime();\n            Logger.Log(\n                Logger.GetStackTrace(),\n                'Received first chunk of file',\n                6\n            );\n        }\n\n        const mimeAsString = new TextDecoder('utf-16').decode(view.slice(1));\n        Logger.Log(Logger.GetStackTrace(), mimeAsString, 6);\n        file.mimetype = mimeAsString;\n    }\n\n    /**\n     * Processes a files contents when received over data channel\n     * @param view - the file contents data\n     */\n    static setContentsFromBytes(view: Uint8Array, file: FileTemplate) {\n        // If we haven't received the initial setup instructions, return\n        if (!file.receiving) return;\n\n        // Extract the total size of the file (across all chunks)\n        file.size = Math.ceil(\n            new DataView(view.slice(1, 5).buffer).getInt32(0, true) /\n                16379 /* The maximum number of payload bits per message*/\n        );\n\n        // Get the file part of the payload\n        const fileBytes = view.slice(1 + 4);\n\n        // Append to existing data that holds the file\n        file.data.push(fileBytes);\n\n        // Uncomment for debug\n        Logger.Log(\n            Logger.GetStackTrace(),\n            `Received file chunk: ${file.data.length}/${file.size}`,\n            6\n        );\n\n        if (file.data.length === file.size) {\n            file.receiving = false;\n            file.valid = true;\n            Logger.Log(Logger.GetStackTrace(), 'Received complete file', 6);\n            const transferDuration = new Date().getTime() - file.timestampStart;\n            const transferBitrate = Math.round(\n                (file.size * 16 * 1024) / transferDuration\n            );\n            Logger.Log(\n                Logger.GetStackTrace(),\n                `Average transfer bitrate: ${transferBitrate}kb/s over ${\n                    transferDuration / 1000\n                } seconds`,\n                6\n            );\n\n            // File reconstruction\n            /**\n             * Example code to reconstruct the file\n             *\n             * This code reconstructs the received data into the original file based on the mime type and extension provided and then downloads the reconstructed file\n             */\n            const received = new Blob(file.data, { type: file.mimetype });\n            const a = document.createElement('a');\n            a.setAttribute('href', URL.createObjectURL(received));\n            a.setAttribute('download', `transfer.${file.extension}`);\n            document.body.append(a);\n            // if you are so inclined to make it auto-download, do something like: a.click();\n            a.remove();\n        } else if (file.data.length > file.size) {\n            file.receiving = false;\n            Logger.Error(\n                Logger.GetStackTrace(),\n                `Received bigger file than advertised: ${file.data.length}/${file.size}`\n            );\n        }\n    }\n}\n\n/**\n * A class that represents a template for a downloaded file\n */\nexport class FileTemplate {\n    mimetype = '';\n    extension = '';\n    receiving = false;\n    size = 0;\n    data: Array<Uint8Array> = [];\n    valid = false;\n    timestampStart: number;\n}\n","export class RTCUtils {\n    static isVideoTransciever(transceiver : RTCRtpTransceiver | undefined) : boolean {\n        return this.canTransceiverReceiveVideo(transceiver) || this.canTransceiverSendVideo(transceiver);\n    }\n\n    static canTransceiverReceiveVideo(transceiver : RTCRtpTransceiver | undefined) : boolean {\n        return !!transceiver &&\n            (transceiver.direction === 'sendrecv' || transceiver.direction === 'recvonly') &&\n            transceiver.receiver &&\n            transceiver.receiver.track &&\n            transceiver.receiver.track.kind === 'video';    \n    }\n\n    static canTransceiverSendVideo(transceiver : RTCRtpTransceiver | undefined) : boolean {\n        return !!transceiver &&\n            (transceiver.direction === 'sendrecv' || transceiver.direction === 'sendonly') &&\n            transceiver.sender &&\n            transceiver.sender.track &&\n            transceiver.sender.track.kind === 'video';    \n    }\n\n    static isAudioTransciever(transceiver : RTCRtpTransceiver | undefined) : boolean {\n        return this.canTransceiverReceiveAudio(transceiver) || this.canTransceiverSendAudio(transceiver);\n    }\n\n    static canTransceiverReceiveAudio(transceiver : RTCRtpTransceiver | undefined) : boolean {\n        return !!transceiver &&\n            (transceiver.direction === 'sendrecv' || transceiver.direction === 'recvonly') &&\n            transceiver.receiver &&\n            transceiver.receiver.track &&\n            transceiver.receiver.track.kind === 'audio';    \n    }\n\n    static canTransceiverSendAudio(transceiver : RTCRtpTransceiver | undefined) : boolean {\n        return !!transceiver &&\n            (transceiver.direction === 'sendrecv' || transceiver.direction === 'sendonly') &&\n            transceiver.sender &&\n            transceiver.sender.track &&\n            transceiver.sender.track.kind === 'audio';\n    }\n}","// Copyright Epic Games, Inc. All Rights Reserved.\n\nexport class WebGLUtils {\n    static vertexShader(): string {\n        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`;\n    }\n\n    static fragmentShader(): string {\n        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`;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nexport class WebXRUtils {\n    /**\n     * Deep copies a gamepad's values by first converting it to a JSON object and then back to a gamepad\n     *\n     * @param gamepad the original gamepad\n     * @returns a new gamepad object, populated with the original gamepads values\n     */\n    static deepCopyGamepad(gamepad: Gamepad): Gamepad {\n        return JSON.parse(\n            JSON.stringify({\n                buttons: gamepad.buttons.map((b) =>\n                    JSON.parse(\n                        JSON.stringify({\n                            pressed: b.pressed,\n                            touched: b.touched\n                        })\n                    )\n                ),\n                axes: gamepad.axes\n            })\n        );\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { MouseController } from '../Inputs/MouseController';\nimport { Logger } from '../Logger/Logger';\nimport { VideoPlayer } from './VideoPlayer';\n\n/**\n * Video Player Controller handles the creation of the video HTML element and all handlers\n */\nexport class StreamController {\n    videoElementProvider: VideoPlayer;\n    audioElement: HTMLAudioElement;\n    mouseController: MouseController;\n\n    /**\n     * @param videoElementProvider Video Player instance\n     */\n    constructor(videoElementProvider: VideoPlayer) {\n        this.videoElementProvider = videoElementProvider;\n        this.audioElement = document.createElement('Audio') as HTMLAudioElement;\n        this.videoElementProvider.setAudioElement(this.audioElement);\n    }\n\n    /**\n     * Handles when the Peer connection has a track event\n     * @param rtcTrackEvent - RTC Track Event\n     */\n    handleOnTrack(rtcTrackEvent: RTCTrackEvent) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'handleOnTrack ' + JSON.stringify(rtcTrackEvent.streams),\n            6\n        );\n        const videoElement = this.videoElementProvider.getVideoElement();\n\n        if (rtcTrackEvent.track) {\n            Logger.Log(\n                Logger.GetStackTrace(),\n                'Got track - ' +\n                    rtcTrackEvent.track.kind +\n                    ' id=' +\n                    rtcTrackEvent.track.id +\n                    ' readyState=' +\n                    rtcTrackEvent.track.readyState,\n                6\n            );\n        }\n\n        if (rtcTrackEvent.track.kind == 'audio') {\n            this.CreateAudioTrack(rtcTrackEvent.streams[0]);\n            return;\n        } else if (\n            rtcTrackEvent.track.kind == 'video' &&\n            videoElement.srcObject !== rtcTrackEvent.streams[0]\n        ) {\n            videoElement.srcObject = rtcTrackEvent.streams[0];\n            Logger.Log(\n                Logger.GetStackTrace(),\n                'Set video source from video track ontrack.'\n            );\n            return;\n        }\n    }\n\n    /**\n     * Creates the audio device when receiving an RTCTrackEvent with the kind of \"audio\"\n     * @param audioMediaStream - Audio Media stream track\n     */\n    CreateAudioTrack(audioMediaStream: MediaStream) {\n        const videoElement = this.videoElementProvider.getVideoElement();\n\n        // do nothing the video has the same media stream as the audio track we have here (they are linked)\n        if (videoElement.srcObject == audioMediaStream) {\n            return;\n        }\n        // video element has some other media stream that is not associated with this audio track\n        else if (\n            videoElement.srcObject &&\n            videoElement.srcObject !== audioMediaStream\n        ) {\n            // create a new audio element\n            this.audioElement.srcObject = audioMediaStream;\n            Logger.Log(\n                Logger.GetStackTrace(),\n                'Created new audio element to play separate audio stream.'\n            );\n        }\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Config, Flags } from '../Config/Config';\nimport { Logger } from '../Logger/Logger';\n\n/**\n * Extra types for the HTMLElement\n */\ndeclare global {\n    interface HTMLElement {\n        mozRequestPointerLock?(): void;\n    }\n}\n\n/**\n * The video player html element\n */\nexport class VideoPlayer {\n    private config: Config;\n    private videoElement: HTMLVideoElement;\n    private audioElement?: HTMLAudioElement;\n    private orientationChangeTimeout: number;\n    private lastTimeResized = new Date().getTime();\n\n    onMatchViewportResolutionCallback: (width: number, height: number) => void;\n    onResizePlayerCallback: () => void;\n    resizeTimeoutHandle: number;\n\n    /**\n     * @param videoElementParent the html div the the video player will be injected into\n     * @param config the applications configuration. We're interested in the startVideoMuted flag\n     */\n    constructor(videoElementParent: HTMLElement, config: Config) {\n        this.videoElement = document.createElement('video');\n        this.config = config;\n        this.videoElement.id = 'streamingVideo';\n        this.videoElement.disablePictureInPicture = true;\n        this.videoElement.playsInline = true;\n        this.videoElement.style.width = '100%';\n        this.videoElement.style.height = '100%';\n        this.videoElement.style.position = 'absolute';\n        this.videoElement.style.pointerEvents = 'all';\n        videoElementParent.appendChild(this.videoElement);\n\n        this.onResizePlayerCallback = () => {\n            console.log(\n                'Resolution changed, restyling player, did you forget to override this function?'\n            );\n        };\n        this.onMatchViewportResolutionCallback = () => {\n            console.log(\n                'Resolution changed and match viewport resolution is turned on, did you forget to override this function?'\n            );\n        };\n\n        // set play for video (and audio)\n        this.videoElement.onclick = () => {\n            if (this.audioElement != undefined && this.audioElement.paused) {\n                this.audioElement.play();\n            }\n            if (this.videoElement.paused) {\n                this.videoElement.play();\n            }\n        };\n\n        this.videoElement.onloadedmetadata = () => {\n            this.onVideoInitialized();\n        };\n\n        // set resize events to the windows if it is resized or its orientation is changed\n        window.addEventListener('resize', () => this.resizePlayerStyle(), true);\n        window.addEventListener('orientationchange', () =>\n            this.onOrientationChange()\n        );\n    }\n\n    public setAudioElement(audioElement: HTMLAudioElement) : void {\n        this.audioElement = audioElement;\n    }\n\n    /**\n     * Sets up the video element with any application config and plays the video element.\n     * @returns A promise for if playing the video was successful or not.\n     */\n    play(): Promise<void> {\n        this.videoElement.muted = this.config.isFlagEnabled(\n            Flags.StartVideoMuted\n        );\n        this.videoElement.autoplay = this.config.isFlagEnabled(\n            Flags.AutoPlayVideo\n        );\n        return this.videoElement.play();\n    }\n\n    /**\n     * @returns True if the video element is paused.\n     */\n    isPaused(): boolean {\n        return this.videoElement.paused;\n    }\n\n    /**\n     * @returns - whether the video element is playing.\n     */\n    isVideoReady(): boolean {\n        return (\n            this.videoElement.readyState !== undefined &&\n            this.videoElement.readyState > 0\n        );\n    }\n\n    /**\n     * @returns True if the video element has a valid video source (srcObject).\n     */\n    hasVideoSource(): boolean {\n        return (\n            this.videoElement.srcObject !== undefined &&\n            this.videoElement.srcObject !== null\n        );\n    }\n\n    /**\n     * Get the current context of the html video element\n     * @returns - the current context of the video element\n     */\n    getVideoElement(): HTMLVideoElement {\n        return this.videoElement;\n    }\n\n    /**\n     * Get the current context of the html video elements parent\n     * @returns - the current context of the video elements parent\n     */\n    getVideoParentElement(): HTMLElement {\n        return this.videoElement.parentElement;\n    }\n\n    /**\n     * Set the Video Elements src object tracks to enable\n     * @param enabled - Enable Tracks on the Src Object\n     */\n    setVideoEnabled(enabled: boolean) {\n        // this is a temporary hack until type scripts video element is updated to reflect the need for tracks on a html video element\n        const videoElement = this.videoElement;\n        (<MediaStream>videoElement.srcObject)\n            .getTracks()\n            .forEach((track: MediaStreamTrack) => (track.enabled = enabled));\n    }\n\n    /**\n     * An override for when the video has been initialized with a srcObject\n     */\n    onVideoInitialized() {\n        // Default Functionality: Do Nothing\n    }\n\n    /**\n     * On the orientation change of a window clear the timeout\n     */\n    onOrientationChange() {\n        clearTimeout(this.orientationChangeTimeout);\n        this.orientationChangeTimeout = window.setTimeout(() => {\n            this.resizePlayerStyle();\n        }, 500);\n    }\n\n    /**\n     * Resizes the player style based on the window height and width\n     * @returns - nil if requirements are satisfied\n     */\n    resizePlayerStyle() {\n        const videoElementParent = this.getVideoParentElement();\n\n        if (!videoElementParent) {\n            return;\n        }\n\n        this.updateVideoStreamSize();\n\n        if (videoElementParent.classList.contains('fixed-size')) {\n            this.onResizePlayerCallback();\n            return;\n        }\n\n        // controls for resizing the player\n        this.resizePlayerStyleToFillParentElement();\n        this.onResizePlayerCallback();\n    }\n\n    /**\n     * Resizes the player element to fill the parent element\n     */\n    resizePlayerStyleToFillParentElement() {\n        const videoElementParent = this.getVideoParentElement();\n\n        //Video is not initialized yet so set videoElementParent to size of parent element\n        const styleWidth = '100%';\n        const styleHeight = '100%';\n        const styleTop = 0;\n        const styleLeft = 0;\n        videoElementParent.setAttribute(\n            'style',\n            'top: ' +\n                styleTop +\n                'px; left: ' +\n                styleLeft +\n                'px; width: ' +\n                styleWidth +\n                '; height: ' +\n                styleHeight +\n                '; cursor: default;'\n        );\n    }\n\n    updateVideoStreamSize() {\n        if (!this.config.isFlagEnabled(Flags.MatchViewportResolution)) {\n            return;\n        }\n\n        const now = new Date().getTime();\n        if (now - this.lastTimeResized > 300) {\n            const videoElementParent = this.getVideoParentElement();\n            if (!videoElementParent) {\n                return;\n            }\n\n            this.onMatchViewportResolutionCallback(\n                videoElementParent.clientWidth,\n                videoElementParent.clientHeight\n            );\n\n            this.lastTimeResized = new Date().getTime();\n        } else {\n            Logger.Log(\n                Logger.GetStackTrace(),\n                'Resizing too often - skipping',\n                6\n            );\n            clearTimeout(this.resizeTimeoutHandle);\n            this.resizeTimeoutHandle = window.setTimeout(\n                () => this.updateVideoStreamSize(),\n                100\n            );\n        }\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { WebSocketController } from '../WebSockets/WebSocketController';\nimport { StreamController } from '../VideoPlayer/StreamController';\nimport {\n    MessageAnswer,\n    MessageOffer,\n    MessageConfig,\n    MessageStreamerList\n} from '../WebSockets/MessageReceive';\nimport { FreezeFrameController } from '../FreezeFrame/FreezeFrameController';\nimport { AFKController } from '../AFK/AFKController';\nimport { DataChannelController } from '../DataChannel/DataChannelController';\nimport { PeerConnectionController } from '../PeerConnectionController/PeerConnectionController';\nimport { KeyboardController } from '../Inputs/KeyboardController';\nimport { AggregatedStats } from '../PeerConnectionController/AggregatedStats';\nimport {\n    Config,\n    Flags,\n    ControlSchemeType,\n    TextParameters,\n    OptionParameters,\n    NumericParameters\n} from '../Config/Config';\nimport {\n    EncoderSettings,\n    InitialSettings,\n    WebRTCSettings\n} from '../DataChannel/InitialSettings';\nimport { LatencyTestResults } from '../DataChannel/LatencyTestResults';\nimport { Logger } from '../Logger/Logger';\nimport { FileTemplate, FileUtil } from '../Util/FileUtil';\nimport { InputClassesFactory } from '../Inputs/InputClassesFactory';\nimport { VideoPlayer } from '../VideoPlayer/VideoPlayer';\nimport {\n    StreamMessageController,\n    MessageDirection\n} from '../UeInstanceMessage/StreamMessageController';\nimport { ResponseController } from '../UeInstanceMessage/ResponseController';\nimport * as MessageReceive from '../WebSockets/MessageReceive';\nimport { MessageOnScreenKeyboard } from '../WebSockets/MessageReceive';\nimport { SendMessageController } from '../UeInstanceMessage/SendMessageController';\nimport { ToStreamerMessagesController } from '../UeInstanceMessage/ToStreamerMessagesController';\nimport { MouseController } from '../Inputs/MouseController';\nimport { GamePadController } from '../Inputs/GamepadController';\nimport { DataChannelSender } from '../DataChannel/DataChannelSender';\nimport {\n    CoordinateConverter,\n    UnquantizedDenormalizedUnsignedCoord\n} from '../Util/CoordinateConverter';\nimport { PixelStreaming } from '../PixelStreaming/PixelStreaming';\nimport { ITouchController } from '../Inputs/ITouchController';\nimport {\n    DataChannelCloseEvent,\n    DataChannelErrorEvent,\n    DataChannelOpenEvent,\n    HideFreezeFrameEvent,\n    LoadFreezeFrameEvent,\n    PlayStreamErrorEvent,\n    PlayStreamEvent,\n    PlayStreamRejectedEvent,\n    StreamerListMessageEvent\n} from '../Util/EventEmitter';\nimport {\n    DataChannelLatencyTestRequest,\n    DataChannelLatencyTestResponse\n} from \"../DataChannel/DataChannelLatencyTestResults\";\n/**\n * Entry point for the WebRTC Player\n */\nexport class WebRtcPlayerController {\n    config: Config;\n    responseController: ResponseController;\n    sdpConstraints: RTCOfferOptions;\n    webSocketController: WebSocketController;\n    // The primary data channel. This is bidirectional when p2p and send only when using an SFU\n    sendrecvDataChannelController: DataChannelController;\n    // A recv only data channel required when using an SFU\n    recvDataChannelController: DataChannelController;\n    dataChannelSender: DataChannelSender;\n    datachannelOptions: RTCDataChannelInit;\n    videoPlayer: VideoPlayer;\n    streamController: StreamController;\n    peerConnectionController: PeerConnectionController;\n    inputClassesFactory: InputClassesFactory;\n    freezeFrameController: FreezeFrameController;\n    shouldShowPlayOverlay = true;\n    afkController: AFKController;\n    videoElementParentClientRect: DOMRect;\n    latencyStartTime: number;\n    pixelStreaming: PixelStreaming;\n    streamMessageController: StreamMessageController;\n    sendMessageController: SendMessageController;\n    toStreamerMessagesController: ToStreamerMessagesController;\n    keyboardController: KeyboardController;\n    mouseController: MouseController;\n    touchController: ITouchController;\n    gamePadController: GamePadController;\n    coordinateConverter: CoordinateConverter;\n    isUsingSFU: boolean;\n    isQualityController: boolean;\n    statsTimerHandle: number;\n    file: FileTemplate;\n    preferredCodec: string;\n    peerConfig: RTCConfiguration;\n    videoAvgQp: number;\n    locallyClosed: boolean;\n    shouldReconnect: boolean;\n    isReconnecting: boolean;\n    reconnectAttempt: number;\n    disconnectMessage: string;\n    subscribedStream: string;\n    signallingUrlBuilder: () => string;\n    autoJoinTimer: ReturnType<typeof setTimeout> = undefined;\n\n    /**\n     *\n     * @param config - the frontend config object\n     * @param pixelStreaming - the PixelStreaming object\n     */\n    constructor(config: Config, pixelStreaming: PixelStreaming) {\n        this.config = config;\n        this.pixelStreaming = pixelStreaming;\n        this.responseController = new ResponseController();\n        this.file = new FileTemplate();\n\n        this.sdpConstraints = {\n            offerToReceiveAudio: true,\n            offerToReceiveVideo: true\n        };\n\n        // set up the afk logic class and connect up its method for closing the signaling server\n        this.afkController = new AFKController(\n            this.config,\n            this.pixelStreaming,\n            this.onAfkTriggered.bind(this)\n        );\n        this.afkController.onAFKTimedOutCallback = () => {\n            this.closeSignalingServer('You have been disconnected due to inactivity');\n        };\n\n        this.freezeFrameController = new FreezeFrameController(\n            this.pixelStreaming.videoElementParent\n        );\n\n        this.videoPlayer = new VideoPlayer(\n            this.pixelStreaming.videoElementParent,\n            this.config\n        );\n        this.videoPlayer.onVideoInitialized = () =>\n            this.handleVideoInitialized();\n\n        // When in match viewport resolution mode, when the browser viewport is resized we send a resize command back to UE.\n        this.videoPlayer.onMatchViewportResolutionCallback = (\n            width: number,\n            height: number\n        ) => {\n            const descriptor = {\n                'Resolution.Width': width,\n                'Resolution.Height': height\n            };\n\n            this.streamMessageController.toStreamerHandlers.get(\n                'Command'\n            )([JSON.stringify(descriptor)]);\n        };\n\n        // Every time video player is resized in browser we need to reinitialize the mouse coordinate conversion and freeze frame sizing logic.\n        this.videoPlayer.onResizePlayerCallback = () => {\n            this.setUpMouseAndFreezeFrame();\n        };\n\n        this.streamController = new StreamController(this.videoPlayer);\n\n        this.coordinateConverter = new CoordinateConverter(this.videoPlayer);\n\n        this.sendrecvDataChannelController = new DataChannelController();\n        this.recvDataChannelController = new DataChannelController();\n        this.registerDataChannelEventEmitters(\n            this.sendrecvDataChannelController\n        );\n        this.registerDataChannelEventEmitters(this.recvDataChannelController);\n        this.dataChannelSender = new DataChannelSender(\n            this.sendrecvDataChannelController\n        );\n        this.dataChannelSender.resetAfkWarningTimerOnDataSend = () =>\n            this.afkController.resetAfkWarningTimer();\n\n        this.streamMessageController = new StreamMessageController();\n\n        // set up websocket methods\n        this.webSocketController = new WebSocketController();\n        this.webSocketController.onConfig = (\n            messageConfig: MessageReceive.MessageConfig\n        ) => this.handleOnConfigMessage(messageConfig);\n        this.webSocketController.onStreamerList = (\n            messageList: MessageReceive.MessageStreamerList\n        ) => this.handleStreamerListMessage(messageList);\n        this.webSocketController.onPlayerCount = (playerCount: MessageReceive.MessagePlayerCount) => { \n            this.pixelStreaming._onPlayerCount(playerCount.count); \n        };\n        this.webSocketController.onOpen.addEventListener('open', () => {\n            const BrowserSendsOffer = this.config.isFlagEnabled(\n                Flags.BrowserSendOffer\n            );\n            if(!BrowserSendsOffer)\n            {\n                this.webSocketController.requestStreamerList();\n            }\n        });\n        this.webSocketController.onClose.addEventListener('close', (event : CustomEvent) => {\n            // when we refresh the page during a stream we get the going away code.\n            // in that case we don't want to reconnect since we're navigating away.\n            // https://developer.mozilla.org/en-US/docs/Web/API/CloseEvent/code\n            // lists all the codes. \n            const CODE_GOING_AWAY = 1001;\n\n            const willTryReconnect = this.shouldReconnect\n               && event.detail.code != CODE_GOING_AWAY\n               && this.config.getNumericSettingValue(NumericParameters.MaxReconnectAttempts) > 0\n\n            const disconnectMessage = this.disconnectMessage ? this.disconnectMessage : event.detail.reason;\n            this.pixelStreaming._onDisconnect(disconnectMessage, !willTryReconnect && !this.isReconnecting);\n\n            this.afkController.stopAfkWarningTimer();\n\n            // stop sending stats on interval if we have closed our connection\n            if (this.statsTimerHandle && this.statsTimerHandle !== undefined) {\n                window.clearInterval(this.statsTimerHandle);\n            }\n\n            // reset the stream quality icon.\n            this.setVideoEncoderAvgQP(0);\n\n            // unregister all input device event handlers on disconnect\n            this.setTouchInputEnabled(false);\n            this.setMouseInputEnabled(false);\n            this.setKeyboardInputEnabled(false);\n            this.setGamePadInputEnabled(false);\n\n            if (willTryReconnect) {\n                // need a small delay here to prevent reconnect spamming\n                setTimeout(() => {\n                    this.isReconnecting = true;\n                    this.reconnectAttempt++;\n                    this.tryReconnect(event.detail.reason);\n                }, 2000);\n            }\n        });\n\n        // set up the final webRtc player controller methods from within our application so a connection can be activated\n        this.sendMessageController = new SendMessageController(\n            this.dataChannelSender,\n            this.streamMessageController\n        );\n        this.toStreamerMessagesController = new ToStreamerMessagesController(\n            this.sendMessageController\n        );\n        this.registerMessageHandlers();\n        this.streamMessageController.populateDefaultProtocol();\n\n        this.inputClassesFactory = new InputClassesFactory(\n            this.streamMessageController,\n            this.videoPlayer,\n            this.coordinateConverter\n        );\n\n        this.isUsingSFU = false;\n        this.isQualityController = false;\n        this.preferredCodec = '';\n        this.shouldReconnect = true;\n        this.isReconnecting = false;\n        this.reconnectAttempt = 0;\n\n        this.config._addOnOptionSettingChangedListener(\n            OptionParameters.StreamerId,\n            (streamerid) => {\n                if(streamerid === \"\") {\n                    return;\n                }\n\n                // close the current peer connection and create a new one\n                this.peerConnectionController.peerConnection.close();\n                this.peerConnectionController.createPeerConnection(\n                    this.peerConfig,\n                    this.preferredCodec\n                );\n                this.subscribedStream = streamerid;\n                this.webSocketController.sendSubscribe(streamerid);\n            }\n        );\n\n        this.setVideoEncoderAvgQP(-1);\n\n        this.signallingUrlBuilder =  () => {\n            let signallingServerUrl = this.config.getTextSettingValue(\n                TextParameters.SignallingServerUrl\n            );\n    \n            // If we are connecting to the SFU add a special url parameter to the url\n            if (this.config.isFlagEnabled(Flags.BrowserSendOffer)) {\n                signallingServerUrl += '?' + Flags.BrowserSendOffer + '=true';\n            }\n    \n            // This code is no longer needed, but is a good example for how subsequent config flags can be appended\n            // if (this.config.isFlagEnabled(Flags.BrowserSendOffer)) {\n            //     signallingServerUrl += (signallingServerUrl.includes('?') ? '&' : '?') + Flags.BrowserSendOffer + '=true';\n            // }\n    \n            return signallingServerUrl;\n        }\n    }\n\n    /**\n     * Make a request to UnquantizedAndDenormalizeUnsigned coordinates\n     * @param x x axis coordinate\n     * @param y y axis coordinate\n     */\n    requestUnquantizedAndDenormalizeUnsigned(\n        x: number,\n        y: number\n    ): UnquantizedDenormalizedUnsignedCoord {\n        return this.coordinateConverter.unquantizeAndDenormalizeUnsigned(x, y);\n    }\n\n    /**\n     * Handles when a message is received\n     * @param event - Message Event\n     */\n    handleOnMessage(event: MessageEvent) {\n        const message = new Uint8Array(event.data);\n        Logger.Log(Logger.GetStackTrace(), 'Message incoming:' + message, 6);\n\n        //try {\n        const messageType =\n            this.streamMessageController.fromStreamerMessages.get(\n                message[0]\n            );\n        this.streamMessageController.fromStreamerHandlers.get(messageType)(\n            event.data\n        );\n        //} catch (e) {\n        //Logger.Error(Logger.GetStackTrace(), `Custom data channel message with message type that is unknown to the Pixel Streaming protocol. Does your PixelStreamingProtocol need updating? The message type was: ${message[0]}`);\n        //}\n    }\n\n    /**\n     * Register message all handlers\n     */\n    registerMessageHandlers() {\n        // From Streamer\n        // Message events from the streamer have a data type of ArrayBuffer as we force this type in the DatachannelController\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'QualityControlOwnership',\n            (data: ArrayBuffer) => this.onQualityControlOwnership(data)\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'Response',\n            (data: ArrayBuffer) => this.responseController.onResponse(data)\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'Command',\n            (data: ArrayBuffer) => {\n                this.onCommand(data);\n            }\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'FreezeFrame',\n            (data: ArrayBuffer) => this.onFreezeFrameMessage(data)\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'UnfreezeFrame',\n            () => this.invalidateFreezeFrameAndEnableVideo()\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'VideoEncoderAvgQP',\n            (data: ArrayBuffer) => this.handleVideoEncoderAvgQP(data)\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'LatencyTest',\n            (data: ArrayBuffer) => this.handleLatencyTestResult(data)\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'DataChannelLatencyTest',\n            (data: ArrayBuffer) => this.handleDataChannelLatencyTestResponse(data)\n        )\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'InitialSettings',\n            (data: ArrayBuffer) => this.handleInitialSettings(data)\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'FileExtension',\n            (data: ArrayBuffer) => this.onFileExtension(data)\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'FileMimeType',\n            (data: ArrayBuffer) => this.onFileMimeType(data)\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'FileContents',\n            (data: ArrayBuffer) => this.onFileContents(data)\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'TestEcho',\n            () => {\n                /* Do nothing */\n            }\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'InputControlOwnership',\n            (data: ArrayBuffer) => this.onInputControlOwnership(data)\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'GamepadResponse',\n            (data: ArrayBuffer) => this.onGamepadResponse(data)\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.FromStreamer,\n            'Protocol',\n            (data: ArrayBuffer) => this.onProtocolMessage(data)\n        );\n\n        // To Streamer\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'IFrameRequest',\n            () =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'IFrameRequest'\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'RequestQualityControl',\n            () =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'RequestQualityControl'\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'FpsRequest',\n            () => this.sendMessageController.sendMessageToStreamer('FpsRequest')\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'AverageBitrateRequest',\n            () =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'AverageBitrateRequest'\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'StartStreaming',\n            () =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'StartStreaming'\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'StopStreaming',\n            () =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'StopStreaming'\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'LatencyTest',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'LatencyTest', data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'RequestInitialSettings',\n            () =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'RequestInitialSettings'\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'TestEcho',\n            () => {\n                /* Do nothing */\n            }\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'UIInteraction',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'UIInteraction', data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'Command',\n            (data: Array<number | string>) => \n                this.sendMessageController.sendMessageToStreamer(\n                    'Command', data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'TextboxEntry',\n            (data: Array<number | string>) => \n                this.sendMessageController.sendMessageToStreamer(\n                    'TextboxEntry', data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'KeyDown',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'KeyDown',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'KeyUp',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer('KeyUp', data)\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'KeyPress',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'KeyPress',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'MouseEnter',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'MouseEnter',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'MouseLeave',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'MouseLeave',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'MouseDown',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'MouseDown',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'MouseUp',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'MouseUp',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'MouseMove',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'MouseMove',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'MouseWheel',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'MouseWheel',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'MouseDouble',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'MouseDouble',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'TouchStart',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'TouchStart',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'TouchEnd',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'TouchEnd',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'TouchMove',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'TouchMove',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'GamepadConnected',\n            () =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'GamepadConnected'\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'GamepadButtonPressed',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'GamepadButtonPressed',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'GamepadButtonReleased',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'GamepadButtonReleased',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'GamepadAnalog',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'GamepadAnalog',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'GamepadDisconnected',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'GamepadDisconnected',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'XRHMDTransform',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'XRHMDTransform',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'XRControllerTransform',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'XRControllerTransform',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'XRSystem',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'XRSystem',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'XRButtonTouched',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'XRButtonTouched',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'XRButtonPressed',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'XRButtonPressed',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'XRButtonReleased',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'XRButtonReleased',\n                    data\n                )\n        );\n        this.streamMessageController.registerMessageHandler(\n            MessageDirection.ToStreamer,\n            'XRAnalog',\n            (data: Array<number | string>) =>\n                this.sendMessageController.sendMessageToStreamer(\n                    'XRAnalog',\n                    data\n                )\n        );\n    }\n\n    /**\n     * Activate the logic associated with a command from UE\n     * @param message\n     */\n    onCommand(message: ArrayBuffer) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'DataChannelReceiveMessageType.Command',\n            6\n        );\n        const commandAsString = new TextDecoder('utf-16').decode(\n            message.slice(1)\n        );\n\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'Data Channel Command: ' + commandAsString,\n            6\n        );\n        const command: MessageOnScreenKeyboard = JSON.parse(commandAsString);\n        if (command.command === 'onScreenKeyboard') {\n            this.pixelStreaming._activateOnScreenKeyboard(command);\n        }\n    }\n\n    /**\n     * Handles a protocol message received from the streamer\n     * @param message the message data from the streamer\n     */\n    onProtocolMessage(message: ArrayBuffer) {\n        try {\n            const protocolString = new TextDecoder('utf-16').decode(\n                message.slice(1)\n            );\n            const protocolJSON = JSON.parse(protocolString);\n            if (\n                !Object.prototype.hasOwnProperty.call(protocolJSON, 'Direction')\n            ) {\n                Logger.Error(\n                    Logger.GetStackTrace(),\n                    'Malformed protocol received. Ensure the protocol message contains a direction'\n                );\n            }\n            const direction = protocolJSON.Direction;\n            delete protocolJSON.Direction;\n            Logger.Log(\n                Logger.GetStackTrace(),\n                `Received new ${\n                    direction == MessageDirection.FromStreamer\n                        ? 'FromStreamer'\n                        : 'ToStreamer'\n                } protocol. Updating existing protocol...`\n            );\n            Object.keys(protocolJSON).forEach((messageType) => {\n                const message = protocolJSON[messageType];\n                switch (direction) {\n                    case MessageDirection.ToStreamer:\n                        // Check that the message contains all the relevant params\n                        if (\n                            !Object.prototype.hasOwnProperty.call(\n                                message,\n                                'id'\n                            )\n                        ) {\n                            Logger.Error(\n                                Logger.GetStackTrace(),\n                                `ToStreamer->${messageType} protocol definition was malformed as it didn't contain at least an id\\n\n                                           Definition was: ${JSON.stringify(\n                                               message,\n                                               null,\n                                               2\n                                           )}`\n                            );\n                            // return in a forEach is equivalent to a continue in a normal for loop\n                            return;\n                        }\n\n                        // UE5.1 and UE5.2 don't send a structure for these message types, but they actually do have a structure so ignore updating them\n                        if((messageType === \"UIInteraction\" || messageType === \"Command\" || messageType === \"LatencyTest\")) {\n                            return;\n                        }\n\n                        if (\n                            this.streamMessageController.toStreamerHandlers.get(\n                                messageType\n                            )\n                        ) {\n                            // If we've registered a handler for this message type we can add it to our supported messages. ie registerMessageHandler(...)\n                            this.streamMessageController.toStreamerMessages.set(\n                                messageType,\n                                message\n                            );\n                        } else {\n                            Logger.Error(\n                                Logger.GetStackTrace(),\n                                `There was no registered handler for \"${messageType}\" - try adding one using registerMessageHandler(MessageDirection.ToStreamer, \"${messageType}\", myHandler)`\n                            );\n                        }\n                        break;\n                    case MessageDirection.FromStreamer:\n                        // Check that the message contains all the relevant params\n                        if (\n                            !Object.prototype.hasOwnProperty.call(message, 'id')\n                        ) {\n                            Logger.Error(\n                                Logger.GetStackTrace(),\n                                `FromStreamer->${messageType} protocol definition was malformed as it didn't contain at least an id\\n\n                            Definition was: ${JSON.stringify(message, null, 2)}`\n                            );\n                            // return in a forEach is equivalent to a continue in a normal for loop\n                            return;\n                        }\n                        if (\n                            this.streamMessageController.fromStreamerHandlers.get(\n                                messageType\n                            )\n                        ) {\n                            // If we've registered a handler for this message type. ie registerMessageHandler(...)\n                            this.streamMessageController.fromStreamerMessages.set(\n                                message.id,\n                                messageType\n                            );\n                        } else {\n                            Logger.Error(\n                                Logger.GetStackTrace(),\n                                `There was no registered handler for \"${message}\" - try adding one using registerMessageHandler(MessageDirection.FromStreamer, \"${messageType}\", myHandler)`\n                            );\n                        }\n                        break;\n                    default:\n                        Logger.Error(\n                            Logger.GetStackTrace(),\n                            `Unknown direction: ${direction}`\n                        );\n                }\n            });\n\n            // Once the protocol has been received, we can send our control messages\n            this.toStreamerMessagesController.SendRequestInitialSettings();\n            this.toStreamerMessagesController.SendRequestQualityControl();\n        } catch (e) {\n            Logger.Log(Logger.GetStackTrace(), e);\n        }\n    }\n\n    /**\n     * Handles an input control message when it is received from the streamer\n     * @param message The input control message\n     */\n    onInputControlOwnership(message: ArrayBuffer) {\n        const view = new Uint8Array(message);\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'DataChannelReceiveMessageType.InputControlOwnership',\n            6\n        );\n        const inputControlOwnership = new Boolean(view[1]).valueOf();\n        Logger.Log(\n            Logger.GetStackTrace(),\n            `Received input controller message - will your input control the stream: ${inputControlOwnership}`\n        );\n        this.pixelStreaming._onInputControlOwnership(inputControlOwnership);\n    }\n\n    /**\n     * \n     * @param message \n     */\n    onGamepadResponse(message: ArrayBuffer) {\n        const responseString = new TextDecoder('utf-16').decode(message.slice(1));\n        const responseJSON = JSON.parse(responseString);\n        this.gamePadController.onGamepadResponseReceived(responseJSON.controllerId);\n    }\n\n    onAfkTriggered(): void {\n        this.afkController.onAfkClick();\n\n        // if the stream is paused play it, if we can\n        if (this.videoPlayer.isPaused() && this.videoPlayer.hasVideoSource()) {\n            this.playStream();\n        }\n    }\n\n    /**\n     * Set whether we should timeout when afk.\n     * @param afkEnabled If true we timeout when idle for some given amount of time.\n     */\n    setAfkEnabled(afkEnabled: boolean): void {\n        if (afkEnabled) {\n            this.onAfkTriggered();\n        } else {\n            this.afkController.stopAfkWarningTimer();\n        }\n    }\n\n    /**\n     * Attempt a reconnection to the signalling server\n     */\n    tryReconnect(message: string) {\n        // if there is no webSocketController return immediately or this will not work\n        if (!this.webSocketController) {\n            Logger.Log(\n                Logger.GetStackTrace(),\n                'The Web Socket Controller does not exist so this will not work right now.'\n            );\n            return;\n        }\n\n        // if the connection is open, first close it. wait some time and try again.\n        this.isReconnecting = true;\n        if (this.webSocketController.webSocket && this.webSocketController.webSocket.readyState != WebSocket.CLOSED) {\n            this.closeSignalingServer(`${message} Restarting stream...`);\n            setTimeout(() => {\n                this.tryReconnect(message);\n            }, 3000);\n        } else {\n            this.pixelStreaming._onWebRtcAutoConnect();\n            this.connectToSignallingServer();\n        }\n    }\n\n    /**\n     * Loads a freeze frame if it is required otherwise shows the play overlay\n     */\n    loadFreezeFrameOrShowPlayOverlay() {\n        this.pixelStreaming.dispatchEvent(\n            new LoadFreezeFrameEvent({\n                shouldShowPlayOverlay: this.shouldShowPlayOverlay,\n                isValid: this.freezeFrameController.valid,\n                jpegData: this.freezeFrameController.jpeg\n            })\n        );\n        if (this.shouldShowPlayOverlay === true) {\n            Logger.Log(Logger.GetStackTrace(), 'showing play overlay');\n            this.resizePlayerStyle();\n        } else {\n            Logger.Log(Logger.GetStackTrace(), 'showing freeze frame');\n            this.freezeFrameController.showFreezeFrame();\n        }\n        setTimeout(() => {\n            this.videoPlayer.setVideoEnabled(false);\n        }, this.freezeFrameController.freezeFrameDelay);\n    }\n\n    /**\n     * Process the freeze frame and load it\n     * @param message The freeze frame data in bytes\n     */\n    onFreezeFrameMessage(message: ArrayBuffer) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'DataChannelReceiveMessageType.FreezeFrame',\n            6\n        );\n        const view = new Uint8Array(message);\n        this.freezeFrameController.processFreezeFrameMessage(view, () =>\n            this.loadFreezeFrameOrShowPlayOverlay()\n        );\n    }\n\n    /**\n     * Enable the video after hiding a freeze frame\n     */\n    invalidateFreezeFrameAndEnableVideo() {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'DataChannelReceiveMessageType.FreezeFrame',\n            6\n        );\n        setTimeout(() => {\n            this.pixelStreaming.dispatchEvent(\n                new HideFreezeFrameEvent()\n            );\n            this.freezeFrameController.hideFreezeFrame();\n        }, this.freezeFrameController.freezeFrameDelay);\n        if (this.videoPlayer.getVideoElement()) {\n            this.videoPlayer.setVideoEnabled(true);\n        }\n    }\n\n    /**\n     * Prep datachannel data for processing file extension\n     * @param data the file extension data\n     */\n    onFileExtension(data: ArrayBuffer) {\n        const view = new Uint8Array(data);\n        FileUtil.setExtensionFromBytes(view, this.file);\n    }\n\n    /**\n     * Prep datachannel data for processing the file mime type\n     * @param data the file mime type data\n     */\n    onFileMimeType(data: ArrayBuffer) {\n        const view = new Uint8Array(data);\n        FileUtil.setMimeTypeFromBytes(view, this.file);\n    }\n\n    /**\n     * Prep datachannel data for processing the file contents\n     * @param data the file contents data\n     */\n    onFileContents(data: ArrayBuffer) {\n        const view = new Uint8Array(data);\n        FileUtil.setContentsFromBytes(view, this.file);\n    }\n\n    /**\n     * Plays the stream audio and video source and sets up other pieces while the stream starts\n     */\n    playStream() {\n        if (!this.videoPlayer.getVideoElement()) {\n            const message =\n                'Could not play video stream because the video player was not initialized correctly.';\n            this.pixelStreaming.dispatchEvent(\n                new PlayStreamErrorEvent({ message })\n            );\n            Logger.Error(Logger.GetStackTrace(), message);\n\n            // close the connection\n            this.closeSignalingServer('Stream not initialized correctly');\n            return;\n        }\n\n        if (!this.videoPlayer.hasVideoSource()) {\n            Logger.Warning(\n                Logger.GetStackTrace(),\n                'Cannot play stream, the video element has no srcObject to play.'\n            );\n            return;\n        }\n\n        this.setTouchInputEnabled(this.config.isFlagEnabled(Flags.TouchInput));\n        this.pixelStreaming.dispatchEvent(new PlayStreamEvent());\n\n        if (this.streamController.audioElement.srcObject) {\n            const startMuted = this.config.isFlagEnabled(Flags.StartVideoMuted)\n            this.streamController.audioElement.muted = startMuted;\n\n            if (startMuted) {\n              this.playVideo();\n            } else {\n                this.streamController.audioElement\n                    .play()\n                    .then(() => {\n                        this.playVideo();\n                    })\n                    .catch((onRejectedReason) => {\n                        Logger.Log(Logger.GetStackTrace(), onRejectedReason);\n                        Logger.Log(\n                            Logger.GetStackTrace(),\n                            'Browser does not support autoplaying video without interaction - to resolve this we are going to show the play button overlay.'\n                        );\n                        this.pixelStreaming.dispatchEvent(\n                            new PlayStreamRejectedEvent({\n                                reason: onRejectedReason\n                            })\n                        );\n                    });\n            }\n        } else {\n            this.playVideo();\n        }\n\n        this.shouldShowPlayOverlay = false;\n        this.freezeFrameController.showFreezeFrame();\n    }\n\n    /**\n     * Plays the video stream\n     */\n    private playVideo() {\n        // handle play() with promise as it is an asynchronous call\n        this.videoPlayer.play().catch((onRejectedReason: string) => {\n            if (this.streamController.audioElement.srcObject) {\n                this.streamController.audioElement.pause();\n            }\n            Logger.Log(Logger.GetStackTrace(), onRejectedReason);\n            Logger.Log(\n                Logger.GetStackTrace(),\n                'Browser does not support autoplaying video without interaction - to resolve this we are going to show the play button overlay.'\n            );\n            this.pixelStreaming.dispatchEvent(\n                new PlayStreamRejectedEvent({ reason: onRejectedReason })\n            );\n        });\n    }\n\n    /**\n     * Enable the video to play automatically if enableAutoplay is true\n     */\n    autoPlayVideoOrSetUpPlayOverlay() {\n        if (this.config.isFlagEnabled(Flags.AutoPlayVideo)) {\n            // attempt to play the video\n            this.playStream();\n        }\n        this.resizePlayerStyle();\n    }\n\n    /**\n     * Connect to the Signaling server\n     */\n    connectToSignallingServer() {\n        this.locallyClosed = false;\n        this.shouldReconnect = true;\n        this.disconnectMessage = null;\n        const signallingUrl = this.signallingUrlBuilder();\n        this.webSocketController.connect(signallingUrl);\n    }\n\n    /**\n     * This will start the handshake to the signalling server\n     * @param peerConfig  - RTC Configuration Options from the Signaling server\n     * @remark RTC Peer Connection on Ice Candidate event have it handled by handle Send Ice Candidate\n     */\n    startSession(peerConfig: RTCConfiguration) {\n        this.peerConfig = peerConfig;\n        // check for forcing turn\n        if (this.config.isFlagEnabled(Flags.ForceTURN)) {\n            // check for a turn server\n            const hasTurnServer = this.checkTurnServerAvailability(peerConfig);\n\n            // close and error if turn is forced and there is no turn server\n            if (!hasTurnServer) {\n                Logger.Info(\n                    Logger.GetStackTrace(),\n                    'No turn server was found in the Peer Connection Options. TURN cannot be forced, closing connection. Please use STUN instead'\n                );\n                this.closeSignalingServer('TURN cannot be forced, closing connection. Please use STUN instead.');\n                return;\n            }\n        }\n\n        // set up the peer connection controller\n        this.peerConnectionController = new PeerConnectionController(\n            this.peerConfig,\n            this.config,\n            this.preferredCodec\n        );\n\n        // set up peer connection controller video stats\n        this.peerConnectionController.onVideoStats = (event: AggregatedStats) =>\n            this.handleVideoStats(event);\n\n        /* When the Peer Connection wants to send an offer have it handled */\n        this.peerConnectionController.onSendWebRTCOffer = (\n            offer: RTCSessionDescriptionInit\n        ) => this.handleSendWebRTCOffer(offer);\n\n        /* When the Peer Connection wants to send an answer have it handled */\n        this.peerConnectionController.onSendWebRTCAnswer = (\n            offer: RTCSessionDescriptionInit\n        ) => this.handleSendWebRTCAnswer(offer);\n\n        /* When the Peer Connection ice candidate is added have it handled */\n        this.peerConnectionController.onPeerIceCandidate = (\n            peerConnectionIceEvent: RTCPeerConnectionIceEvent\n        ) => this.handleSendIceCandidate(peerConnectionIceEvent);\n\n        /* When the Peer Connection has a data channel created for it by the browser, handle it */\n        this.peerConnectionController.onDataChannel = (\n            datachannelEvent: RTCDataChannelEvent\n        ) => this.handleDataChannel(datachannelEvent);\n\n        // set up webRtc text overlays\n        this.peerConnectionController.showTextOverlayConnecting = () =>\n            this.pixelStreaming._onWebRtcConnecting();\n        this.peerConnectionController.showTextOverlaySetupFailure = () =>\n            this.pixelStreaming._onWebRtcFailed();\n        let webRtcConnectedSent = false;\n        this.peerConnectionController.onIceConnectionStateChange = () => {\n            // Browsers emit \"connected\" when getting first connection and \"completed\" when finishing\n            // candidate checking. However, sometimes browsers can skip \"connected\" and only emit \"completed\".\n            // Therefore need to check both cases and emit onWebRtcConnected only once on the first hit.\n            if (!webRtcConnectedSent && \n                [\"connected\", \"completed\"].includes(this.peerConnectionController.peerConnection.iceConnectionState)) {\n                this.pixelStreaming._onWebRtcConnected();\n                webRtcConnectedSent = true;\n            }\n        };\n\n        /* RTC Peer Connection on Track event -> handle on track */\n        this.peerConnectionController.onTrack = (trackEvent: RTCTrackEvent) =>\n            this.streamController.handleOnTrack(trackEvent);\n\n        /* Start the Hand shake process by creating an Offer */\n        const BrowserSendsOffer = this.config.isFlagEnabled(\n            Flags.BrowserSendOffer\n        );\n        if (BrowserSendsOffer) {\n            // If browser is sending the offer, create an offer and send it to the streamer\n            this.sendrecvDataChannelController.createDataChannel(\n                this.peerConnectionController.peerConnection,\n                'cirrus',\n                this.datachannelOptions\n            );\n            this.sendrecvDataChannelController.handleOnMessage = (\n                ev: MessageEvent<ArrayBuffer>\n            ) => this.handleOnMessage(ev);\n            this.peerConnectionController.createOffer(\n                this.sdpConstraints,\n                this.config\n            );\n        }\n    }\n\n    /**\n     * Checks the peer connection options for a turn server and returns true or false\n     */\n    checkTurnServerAvailability(options: RTCConfiguration) {\n        // if iceServers is empty return false this should not be the general use case but is here incase\n        if (!options.iceServers) {\n            Logger.Info(Logger.GetStackTrace(), 'A turn sever was not found');\n            return false;\n        }\n\n        // loop through the ice servers to check for a turn url\n        for (const iceServer of options.iceServers) {\n            for (const url of iceServer.urls) {\n                if (url.includes('turn')) {\n                    Logger.Log(\n                        Logger.GetStackTrace(),\n                        `A turn sever was found at ${url}`\n                    );\n                    return true;\n                }\n            }\n        }\n\n        Logger.Info(Logger.GetStackTrace(), 'A turn sever was not found');\n        return false;\n    }\n\n    /**\n     * Handles when a Config Message is received contains the Peer Connection Options required (STUN and TURN Server Info)\n     * @param messageConfig - Config Message received from the signaling server\n     */\n    handleOnConfigMessage(messageConfig: MessageConfig) {\n        this.resizePlayerStyle();\n\n        // Tell the WebRtcController to start a session with the peer options sent from the signaling server\n        this.startSession(messageConfig.peerConnectionOptions);\n\n        // When the signaling server sends a WebRTC Answer over the websocket connection have the WebRtcController handle the message\n        this.webSocketController.onWebRtcAnswer = (\n            messageAnswer: MessageReceive.MessageAnswer\n        ) => this.handleWebRtcAnswer(messageAnswer);\n        this.webSocketController.onWebRtcOffer = (\n            messageOffer: MessageReceive.MessageOffer\n        ) => this.handleWebRtcOffer(messageOffer);\n        this.webSocketController.onWebRtcPeerDataChannels = (\n            messageDataChannels: MessageReceive.MessagePeerDataChannels\n        ) => this.handleWebRtcSFUPeerDatachannels(messageDataChannels);\n\n        // When the signaling server sends a IceCandidate over the websocket connection have the WebRtcController handle the message\n        this.webSocketController.onIceCandidate = (\n            iceCandidate: RTCIceCandidateInit\n        ) => this.handleIceCandidate(iceCandidate);\n    }\n\n    /**\n     * Handles when the signalling server gives us the list of streamer ids.\n     */\n    handleStreamerListMessage(messageStreamerList: MessageStreamerList) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            `Got streamer list ${messageStreamerList.ids}`,\n            6\n        );\n\n        // add the streamers to the UI\n        const settingOptions = [...messageStreamerList.ids]; // copy the original messageStreamerList.ids\n        settingOptions.unshift(''); // add an empty option at the top\n        this.config.setOptionSettingOptions(\n            OptionParameters.StreamerId,\n            settingOptions\n        );\n\n        let wantedStreamerId: string = null;\n        let autoSelectedStreamerId: string  = null;\n        const waitForStreamer = this.config.isFlagEnabled(Flags.WaitForStreamer);\n        const reconnectLimit = this.config.getNumericSettingValue(NumericParameters.MaxReconnectAttempts);\n        const reconnectDelay = this.config.getNumericSettingValue(NumericParameters.StreamerAutoJoinInterval);\n\n        // first we figure out a wanted streamer id through various means\n        const urlParams = new URLSearchParams(window.location.search);\n        if (urlParams.has(OptionParameters.StreamerId)) {\n            // if we've set the streamer id on the url we only want that streamer id\n            wantedStreamerId = urlParams.get(OptionParameters.StreamerId);\n        } else if (this.subscribedStream) {\n            // we were previously subscribed to a streamer, we want that\n            wantedStreamerId = this.subscribedStream;\n        }\n\n        // now lets see if we can pick it.\n        if (wantedStreamerId && messageStreamerList.ids.includes(wantedStreamerId)) {\n            // if the wanted stream is in the list. we pick that\n            autoSelectedStreamerId = wantedStreamerId;\n        } else if ((!wantedStreamerId || !waitForStreamer) && messageStreamerList.ids.length == 1) {\n            // otherwise, if we're not waiting for the wanted streamer and there's only one streamer, connect to it\n            autoSelectedStreamerId = messageStreamerList.ids[0];\n        }\n\n        // if we found a streamer id to auto select, select it\n        if (autoSelectedStreamerId) {\n            this.isReconnecting = false;\n            this.reconnectAttempt = 0;\n            this.config.setOptionSettingValue(\n                OptionParameters.StreamerId,\n                autoSelectedStreamerId\n            );\n        } else {\n            // no auto selected streamer.\n            // if we're waiting for a streamer then try reconnecting\n            if (waitForStreamer) {\n                if (this.reconnectAttempt < reconnectLimit) {\n                    // still reconnects available\n                    this.isReconnecting = true;\n                    this.reconnectAttempt++;\n                    setTimeout(() => {\n                        this.webSocketController.requestStreamerList();\n                    }, reconnectDelay);\n                } else {\n                    // We've exhausted our reconnect attempts, return to main screen\n                    this.reconnectAttempt = 0;\n                    this.isReconnecting = false;\n                    this.shouldReconnect = false;\n                }\n            }\n        }\n\n        // dispatch this event finally\n        this.pixelStreaming.dispatchEvent(\n            new StreamerListMessageEvent({\n                messageStreamerList,\n                autoSelectedStreamerId,\n                wantedStreamerId\n            })\n        );\n    }\n\n    /**\n     * Handle the RTC Answer from the signaling server\n     * @param Answer - Answer SDP from the peer.\n     */\n    handleWebRtcAnswer(Answer: MessageAnswer) {\n        Logger.Log(Logger.GetStackTrace(), `Got answer sdp ${Answer.sdp}`, 6);\n\n        const sdpAnswer: RTCSessionDescriptionInit = {\n            sdp: Answer.sdp,\n            type: 'answer'\n        };\n\n        this.peerConnectionController.receiveAnswer(sdpAnswer);\n        this.handlePostWebrtcNegotiation();\n    }\n\n    /**\n     * Handle the RTC offer from a WebRTC peer (received through the signalling server).\n     * @param Offer - Offer SDP from the peer.\n     */\n    handleWebRtcOffer(Offer: MessageOffer) {\n        Logger.Log(Logger.GetStackTrace(), `Got offer sdp ${Offer.sdp}`, 6);\n\n        this.isUsingSFU = Offer.sfu ? Offer.sfu : false;\n        if (this.isUsingSFU) {\n            // Disable negotiating with the sfu as the sfu only supports one codec at a time\n            this.peerConnectionController.preferredCodec = '';\n        }\n\n        const sdpOffer: RTCSessionDescriptionInit = {\n            sdp: Offer.sdp,\n            type: 'offer'\n        };\n\n        this.peerConnectionController.receiveOffer(sdpOffer, this.config);\n        this.handlePostWebrtcNegotiation();\n    }\n\n    /**\n     * Handle when the SFU provides the peer with its data channels\n     * @param DataChannels - The message from the SFU containing the data channels ids\n     */\n    handleWebRtcSFUPeerDatachannels(\n        DataChannels: MessageReceive.MessagePeerDataChannels\n    ) {\n        const SendOptions: RTCDataChannelInit = {\n            ordered: true,\n            negotiated: true,\n            id: DataChannels.sendStreamId\n        };\n\n        const unidirectional =\n            DataChannels.sendStreamId != DataChannels.recvStreamId;\n\n        this.sendrecvDataChannelController.createDataChannel(\n            this.peerConnectionController.peerConnection,\n            unidirectional ? 'send-datachannel' : 'datachannel',\n            SendOptions\n        );\n\n        if (unidirectional) {\n            const RecvOptions: RTCDataChannelInit = {\n                ordered: true,\n                negotiated: true,\n                id: DataChannels.recvStreamId\n            };\n\n            this.recvDataChannelController.createDataChannel(\n                this.peerConnectionController.peerConnection,\n                'recv-datachannel',\n                RecvOptions\n            );\n            this.recvDataChannelController.handleOnOpen = () =>\n                this.webSocketController.sendSFURecvDataChannelReady();\n            // If we're uni-directional, only the recv data channel should handle incoming messages\n            this.recvDataChannelController.handleOnMessage = (\n                ev: MessageEvent\n            ) => this.handleOnMessage(ev);\n        } else {\n            // else our primary datachannel is send/recv so it can handle incoming messages\n            this.sendrecvDataChannelController.handleOnMessage = (\n                ev: MessageEvent\n            ) => this.handleOnMessage(ev);\n        }\n    }\n\n    handlePostWebrtcNegotiation() {\n        // start the afk warning timer as PS is now running\n        this.afkController.startAfkWarningTimer();\n        // show the overlay that we have negotiated a connection\n        this.pixelStreaming._onWebRtcSdp();\n\n        if (this.statsTimerHandle && this.statsTimerHandle !== undefined) {\n            window.clearInterval(this.statsTimerHandle);\n        }\n\n        this.statsTimerHandle = window.setInterval(() => this.getStats(), 1000);\n\n        /*  */\n        this.setMouseInputEnabled(this.config.isFlagEnabled(Flags.MouseInput));\n        this.setKeyboardInputEnabled(this.config.isFlagEnabled(Flags.KeyboardInput));\n        this.setGamePadInputEnabled(this.config.isFlagEnabled(Flags.GamepadInput));\n    }\n\n    /**\n     * When an ice Candidate is received from the Signaling server add it to the Peer Connection Client\n     * @param iceCandidate - Ice Candidate from Server\n     */\n    handleIceCandidate(iceCandidate: RTCIceCandidateInit) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'Web RTC Controller: onWebRtcIce',\n            6\n        );\n\n        const candidate = new RTCIceCandidate(iceCandidate);\n        this.peerConnectionController.handleOnIce(candidate);\n    }\n\n    /**\n     * Send the ice Candidate to the signaling server via websocket\n     * @param iceEvent - RTC Peer ConnectionIceEvent) {\n     */\n    handleSendIceCandidate(iceEvent: RTCPeerConnectionIceEvent) {\n        Logger.Log(Logger.GetStackTrace(), 'OnIceCandidate', 6);\n        if (iceEvent.candidate && iceEvent.candidate.candidate) {\n            this.webSocketController.sendIceCandidate(iceEvent.candidate);\n        }\n    }\n\n    /**\n     * Send the ice Candidate to the signaling server via websocket\n     * @param iceEvent - RTC Peer ConnectionIceEvent) {\n     */\n    handleDataChannel(datachannelEvent: RTCDataChannelEvent) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'Data channel created for us by browser as we are a receiving peer.',\n            6\n        );\n        this.sendrecvDataChannelController.dataChannel =\n            datachannelEvent.channel;\n        // Data channel was created for us, so we just need to setup its callbacks and array type\n        this.sendrecvDataChannelController.setupDataChannel();\n        this.sendrecvDataChannelController.handleOnMessage = (\n            ev: MessageEvent<ArrayBuffer>\n        ) => this.handleOnMessage(ev);\n    }\n\n    /**\n     * Send the RTC Offer Session to the Signaling server via websocket\n     * @param offer - RTC Session Description\n     */\n    handleSendWebRTCOffer(offer: RTCSessionDescriptionInit) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'Sending the offer to the Server',\n            6\n        );\n        this.webSocketController.sendWebRtcOffer(offer);\n    }\n\n    /**\n     * Send the RTC Offer Session to the Signaling server via websocket\n     * @param answer - RTC Session Description\n     */\n    handleSendWebRTCAnswer(answer: RTCSessionDescriptionInit) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'Sending the answer to the Server',\n            6\n        );\n        this.webSocketController.sendWebRtcAnswer(answer);\n\n        if (this.isUsingSFU) {\n            this.webSocketController.sendWebRtcDatachannelRequest();\n        }\n    }\n\n    /**\n     * Set the freeze frame overlay to the player div\n     */\n    setUpMouseAndFreezeFrame() {\n        // Calculating and normalizing positions depends on the width and height of the player.\n        this.videoElementParentClientRect = this.videoPlayer\n            .getVideoParentElement()\n            .getBoundingClientRect();\n        this.coordinateConverter.setupNormalizeAndQuantize();\n        this.freezeFrameController.freezeFrame.resize();\n    }\n\n    /**\n     * Close the Connection to the signaling server\n     */\n    closeSignalingServer(message: string) {\n        // We explicitly called close, therefore we don't want to trigger auto reconnect\n        this.locallyClosed = true;\n        this.shouldReconnect = false;\n        this.disconnectMessage = message;\n        this.webSocketController?.close();\n    }\n\n    /**\n     * Close the peer connection\n     */\n    closePeerConnection() {\n        this.peerConnectionController?.close();\n    }\n\n    /**\n     * Close all connections\n     */\n    close() {\n        this.closeSignalingServer('');\n        this.closePeerConnection();\n    }\n\n    /**\n     * Fires a Video Stats Event in the RTC Peer Connection\n     */\n    getStats() {\n        this.peerConnectionController.generateStats();\n    }\n\n    /**\n     * Send a Latency Test Request to the UE Instance\n     */\n    sendLatencyTest() {\n        this.latencyStartTime = Date.now();\n\n        this.streamMessageController.toStreamerHandlers.get(\n            'LatencyTest'\n        )([JSON.stringify({\n            StartTime: this.latencyStartTime\n        })]);\n    }\n\n    /**\n     * Send a Data Channel Latency Test Request to the UE Instance\n     */\n    sendDataChannelLatencyTest(descriptor: DataChannelLatencyTestRequest) {\n        this.streamMessageController.toStreamerHandlers.get(\n            'DataChannelLatencyTest'\n        )([JSON.stringify(descriptor)]);\n    }\n\n    /**\n     * Send the MinQP encoder setting to the UE Instance.\n     * @param minQP - The lower bound for QP when encoding\n     * valid values are (1-51) where:\n     * 1 = Best quality but highest bitrate.\n     * 51 = Worst quality but lowest bitrate.\n     * By default the minQP is 1 meaning the encoder is free\n     * to aim for the best quality it can on the given network link.\n     */\n    sendEncoderMinQP(minQP: number) {\n        Logger.Log(Logger.GetStackTrace(), `MinQP=${minQP}\\n`, 6);\n\n        if (minQP != null) {\n            this.streamMessageController.toStreamerHandlers.get(\n                'Command'\n            )([JSON.stringify({\n                'Encoder.MinQP': minQP\n            })]);\n        }\n    }\n\n    /**\n     * Send the MaxQP encoder setting to the UE Instance.\n     * @param maxQP - The upper bound for QP when encoding\n     * valid values are (1-51) where:\n     * 1 = Best quality but highest bitrate.\n     * 51 = Worst quality but lowest bitrate.\n     * By default the maxQP is 51 meaning the encoder is free\n     * to drop quality as low as needed on the given network link.\n     */\n     sendEncoderMaxQP(maxQP: number) {\n        Logger.Log(Logger.GetStackTrace(), `MaxQP=${maxQP}\\n`, 6);\n\n        if (maxQP != null) {\n            this.streamMessageController.toStreamerHandlers.get(\n                'Command'\n            )([JSON.stringify({\n                'Encoder.MaxQP': maxQP\n            })]);\n        }\n    }\n\n    /**\n     * Send the { WebRTC.MinBitrate: SomeNumber }} command to UE to set \n     * the minimum bitrate that we allow WebRTC to use \n     * (note setting this too high in poor networks can be problematic).\n     * @param minBitrate - The minimum bitrate we would like WebRTC to not fall below.\n     */\n    sendWebRTCMinBitrate(minBitrate: number) {\n        Logger.Log(Logger.GetStackTrace(), `WebRTC Min Bitrate=${minBitrate}`, 6);\n        if (minBitrate != null) {\n            this.streamMessageController.toStreamerHandlers.get(\n                'Command'\n            )([JSON.stringify({\n                'WebRTC.MinBitrate': minBitrate\n            })]);\n        }\n    }\n\n    /**\n     * Send the { WebRTC.MaxBitrate: SomeNumber }} command to UE to set \n     * the minimum bitrate that we allow WebRTC to use \n     * (note setting this too low could result in blocky video).\n     * @param minBitrate - The minimum bitrate we would like WebRTC to not fall below.\n     */\n     sendWebRTCMaxBitrate(maxBitrate: number) {\n        Logger.Log(Logger.GetStackTrace(), `WebRTC Max Bitrate=${maxBitrate}`, 6);\n        if (maxBitrate != null) {\n            this.streamMessageController.toStreamerHandlers.get(\n                'Command'\n            )([JSON.stringify({\n                'WebRTC.MaxBitrate': maxBitrate\n            })]);\n        }\n    }\n\n    /**\n     * Send the { WebRTC.Fps: SomeNumber }} UE 5.0+\n     * and { WebRTC.MaxFps } UE 4.27 command to set \n     * the maximum fps we would like WebRTC to stream at. \n     * @param fps - The maximum stream fps.\n     */\n     sendWebRTCFps(fps: number) {\n        Logger.Log(Logger.GetStackTrace(), `WebRTC FPS=${fps}`, 6);\n        if (fps != null) {\n            this.streamMessageController.toStreamerHandlers.get(\n                'Command'\n            )([JSON.stringify({'WebRTC.Fps': fps})]);\n\n            /* TODO: Remove when UE 4.27 unsupported. */\n            this.streamMessageController.toStreamerHandlers.get(\n                'Command'\n            )([JSON.stringify({'WebRTC.MaxFps': fps})]); \n        }\n    }\n\n    /**\n     * Sends the UI Descriptor `stat fps` to the UE Instance\n     */\n    sendShowFps(): void {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            '----   Sending show stat to UE   ----',\n            6\n        );\n\n        this.streamMessageController.toStreamerHandlers.get(\n            'Command'\n        )([JSON.stringify({ 'stat.fps': '' })]);\n    }\n\n    /**\n     * Send an Iframe request to the streamer\n     */\n    sendIframeRequest(): void {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            '----   Sending Request for an IFrame  ----',\n            6\n        );\n        this.streamMessageController.toStreamerHandlers.get('IFrameRequest')();\n    }\n\n    /**\n     * Send a UIInteraction message\n     */\n    emitUIInteraction(descriptor: object | string) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            '----   Sending custom UIInteraction message   ----',\n            6\n        );\n\n        this.streamMessageController.toStreamerHandlers.get(\n            'UIInteraction'\n        )([JSON.stringify(descriptor)]);\n    }\n\n    /**\n     * Send a Command message\n     */\n    emitCommand(descriptor: object) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            '----   Sending custom Command message   ----',\n            6\n        );\n        \n        this.streamMessageController.toStreamerHandlers.get(\n            'Command'\n        )([JSON.stringify(descriptor)]);\n    }\n\n    /**\n     * Send a console command message\n     */\n    emitConsoleCommand(command: string) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            '----   Sending custom Command:ConsoleCommand message   ----',\n            6\n        );\n\n        this.streamMessageController.toStreamerHandlers.get(\n            'Command'\n        )([JSON.stringify({\n            ConsoleCommand: command,\n        })]);\n    }\n\n    /**\n     * Sends a request to the UE Instance to have ownership of Quality\n     */\n    sendRequestQualityControlOwnership(): void {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            '----   Sending Request to Control Quality  ----',\n            6\n        );\n        this.toStreamerMessagesController.SendRequestQualityControl();\n    }\n\n    /**\n     * Handles when a Latency Test Result are received from the UE Instance\n     * @param message - Latency Test Timings\n     */\n    handleLatencyTestResult(message: ArrayBuffer) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'DataChannelReceiveMessageType.latencyTest',\n            6\n        );\n        const latencyAsString = new TextDecoder('utf-16').decode(\n            message.slice(1)\n        );\n        const latencyTestResults: LatencyTestResults = new LatencyTestResults();\n        Object.assign(latencyTestResults, JSON.parse(latencyAsString));\n        latencyTestResults.processFields();\n\n        latencyTestResults.testStartTimeMs = this.latencyStartTime;\n        latencyTestResults.browserReceiptTimeMs = Date.now();\n\n        latencyTestResults.latencyExcludingDecode = ~~(\n            latencyTestResults.browserReceiptTimeMs -\n            latencyTestResults.testStartTimeMs\n        );\n        latencyTestResults.testDuration = ~~(\n            latencyTestResults.TransmissionTimeMs -\n            latencyTestResults.ReceiptTimeMs\n        );\n        latencyTestResults.networkLatency = ~~(\n            latencyTestResults.latencyExcludingDecode -\n            latencyTestResults.testDuration\n        );\n\n        if (\n            latencyTestResults.frameDisplayDeltaTimeMs &&\n            latencyTestResults.browserReceiptTimeMs\n        ) {\n            latencyTestResults.endToEndLatency =\n                ~~(latencyTestResults.frameDisplayDeltaTimeMs +\n                    latencyTestResults.networkLatency,\n                +latencyTestResults.CaptureToSendMs);\n        }\n        this.pixelStreaming._onLatencyTestResult(latencyTestResults);\n    }\n\n    /**\n     * Handles when a Data Channel Latency Test Response is received from the UE Instance\n     * @param message - Data Channel Latency Test Response\n     */\n    handleDataChannelLatencyTestResponse(message: ArrayBuffer) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'DataChannelReceiveMessageType.dataChannelLatencyResponse',\n            6\n        );\n        const responseAsString = new TextDecoder('utf-16').decode(\n            message.slice(1)\n        );\n        const latencyTestResponse: DataChannelLatencyTestResponse = JSON.parse(responseAsString);\n        this.pixelStreaming._onDataChannelLatencyTestResponse(latencyTestResponse);\n    }\n\n    /**\n     * Handles when the Encoder and Web RTC Settings are received from the UE Instance\n     * @param message - Initial Encoder and Web RTC Settings\n     */\n    handleInitialSettings(message: ArrayBuffer) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'DataChannelReceiveMessageType.InitialSettings',\n            6\n        );\n        const payloadAsString = new TextDecoder('utf-16').decode(\n            message.slice(1)\n        );\n        const parsedInitialSettings = JSON.parse(payloadAsString);\n\n        const initialSettings: InitialSettings = new InitialSettings();\n\n        if (parsedInitialSettings.Encoder) {\n            initialSettings.EncoderSettings = parsedInitialSettings.Encoder;\n        }\n\n        if (parsedInitialSettings.WebRTC) {\n            initialSettings.WebRTCSettings = parsedInitialSettings.WebRTC;\n        }\n\n        if (parsedInitialSettings.PixelStreaming) {\n            initialSettings.PixelStreamingSettings =\n                parsedInitialSettings.PixelStreaming;\n        }\n\n        if (parsedInitialSettings.ConfigOptions && parsedInitialSettings.ConfigOptions.DefaultToHover !== undefined) {\n            this.config.setFlagEnabled(\n                Flags.HoveringMouseMode,\n                !!parsedInitialSettings.ConfigOptions.DefaultToHover\n            );\n        }\n\n        initialSettings.ueCompatible();\n        Logger.Log(Logger.GetStackTrace(), payloadAsString, 6);\n\n        this.pixelStreaming._onInitialSettings(initialSettings);\n    }\n\n    /**\n     * Handles when the Quantization Parameter are received from the UE Instance\n     * @param message - Encoders Quantization Parameter\n     */\n    handleVideoEncoderAvgQP(message: ArrayBuffer) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'DataChannelReceiveMessageType.VideoEncoderAvgQP',\n            6\n        );\n        const AvgQP = Number(\n            new TextDecoder('utf-16').decode(message.slice(1))\n        );\n        this.setVideoEncoderAvgQP(AvgQP);\n    }\n\n    /**\n     * Handles when the video element has been loaded with a srcObject\n     */\n    handleVideoInitialized() {\n        this.pixelStreaming._onVideoInitialized();\n\n        // either autoplay the video or set up the play overlay\n        this.autoPlayVideoOrSetUpPlayOverlay();\n        this.resizePlayerStyle();\n        this.videoPlayer.updateVideoStreamSize();\n    }\n\n    /**\n     * Flag set if the user has Quality Ownership\n     * @param message - Does the current client have Quality Ownership\n     */\n    onQualityControlOwnership(message: ArrayBuffer) {\n        const view = new Uint8Array(message);\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'DataChannelReceiveMessageType.QualityControlOwnership',\n            6\n        );\n        this.isQualityController = new Boolean(view[1]).valueOf();\n        Logger.Log(\n            Logger.GetStackTrace(),\n            `Received quality controller message, will control quality: ${this.isQualityController}`\n        );\n        this.pixelStreaming._onQualityControlOwnership(\n            this.isQualityController\n        );\n    }\n\n    /**\n     * Handles when the Aggregated stats are Collected\n     * @param stats - Aggregated Stats\n     */\n    handleVideoStats(stats: AggregatedStats) {\n        this.pixelStreaming._onVideoStats(stats);\n    }\n\n    /**\n     * To Resize the Video Player element\n     */\n    resizePlayerStyle(): void {\n        this.videoPlayer.resizePlayerStyle();\n    }\n\n    setPreferredCodec(codec: string) {\n        this.preferredCodec = codec;\n        if (this.peerConnectionController) {\n            this.peerConnectionController.preferredCodec = codec;\n            this.peerConnectionController.updateCodecSelection = false;\n        }\n    }\n\n    setVideoEncoderAvgQP(avgQP: number) {\n        this.videoAvgQp = avgQP;\n        this.pixelStreaming._onVideoEncoderAvgQP(this.videoAvgQp);\n    }\n\n    /**\n     * enables/disables keyboard event listeners\n     */\n    setKeyboardInputEnabled(isEnabled: boolean) {\n        this.keyboardController?.unregisterKeyBoardEvents();\n        if (isEnabled) {\n            this.keyboardController = this.inputClassesFactory.registerKeyBoard(\n                this.config\n            );\n        }\n    }\n\n    /**\n     * enables/disables mouse event listeners\n     */\n    setMouseInputEnabled(isEnabled: boolean) {\n        this.mouseController?.unregisterMouseEvents();\n        if (isEnabled) {\n            const mouseMode = this.config.isFlagEnabled(Flags.HoveringMouseMode)\n            ? ControlSchemeType.HoveringMouse\n            : ControlSchemeType.LockedMouse;\n            this.mouseController =\n            this.inputClassesFactory.registerMouse(mouseMode);\n        }\n    }\n\n    /**\n     * enables/disables touch event listeners\n     */\n    setTouchInputEnabled(isEnabled: boolean) {\n        this.touchController?.unregisterTouchEvents();\n        if (isEnabled) {\n            this.touchController = this.inputClassesFactory.registerTouch(\n                this.config.isFlagEnabled(Flags.FakeMouseWithTouches),\n                this.videoElementParentClientRect\n            );\n        }\n    }\n\n    /**\n     * enables/disables game pad event listeners\n     */\n    setGamePadInputEnabled(isEnabled: boolean) {\n        this.gamePadController?.unregisterGamePadEvents();\n        if (isEnabled) {\n            this.gamePadController = this.inputClassesFactory.registerGamePad();\n            this.gamePadController.onGamepadConnected = () => {\n                this.streamMessageController.toStreamerHandlers.get('GamepadConnected')();\n            }\n            this.gamePadController.onGamepadDisconnected = (controllerIdx: number) => {\n                this.streamMessageController.toStreamerHandlers.get('GamepadDisconnected')([controllerIdx]);\n            }\n        }\n    }\n\n    registerDataChannelEventEmitters(dataChannel: DataChannelController) {\n        dataChannel.onOpen = (label, event) =>\n            this.pixelStreaming.dispatchEvent(\n                new DataChannelOpenEvent({ label, event })\n            );\n        dataChannel.onClose = (label, event) =>\n            this.pixelStreaming.dispatchEvent(\n                new DataChannelCloseEvent({ label, event })\n            );\n        dataChannel.onError = (label, event) =>\n            this.pixelStreaming.dispatchEvent(\n                new DataChannelErrorEvent({ label, event })\n            );\n    }\n\n    public registerMessageHandler(name: string, direction: MessageDirection, handler?: (data: ArrayBuffer | Array<number | string>) => void) {\n        if(direction === MessageDirection.FromStreamer && typeof handler === 'undefined') {\n            Logger.Warning(\n                Logger.GetStackTrace(),\n                `Unable to register handler for ${name} as no handler was passed`\n            );\n        }\n\n        \n        this.streamMessageController.registerMessageHandler(\n            direction,\n            name,\n            (data: Array<number | string>) => (typeof handler === 'undefined' && direction === MessageDirection.ToStreamer) ? \n                this.sendMessageController.sendMessageToStreamer(\n                    name,\n                    data\n                ) :   \n                handler(data)\n        );\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * The Types of Messages that will be received\n */\nexport enum MessageRecvTypes {\n    CONFIG = 'config',\n    STREAMER_LIST = 'streamerList',\n    PLAYER_COUNT = 'playerCount',\n    OFFER = 'offer',\n    ANSWER = 'answer',\n    ICE_CANDIDATE = 'iceCandidate',\n    PEER_DATA_CHANNELS = 'peerDataChannels',\n    PING = 'ping',\n    WARNING = 'warning'\n}\n\n/**\n * Concrete Received Message wrapper\n */\nexport class MessageRecv {\n    type: string;\n    id: string;\n}\n\n/**\n * Authentication Required Message wrapper\n */\nexport class MessageAuthRequired extends MessageRecv {}\n\n/**\n * Config Message Wrapper\n */\nexport class MessageConfig extends MessageRecv {\n    peerConnectionOptions: RTCConfiguration;\n}\n\n/**\n * Streamer List Message Wrapper\n */\nexport class MessageStreamerList extends MessageRecv {\n    ids: string[];\n}\n\n/**\n * Player Count Message wrapper\n */\nexport class MessagePlayerCount extends MessageRecv {\n    count: number;\n}\n\n/**\n * Web RTC offer Answer Message wrapper\n */\nexport class MessageAnswer extends MessageRecv {\n    sdp: string;\n}\n\n/**\n * WebRTC sdp offer Message wrapper.\n */\nexport class MessageOffer extends MessageRecv {\n    sdp: string;\n    sfu?: boolean;\n    defaultToHover?: string;\n}\n\n/**\n * Ice Candidate Message wrapper\n */\nexport class MessageIceCandidate extends MessageRecv {\n    candidate: RTCIceCandidateInit;\n}\n\n/**\n * Peer Data Channels Message wrapper\n */\nexport class MessagePeerDataChannels extends MessageRecv {\n    recvStreamId: number;\n    sendStreamId: number;\n    type: string;\n}\n\nexport class MessageOnScreenKeyboard {\n    command: string;\n    showOnScreenKeyboard: boolean;\n    x: number;\n    y: number;\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\n\n/**\n * The Send Types that are pushed from the signaling server\n */\nexport enum MessageSendTypes {\n    LIST_STREAMERS = 'listStreamers',\n    SUBSCRIBE = 'subscribe',\n    UNSUBSCRIBE = 'unsubscribe',\n    ICE_CANDIDATE = 'iceCandidate',\n    OFFER = 'offer',\n    ANSWER = 'answer',\n    DATACHANNELREQUEST = 'dataChannelRequest',\n    SFURECVDATACHANNELREADY = 'peerDataChannelsReady',\n    PONG = 'pong'\n}\n\n/**\n * A Wrapper for the message to send to the signaling server\n */\nexport class MessageSend implements Send {\n    type: string;\n    peerConnectionOptions: object;\n\n    /**\n     * Turns the wrapper into a JSON String\n     * @returns - JSON String of the Message to send\n     */\n    payload() {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'Sending => \\n' + JSON.stringify(this, undefined, 4),\n            6\n        );\n        return JSON.stringify(this);\n    }\n}\n\nexport interface Send {\n    /**\n     * Turns the wrapper into a JSON String\n     * @returns - JSON String of the Message to send\n     */\n    payload: () => string;\n}\n\nexport class MessageListStreamers extends MessageSend {\n    constructor() {\n        super();\n        this.type = MessageSendTypes.LIST_STREAMERS;\n    }\n}\n\nexport class MessageSubscribe extends MessageSend {\n    streamerId: string;\n\n    constructor(streamerid: string) {\n        super();\n        this.type = MessageSendTypes.SUBSCRIBE;\n        this.streamerId = streamerid;\n    }\n}\n\nexport class MessageUnsubscribe extends MessageSend {\n    constructor() {\n        super();\n        this.type = MessageSendTypes.UNSUBSCRIBE;\n    }\n}\n\n/**\n * Instance Request Message Wrapper\n */\nexport class MessagePong extends MessageSend {\n    time: number;\n\n    constructor(time: number) {\n        super();\n        this.type = MessageSendTypes.PONG;\n        this.time = time;\n    }\n}\n\n/**\n *  Web RTC Offer message wrapper\n */\nexport class MessageWebRTCOffer extends MessageSend {\n    sdp: string;\n\n    /**\n     * @param offer - Generated Web RTC Offer\n     */\n    constructor(offer?: RTCSessionDescriptionInit) {\n        super();\n        this.type = MessageSendTypes.OFFER;\n\n        if (offer) {\n            this.type = offer.type as MessageSendTypes;\n            this.sdp = offer.sdp;\n        }\n    }\n}\n\n/**\n *  Web RTC Answer message wrapper\n */\nexport class MessageWebRTCAnswer extends MessageSend {\n    sdp: string;\n\n    /**\n     * @param answer - Generated Web RTC Offer\n     */\n    constructor(answer?: RTCSessionDescriptionInit) {\n        super();\n        this.type = MessageSendTypes.ANSWER;\n\n        if (answer) {\n            this.type = answer.type as MessageSendTypes;\n            this.sdp = answer.sdp;\n        }\n    }\n}\n\n/**\n *  Web RTC Data channel request message wrapper\n */\nexport class MessageWebRTCDatachannelRequest extends MessageSend {\n    constructor() {\n        super();\n        this.type = MessageSendTypes.DATACHANNELREQUEST;\n    }\n}\n\n/**\n *  Web RTC SFU Data channel ready message wrapper\n */\nexport class MessageSFURecvDataChannelReady extends MessageSend {\n    constructor() {\n        super();\n        this.type = MessageSendTypes.SFURECVDATACHANNELREADY;\n    }\n}\n\n/**\n * RTC Ice Candidate Wrapper\n */\nexport class MessageIceCandidate implements Send {\n    candidate: RTCIceCandidate;\n    type: MessageSendTypes;\n\n    /**\n     * @param candidate - RTC Ice Candidate\n     */\n    constructor(candidate: RTCIceCandidate) {\n        this.type = MessageSendTypes.ICE_CANDIDATE;\n        this.candidate = candidate;\n    }\n\n    /**\n     * Turns the wrapper into a JSON String\n     * @returns - JSON String of the Message to send\n     */\n    payload() {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'Sending => \\n' + JSON.stringify(this, undefined, 4),\n            6\n        );\n        return JSON.stringify(this);\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\nimport { WebSocketController } from './WebSocketController';\nimport {\n    MessageRecvTypes,\n    MessageConfig,\n    MessageStreamerList,\n    MessagePlayerCount,\n    MessageAnswer,\n    MessageOffer,\n    MessageIceCandidate,\n    MessagePeerDataChannels\n} from './MessageReceive';\nimport { MessagePong } from './MessageSend';\n\n/**\n * Signalling protocol for handling messages from the signalling server.\n */\nexport class SignallingProtocol {\n    private FromUEMessageHandlers: Map<string, (payload: string) => void>;\n\n    constructor() {\n        this.FromUEMessageHandlers = new Map<\n            string,\n            (payload: string) => void\n        >();\n    }\n\n    addMessageHandler(\n        messageId: string,\n        messageHandler: (payload: string) => void\n    ) {\n        this.FromUEMessageHandlers.set(messageId, messageHandler);\n    }\n\n    handleMessage(messageId: string, messageData: string) {\n        if (this.FromUEMessageHandlers.has(messageId)) {\n            this.FromUEMessageHandlers.get(messageId)(messageData);\n        } else {\n            Logger.Error(\n                Logger.GetStackTrace(),\n                `Message type of ${messageId} does not have a message handler registered on the frontend - ignoring message.`\n            );\n        }\n    }\n\n    /**\n     * Setup any default signalling message handling, these can be overridden or additional handlers added with `addMessageHandler`.\n     * @param websocketController The controller to setup these handlers on.\n     */\n    static setupDefaultHandlers(websocketController: WebSocketController) {\n        // PING\n        websocketController.signallingProtocol.addMessageHandler(\n            MessageRecvTypes.PING,\n            (pingPayload: string) => {\n                // send our pong payload back to the signalling server\n                const pongPayload = new MessagePong(\n                    new Date().getTime()\n                ).payload();\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    MessageRecvTypes.PING + ': ' + pingPayload,\n                    6\n                );\n                websocketController.webSocket.send(pongPayload);\n            }\n        );\n\n        // CONFIG\n        websocketController.signallingProtocol.addMessageHandler(\n            MessageRecvTypes.CONFIG,\n            (configPayload: string) => {\n                Logger.Log(Logger.GetStackTrace(), MessageRecvTypes.CONFIG, 6);\n                const config: MessageConfig = JSON.parse(configPayload);\n                websocketController.onConfig(config);\n            }\n        );\n\n        // STREAMER_LIST\n        websocketController.signallingProtocol.addMessageHandler(\n            MessageRecvTypes.STREAMER_LIST,\n            (listPayload: string) => {\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    MessageRecvTypes.STREAMER_LIST,\n                    6\n                );\n                const streamerList: MessageStreamerList =\n                    JSON.parse(listPayload);\n                websocketController.onStreamerList(streamerList);\n            }\n        );\n\n        // PLAYER_COUNT\n        websocketController.signallingProtocol.addMessageHandler(\n            MessageRecvTypes.PLAYER_COUNT,\n            (playerCountPayload: string) => {\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    MessageRecvTypes.PLAYER_COUNT,\n                    6\n                );\n                const playerCount: MessagePlayerCount =\n                    JSON.parse(playerCountPayload);\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    'Player Count: ' + playerCount.count,\n                    6\n                );\n                websocketController.onPlayerCount(playerCount)\n            }\n        );\n\n        // ANSWER\n        websocketController.signallingProtocol.addMessageHandler(\n            MessageRecvTypes.ANSWER,\n            (answerPayload: string) => {\n                // send our pong payload back to the signalling server\n                Logger.Log(Logger.GetStackTrace(), MessageRecvTypes.ANSWER, 6);\n                const answer: MessageAnswer = JSON.parse(answerPayload);\n                websocketController.onWebRtcAnswer(answer);\n            }\n        );\n\n        // OFFER\n        websocketController.signallingProtocol.addMessageHandler(\n            MessageRecvTypes.OFFER,\n            (offerPayload: string) => {\n                // send our pong payload back to the signalling server\n                Logger.Log(Logger.GetStackTrace(), MessageRecvTypes.OFFER, 6);\n                const offer: MessageOffer = JSON.parse(offerPayload);\n                websocketController.onWebRtcOffer(offer);\n            }\n        );\n\n        // ICE CANDIDATE\n        websocketController.signallingProtocol.addMessageHandler(\n            MessageRecvTypes.ICE_CANDIDATE,\n            (iceCandidatePayload: string) => {\n                // send our pong payload back to the signalling server\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    MessageRecvTypes.ICE_CANDIDATE,\n                    6\n                );\n                const iceCandidate: MessageIceCandidate =\n                    JSON.parse(iceCandidatePayload);\n                websocketController.onIceCandidate(iceCandidate.candidate);\n            }\n        );\n\n        // WARNING\n        websocketController.signallingProtocol.addMessageHandler(\n            MessageRecvTypes.WARNING,\n            (warningPayload: string) => {\n                Logger.Warning(\n                    Logger.GetStackTrace(),\n                    `Warning received: ${warningPayload}`\n                );\n            }\n        );\n\n        // PEER DATA CHANNELS\n        websocketController.signallingProtocol.addMessageHandler(\n            MessageRecvTypes.PEER_DATA_CHANNELS,\n            (peerDataChannelsPayload: string) => {\n                Logger.Log(\n                    Logger.GetStackTrace(),\n                    MessageRecvTypes.PEER_DATA_CHANNELS,\n                    6\n                );\n                const peerDataChannels: MessagePeerDataChannels = JSON.parse(\n                    peerDataChannelsPayload\n                );\n                websocketController.onWebRtcPeerDataChannels(peerDataChannels);\n            }\n        );\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\nimport * as MessageReceive from './MessageReceive';\nimport * as MessageSend from './MessageSend';\nimport { SignallingProtocol } from './SignallingProtocol';\n\n// declare the new method for the websocket interface\ndeclare global {\n    interface WebSocket {\n        onmessagebinary?(event?: MessageEvent): void;\n    }\n}\n\n/**\n * The controller for the WebSocket and all associated methods\n */\nexport class WebSocketController {\n    WS_OPEN_STATE = 1;\n    webSocket: WebSocket;\n    onOpen: EventTarget;\n    onClose: EventTarget;\n    signallingProtocol: SignallingProtocol;\n\n    constructor() {\n        this.onOpen = new EventTarget();\n        this.onClose = new EventTarget();\n        this.signallingProtocol = new SignallingProtocol();\n        SignallingProtocol.setupDefaultHandlers(this);\n    }\n\n    /**\n     * Connect to the signaling server\n     * @param connectionURL - The Address of the signaling server\n     * @returns - If there is a connection\n     */\n    connect(connectionURL: string): boolean {\n        Logger.Log(Logger.GetStackTrace(), connectionURL, 6);\n        try {\n            this.webSocket = new WebSocket(connectionURL);\n            this.webSocket.onopen = (event) => this.handleOnOpen(event);\n            this.webSocket.onerror = () => this.handleOnError();\n            this.webSocket.onclose = (event) => this.handleOnClose(event);\n            this.webSocket.onmessage = (event) => this.handleOnMessage(event);\n            this.webSocket.onmessagebinary = (event) =>\n                this.handleOnMessageBinary(event);\n            return true;\n        } catch (error) {\n            Logger.Error(error, error);\n            return false;\n        }\n    }\n\n    /**\n     * Handles what happens when a message is received in binary form\n     * @param event - Message Received\n     */\n    handleOnMessageBinary(event: MessageEvent) {\n        // if the event is empty return\n        if (!event || !event.data) {\n            return;\n        }\n\n        // handle the binary and then handle the message\n        event.data\n            .text()\n            .then((messageString: unknown) => {\n                // build a new message\n                const constructedMessage = new MessageEvent(\n                    'messageFromBinary',\n                    {\n                        data: messageString\n                    }\n                );\n\n                // send the new stringified event back into `onmessage`\n                this.handleOnMessage(constructedMessage);\n            })\n            .catch((error: Error) => {\n                Logger.Error(\n                    Logger.GetStackTrace(),\n                    `Failed to parse binary blob from websocket, reason: ${error}`\n                );\n            });\n    }\n\n    /**\n     * Handles what happens when a message is received\n     * @param event - Message Received\n     */\n    handleOnMessage(event: MessageEvent) {\n        // Check if websocket message is binary, if so, stringify it.\n        if (event.data && event.data instanceof Blob) {\n            this.handleOnMessageBinary(event);\n            return;\n        }\n\n        const message: MessageReceive.MessageRecv = JSON.parse(event.data);\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'received => \\n' +\n                JSON.stringify(JSON.parse(event.data), undefined, 4),\n            6\n        );\n\n        // Send to our signalling protocol to handle the incoming message\n        this.signallingProtocol.handleMessage(message.type, event.data);\n    }\n\n    /**\n     * Handles when the Websocket is opened\n     * @param event - Not Used\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars\n    handleOnOpen(event: Event) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'Connected to the signalling server via WebSocket',\n            6\n        );\n        this.onOpen.dispatchEvent(new Event('open'));\n    }\n\n    /**\n     * Handles when there is an error on the websocket\n     * @param event - Error Payload\n     */\n    handleOnError() {\n        Logger.Error(Logger.GetStackTrace(), 'WebSocket error');\n    }\n\n    /**\n     * Handles when the Websocket is closed\n     * @param event - Close Event\n     */\n    handleOnClose(event: CloseEvent) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            'Disconnected to the signalling server via WebSocket: ' +\n                JSON.stringify(event.code) +\n                ' - ' +\n                event.reason\n        );\n        this.onClose.dispatchEvent(new CustomEvent('close', { 'detail': event }));\n    }\n\n    requestStreamerList() {\n        const payload = new MessageSend.MessageListStreamers();\n        this.webSocket.send(payload.payload());\n    }\n\n    sendSubscribe(streamerid: string) {\n        const payload = new MessageSend.MessageSubscribe(streamerid);\n        this.webSocket.send(payload.payload());\n    }\n\n    sendUnsubscribe() {\n        const payload = new MessageSend.MessageUnsubscribe();\n        this.webSocket.send(payload.payload());\n    }\n\n    sendWebRtcOffer(offer: RTCSessionDescriptionInit) {\n        const payload = new MessageSend.MessageWebRTCOffer(offer);\n        this.webSocket.send(payload.payload());\n    }\n\n    sendWebRtcAnswer(answer: RTCSessionDescriptionInit) {\n        const payload = new MessageSend.MessageWebRTCAnswer(answer);\n        this.webSocket.send(payload.payload());\n    }\n\n    sendWebRtcDatachannelRequest() {\n        const payload = new MessageSend.MessageWebRTCDatachannelRequest();\n        this.webSocket.send(payload.payload());\n    }\n\n    sendSFURecvDataChannelReady() {\n        const payload = new MessageSend.MessageSFURecvDataChannelReady();\n        this.webSocket.send(payload.payload());\n    }\n\n    /**\n     * Sends an RTC Ice Candidate to the Server\n     * @param candidate - RTC Ice Candidate\n     */\n    sendIceCandidate(candidate: RTCIceCandidate) {\n        Logger.Log(Logger.GetStackTrace(), 'Sending Ice Candidate');\n        if (\n            this.webSocket &&\n            this.webSocket.readyState === this.WS_OPEN_STATE\n        ) {\n            //ws.send(JSON.stringify({ type: 'iceCandidate', candidate: candidate }));\n            const IceCandidate = new MessageSend.MessageIceCandidate(candidate);\n\n            this.webSocket.send(IceCandidate.payload());\n        }\n    }\n\n    /**\n     * Closes the Websocket connection\n     */\n    close() {\n        this.webSocket?.close();\n    }\n\n    /**\n     * The Message Contains the payload of the peer connection options used for the RTC Peer hand shake\n     * @param messageConfig - Config Message received from he signaling server\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function\n    onConfig(messageConfig: MessageReceive.MessageConfig) {}\n\n    /**\n     * The Message Contains the payload of the peer connection options used for the RTC Peer hand shake\n     * @param messageConfig - Config Message received from he signaling server\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function\n    onStreamerList(messageStreamerList: MessageReceive.MessageStreamerList) {}\n\n    /**\n     * @param iceCandidate - Ice Candidate sent from the Signaling server server's RTC hand shake\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function\n    onIceCandidate(iceCandidate: RTCIceCandidateInit) {}\n\n    /**\n     * Event is fired when the websocket receives the answer for the RTC peer Connection\n     * @param messageAnswer - The RTC Answer payload from the signaling server\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function\n    onWebRtcAnswer(messageAnswer: MessageReceive.MessageAnswer) {}\n\n    /**\n     * Event is fired when the websocket receives the offer for the RTC peer Connection\n     * @param messageOffer - The sdp offer\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function\n    onWebRtcOffer(messageOffer: MessageReceive.MessageOffer) {}\n\n    /**\n     * Event is fired when the websocket receives the data channels for the RTC peer Connection from the SFU\n     * @param messageDataChannels - The data channels details\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function\n    onWebRtcPeerDataChannels(messageDataChannels: MessageReceive.MessagePeerDataChannels) {}\n\n    /**\n     * Event is fired when the websocket receives the an updated player count from cirrus\n     * @param MessagePlayerCount - The new player count\n     */\n    // eslint-disable-next-line @typescript-eslint/no-unused-vars, @typescript-eslint/no-empty-function\n    onPlayerCount(playerCount: MessageReceive.MessagePlayerCount) {}\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '../Logger/Logger';\nimport { WebRtcPlayerController } from '../WebRtcPlayer/WebRtcPlayerController';\nimport { WebGLUtils } from '../Util/WebGLUtils';\nimport { Controller } from '../Inputs/GamepadTypes';\nimport { XRGamepadController } from '../Inputs/XRGamepadController';\nimport { XrFrameEvent } from '../Util/EventEmitter'\nimport { Flags } from '../pixelstreamingfrontend';\n\nexport class WebXRController {\n    private xrSession: XRSession;\n    private xrRefSpace: XRReferenceSpace;\n    private gl: WebGL2RenderingContext;\n\n    private positionLocation: number;\n    private texcoordLocation: number;\n    private resolutionLocation: WebGLUniformLocation;\n    private offsetLocation: WebGLUniformLocation;\n\n    private positionBuffer: WebGLBuffer;\n    private texcoordBuffer: WebGLBuffer;\n\n    private webRtcController: WebRtcPlayerController;\n    private xrGamepadController: XRGamepadController;\n    private xrControllers: Array<Controller>;\n\n    onSessionStarted: EventTarget;\n    onSessionEnded: EventTarget;\n    onFrame: EventTarget;\n\n    constructor(webRtcPlayerController: WebRtcPlayerController) {\n        this.xrSession = null;\n        this.webRtcController = webRtcPlayerController;\n        this.xrControllers = [];\n        this.xrGamepadController = new XRGamepadController(\n            this.webRtcController.streamMessageController\n        );\n        this.onSessionEnded = new EventTarget();\n        this.onSessionStarted = new EventTarget();\n        this.onFrame = new EventTarget();\n    }\n\n    public xrClicked() {\n        if (!this.xrSession) {\n            navigator.xr\n                .requestSession('immersive-vr')\n                .then((session: XRSession) => {\n                    this.onXrSessionStarted(session);\n                });\n        } else {\n            this.xrSession.end();\n        }\n    }\n\n    onXrSessionEnded() {\n        Logger.Log(Logger.GetStackTrace(), 'XR Session ended');\n        this.xrSession = null;\n        this.onSessionEnded.dispatchEvent(new Event('xrSessionEnded'));\n    }\n\n    onXrSessionStarted(session: XRSession) {\n        Logger.Log(Logger.GetStackTrace(), 'XR Session started');\n\n        this.xrSession = session;\n        this.xrSession.addEventListener('end', () => {\n            this.onXrSessionEnded();\n        });\n\n        const canvas = document.createElement('canvas');\n        this.gl = canvas.getContext('webgl2', {\n            xrCompatible: true\n        });\n\n        this.xrSession.updateRenderState({\n            baseLayer: new XRWebGLLayer(this.xrSession, this.gl)\n        });\n\n        // setup vertex shader\n        const vertexShader = this.gl.createShader(this.gl.VERTEX_SHADER);\n        this.gl.shaderSource(vertexShader, WebGLUtils.vertexShader());\n        this.gl.compileShader(vertexShader);\n\n        // setup fragment shader\n        const fragmentShader = this.gl.createShader(this.gl.FRAGMENT_SHADER);\n        this.gl.shaderSource(fragmentShader, WebGLUtils.fragmentShader());\n        this.gl.compileShader(fragmentShader);\n\n        // setup GLSL program\n        const shaderProgram = this.gl.createProgram();\n        this.gl.attachShader(shaderProgram, vertexShader);\n        this.gl.attachShader(shaderProgram, fragmentShader);\n        this.gl.linkProgram(shaderProgram);\n        this.gl.useProgram(shaderProgram);\n\n        // look up where vertex data needs to go\n        this.positionLocation = this.gl.getAttribLocation(\n            shaderProgram,\n            'a_position'\n        );\n        this.texcoordLocation = this.gl.getAttribLocation(\n            shaderProgram,\n            'a_texCoord'\n        );\n        // Create a buffer to put three 2d clip space points in\n        this.positionBuffer = this.gl.createBuffer();\n        // Bind it to ARRAY_BUFFER (think of it as ARRAY_BUFFER = positionBuffer)\n        this.gl.bindBuffer(this.gl.ARRAY_BUFFER, this.positionBuffer);\n\n        // Turn on the position attribute\n        this.gl.enableVertexAttribArray(this.positionLocation);\n        // Create a texture.\n        const texture = this.gl.createTexture();\n        this.gl.bindTexture(this.gl.TEXTURE_2D, texture);\n        // Set the parameters so we can render any size image.\n        this.gl.texParameteri(\n            this.gl.TEXTURE_2D,\n            this.gl.TEXTURE_WRAP_S,\n            this.gl.CLAMP_TO_EDGE\n        );\n        this.gl.texParameteri(\n            this.gl.TEXTURE_2D,\n            this.gl.TEXTURE_WRAP_T,\n            this.gl.CLAMP_TO_EDGE\n        );\n        this.gl.texParameteri(\n            this.gl.TEXTURE_2D,\n            this.gl.TEXTURE_MIN_FILTER,\n            this.gl.NEAREST\n        );\n        this.gl.texParameteri(\n            this.gl.TEXTURE_2D,\n            this.gl.TEXTURE_MAG_FILTER,\n            this.gl.NEAREST\n        );\n\n        this.texcoordBuffer = this.gl.createBuffer();\n        // lookup uniforms\n        this.resolutionLocation = this.gl.getUniformLocation(\n            shaderProgram,\n            'u_resolution'\n        );\n        this.offsetLocation = this.gl.getUniformLocation(\n            shaderProgram,\n            'u_offset'\n        );\n\n        session.requestReferenceSpace('local').then((refSpace) => {\n            this.xrRefSpace = refSpace;\n            this.xrSession.requestAnimationFrame(\n                (time: DOMHighResTimeStamp, frame: XRFrame) =>\n                    this.onXrFrame(time, frame)\n            );\n        });\n\n        this.onSessionStarted.dispatchEvent(new Event('xrSessionStarted'));\n    }\n\n    onXrFrame(time: DOMHighResTimeStamp, frame: XRFrame) {\n        const pose = frame.getViewerPose(this.xrRefSpace);\n        if (pose) {\n            const matrix = pose.transform.matrix;\n            const mat = [];\n            for (let i = 0; i < 16; i++) {\n                mat[i] = new Float32Array([matrix[i]])[0];\n            }\n\n            // prettier-ignore\n            this.webRtcController.streamMessageController.toStreamerHandlers.get('XRHMDTransform')([\n                mat[0], mat[4], mat[8], mat[12],\n                mat[1], mat[5], mat[9], mat[13], \n                mat[2], mat[6], mat[10], mat[14], \n                mat[3], mat[7], mat[11], mat[15]\n            ]);\n\n            const glLayer = this.xrSession.renderState.baseLayer;\n            // If we do have a valid pose, bind the WebGL layer's framebuffer,\n            // which is where any content to be displayed on the XRDevice must be\n            // rendered.\n            this.gl.bindFramebuffer(this.gl.FRAMEBUFFER, glLayer.framebuffer);\n\n            // Upload the image into the texture. WebGL knows how to extract the current frame from the video element\n            this.gl.texImage2D(\n                this.gl.TEXTURE_2D,\n                0,\n                this.gl.RGBA,\n                this.gl.RGBA,\n                this.gl.UNSIGNED_BYTE,\n                this.webRtcController.videoPlayer.getVideoElement()\n            );\n            this.render(this.webRtcController.videoPlayer.getVideoElement());\n        }\n\n        if (this.webRtcController.config.isFlagEnabled(Flags.XRControllerInput)) {\n            this.xrSession.inputSources.forEach(\n                (source: XRInputSource, index: number, array: XRInputSource[]) => {\n                    this.xrGamepadController.updateStatus(\n                        source,\n                        frame,\n                        this.xrRefSpace\n                    );\n                },\n                this\n            );\n        }\n\n        this.xrSession.requestAnimationFrame(\n            (time: DOMHighResTimeStamp, frame: XRFrame) =>\n                this.onXrFrame(time, frame)\n        );\n\n        this.onFrame.dispatchEvent(new XrFrameEvent({\n            time,\n            frame\n        }));\n    }\n\n    private render(videoElement: HTMLVideoElement) {\n        if (!this.gl) {\n            return;\n        }\n\n        const glLayer = this.xrSession.renderState.baseLayer;\n        this.gl.viewport(\n            0,\n            0,\n            glLayer.framebufferWidth,\n            glLayer.framebufferHeight\n        );\n        this.gl.uniform4f(this.offsetLocation, 1.0, 1.0, 0.0, 0.0);\n\n        // Set rectangle\n        // prettier-ignore\n        this.gl.bufferData(\n            this.gl.ARRAY_BUFFER,\n            new Float32Array([\n                0, 0, \n                videoElement.videoWidth, 0,\n                0, videoElement.videoHeight, \n                0, videoElement.videoHeight,\n                videoElement.videoWidth, 0,\n                videoElement.videoWidth, videoElement.videoHeight\n            ]),\n            this.gl.STATIC_DRAW\n        );\n\n        // Provide texture coordinates for the rectangle\n        this.gl.bindBuffer(this.gl.ARRAY_BUFFER, this.texcoordBuffer);\n        this.gl.bufferData(\n            this.gl.ARRAY_BUFFER,\n            new Float32Array([\n                0.0, 0.0, 1.0, 0.0, 0.0, 1.0, 0.0, 1.0, 1.0, 0.0, 1.0, 1.0\n            ]),\n            this.gl.STATIC_DRAW\n        );\n\n        let size; // components per iteration\n        let type; // the data type\n        let normalize; // normalize the data\n        let stride; // 0 = move forward size * sizeof(type) each iteration to get the next position\n        let offset; // start position of the buffer\n\n        // Bind the position buffer.\n        this.gl.bindBuffer(this.gl.ARRAY_BUFFER, this.positionBuffer);\n        // Tell the position attribute how to get data out of positionBuffer (ARRAY_BUFFER)\n        size = 2; // 2 components per iteration\n        type = this.gl.FLOAT; // the data is 32bit floats\n        normalize = false; // don't normalize the data\n        stride = 0; // 0 = move forward size * sizeof(type) each iteration to get the next position\n        offset = 0; // start at the beginning of the buffer\n        this.gl.vertexAttribPointer(\n            this.positionLocation,\n            size,\n            type,\n            normalize,\n            stride,\n            offset\n        );\n        // Turn on the texcoord attribute\n        this.gl.enableVertexAttribArray(this.texcoordLocation);\n        // bind the texcoord buffer.\n        this.gl.bindBuffer(this.gl.ARRAY_BUFFER, this.texcoordBuffer);\n        // Tell the texcoord attribute how to get data out of texcoordBuffer (ARRAY_BUFFER)\n        size = 2; // 2 components per iteration\n        type = this.gl.FLOAT; // the data is 32bit floats\n        normalize = false; // don't normalize the data\n        stride = 0; // 0 = move forward size * sizeof(type) each iteration to get the next position\n        offset = 0; // start at the beginning of the buffer\n        this.gl.vertexAttribPointer(\n            this.texcoordLocation,\n            size,\n            type,\n            normalize,\n            stride,\n            offset\n        );\n        // set the resolution\n        this.gl.uniform2f(\n            this.resolutionLocation,\n            videoElement.videoWidth,\n            videoElement.videoHeight\n        );\n        // draw the rectangle.\n        const primitiveType = this.gl.TRIANGLES;\n        const count = 6;\n        offset = 0;\n        this.gl.drawArrays(primitiveType, offset, count);\n    }\n\n    static isSessionSupported(mode: XRSessionMode): Promise<boolean> {\n        if (navigator.xr) {\n            return navigator.xr.isSessionSupported(mode);\n        } else {\n            return new Promise<boolean>(() => {\n                return false;\n            });\n        }\n    }\n}\n","module.exports = __WEBPACK_EXTERNAL_MODULE_sdp__;","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\n};","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","// Copyright Epic Games, Inc. All Rights Reserved.\n\nexport { WebRtcPlayerController } from './WebRtcPlayer/WebRtcPlayerController';\nexport { WebXRController } from './WebXR/WebXRController';\nexport {\n    Config,\n    ControlSchemeType,\n    Flags,\n    NumericParameters,\n    TextParameters,\n    OptionParameters,\n    FlagsIds,\n    NumericParametersIds,\n    TextParametersIds,\n    OptionParametersIds,\n    AllSettings\n} from './Config/Config';\nexport { SettingBase } from './Config/SettingBase';\nexport { SettingFlag } from './Config/SettingFlag';\nexport { SettingNumber } from './Config/SettingNumber';\nexport { SettingOption } from './Config/SettingOption';\nexport { SettingText } from './Config/SettingText';\nexport { PixelStreaming } from './PixelStreaming/PixelStreaming';\n\nexport { AFKController as AfkLogic } from './AFK/AFKController';\n\nexport { LatencyTestResults } from './DataChannel/LatencyTestResults';\nexport {\n    EncoderSettings,\n    InitialSettings,\n    WebRTCSettings\n} from './DataChannel/InitialSettings';\nexport { AggregatedStats } from './PeerConnectionController/AggregatedStats';\nexport { Logger } from './Logger/Logger';\nexport { UnquantizedDenormalizedUnsignedCoord as UnquantizedAndDenormalizeUnsigned } from './Util/CoordinateConverter';\nexport { MessageDirection } from './UeInstanceMessage/StreamMessageController';\nexport { MessageSend } from './WebSockets/MessageSend';\nexport { MessageRecv, MessageStreamerList } from './WebSockets/MessageReceive';\nexport { WebSocketController } from './WebSockets/WebSocketController';\nexport { SignallingProtocol } from './WebSockets/SignallingProtocol';\n\nexport { CandidatePairStats } from './PeerConnectionController/CandidatePairStats';\nexport { CandidateStat } from './PeerConnectionController/CandidateStat';\nexport { DataChannelStats } from './PeerConnectionController/DataChannelStats';\nexport {\n    InboundAudioStats,\n    InboundVideoStats\n} from './PeerConnectionController/InboundRTPStats';\nexport { OutBoundVideoStats } from './PeerConnectionController/OutBoundRTPStats';\nexport * from './Util/EventEmitter';\n"],"names":[],"sourceRoot":""}