import * as __WEBPACK_EXTERNAL_MODULE__epicgames_ps_lib_pixelstreamingfrontend_ue5_3_512f3c9b__ from "@epicgames-ps/lib-pixelstreamingfrontend-ue5.3"; import * as __WEBPACK_EXTERNAL_MODULE_jss__ from "jss"; import * as __WEBPACK_EXTERNAL_MODULE_jss_plugin_camel_case_de113355__ from "jss-plugin-camel-case"; import * as __WEBPACK_EXTERNAL_MODULE_jss_plugin_global_ef86f421__ from "jss-plugin-global"; /******/ var __webpack_modules__ = ({ /***/ "./src/Application/Application.ts": /*!****************************************!*\ !*** ./src/Application/Application.ts ***! \****************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Application": () => (/* binding */ Application) /* harmony export */ }); /* harmony import */ var _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @epicgames-ps/lib-pixelstreamingfrontend-ue5.3 */ "@epicgames-ps/lib-pixelstreamingfrontend-ue5.3"); /* harmony import */ var _Overlay_ConnectOverlay__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ../Overlay/ConnectOverlay */ "./src/Overlay/ConnectOverlay.ts"); /* harmony import */ var _Overlay_DisconnectOverlay__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ../Overlay/DisconnectOverlay */ "./src/Overlay/DisconnectOverlay.ts"); /* harmony import */ var _Overlay_PlayOverlay__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ../Overlay/PlayOverlay */ "./src/Overlay/PlayOverlay.ts"); /* harmony import */ var _Overlay_InfoOverlay__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! ../Overlay/InfoOverlay */ "./src/Overlay/InfoOverlay.ts"); /* harmony import */ var _Overlay_ErrorOverlay__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! ../Overlay/ErrorOverlay */ "./src/Overlay/ErrorOverlay.ts"); /* harmony import */ var _Overlay_AFKOverlay__WEBPACK_IMPORTED_MODULE_11__ = __webpack_require__(/*! ../Overlay/AFKOverlay */ "./src/Overlay/AFKOverlay.ts"); /* harmony import */ var _UI_Controls__WEBPACK_IMPORTED_MODULE_12__ = __webpack_require__(/*! ../UI/Controls */ "./src/UI/Controls.ts"); /* harmony import */ var _UI_LabelledButton__WEBPACK_IMPORTED_MODULE_14__ = __webpack_require__(/*! ../UI/LabelledButton */ "./src/UI/LabelledButton.ts"); /* harmony import */ var _UI_SettingsPanel__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ../UI/SettingsPanel */ "./src/UI/SettingsPanel.ts"); /* harmony import */ var _UI_StatsPanel__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../UI/StatsPanel */ "./src/UI/StatsPanel.ts"); /* harmony import */ var _UI_VideoQpIndicator__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ../UI/VideoQpIndicator */ "./src/UI/VideoQpIndicator.ts"); /* harmony import */ var _Config_ConfigUI__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../Config/ConfigUI */ "./src/Config/ConfigUI.ts"); /* harmony import */ var _UI_UIConfigurationTypes__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ../UI/UIConfigurationTypes */ "./src/UI/UIConfigurationTypes.ts"); /* harmony import */ var _UI_FullscreenIcon__WEBPACK_IMPORTED_MODULE_13__ = __webpack_require__(/*! ../UI/FullscreenIcon */ "./src/UI/FullscreenIcon.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * An Application is a combination of UI elements to display and manage a WebRTC Pixel Streaming * connection. It includes features for controlling a stream with mouse and keyboard, * managing connection endpoints, as well as displaying stats and other information about it. */ class Application { /** * @param options - Initialization options */ constructor(options) { this._options = options; this.stream = options.stream; this.onColorModeChanged = options.onColorModeChanged; this.configUI = new _Config_ConfigUI__WEBPACK_IMPORTED_MODULE_1__.ConfigUI(this.stream.config); this.createOverlays(); if ((0,_UI_UIConfigurationTypes__WEBPACK_IMPORTED_MODULE_2__.isPanelEnabled)(options.statsPanelConfig)) { // Add stats panel this.statsPanel = new _UI_StatsPanel__WEBPACK_IMPORTED_MODULE_3__.StatsPanel(); this.uiFeaturesElement.appendChild(this.statsPanel.rootElement); } if ((0,_UI_UIConfigurationTypes__WEBPACK_IMPORTED_MODULE_2__.isPanelEnabled)(options.settingsPanelConfig)) { // Add settings panel this.settingsPanel = new _UI_SettingsPanel__WEBPACK_IMPORTED_MODULE_4__.SettingsPanel(); this.uiFeaturesElement.appendChild(this.settingsPanel.rootElement); this.configureSettings(); } if (!options.videoQpIndicatorConfig || !options.videoQpIndicatorConfig.disableIndicator) { // Add the video stream QP indicator this.videoQpIndicator = new _UI_VideoQpIndicator__WEBPACK_IMPORTED_MODULE_5__.VideoQpIndicator(); this.uiFeaturesElement.appendChild(this.videoQpIndicator.rootElement); } this.createButtons(); this.registerCallbacks(); this.showConnectOrAutoConnectOverlays(); this.setColorMode(this.configUI.isCustomFlagEnabled(_Config_ConfigUI__WEBPACK_IMPORTED_MODULE_1__.LightMode)); } createOverlays() { // build all of the overlays this.disconnectOverlay = new _Overlay_DisconnectOverlay__WEBPACK_IMPORTED_MODULE_6__.DisconnectOverlay(this.stream.videoElementParent); this.connectOverlay = new _Overlay_ConnectOverlay__WEBPACK_IMPORTED_MODULE_7__.ConnectOverlay(this.stream.videoElementParent); this.playOverlay = new _Overlay_PlayOverlay__WEBPACK_IMPORTED_MODULE_8__.PlayOverlay(this.stream.videoElementParent); this.infoOverlay = new _Overlay_InfoOverlay__WEBPACK_IMPORTED_MODULE_9__.InfoOverlay(this.stream.videoElementParent); this.errorOverlay = new _Overlay_ErrorOverlay__WEBPACK_IMPORTED_MODULE_10__.ErrorOverlay(this.stream.videoElementParent); this.afkOverlay = new _Overlay_AFKOverlay__WEBPACK_IMPORTED_MODULE_11__.AFKOverlay(this.stream.videoElementParent); this.disconnectOverlay.onAction(() => this.stream.reconnect()); // Build the webRtc connect overlay Event Listener and show the connect overlay this.connectOverlay.onAction(() => this.stream.connect()); // set up the play overlays action this.playOverlay.onAction(() => this.stream.play()); } /** * Set up button click functions and button functionality */ createButtons() { const controlsUIConfig = { statsButtonType: !!this._options.statsPanelConfig ? this._options.statsPanelConfig.visibilityButtonConfig : undefined, settingsButtonType: !!this._options.settingsPanelConfig ? this._options.settingsPanelConfig.visibilityButtonConfig : undefined, fullscreenButtonType: this._options.fullScreenControlsConfig, xrIconType: this._options.xrControlsConfig }; // Setup controls const controls = new _UI_Controls__WEBPACK_IMPORTED_MODULE_12__.Controls(controlsUIConfig); this.uiFeaturesElement.appendChild(controls.rootElement); // When we fullscreen we want this element to be the root const fullScreenButton = // Depending on if we're creating an internal button, or using an external one (!!this._options.fullScreenControlsConfig && this._options.fullScreenControlsConfig.creationMode === _UI_UIConfigurationTypes__WEBPACK_IMPORTED_MODULE_2__.UIElementCreationMode.UseCustomElement) // Either create a fullscreen class based on the external button ? new _UI_FullscreenIcon__WEBPACK_IMPORTED_MODULE_13__.FullScreenIconExternal(this._options.fullScreenControlsConfig.customElement) // Or use the one created by the Controls initializer earlier : controls.fullscreenIcon; if (fullScreenButton) { fullScreenButton.fullscreenElement = /iPad|iPhone|iPod/.test(navigator.userAgent) ? this.stream.videoElementParent.getElementsByTagName("video")[0] : this.rootElement; } // Add settings button to controls const settingsButton = !!controls.settingsIcon ? controls.settingsIcon.rootElement : this._options.settingsPanelConfig.visibilityButtonConfig.customElement; if (!!settingsButton) settingsButton.onclick = () => this.settingsClicked(); if (!!this.settingsPanel) this.settingsPanel.settingsCloseButton.onclick = () => this.settingsClicked(); // Add WebXR button to controls const xrButton = !!controls.xrIcon ? controls.xrIcon.rootElement : this._options.xrControlsConfig.creationMode === _UI_UIConfigurationTypes__WEBPACK_IMPORTED_MODULE_2__.UIElementCreationMode.UseCustomElement ? this._options.xrControlsConfig.customElement : undefined; if (!!xrButton) xrButton.onclick = () => this.stream.toggleXR(); // setup the stats/info button const statsButton = !!controls.statsIcon ? controls.statsIcon.rootElement : this._options.statsPanelConfig.visibilityButtonConfig.customElement; if (!!statsButton) statsButton.onclick = () => this.statsClicked(); if (!!this.statsPanel) { this.statsPanel.statsCloseButton.onclick = () => this.statsClicked(); } // Add command buttons (if we have somewhere to add them to) if (!!this.settingsPanel) { // Add button for toggle fps const showFPSButton = new _UI_LabelledButton__WEBPACK_IMPORTED_MODULE_14__.LabelledButton('Show FPS', 'Toggle'); showFPSButton.addOnClickListener(() => { this.stream.requestShowFps(); }); // Add button for restart stream const restartStreamButton = new _UI_LabelledButton__WEBPACK_IMPORTED_MODULE_14__.LabelledButton('Restart Stream', 'Restart'); restartStreamButton.addOnClickListener(() => { this.stream.reconnect(); }); // Add button for request keyframe const requestKeyframeButton = new _UI_LabelledButton__WEBPACK_IMPORTED_MODULE_14__.LabelledButton('Request keyframe', 'Request'); requestKeyframeButton.addOnClickListener(() => { this.stream.requestIframe(); }); const commandsSectionElem = this.configUI.buildSectionWithHeading(this.settingsPanel.settingsContentElement, 'Commands'); commandsSectionElem.appendChild(showFPSButton.rootElement); commandsSectionElem.appendChild(requestKeyframeButton.rootElement); commandsSectionElem.appendChild(restartStreamButton.rootElement); } } /** * Configure the settings with on change listeners and any additional per experience settings. */ configureSettings() { // This builds all the settings sections and flags under this `settingsContent` element. this.configUI.populateSettingsElement(this.settingsPanel.settingsContentElement); this.configUI.addCustomFlagOnSettingChangedListener(_Config_ConfigUI__WEBPACK_IMPORTED_MODULE_1__.LightMode, (isLightMode) => { this.configUI.setCustomFlagLabel(_Config_ConfigUI__WEBPACK_IMPORTED_MODULE_1__.LightMode, `Color Scheme: ${isLightMode ? 'Light' : 'Dark'} Mode`); this.setColorMode(isLightMode); }); } registerCallbacks() { this.stream.addEventListener('afkWarningActivate', ({ data: { countDown, dismissAfk } }) => this.showAfkOverlay(countDown, dismissAfk)); this.stream.addEventListener('afkWarningUpdate', ({ data: { countDown } }) => this.afkOverlay.updateCountdown(countDown)); this.stream.addEventListener('afkWarningDeactivate', () => this.afkOverlay.hide()); this.stream.addEventListener('afkTimedOut', () => this.afkOverlay.hide()); this.stream.addEventListener('videoEncoderAvgQP', ({ data: { avgQP } }) => this.onVideoEncoderAvgQP(avgQP)); this.stream.addEventListener('webRtcSdp', () => this.onWebRtcSdp()); this.stream.addEventListener('webRtcAutoConnect', () => this.onWebRtcAutoConnect()); this.stream.addEventListener('webRtcConnecting', () => this.onWebRtcConnecting()); this.stream.addEventListener('webRtcConnected', () => this.onWebRtcConnected()); this.stream.addEventListener('webRtcFailed', () => this.onWebRtcFailed()); this.stream.addEventListener('webRtcDisconnected', ({ data: { eventString, allowClickToReconnect } }) => this.onDisconnect(eventString, allowClickToReconnect)); this.stream.addEventListener('videoInitialized', () => this.onVideoInitialized()); this.stream.addEventListener('streamLoading', () => this.onStreamLoading()); this.stream.addEventListener('playStreamError', ({ data: { message } }) => this.onPlayStreamError(message)); this.stream.addEventListener('playStream', () => this.onPlayStream()); this.stream.addEventListener('playStreamRejected', ({ data: { reason } }) => this.onPlayStreamRejected(reason)); this.stream.addEventListener('loadFreezeFrame', ({ data: { shouldShowPlayOverlay } }) => this.onLoadFreezeFrame(shouldShowPlayOverlay)); this.stream.addEventListener('statsReceived', ({ data: { aggregatedStats } }) => this.onStatsReceived(aggregatedStats)); this.stream.addEventListener('latencyTestResult', ({ data: { latencyTimings } }) => this.onLatencyTestResults(latencyTimings)); this.stream.addEventListener('dataChannelLatencyTestResult', ({ data: { result } }) => this.onDataChannelLatencyTestResults(result)); this.stream.addEventListener('streamerListMessage', ({ data: { messageStreamerList, autoSelectedStreamerId, wantedStreamerId } }) => this.handleStreamerListMessage(messageStreamerList, autoSelectedStreamerId, wantedStreamerId)); this.stream.addEventListener('settingsChanged', (event) => this.configUI.onSettingsChanged(event)); this.stream.addEventListener('playerCount', ({ data: { count } }) => this.onPlayerCount(count)); } /** * Gets the rootElement of the application, video stream and all UI are children of this element. */ get rootElement() { if (!this._rootElement) { this._rootElement = document.createElement('div'); this._rootElement.id = 'playerUI'; this._rootElement.classList.add('noselect'); this._rootElement.appendChild(this.stream.videoElementParent); this._rootElement.appendChild(this.uiFeaturesElement); } return this._rootElement; } /** * Gets the element that contains all the UI features, like the stats and settings panels. */ get uiFeaturesElement() { if (!this._uiFeatureElement) { this._uiFeatureElement = document.createElement('div'); this._uiFeatureElement.id = 'uiFeatures'; } return this._uiFeatureElement; } /** * Shows the disconnect overlay * @param updateText - the text that will be displayed in the overlay */ showDisconnectOverlay(updateText) { this.hideCurrentOverlay(); this.updateDisconnectOverlay(updateText); this.disconnectOverlay.show(); this.currentOverlay = this.disconnectOverlay; } /** * Update the disconnect overlays span text * @param updateText - the new countdown number */ updateDisconnectOverlay(updateText) { this.disconnectOverlay.update(updateText); } /** * Activates the disconnect overlays action */ onDisconnectionAction() { this.disconnectOverlay.activate(); } /** * Hides the current overlay */ hideCurrentOverlay() { if (this.currentOverlay != null) { this.currentOverlay.hide(); this.currentOverlay = null; } } /** * Shows the connect overlay */ showConnectOverlay() { this.hideCurrentOverlay(); this.connectOverlay.show(); this.currentOverlay = this.connectOverlay; } /** * Shows the play overlay */ showPlayOverlay() { this.hideCurrentOverlay(); this.playOverlay.show(); this.currentOverlay = this.playOverlay; } /** * Shows the text overlay * @param text - the text that will be shown in the overlay */ showTextOverlay(text) { this.hideCurrentOverlay(); this.infoOverlay.update(text); this.infoOverlay.show(); this.currentOverlay = this.infoOverlay; } /** * Shows the error overlay * @param text - the text that will be shown in the overlay */ showErrorOverlay(text) { this.hideCurrentOverlay(); this.errorOverlay.update(text); this.errorOverlay.show(); this.currentOverlay = this.errorOverlay; } /** * Shows or hides the settings panel if clicked */ settingsClicked() { var _a; (_a = this.statsPanel) === null || _a === void 0 ? void 0 : _a.hide(); this.settingsPanel.toggleVisibility(); } /** * Shows or hides the stats panel if clicked */ statsClicked() { var _a; (_a = this.settingsPanel) === null || _a === void 0 ? void 0 : _a.hide(); this.statsPanel.toggleVisibility(); } /** * Activates the connect overlays action */ onConnectAction() { this.connectOverlay.activate(); } /** * Activates the play overlays action */ onPlayAction() { this.playOverlay.activate(); } /** * Shows the afk overlay * @param countDown - the countdown number for the afk countdown */ showAfkOverlay(countDown, dismissAfk) { this.hideCurrentOverlay(); this.afkOverlay.updateCountdown(countDown); this.afkOverlay.onAction(() => dismissAfk()); this.afkOverlay.show(); this.currentOverlay = this.afkOverlay; } /** * Show the Connect Overlay or auto connect */ showConnectOrAutoConnectOverlays() { // set up if the auto play will be used or regular click to start if (!this.stream.config.isFlagEnabled(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.AutoConnect)) { this.showConnectOverlay(); } } /** * Show the webRtcAutoConnect Overlay and connect */ onWebRtcAutoConnect() { this.showTextOverlay('Auto Connecting Now'); } /** * Set up functionality to happen when receiving a webRTC answer */ onWebRtcSdp() { this.showTextOverlay('WebRTC Connection Negotiated'); } /** * Shows a text overlay to alert the user the stream is currently loading */ onStreamLoading() { // build the spinner span const spinnerSpan = document.createElement('span'); spinnerSpan.className = 'visually-hidden'; spinnerSpan.innerHTML = 'Loading...'; // build the spinner div const spinnerDiv = document.createElement('div'); spinnerDiv.id = 'loading-spinner'; spinnerDiv.className = 'spinner-border ms-2'; spinnerDiv.setAttribute('role', 'status'); // append the spinner to the element spinnerDiv.appendChild(spinnerSpan); this.showTextOverlay('Loading Stream ' + spinnerDiv.outerHTML); } /** * Event fired when the video is disconnected - displays the error overlay and resets the buttons stream tools upon disconnect * @param eventString - the event text that will be shown in the overlay * @param allowClickToReconnect - true if we want to allow the user to click to reconnect. Otherwise it's just a message. */ onDisconnect(eventString, allowClickToReconnect) { var _a; const overlayMessage = 'Disconnected' + (eventString ? `: ${eventString}` : ''); if (allowClickToReconnect) { this.showDisconnectOverlay(`${overlayMessage} Click To Restart.`); } else { this.showErrorOverlay(overlayMessage); } // disable starting a latency checks (_a = this.statsPanel) === null || _a === void 0 ? void 0 : _a.onDisconnect(); } /** * Handles when Web Rtc is connecting */ onWebRtcConnecting() { this.showTextOverlay('Starting connection to server, please wait'); } /** * Handles when Web Rtc has connected */ onWebRtcConnected() { this.showTextOverlay('WebRTC connected, waiting for video'); } /** * Handles when Web Rtc fails to connect */ onWebRtcFailed() { this.showErrorOverlay('Unable to setup video'); } onLoadFreezeFrame(shouldShowPlayOverlay) { if (shouldShowPlayOverlay === true) { _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'showing play overlay'); this.showPlayOverlay(); } } onPlayStream() { this.hideCurrentOverlay(); } onPlayStreamError(message) { this.showErrorOverlay(message); } onPlayStreamRejected(onRejectedReason) { this.showPlayOverlay(); } onVideoInitialized() { var _a; if (!this.stream.config.isFlagEnabled(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.AutoPlayVideo)) { this.showPlayOverlay(); } (_a = this.statsPanel) === null || _a === void 0 ? void 0 : _a.onVideoInitialized(this.stream); } /** * Set up functionality to happen when calculating the average video encoder qp * @param QP - the quality number of the stream */ onVideoEncoderAvgQP(QP) { // Update internal QP indicator if one is present if (!!this.videoQpIndicator) { this.videoQpIndicator.updateQpTooltip(QP); } } onInitialSettings(settings) { var _a; if (settings.PixelStreamingSettings) { (_a = this.statsPanel) === null || _a === void 0 ? void 0 : _a.configure(settings.PixelStreamingSettings); } } onStatsReceived(aggregatedStats) { var _a; // Grab all stats we can off the aggregated stats (_a = this.statsPanel) === null || _a === void 0 ? void 0 : _a.handleStats(aggregatedStats); } onLatencyTestResults(latencyTimings) { var _a; (_a = this.statsPanel) === null || _a === void 0 ? void 0 : _a.latencyTest.handleTestResult(latencyTimings); } onDataChannelLatencyTestResults(result) { var _a; (_a = this.statsPanel) === null || _a === void 0 ? void 0 : _a.dataChannelLatencyTest.handleTestResult(result); } onPlayerCount(playerCount) { var _a; (_a = this.statsPanel) === null || _a === void 0 ? void 0 : _a.handlePlayerCount(playerCount); } handleStreamerListMessage(messageStreamingList, autoSelectedStreamerId, wantedStreamerId) { const waitForStreamer = this.stream.config.isFlagEnabled(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.WaitForStreamer); const isReconnecting = this.stream.isReconnecting(); let message = null; let allowRestart = true; if (!autoSelectedStreamerId) { if (waitForStreamer && wantedStreamerId) { if (isReconnecting) { message = `Waiting for ${wantedStreamerId} to become available.`; allowRestart = false; } else { message = `Gave up waiting for ${wantedStreamerId} to become available. Click to try again`; if (messageStreamingList.ids.length > 0) { message += ` or select a streamer from the settings menu.`; } allowRestart = true; } } else if (messageStreamingList.ids.length == 0) { if (isReconnecting) { message = `Waiting for a streamer to become available.`; allowRestart = false; } else { message = `No streamers available. Click to try again.`; allowRestart = true; } } else { message = `Multiple streamers available. Select one from the settings menu.`; allowRestart = false; } if (allowRestart) { this.showDisconnectOverlay(message); } else { this.showTextOverlay(message); } } } /** * Set light/dark color mode * @param isLightMode - should we use a light or dark color scheme */ setColorMode(isLightMode) { if (this.onColorModeChanged) { this.onColorModeChanged(isLightMode); } } } /***/ }), /***/ "./src/Config/ConfigUI.ts": /*!********************************!*\ !*** ./src/Config/ConfigUI.ts ***! \********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "ConfigUI": () => (/* binding */ ConfigUI), /* harmony export */ "LightMode": () => (/* binding */ LightMode) /* harmony export */ }); /* harmony import */ var _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @epicgames-ps/lib-pixelstreamingfrontend-ue5.3 */ "@epicgames-ps/lib-pixelstreamingfrontend-ue5.3"); /* harmony import */ var _SettingUIFlag__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./SettingUIFlag */ "./src/Config/SettingUIFlag.ts"); /* harmony import */ var _SettingUINumber__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./SettingUINumber */ "./src/Config/SettingUINumber.ts"); /* harmony import */ var _SettingUIText__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./SettingUIText */ "./src/Config/SettingUIText.ts"); /* harmony import */ var _SettingUIOption__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./SettingUIOption */ "./src/Config/SettingUIOption.ts"); // Copyright Epic Games, Inc. All Rights Reserved. const LightMode = 'LightMode'; class ConfigUI { // ------------ Settings ----------------- constructor(config) { this.customFlags = new Map(); /* A map of flags that can be toggled - options that can be set in the application - e.g. Use Mic? */ this.flagsUi = new Map(); /* A map of numerical settings - options that can be in the application - e.g. MinBitrate */ this.numericParametersUi = new Map(); /* A map of text settings - e.g. signalling server url */ this.textParametersUi = new Map(); /* A map of enum based settings - e.g. preferred codec */ this.optionParametersUi = new Map(); this.createCustomUISettings(config.useUrlParams); this.registerSettingsUIComponents(config); } /** * Create custom UI settings that are not provided by the Pixel Streaming library. */ createCustomUISettings(useUrlParams) { this.customFlags.set(LightMode, new _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.SettingFlag(LightMode, 'Color Scheme: Dark Mode', 'Page styling will be either light or dark', false /*if want to use system pref: (window.matchMedia && window.matchMedia('(prefers-color-scheme: light)').matches)*/, useUrlParams, (isLightMode, setting) => { setting.label = `Color Scheme: ${isLightMode ? 'Light' : 'Dark'} Mode`; })); } /** * Creates UI wrapper components for each setting element in config. * @param config */ registerSettingsUIComponents(config) { for (const setting of config.getFlags()) { this.flagsUi.set(setting.id, new _SettingUIFlag__WEBPACK_IMPORTED_MODULE_1__.SettingUIFlag(setting)); } for (const setting of Array.from(this.customFlags.values())) { this.flagsUi.set(setting.id, new _SettingUIFlag__WEBPACK_IMPORTED_MODULE_1__.SettingUIFlag(setting)); } for (const setting of config.getTextSettings()) { this.textParametersUi.set(setting.id, new _SettingUIText__WEBPACK_IMPORTED_MODULE_2__.SettingUIText(setting)); } for (const setting of config.getNumericSettings()) { this.numericParametersUi.set(setting.id, new _SettingUINumber__WEBPACK_IMPORTED_MODULE_3__.SettingUINumber(setting)); } for (const setting of config.getOptionSettings()) { this.optionParametersUi.set(setting.id, new _SettingUIOption__WEBPACK_IMPORTED_MODULE_4__.SettingUIOption(setting)); } } /** * Make DOM elements for a settings section with a heading. * @param settingsElem The parent container for our DOM elements. * @param sectionHeading The heading element to go into the section. * @returns The constructed DOM element for the section. */ buildSectionWithHeading(settingsElem, sectionHeading) { // make section element const sectionElem = document.createElement('section'); sectionElem.classList.add('settingsContainer'); // make section heading const psSettingsHeader = document.createElement('div'); psSettingsHeader.classList.add('settingsHeader'); psSettingsHeader.classList.add('settings-text'); psSettingsHeader.textContent = sectionHeading; // add section and heading to parent settings element sectionElem.appendChild(psSettingsHeader); settingsElem.appendChild(sectionElem); return sectionElem; } /** * Setup flags with their default values and add them to the `Config.flags` map. * @param settingsElem - The element that contains all the individual settings sections, flags, and so on. */ populateSettingsElement(settingsElem) { /* Setup all Pixel Streaming specific settings */ const psSettingsSection = this.buildSectionWithHeading(settingsElem, 'Pixel Streaming'); // make settings show up in DOM this.addSettingText(psSettingsSection, this.textParametersUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.TextParameters.SignallingServerUrl)); this.addSettingOption(psSettingsSection, this.optionParametersUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.OptionParameters.StreamerId)); this.addSettingFlag(psSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.AutoConnect)); this.addSettingFlag(psSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.AutoPlayVideo)); this.addSettingFlag(psSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.BrowserSendOffer)); this.addSettingFlag(psSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.UseMic)); this.addSettingFlag(psSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.StartVideoMuted)); this.addSettingFlag(psSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.IsQualityController)); this.addSettingFlag(psSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.ForceMonoAudio)); this.addSettingFlag(psSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.ForceTURN)); this.addSettingFlag(psSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.SuppressBrowserKeys)); this.addSettingFlag(psSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.AFKDetection)); this.addSettingFlag(psSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.WaitForStreamer)); this.addSettingNumeric(psSettingsSection, this.numericParametersUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.NumericParameters.AFKTimeoutSecs)); this.addSettingNumeric(psSettingsSection, this.numericParametersUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.NumericParameters.MaxReconnectAttempts)); this.addSettingNumeric(psSettingsSection, this.numericParametersUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.NumericParameters.StreamerAutoJoinInterval)); /* Setup all view/ui related settings under this section */ const viewSettingsSection = this.buildSectionWithHeading(settingsElem, 'UI'); this.addSettingFlag(viewSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.MatchViewportResolution)); this.addSettingFlag(viewSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.HoveringMouseMode)); this.addSettingFlag(viewSettingsSection, this.flagsUi.get(LightMode)); /* Setup all encoder related settings under this section */ const inputSettingsSection = this.buildSectionWithHeading(settingsElem, 'Input'); this.addSettingFlag(inputSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.KeyboardInput)); this.addSettingFlag(inputSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.MouseInput)); this.addSettingFlag(inputSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.TouchInput)); this.addSettingFlag(inputSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.GamepadInput)); this.addSettingFlag(inputSettingsSection, this.flagsUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Flags.XRControllerInput)); /* Setup all encoder related settings under this section */ const encoderSettingsSection = this.buildSectionWithHeading(settingsElem, 'Encoder'); this.addSettingNumeric(encoderSettingsSection, this.numericParametersUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.NumericParameters.MinQP)); this.addSettingNumeric(encoderSettingsSection, this.numericParametersUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.NumericParameters.MaxQP)); const preferredCodecOption = this.optionParametersUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.OptionParameters.PreferredCodec); this.addSettingOption(encoderSettingsSection, this.optionParametersUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.OptionParameters.PreferredCodec)); if (preferredCodecOption && [...preferredCodecOption.selector.options] .map((o) => o.value) .includes('Only available on Chrome')) { preferredCodecOption.disable(); } /* Setup all webrtc related settings under this section */ const webrtcSettingsSection = this.buildSectionWithHeading(settingsElem, 'WebRTC'); this.addSettingNumeric(webrtcSettingsSection, this.numericParametersUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.NumericParameters.WebRTCFPS)); this.addSettingNumeric(webrtcSettingsSection, this.numericParametersUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.NumericParameters.WebRTCMinBitrate)); this.addSettingNumeric(webrtcSettingsSection, this.numericParametersUi.get(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.NumericParameters.WebRTCMaxBitrate)); } /** * Add a SettingText element to a particular settings section in the DOM and registers that text in the text settings map. * @param settingsSection The settings section HTML element. * @param settingText The textual settings object. */ addSettingText(settingsSection, settingText) { if (settingText) { settingsSection.appendChild(settingText.rootElement); this.textParametersUi.set(settingText.setting.id, settingText); } } /** * Add a SettingFlag element to a particular settings section in the DOM and registers that flag in the Config.flag map. * @param settingsSection The settings section HTML element. * @param settingFlag The settings flag object. */ addSettingFlag(settingsSection, settingFlag) { if (settingFlag) { settingsSection.appendChild(settingFlag.rootElement); this.flagsUi.set(settingFlag.setting.id, settingFlag); } } /** * Add a numeric setting element to a particular settings section in the DOM and registers that flag in the Config.numericParameters map. * @param settingsSection The settings section HTML element. * @param settingFlag The settings flag object. */ addSettingNumeric(settingsSection, setting) { if (setting) { settingsSection.appendChild(setting.rootElement); this.numericParametersUi.set(setting.setting.id, setting); } } /** * Add an enum based settings element to a particular settings section in the DOM and registers that flag in the Config.enumParameters map. * @param settingsSection The settings section HTML element. * @param settingFlag The settings flag object. */ addSettingOption(settingsSection, setting) { if (setting) { settingsSection.appendChild(setting.rootElement); this.optionParametersUi.set(setting.setting.id, setting); } } onSettingsChanged({ data: { id, target, type } }) { if (type === 'flag') { const _id = id; const _target = target; const setting = this.flagsUi.get(_id); if (setting) { if (setting.flag !== _target.flag) { setting.flag = _target.flag; } if (setting.label !== _target.label) { setting.label = _target.label; } } } else if (type === 'number') { const _id = id; const _target = target; const setting = this.numericParametersUi.get(_id); if (setting) { if (setting.number !== _target.number) { setting.number = _target.number; } if (setting.label !== _target.label) { setting.label = _target.label; } } } else if (type === 'text') { const _id = id; const _target = target; const setting = this.textParametersUi.get(_id); if (setting) { if (setting.text !== _target.text) { setting.text = _target.text; } if (setting.label !== _target.label) { setting.label = _target.label; } } } else if (type === 'option') { const _id = id; const _target = target; const setting = this.optionParametersUi.get(_id); if (setting) { const uiOptions = setting.options; const targetOptions = _target.options; if (uiOptions.length !== targetOptions.length || !uiOptions.every((value) => targetOptions.includes(value))) { setting.options = _target.options; } if (setting.selected !== _target.selected) { setting.selected = _target.selected; } if (setting.label !== _target.label) { setting.label = _target.label; } } } } /** * 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. */ addCustomFlagOnSettingChangedListener(id, onChangeListener) { if (this.customFlags.has(id)) { this.customFlags.get(id).onChange = onChangeListener; } } /** * Set the label for the flag. * @param id The id of the flag. * @param label The new label to use for the flag. */ setCustomFlagLabel(id, label) { if (!this.customFlags.has(id)) { _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.Warning(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Cannot set label for flag called ${id} - it does not exist in the Config.flags map.`); } else { this.customFlags.get(id).label = label; this.flagsUi.get(id).label = label; } } /** * 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. */ isCustomFlagEnabled(id) { return this.customFlags.get(id).flag; } } /***/ }), /***/ "./src/Config/SettingUIBase.ts": /*!*************************************!*\ !*** ./src/Config/SettingUIBase.ts ***! \*************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SettingUIBase": () => (/* binding */ SettingUIBase) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Base class for a setting that has a text label, an arbitrary setting value it stores, an a HTML element that represents this setting. */ class SettingUIBase { constructor(setting) { this._setting = setting; } /** * @returns The setting component. */ get setting() { return this._setting; } /** * @returns Return or creates a HTML element that represents this setting in the DOM. */ get rootElement() { if (!this._rootElement) { this._rootElement = document.createElement('div'); } return this._rootElement; } } /***/ }), /***/ "./src/Config/SettingUIFlag.ts": /*!*************************************!*\ !*** ./src/Config/SettingUIFlag.ts ***! \*************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SettingUIFlag": () => (/* binding */ SettingUIFlag) /* harmony export */ }); /* harmony import */ var _SettingUIBase__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./SettingUIBase */ "./src/Config/SettingUIBase.ts"); // Copyright Epic Games, Inc. All Rights Reserved. class SettingUIFlag extends _SettingUIBase__WEBPACK_IMPORTED_MODULE_0__.SettingUIBase { constructor(setting) { super(setting); this.label = setting.label; this.flag = setting.flag; } /** * @returns The setting component. */ get setting() { return this._setting; } get settingsTextElem() { if (!this._settingsTextElem) { this._settingsTextElem = document.createElement('div'); this._settingsTextElem.innerText = this.setting._label; this._settingsTextElem.title = this.setting.description; } return this._settingsTextElem; } get checkbox() { if (!this._checkbox) { this._checkbox = document.createElement('input'); this._checkbox.type = 'checkbox'; } return this._checkbox; } /** * @returns Return or creates a HTML element that represents this setting in the DOM. */ get rootElement() { if (!this._rootElement) { // create root div with "setting" css class this._rootElement = document.createElement('div'); this._rootElement.id = this.setting.id; this._rootElement.classList.add('setting'); // create div element to contain our setting's text this._rootElement.appendChild(this.settingsTextElem); // create label element to wrap out input type const wrapperLabel = document.createElement('label'); wrapperLabel.classList.add('tgl-switch'); this._rootElement.appendChild(wrapperLabel); // create input type=checkbox this.checkbox.title = this.setting.description; this.checkbox.classList.add('tgl'); this.checkbox.classList.add('tgl-flat'); const slider = document.createElement('div'); slider.classList.add('tgl-slider'); wrapperLabel.appendChild(this.checkbox); wrapperLabel.appendChild(slider); // setup on change from checkbox this.checkbox.addEventListener('change', () => { if (this.setting.flag !== this.checkbox.checked) { this.setting.flag = this.checkbox.checked; this.setting.updateURLParams(); } }); } return this._rootElement; } /** * Update the setting's stored value. * @param inValue The new value for the setting. */ set flag(inValue) { this.checkbox.checked = inValue; } /** * Get value */ get flag() { return this.checkbox.checked; } /** * Set the label text for the setting. * @param label setting label. */ set label(inLabel) { this.settingsTextElem.innerText = inLabel; } /** * Get label */ get label() { return this.settingsTextElem.innerText; } } /***/ }), /***/ "./src/Config/SettingUINumber.ts": /*!***************************************!*\ !*** ./src/Config/SettingUINumber.ts ***! \***************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SettingUINumber": () => (/* binding */ SettingUINumber) /* harmony export */ }); /* harmony import */ var _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @epicgames-ps/lib-pixelstreamingfrontend-ue5.3 */ "@epicgames-ps/lib-pixelstreamingfrontend-ue5.3"); /* harmony import */ var _SettingUIBase__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./SettingUIBase */ "./src/Config/SettingUIBase.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * A number spinner with a text label beside it. */ class SettingUINumber extends _SettingUIBase__WEBPACK_IMPORTED_MODULE_1__.SettingUIBase { constructor(setting) { super(setting); this.label = this.setting.label; this.number = this.setting.number; } /** * @returns The setting component. */ get setting() { return this._setting; } get settingsTextElem() { if (!this._settingsTextElem) { this._settingsTextElem = document.createElement('label'); this._settingsTextElem.innerText = this.setting.label; this._settingsTextElem.title = this.setting.description; } return this._settingsTextElem; } /** * Get the HTMLInputElement for the button. */ get spinner() { if (!this._spinner) { this._spinner = document.createElement('input'); this._spinner.type = 'number'; this._spinner.min = this.setting.min.toString(); this._spinner.max = this.setting.max.toString(); this._spinner.value = this.setting.number.toString(); this._spinner.title = this.setting.description; this._spinner.classList.add('form-control'); } return this._spinner; } /** * @returns Return or creates a HTML element that represents this setting in the DOM. */ get rootElement() { if (!this._rootElement) { // create root div with "setting" css class this._rootElement = document.createElement('div'); this._rootElement.classList.add('setting'); this._rootElement.classList.add('form-group'); // create div element to contain our setting's text this._rootElement.appendChild(this.settingsTextElem); // create label element to wrap out input type this._rootElement.appendChild(this.spinner); // setup onchange this.spinner.onchange = (event) => { const inputElem = event.target; const parsedValue = Number.parseInt(inputElem.value); if (Number.isNaN(parsedValue)) { _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.Warning(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `Could not parse value change into a valid number - value was ${inputElem.value}, resetting value to ${this.setting.min}`); if (this.setting.number !== this.setting.min) { this.setting.number = this.setting.min; } } else { if (this.setting.number !== parsedValue) { this.setting.number = parsedValue; this.setting.updateURLParams(); } } }; } return this._rootElement; } /** * Set the number in the spinner (will be clamped within range). */ set number(newNumber) { this.spinner.value = this.setting.clamp(newNumber).toString(); } /** * Get value */ get number() { return +this.spinner.value; } /** * Set the label text for the setting. * @param label setting label. */ set label(inLabel) { this.settingsTextElem.innerText = inLabel; } /** * Get label */ get label() { return this.settingsTextElem.innerText; } } /***/ }), /***/ "./src/Config/SettingUIOption.ts": /*!***************************************!*\ !*** ./src/Config/SettingUIOption.ts ***! \***************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SettingUIOption": () => (/* binding */ SettingUIOption) /* harmony export */ }); /* harmony import */ var _SettingUIBase__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./SettingUIBase */ "./src/Config/SettingUIBase.ts"); // Copyright Epic Games, Inc. All Rights Reserved. class SettingUIOption extends _SettingUIBase__WEBPACK_IMPORTED_MODULE_0__.SettingUIBase { constructor(setting) { super(setting); this.label = this.setting.label; this.options = this.setting.options; this.selected = this.setting.selected; } /** * @returns The setting component. */ get setting() { return this._setting; } get selector() { if (!this._selector) { this._selector = document.createElement('select'); this._selector.classList.add('form-control'); this._selector.classList.add('settings-option'); } return this._selector; } get settingsTextElem() { if (!this._settingsTextElem) { this._settingsTextElem = document.createElement('div'); this._settingsTextElem.innerText = this.setting.label; this._settingsTextElem.title = this.setting.description; } return this._settingsTextElem; } /** * Set the label text for the setting. * @param label setting label. */ set label(inLabel) { this.settingsTextElem.innerText = inLabel; } /** * Get label */ get label() { return this.settingsTextElem.innerText; } /** * @returns Return or creates a HTML element that represents this setting in the DOM. */ get rootElement() { if (!this._rootElement) { // create root div with "setting" css class this._rootElement = document.createElement('div'); this._rootElement.id = this.setting.id; this._rootElement.classList.add('setting'); this._rootElement.classList.add('form-group'); // create div element to contain our setting's text this._rootElement.appendChild(this.settingsTextElem); // create label element to wrap out input type const wrapperLabel = document.createElement('label'); this._rootElement.appendChild(wrapperLabel); // create select element this.selector.title = this.setting.description; wrapperLabel.appendChild(this.selector); // setup on change from selector this.selector.onchange = () => { if (this.setting.selected !== this.selector.value) { this.setting.selected = this.selector.value; this.setting.updateURLParams(); } }; } return this._rootElement; } set options(values) { for (let i = this.selector.options.length - 1; i >= 0; i--) { this.selector.remove(i); } values.forEach((value) => { const opt = document.createElement('option'); opt.value = value; opt.innerHTML = value; this.selector.appendChild(opt); }); } get options() { return [...this.selector.options].map((o) => o.value); } 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' const filteredList = this.options.filter((option) => option.indexOf(value) !== -1); if (filteredList.length) { this.selector.value = filteredList[0]; } } get selected() { return this.selector.value; } disable() { this.selector.disabled = true; } enable() { this.selector.disabled = false; } } /***/ }), /***/ "./src/Config/SettingUIText.ts": /*!*************************************!*\ !*** ./src/Config/SettingUIText.ts ***! \*************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SettingUIText": () => (/* binding */ SettingUIText) /* harmony export */ }); /* harmony import */ var _SettingUIBase__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./SettingUIBase */ "./src/Config/SettingUIBase.ts"); // Copyright Epic Games, Inc. All Rights Reserved. class SettingUIText extends _SettingUIBase__WEBPACK_IMPORTED_MODULE_0__.SettingUIBase { constructor(setting) { super(setting); this.label = this.setting.label; this.text = this.setting.text; } /** * @returns The setting component. */ get setting() { return this._setting; } get settingsTextElem() { if (!this._settingsTextElem) { this._settingsTextElem = document.createElement('div'); this._settingsTextElem.innerText = this.setting.label; this._settingsTextElem.title = this.setting.description; } return this._settingsTextElem; } get textbox() { if (!this._textbox) { this._textbox = document.createElement('input'); this._textbox.classList.add('form-control'); this._textbox.type = 'textbox'; } return this._textbox; } /** * @returns Return or creates a HTML element that represents this setting in the DOM. */ get rootElement() { if (!this._rootElement) { // create root div with "setting" css class this._rootElement = document.createElement('div'); this._rootElement.id = this.setting.id; this._rootElement.classList.add('setting'); // create div element to contain our setting's text this._rootElement.appendChild(this.settingsTextElem); // create label element to wrap out input type const wrapperLabel = document.createElement('label'); this._rootElement.appendChild(wrapperLabel); // create input type=checkbox this.textbox.title = this.setting.description; wrapperLabel.appendChild(this.textbox); // setup on change from checkbox this.textbox.addEventListener('input', () => { if (this.setting.text !== this.textbox.value) { this.setting.text = this.textbox.value; this.setting.updateURLParams(); } }); } return this._rootElement; } /** * Update the setting's stored value. * @param inValue The new value for the setting. */ set text(inValue) { this.textbox.value = inValue; } /** * Get value */ get text() { return this.textbox.value; } /** * Set the label text for the setting. * @param label setting label. */ set label(inLabel) { this.settingsTextElem.innerText = inLabel; } /** * Get label */ get label() { return this.settingsTextElem.innerText; } } /***/ }), /***/ "./src/Overlay/AFKOverlay.ts": /*!***********************************!*\ !*** ./src/Overlay/AFKOverlay.ts ***! \***********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "AFKOverlay": () => (/* binding */ AFKOverlay) /* harmony export */ }); /* harmony import */ var _ActionOverlay__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./ActionOverlay */ "./src/Overlay/ActionOverlay.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Show an overlay for when the session is unattended, it begins a countdown timer, which when elapsed will disconnect the stream. */ class AFKOverlay extends _ActionOverlay__WEBPACK_IMPORTED_MODULE_0__.ActionOverlay { /** * @returns The created root element of this overlay. */ static createRootElement() { const afkOverlayHtml = document.createElement('div'); afkOverlayHtml.id = 'afkOverlay'; afkOverlayHtml.className = 'clickableState'; return afkOverlayHtml; } /** * @returns The created content element of this overlay, which contain some text for an afk count down. */ static createContentElement() { const afkOverlayHtmlInner = document.createElement('div'); afkOverlayHtmlInner.id = 'afkOverlayInner'; afkOverlayHtmlInner.innerHTML = '
No activity detected
Disconnecting in seconds
Click to continue
'; return afkOverlayHtmlInner; } /** * Construct an Afk overlay * @param parentElement the element this overlay will be inserted into */ constructor(rootDiv) { super(rootDiv, AFKOverlay.createRootElement(), AFKOverlay.createContentElement()); this.rootElement.addEventListener('click', () => { this.activate(); }); } /** * Update the count down spans number for the overlay * @param countdown the count down number to be inserted into the span for updating */ updateCountdown(countdown) { this.textElement.innerHTML = `
No activity detected
Disconnecting in ${countdown} seconds
Click to continue
`; } } /***/ }), /***/ "./src/Overlay/ActionOverlay.ts": /*!**************************************!*\ !*** ./src/Overlay/ActionOverlay.ts ***! \**************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "ActionOverlay": () => (/* binding */ ActionOverlay) /* harmony export */ }); /* harmony import */ var _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @epicgames-ps/lib-pixelstreamingfrontend-ue5.3 */ "@epicgames-ps/lib-pixelstreamingfrontend-ue5.3"); /* harmony import */ var _BaseOverlay__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./BaseOverlay */ "./src/Overlay/BaseOverlay.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Class for the base action overlay structure */ class ActionOverlay extends _BaseOverlay__WEBPACK_IMPORTED_MODULE_1__.OverlayBase { /** * Construct an action overlay * @param rootDiv the root element this overlay will be inserted into * @param rootElement the root element that is the overlay * @param contentElement an element that contains text for the action overlay */ constructor(rootDiv, rootElement, contentElement) { super(rootDiv, rootElement, contentElement); this.onActionCallback = () => { /* do nothing */ _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.Info(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), 'Did you forget to set the onAction callback in your overlay?'); }; } /** * Update the text overlays inner text * @param text the update text to be inserted into the overlay */ update(text) { if (text != null || text != undefined) { this.textElement.innerHTML = text; } } /** * Set a method as an event emitter callback * @param callBack the method that is to be called when the event is emitted */ onAction(callBack) { this.onActionCallback = callBack; } /** * Activate an event that is attached to the event emitter */ activate() { this.onActionCallback(); } } /***/ }), /***/ "./src/Overlay/BaseOverlay.ts": /*!************************************!*\ !*** ./src/Overlay/BaseOverlay.ts ***! \************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "OverlayBase": () => (/* binding */ OverlayBase) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Class for the base overlay structure */ class OverlayBase { /** * Construct an overlay * @param rootDiv the root element this overlay will be inserted into * @param rootElement the root element that is the overlay */ constructor(rootDiv, rootElement, textElement) { this.rootDiv = rootDiv; this.rootElement = rootElement; this.textElement = textElement; this.rootElement.appendChild(this.textElement); this.hide(); this.rootDiv.appendChild(this.rootElement); } /** * Show the overlay */ show() { this.rootElement.classList.remove('hiddenState'); } /** * Hide the overlay */ hide() { this.rootElement.classList.add('hiddenState'); } } /***/ }), /***/ "./src/Overlay/ConnectOverlay.ts": /*!***************************************!*\ !*** ./src/Overlay/ConnectOverlay.ts ***! \***************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "ConnectOverlay": () => (/* binding */ ConnectOverlay) /* harmony export */ }); /* harmony import */ var _ActionOverlay__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./ActionOverlay */ "./src/Overlay/ActionOverlay.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Overlay shown during connection, has a button that can be clicked to initiate a connection. */ class ConnectOverlay extends _ActionOverlay__WEBPACK_IMPORTED_MODULE_0__.ActionOverlay { /** * @returns The created root element of this overlay. */ static createRootElement() { const connectElem = document.createElement('div'); connectElem.id = 'connectOverlay'; connectElem.className = 'clickableState'; return connectElem; } /** * @returns The created content element of this overlay, which contain whatever content this element contains, like text or a button. */ static createContentElement() { const connectContentElem = document.createElement('div'); connectContentElem.id = 'connectButton'; connectContentElem.innerHTML = 'Click to start'; return connectContentElem; } /** * Construct a connect overlay with a connection button. * @param parentElem the parent element this overlay will be inserted into. */ constructor(parentElem) { super(parentElem, ConnectOverlay.createRootElement(), ConnectOverlay.createContentElement()); // add the new event listener this.rootElement.addEventListener('click', () => { this.activate(); }); } } /***/ }), /***/ "./src/Overlay/DisconnectOverlay.ts": /*!******************************************!*\ !*** ./src/Overlay/DisconnectOverlay.ts ***! \******************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "DisconnectOverlay": () => (/* binding */ DisconnectOverlay) /* harmony export */ }); /* harmony import */ var _ActionOverlay__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./ActionOverlay */ "./src/Overlay/ActionOverlay.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Overlay shown during disconnection, has a reconnection element that can be clicked to reconnect. */ class DisconnectOverlay extends _ActionOverlay__WEBPACK_IMPORTED_MODULE_0__.ActionOverlay { /** * @returns The created root element of this overlay. */ static createRootElement() { const disconnectOverlayHtml = document.createElement('div'); disconnectOverlayHtml.id = 'disconnectOverlay'; disconnectOverlayHtml.className = 'clickableState'; return disconnectOverlayHtml; } /** * @returns The created content element of this overlay, which contain whatever content this element contains, like text or a button. */ static createContentElement() { // build the inner html container const disconnectOverlayHtmlContainer = document.createElement('div'); disconnectOverlayHtmlContainer.id = 'disconnectButton'; disconnectOverlayHtmlContainer.innerHTML = 'Click To Restart'; return disconnectOverlayHtmlContainer; } /** * Construct a disconnect overlay with a retry connection icon. * @param parentElem the parent element this overlay will be inserted into. */ constructor(parentElem) { super(parentElem, DisconnectOverlay.createRootElement(), DisconnectOverlay.createContentElement()); // add the new event listener this.rootElement.addEventListener('click', () => { this.activate(); }); } } /***/ }), /***/ "./src/Overlay/ErrorOverlay.ts": /*!*************************************!*\ !*** ./src/Overlay/ErrorOverlay.ts ***! \*************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "ErrorOverlay": () => (/* binding */ ErrorOverlay) /* harmony export */ }); /* harmony import */ var _TextOverlay__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./TextOverlay */ "./src/Overlay/TextOverlay.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Generic overlay used to show textual error info to the user. */ class ErrorOverlay extends _TextOverlay__WEBPACK_IMPORTED_MODULE_0__.TextOverlay { /** * @returns The created root element of this overlay. */ static createRootElement() { const errorOverlayHtml = document.createElement('div'); errorOverlayHtml.id = 'errorOverlay'; errorOverlayHtml.className = 'textDisplayState'; return errorOverlayHtml; } /** * @returns The created content element of this overlay, which contain whatever content this element contains, like text or a button. */ static createContentElement() { const errorOverlayHtmlInner = document.createElement('div'); errorOverlayHtmlInner.id = 'errorOverlayInner'; return errorOverlayHtmlInner; } /** * Construct a connect overlay with a connection button. * @param parentElem the parent element this overlay will be inserted into. */ constructor(parentElem) { super(parentElem, ErrorOverlay.createRootElement(), ErrorOverlay.createContentElement()); } } /***/ }), /***/ "./src/Overlay/InfoOverlay.ts": /*!************************************!*\ !*** ./src/Overlay/InfoOverlay.ts ***! \************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "InfoOverlay": () => (/* binding */ InfoOverlay) /* harmony export */ }); /* harmony import */ var _TextOverlay__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./TextOverlay */ "./src/Overlay/TextOverlay.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Generic overlay used to show textual info to the user. */ class InfoOverlay extends _TextOverlay__WEBPACK_IMPORTED_MODULE_0__.TextOverlay { /** * @returns The created root element of this overlay. */ static createRootElement() { const infoOverlayHtml = document.createElement('div'); infoOverlayHtml.id = 'infoOverlay'; infoOverlayHtml.className = 'textDisplayState'; return infoOverlayHtml; } /** * @returns The created content element of this overlay, which contain whatever content this element contains, like text or a button. */ static createContentElement() { const infoOverlayHtmlInner = document.createElement('div'); infoOverlayHtmlInner.id = 'messageOverlayInner'; return infoOverlayHtmlInner; } /** * Construct a connect overlay with a connection button. * @param parentElem the parent element this overlay will be inserted into. */ constructor(parentElem) { super(parentElem, InfoOverlay.createRootElement(), InfoOverlay.createContentElement()); } } /***/ }), /***/ "./src/Overlay/PlayOverlay.ts": /*!************************************!*\ !*** ./src/Overlay/PlayOverlay.ts ***! \************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "PlayOverlay": () => (/* binding */ PlayOverlay) /* harmony export */ }); /* harmony import */ var _ActionOverlay__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./ActionOverlay */ "./src/Overlay/ActionOverlay.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Overlay shown when stream is ready to play. */ class PlayOverlay extends _ActionOverlay__WEBPACK_IMPORTED_MODULE_0__.ActionOverlay { /** * @returns The created root element of this overlay. */ static createRootElement() { const playElem = document.createElement('div'); playElem.id = 'playOverlay'; playElem.className = 'clickableState'; return playElem; } /** * @returns The created content element of this overlay, which contain whatever content this element contains, like text or a button. */ static createContentElement() { // todo: change this to an svg const playOverlayHtmlInner = document.createElement('img'); playOverlayHtmlInner.id = 'playButton'; playOverlayHtmlInner.src = 'data:image/png;base64,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'; playOverlayHtmlInner.alt = 'Start Streaming'; return playOverlayHtmlInner; } /** * Construct a connect overlay with a connection button. * @param parentElem the parent element this overlay will be inserted into. */ constructor(parentElem) { super(parentElem, PlayOverlay.createRootElement(), PlayOverlay.createContentElement()); // add the new event listener this.rootElement.addEventListener('click', () => { this.activate(); }); } } /***/ }), /***/ "./src/Overlay/TextOverlay.ts": /*!************************************!*\ !*** ./src/Overlay/TextOverlay.ts ***! \************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "TextOverlay": () => (/* binding */ TextOverlay) /* harmony export */ }); /* harmony import */ var _BaseOverlay__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./BaseOverlay */ "./src/Overlay/BaseOverlay.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Class for the text overlay base */ class TextOverlay extends _BaseOverlay__WEBPACK_IMPORTED_MODULE_0__.OverlayBase { /** * Construct a text overlay * @param rootDiv the root element this overlay will be inserted into * @param rootElement the root element that is the overlay * @param textElement an element that contains text for the action overlay */ constructor(rootDiv, rootElement, textElement) { super(rootDiv, rootElement, textElement); } /** * Update the text overlays inner text * @param text the update text to be inserted into the overlay */ update(text) { if (text != null || text != undefined) { this.textElement.innerHTML = text; } } } /***/ }), /***/ "./src/Styles/PixelStreamingApplicationStyles.ts": /*!*******************************************************!*\ !*** ./src/Styles/PixelStreamingApplicationStyles.ts ***! \*******************************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "PixelStreamingApplicationStyle": () => (/* binding */ PixelStreamingApplicationStyle) /* harmony export */ }); /* harmony import */ var jss__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! jss */ "jss"); /* harmony import */ var jss_plugin_global__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! jss-plugin-global */ "jss-plugin-global"); /* harmony import */ var jss_plugin_camel_case__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! jss-plugin-camel-case */ "jss-plugin-camel-case"); /* Copyright Epic Games, Inc. All Rights Reserved. */ class PixelStreamingApplicationStyle { constructor(options) { this.defaultLightModePalette = { '--color0': '#e2e0dd80', '--color1': '#FFFFFF', '--color2': '#000000', '--color3': '#0585fe', '--color4': '#35b350', '--color5': '#ffab00', '--color6': '#e1e2dd', '--color7': '#c3c4bf' }; this.defaultDarkModePalette = { '--color0': '#1D1F2280', '--color1': '#000000', '--color2': '#FFFFFF', '--color3': '#0585fe', '--color4': '#35b350', '--color5': '#ffab00', '--color6': '#1e1d22', '--color7': '#3c3b40' }; this.defaultStyles = { ':root': { '--color0': '#1D1F2280', '--color1': '#000000', '--color2': '#FFFFFF', '--color3': '#0585fe', '--color4': '#35b350', '--color5': '#ffab00', '--color6': '#1e1d22', '--color7': '#3c3b40', '--color8': '#41008c', '--color9': '#3e0070', '--color10': '#2e0052', '--color11': 'rgba(65,0,139,1)' }, '.noselect': { userSelect: 'none' }, '#playerUI': { width: '100%', height: '100%', position: 'relative' }, '#videoElementParent': { width: '100%', height: '100%', position: 'absolute', backgroundColor: 'var(--color1)' }, '#uiFeatures': { width: '100%', height: '100%', zIndex: '30', position: 'relative', color: 'var(--color2)', pointerEvents: 'none', overflow: 'hidden' }, '.UiTool .tooltiptext': { visibility: 'hidden', width: 'auto', color: 'var(--color2)', textAlign: 'center', borderRadius: '15px', padding: '0px 10px', fontFamily: "'Montserrat', sans-serif", fontSize: '0.75rem', letterSpacing: '0.75px', position: 'absolute', top: '0', transform: 'translateY(25%)', left: '125%', zIndex: '20' }, '.UiTool:hover .tooltiptext': { visibility: 'visible', backgroundColor: 'var(--color7)' }, '#connection .tooltiptext': { top: '125%', transform: 'translateX(-25%)', left: '0', zIndex: '20', padding: '5px 10px' }, '#connection': { position: 'absolute', bottom: '8%', left: '5%', fontFamily: "'Michroma', sans-serif", height: '3rem', width: '3rem', pointerEvents: 'all' }, '#settings-panel .tooltiptext': { display: 'block', top: '125%', transform: 'translateX(-50%)', left: '0', zIndex: '20', padding: '5px 10px', border: '3px solid var(--color3)', width: 'max-content', fallbacks: [ { width: 'max-content' }, { border: '3px solid var(--color3)' }, { padding: '5px 10px' }, { zIndex: '20' }, { left: '0' }, { transform: 'translateX(-50%)' }, { top: '125%' }, { display: 'block' } ] }, '#controls': { position: 'absolute', top: '3%', left: '2%', fontFamily: "'Michroma', sans-serif", pointerEvents: 'all', display: 'block' }, '#controls>*': { marginBottom: '0.5rem', borderRadius: '50%', display: 'block', height: '2rem', lineHeight: '1.75rem', padding: '0.5rem' }, '#controls #additionalinfo': { textAlign: 'center', fontFamily: "'Montserrat', sans-serif" }, '#fullscreen-btn': { padding: '0.6rem !important' }, '#minimizeIcon': { display: 'none' }, '#settingsBtn, #statsBtn': { cursor: 'pointer' }, '#uiFeatures button': { backgroundColor: 'var(--color7)', border: '1px solid var(--color7)', color: 'var(--color2)', position: 'relative', width: '3rem', height: '3rem', padding: '0.5rem', textAlign: 'center' }, '#uiFeatures button:hover': { backgroundColor: 'var(--color3)', border: '3px solid var(--color3)', transition: '0.25s ease', paddingLeft: '0.55rem', paddingTop: '0.55rem' }, '#uiFeatures button:active': { border: '3px solid var(--color3)', backgroundColor: 'var(--color7)', paddingLeft: '0.55rem', paddingTop: '0.55rem' }, '.btn-flat': { backgroundColor: 'transparent', color: 'var(--color2)', fontFamily: "'Montserrat'", fontWeight: 'bold', border: '3px solid var(--color3)', borderRadius: '1rem', fontSize: '0.75rem', paddingLeft: '0.5rem', paddingRight: '0.5rem', cursor: 'pointer', textAlign: 'center' }, '.btn-flat:hover': { backgroundColor: 'var(--color3)', transition: 'ease 0.3s' }, '.btn-flat:disabled': { background: 'var(--color7)', borderColor: 'var(--color3)', color: 'var(--color3)', cursor: 'default' }, '.btn-flat:active': { backgroundColor: 'transparent' }, '.btn-flat:focus': { outline: 'none' }, '#uiFeatures img': { width: '100%', height: '100%' }, '.panel-wrap': { position: 'absolute', top: '0', bottom: '0', right: '0', height: '100%', minWidth: '20vw', maxWidth: '90vw', transform: 'translateX(100%)', transition: '.3s ease-out', pointerEvents: 'all', backdropFilter: 'blur(10px)', '-webkit-backdrop-filter': 'blur(10px)', overflowY: 'auto', overflowX: 'hidden', backgroundColor: 'var(--color0)' }, '.panel-wrap-visible': { transform: 'translateX(0%)' }, '.panel': { overflowY: 'auto', padding: '1em' }, '#settingsHeading, #statsHeading': { display: 'inline-block', fontSize: '2em', marginBlockStart: '0.67em', marginBlockEnd: '0.67em', marginInlineStart: '0px', marginInlineEnd: '0px', position: 'relative', padding: '0 0 0 2rem' }, '#settingsClose, #statsClose': { margin: '0.5rem', paddingTop: '0.5rem', paddingBottom: '0.5rem', paddingRight: '0.5rem', fontSize: '2em', float: 'right' }, '#settingsClose:after, #statsClose:after': { paddingLeft: '0.5rem', display: 'inline-block', content: '"\\00d7"' }, '#settingsClose:hover, #statsClose:hover': { color: 'var(--color3)', transition: 'ease 0.3s' }, '#settingsContent, #statsContent': { marginLeft: '2rem', marginRight: '2rem' }, '.setting': { display: 'flex', flexDirection: 'row', justifyContent: 'space-between', padding: '0.15rem 10px 0.15rem 10px' }, '.settings-text': { color: 'var(--color2)', verticalAlign: 'middle', fontWeight: 'normal' }, '.settings-option': { width: '100%', textOverflow: 'ellipsis', whiteSpace: 'nowrap' }, '#connectOverlay, #playOverlay, #infoOverlay, #errorOverlay, #afkOverlay, #disconnectOverlay': { zIndex: '30', position: 'absolute', color: 'var(--color2)', fontSize: '1.8em', width: '100%', height: '100%', backgroundColor: 'var(--color1)', alignItems: 'center', justifyContent: 'center', textTransform: 'uppercase' }, '.clickableState': { alignItems: 'center', justifyContent: 'center', display: 'flex', cursor: 'pointer' }, '.textDisplayState': { display: 'flex' }, '.hiddenState': { display: 'none' }, '#playButton, #connectButton': { display: 'inline-block', height: 'auto', zIndex: '30' }, 'img#playButton': { maxWidth: '241px', width: '10%' }, '#uiInteraction': { position: 'fixed' }, '#UIInteractionButtonBoundary': { padding: '2px' }, '#UIInteractionButton': { cursor: 'pointer' }, '#hiddenInput': { position: 'absolute', left: '-10%', width: '0px', opacity: '0' }, '#editTextButton': { position: 'absolute', height: '40px', width: '40px' }, '.btn-overlay': { verticalAlign: 'middle', display: 'inline-block' }, '.tgl-switch': { verticalAlign: 'middle', display: 'inline-block' }, '.tgl-switch .tgl': { display: 'none' }, '.tgl, .tgl:after, .tgl:before, .tgl *, .tgl *:after, .tgl *:before, .tgl+.tgl-slider': { '-webkit-box-sizing': 'border-box', boxSizing: 'border-box' }, '.tgl::-moz-selection, .tgl:after::-moz-selection, .tgl:before::-moz-selection, .tgl *::-moz-selection, .tgl *:after::-moz-selection, .tgl *:before::-moz-selection, .tgl+.tgl-slider::-moz-selection': { background: 'none' }, '.tgl::selection, .tgl:after::selection, .tgl:before::selection, .tgl *::selection, .tgl *:after::selection, .tgl *:before::selection, .tgl+.tgl-slider::selection': { background: 'none' }, '.tgl-slider': {}, '.tgl+.tgl-slider': { outline: '0', display: 'block', width: '40px', height: '18px', position: 'relative', cursor: 'pointer', userSelect: 'none' }, '.tgl+.tgl-slider:after, .tgl+.tgl-slider:before': { position: 'relative', display: 'block', content: '""', width: '50%', height: '100%' }, '.tgl+.tgl-slider:after': { left: '0' }, '.tgl+.tgl-slider:before': { display: 'none' }, '.tgl-flat+.tgl-slider': { padding: '2px', '-webkit-transition': 'all .2s ease', transition: 'all .2s ease', background: 'var(--color6)', border: '3px solid var(--color7)', borderRadius: '2em' }, '.tgl-flat+.tgl-slider:after': { '-webkit-transition': 'all .2s ease', transition: 'all .2s ease', background: 'var(--color7)', content: '""', borderRadius: '1em' }, '.tgl-flat:checked+.tgl-slider': { border: '3px solid var(--color3)' }, '.tgl-flat:checked+.tgl-slider:after': { left: '50%', background: 'var(--color3)' }, '.btn-apply': { display: 'block !important', marginLeft: 'auto', marginRight: 'auto', width: '40%' }, '.form-control': { backgroundColor: 'var(--color7)', border: '2px solid var(--color7)', borderRadius: '4px', color: 'var(--color2)', textAlign: 'right', fontFamily: 'inherit' }, '.form-control:hover': { borderColor: 'var(--color7)' }, '.form-group': { paddingTop: '4px', display: 'grid', gridTemplateColumns: '80% 20%', rowGap: '4px', paddingRight: '10px', paddingLeft: '10px' }, '.form-group label': { verticalAlign: 'middle', fontWeight: 'normal' }, '.settingsContainer': { display: 'flex', flexDirection: 'column', borderBottom: '1px solid var(--color7)', paddingTop: '10px', paddingBottom: '10px' }, '.settingsContainer> :first-child': { marginTop: '4px', marginBottom: '4px', fontWeight: 'bold', justifyContent: 'space-between', display: 'flex', flexDirection: 'row', alignItems: 'baseline' }, '.collapse': { paddingLeft: '5%' }, '#streamTools': { borderBottomRightRadius: '5px', borderBottomLeftRadius: '5px', userSelect: 'none', position: 'absolute', top: '0', right: '2%', zIndex: '100', border: '4px solid var(--colour8)', borderTopWidth: '0px' }, '.settingsHeader': { fontStyle: 'italic' }, '#streamToolsHeader': { display: 'flex', flexDirection: 'row', justifyContent: 'space-between', borderBottom: '1px solid var(--colour8)', backgroundColor: 'var(--color7)' }, '.streamTools': { backgroundColor: 'var(--color2)', fontFamily: 'var(--buttonFont)', fontWeight: 'lighter', color: 'var(--color7)' }, '.streamTools-shown>#streamToolsSettings, .streamTools-shown>#streamToolsStats': { display: 'block' }, '#streamToolsToggle': { width: '100%' }, '#qualityStatus': { fontSize: '37px', paddingRight: '4px' }, '.svgIcon': { fill: 'var(--color2)' } }; const { customStyles, lightModePalette, darkModePalette, jssInsertionPoint } = options !== null && options !== void 0 ? options : {}; // One time setup with default plugins and settings. const jssOptions = { // JSS has many interesting plugins we may wish to turn on //plugins: [functions(), template(), global(), extend(), nested(), compose(), camelCase(), defaultUnit(options.defaultUnit), expand(), vendorPrefixer(), propsSort()] plugins: [(0,jss_plugin_global__WEBPACK_IMPORTED_MODULE_1__["default"])(), (0,jss_plugin_camel_case__WEBPACK_IMPORTED_MODULE_2__["default"])()], insertionPoint: jssInsertionPoint }; jss__WEBPACK_IMPORTED_MODULE_0__["default"].setup(jssOptions); this.customStyles = customStyles; this.lightModePalette = lightModePalette !== null && lightModePalette !== void 0 ? lightModePalette : this.defaultLightModePalette; this.darkModePalette = darkModePalette !== null && darkModePalette !== void 0 ? darkModePalette : this.defaultDarkModePalette; } applyStyleSheet() { // Todo: refactor codebase to use jss at a component level, classes can be grabbed like so: //const {pixelStreamingClasses} = jss.createStyleSheet(styles).attach(); // attach generated style sheet to page jss__WEBPACK_IMPORTED_MODULE_0__["default"].createStyleSheet({ '@global': Object.assign(Object.assign({}, this.defaultStyles), this.customStyles) }).attach(); } applyPalette(palette) { const rootElement = document.querySelector(':root'); rootElement.style.setProperty('--color0', palette['--color0']); rootElement.style.setProperty('--color1', palette['--color1']); rootElement.style.setProperty('--color2', palette['--color2']); rootElement.style.setProperty('--color3', palette['--color3']); rootElement.style.setProperty('--color4', palette['--color4']); rootElement.style.setProperty('--color5', palette['--color5']); rootElement.style.setProperty('--color6', palette['--color6']); rootElement.style.setProperty('--color7', palette['--color7']); } /** * Update the players color variables * @param isLightMode - should we use a light or dark color scheme */ setColorMode(isLightMode) { if (isLightMode) { this.applyPalette(this.lightModePalette); } else { this.applyPalette(this.darkModePalette); } } } /***/ }), /***/ "./src/UI/Controls.ts": /*!****************************!*\ !*** ./src/UI/Controls.ts ***! \****************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Controls": () => (/* binding */ Controls) /* harmony export */ }); /* harmony import */ var _FullscreenIcon__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./FullscreenIcon */ "./src/UI/FullscreenIcon.ts"); /* harmony import */ var _SettingsIcon__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./SettingsIcon */ "./src/UI/SettingsIcon.ts"); /* harmony import */ var _StatsIcon__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./StatsIcon */ "./src/UI/StatsIcon.ts"); /* harmony import */ var _XRIcon__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./XRIcon */ "./src/UI/XRIcon.ts"); /* harmony import */ var _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @epicgames-ps/lib-pixelstreamingfrontend-ue5.3 */ "@epicgames-ps/lib-pixelstreamingfrontend-ue5.3"); /* harmony import */ var _UI_UIConfigurationTypes__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../UI/UIConfigurationTypes */ "./src/UI/UIConfigurationTypes.ts"); // Copyright Epic Games, Inc. All Rights Reserved. // If there isn't a type provided, default behaviour is to create the element. function shouldCreateButton(type) { return (type == undefined) ? true : (type.creationMode === _UI_UIConfigurationTypes__WEBPACK_IMPORTED_MODULE_1__.UIElementCreationMode.CreateDefaultElement); } /** * Element containing various controls like stats, settings, fullscreen. */ class Controls { /** * Construct the controls */ constructor(config) { if (!config || shouldCreateButton(config.statsButtonType)) { this.statsIcon = new _StatsIcon__WEBPACK_IMPORTED_MODULE_2__.StatsIcon(); } if (!config || shouldCreateButton(config.settingsButtonType)) { this.settingsIcon = new _SettingsIcon__WEBPACK_IMPORTED_MODULE_3__.SettingsIcon(); } if (!config || shouldCreateButton(config.fullscreenButtonType)) { this.fullscreenIcon = new _FullscreenIcon__WEBPACK_IMPORTED_MODULE_4__.FullScreenIcon(); } if (!config || shouldCreateButton(config.xrIconType)) { this.xrIcon = new _XRIcon__WEBPACK_IMPORTED_MODULE_5__.XRIcon(); } } /** * Get the element containing the controls. */ get rootElement() { if (!this._rootElement) { this._rootElement = document.createElement('div'); this._rootElement.id = 'controls'; if (!!this.fullscreenIcon) { this._rootElement.appendChild(this.fullscreenIcon.rootElement); } if (!!this.settingsIcon) { this._rootElement.appendChild(this.settingsIcon.rootElement); } if (!!this.statsIcon) { this._rootElement.appendChild(this.statsIcon.rootElement); } if (!!this.xrIcon) { _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.WebXRController.isSessionSupported('immersive-vr').then((supported) => { if (supported) { this._rootElement.appendChild(this.xrIcon.rootElement); } }); } ; } return this._rootElement; } } /***/ }), /***/ "./src/UI/DataChannelLatencyTest.ts": /*!******************************************!*\ !*** ./src/UI/DataChannelLatencyTest.ts ***! \******************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "DataChannelLatencyTest": () => (/* binding */ DataChannelLatencyTest) /* harmony export */ }); /* harmony import */ var _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @epicgames-ps/lib-pixelstreamingfrontend-ue5.3 */ "@epicgames-ps/lib-pixelstreamingfrontend-ue5.3"); // Copyright Epic Games, Inc. All Rights Reserved. /** * DataChannel Latency test UI elements and results handling. */ class DataChannelLatencyTest { /** * Get the button containing the stats icon. */ get rootElement() { if (!this._rootElement) { this._rootElement = document.createElement('section'); this._rootElement.classList.add('settingsContainer'); // make heading const heading = document.createElement('div'); heading.id = 'dataChannelLatencyTestHeader'; heading.classList.add('settings-text'); heading.classList.add('settingsHeader'); this._rootElement.appendChild(heading); const headingText = document.createElement('div'); headingText.innerHTML = 'Data Channel Latency Test'; heading.appendChild(headingText); heading.appendChild(this.latencyTestButton); // make test results element const resultsParentElem = document.createElement('div'); resultsParentElem.id = 'dataChannelLatencyTestContainer'; resultsParentElem.classList.add('d-none'); this._rootElement.appendChild(resultsParentElem); resultsParentElem.appendChild(this.latencyTestResultsElement); } return this._rootElement; } get latencyTestResultsElement() { if (!this._latencyTestResultsElement) { this._latencyTestResultsElement = document.createElement('div'); this._latencyTestResultsElement.id = 'dataChannelLatencyStatsResults'; this._latencyTestResultsElement.classList.add('StatsResult'); } return this._latencyTestResultsElement; } get latencyTestButton() { if (!this._latencyTestButton) { this._latencyTestButton = document.createElement('input'); this._latencyTestButton.type = 'button'; this._latencyTestButton.value = 'Run Test'; this._latencyTestButton.id = 'btn-start-data-channel-latency-test'; this._latencyTestButton.classList.add('streamTools-button'); this._latencyTestButton.classList.add('btn-flat'); } return this._latencyTestButton; } /** * Populate the UI based on the latency test's results. * @param result The latency test results. */ handleTestResult(result) { _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), result.toString(), 6); /** * Check we have results, NaN would mean that UE version we talk to doesn't support our test */ if (isNaN(result.dataChannelRtt)) { this.latencyTestResultsElement.innerHTML = '
Not supported
'; return; } let latencyStatsInnerHTML = ''; latencyStatsInnerHTML += '
Data channel RTT (ms): ' + result.dataChannelRtt + '
'; /** * Separate path time discovery works only when UE and Player clocks have been synchronized. */ if (result.playerToStreamerTime >= 0 && result.streamerToPlayerTime >= 0) { latencyStatsInnerHTML += '
Player to Streamer path (ms): ' + result.playerToStreamerTime + '
'; latencyStatsInnerHTML += '
Streamer to Player path (ms): ' + result.streamerToPlayerTime + '
'; } this.latencyTestResultsElement.innerHTML = latencyStatsInnerHTML; //setup button to download the detailed results let downloadButton = document.createElement('input'); downloadButton.type = 'button'; downloadButton.value = 'Download'; downloadButton.classList.add('streamTools-button'); downloadButton.classList.add('btn-flat'); downloadButton.onclick = () => { let file = new Blob([result.exportLatencyAsCSV()], { type: 'text/plain' }); let a = document.createElement("a"), url = URL.createObjectURL(file); a.href = url; a.download = "data_channel_latency_test_results.csv"; document.body.appendChild(a); a.click(); setTimeout(function () { document.body.removeChild(a); window.URL.revokeObjectURL(url); }, 0); }; this.latencyTestResultsElement.appendChild(downloadButton); } handleTestStart() { this.latencyTestResultsElement.innerHTML = '
Test in progress
'; } } /***/ }), /***/ "./src/UI/FullscreenIcon.ts": /*!**********************************!*\ !*** ./src/UI/FullscreenIcon.ts ***! \**********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "FullScreenIcon": () => (/* binding */ FullScreenIcon), /* harmony export */ "FullScreenIconBase": () => (/* binding */ FullScreenIconBase), /* harmony export */ "FullScreenIconExternal": () => (/* binding */ FullScreenIconExternal) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Base class for an element (i.e. button) that, when clicked, will toggle fullscreen of a given element. * Can be initialized with any HTMLElement, if it is set as rootElement in the constructor. */ class FullScreenIconBase { get rootElement() { return this._rootElement; } set rootElement(element) { element.onclick = () => this.toggleFullscreen(); this._rootElement = element; } constructor() { this.isFullscreen = false; // set up the full screen events document.addEventListener('webkitfullscreenchange', () => this.onFullscreenChange(), false); document.addEventListener('mozfullscreenchange', () => this.onFullscreenChange(), false); document.addEventListener('fullscreenchange', () => this.onFullscreenChange(), false); document.addEventListener('MSFullscreenChange', () => this.onFullscreenChange(), false); } /** * Makes the document or fullscreenElement fullscreen. */ toggleFullscreen() { // if already full screen; exit // else go fullscreen if (document.fullscreenElement || document.webkitFullscreenElement || document.mozFullScreenElement || document.msFullscreenElement) { if (document.exitFullscreen) { document.exitFullscreen(); } else if (document.mozCancelFullScreen) { document.mozCancelFullScreen(); } else if (document.webkitExitFullscreen) { document.webkitExitFullscreen(); } else if (document.msExitFullscreen) { document.msExitFullscreen(); } } else { const element = this.fullscreenElement; if (!element) { return; } if (element.requestFullscreen) { element.requestFullscreen(); } else if (element.mozRequestFullscreen) { element.mozRequestFullscreen(); } else if (element.webkitRequestFullscreen) { element.webkitRequestFullscreen(); } else if (element.msRequestFullscreen) { element.msRequestFullscreen(); } else if (element.webkitEnterFullscreen) { element.webkitEnterFullscreen(); //for iphone this code worked } } this.onFullscreenChange(); } /** * Handles the fullscreen button on change */ onFullscreenChange() { this.isFullscreen = document.webkitIsFullScreen || document.mozFullScreen || (document.msFullscreenElement && document.msFullscreenElement !== null) || (document.fullscreenElement && document.fullscreenElement !== null); } } /** * An implementation of FullScreenIconBase that uses an externally * provided HTMLElement for toggling full screen. */ class FullScreenIconExternal extends FullScreenIconBase { constructor(externalButton) { super(); this.rootElement = externalButton; } } /** * The default fullscreen icon that contains a button and svgs for each state. */ class FullScreenIcon extends FullScreenIconBase { constructor() { super(); const createdButton = document.createElement('button'); createdButton.type = 'button'; createdButton.classList.add('UiTool'); createdButton.id = 'fullscreen-btn'; createdButton.appendChild(this.maximizeIcon); createdButton.appendChild(this.minimizeIcon); createdButton.appendChild(this.tooltipText); this.rootElement = createdButton; } get tooltipText() { if (!this._tooltipText) { this._tooltipText = document.createElement('span'); this._tooltipText.classList.add('tooltiptext'); this._tooltipText.innerHTML = 'Fullscreen'; } return this._tooltipText; } get maximizeIcon() { if (!this._maximizeIcon) { this._maximizeIcon = document.createElementNS('http://www.w3.org/2000/svg', 'svg'); this._maximizeIcon.setAttributeNS(null, 'id', 'maximizeIcon'); this._maximizeIcon.setAttributeNS(null, 'x', '0px'); this._maximizeIcon.setAttributeNS(null, 'y', '0px'); this._maximizeIcon.setAttributeNS(null, 'viewBox', '0 0 384.97 384.97'); // create svg group for the paths const svgGroup = document.createElementNS('http://www.w3.org/2000/svg', 'g'); svgGroup.classList.add('svgIcon'); this._maximizeIcon.appendChild(svgGroup); // create paths for the icon itself, one for each corner const path1 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path1.setAttributeNS(null, 'd', 'M384.97,12.03c0-6.713-5.317-12.03-12.03-12.03H264.847c-6.833,0-11.922,5.39-11.934,12.223c0,6.821,5.101,11.838,11.934,11.838h96.062l-0.193,96.519c0,6.833,5.197,12.03,12.03,12.03c6.833-0.012,12.03-5.197,12.03-12.03l0.193-108.369c0-0.036-0.012-0.06-0.012-0.084C384.958,12.09,384.97,12.066,384.97,12.03z'); const path2 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path2.setAttributeNS(null, 'd', 'M120.496,0H12.403c-0.036,0-0.06,0.012-0.096,0.012C12.283,0.012,12.247,0,12.223,0C5.51,0,0.192,5.317,0.192,12.03L0,120.399c0,6.833,5.39,11.934,12.223,11.934c6.821,0,11.838-5.101,11.838-11.934l0.192-96.339h96.242c6.833,0,12.03-5.197,12.03-12.03C132.514,5.197,127.317,0,120.496,0z'); const path3 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path3.setAttributeNS(null, 'd', 'M120.123,360.909H24.061v-96.242c0-6.833-5.197-12.03-12.03-12.03S0,257.833,0,264.667v108.092c0,0.036,0.012,0.06,0.012,0.084c0,0.036-0.012,0.06-0.012,0.096c0,6.713,5.317,12.03,12.03,12.03h108.092c6.833,0,11.922-5.39,11.934-12.223C132.057,365.926,126.956,360.909,120.123,360.909z'); const path4 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path4.setAttributeNS(null, 'd', 'M372.747,252.913c-6.833,0-11.85,5.101-11.838,11.934v96.062h-96.242c-6.833,0-12.03,5.197-12.03,12.03s5.197,12.03,12.03,12.03h108.092c0.036,0,0.06-0.012,0.084-0.012c0.036-0.012,0.06,0.012,0.096,0.012c6.713,0,12.03-5.317,12.03-12.03V264.847C384.97,258.014,379.58,252.913,372.747,252.913z'); svgGroup.appendChild(path1); svgGroup.appendChild(path2); svgGroup.appendChild(path3); svgGroup.appendChild(path4); } return this._maximizeIcon; } get minimizeIcon() { if (!this._minimizeIcon) { this._minimizeIcon = document.createElementNS('http://www.w3.org/2000/svg', 'svg'); this._minimizeIcon.setAttributeNS(null, 'id', 'minimizeIcon'); this._minimizeIcon.setAttributeNS(null, 'x', '0px'); this._minimizeIcon.setAttributeNS(null, 'y', '0px'); this._minimizeIcon.setAttributeNS(null, 'viewBox', '0 0 385.331 385.331'); // create svg group for the paths const svgGroup = document.createElementNS('http://www.w3.org/2000/svg', 'g'); svgGroup.classList.add('svgIcon'); this._minimizeIcon.appendChild(svgGroup); // create paths for the icon itself, one for each corner const path1 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path1.setAttributeNS(null, 'd', 'M264.943,156.665h108.273c6.833,0,11.934-5.39,11.934-12.211c0-6.833-5.101-11.85-11.934-11.838h-96.242V36.181c0-6.833-5.197-12.03-12.03-12.03s-12.03,5.197-12.03,12.03v108.273c0,0.036,0.012,0.06,0.012,0.084c0,0.036-0.012,0.06-0.012,0.096C252.913,151.347,258.23,156.677,264.943,156.665z'); const path2 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path2.setAttributeNS(null, 'd', 'M120.291,24.247c-6.821,0-11.838,5.113-11.838,11.934v96.242H12.03c-6.833,0-12.03,5.197-12.03,12.03c0,6.833,5.197,12.03,12.03,12.03h108.273c0.036,0,0.06-0.012,0.084-0.012c0.036,0,0.06,0.012,0.096,0.012c6.713,0,12.03-5.317,12.03-12.03V36.181C132.514,29.36,127.124,24.259,120.291,24.247z'); const path3 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path3.setAttributeNS(null, 'd', 'M120.387,228.666H12.115c-6.833,0.012-11.934,5.39-11.934,12.223c0,6.833,5.101,11.85,11.934,11.838h96.242v96.423c0,6.833,5.197,12.03,12.03,12.03c6.833,0,12.03-5.197,12.03-12.03V240.877c0-0.036-0.012-0.06-0.012-0.084c0-0.036,0.012-0.06,0.012-0.096C132.418,233.983,127.1,228.666,120.387,228.666z'); const path4 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path4.setAttributeNS(null, 'd', 'M373.3,228.666H265.028c-0.036,0-0.06,0.012-0.084,0.012c-0.036,0-0.06-0.012-0.096-0.012c-6.713,0-12.03,5.317-12.03,12.03v108.273c0,6.833,5.39,11.922,12.223,11.934c6.821,0.012,11.838-5.101,11.838-11.922v-96.242H373.3c6.833,0,12.03-5.197,12.03-12.03S380.134,228.678,373.3,228.666z'); svgGroup.appendChild(path1); svgGroup.appendChild(path2); svgGroup.appendChild(path3); svgGroup.appendChild(path4); } return this._minimizeIcon; } onFullscreenChange() { super.onFullscreenChange(); const minimize = this.minimizeIcon; const maximize = this.maximizeIcon; if (this.isFullscreen) { minimize.style.display = 'inline'; //ios disappearing svg fix minimize.style.transform = 'translate(0, 0)'; maximize.style.display = 'none'; } else { minimize.style.display = 'none'; maximize.style.display = 'inline'; //ios disappearing svg fix maximize.style.transform = 'translate(0, 0)'; } } } /***/ }), /***/ "./src/UI/LabelledButton.ts": /*!**********************************!*\ !*** ./src/UI/LabelledButton.ts ***! \**********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "LabelledButton": () => (/* binding */ LabelledButton) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * A button with a text label beside it. */ class LabelledButton { constructor(label, buttonText) { this._label = label; this._buttonText = buttonText; } /** * Add a click listener to the button element. */ addOnClickListener(onClickFunc) { this.button.addEventListener('click', onClickFunc); } /** * Get the HTMLInputElement for the button. */ get button() { if (!this._button) { this._button = document.createElement('input'); this._button.type = 'button'; this._button.value = this._buttonText; this._button.classList.add('overlay-button'); this._button.classList.add('btn-flat'); } return this._button; } /** * @returns Return or creates a HTML element that represents this setting in the DOM. */ get rootElement() { if (!this._rootElement) { // create root div with "setting" css class this._rootElement = document.createElement('div'); this._rootElement.classList.add('setting'); // create div element to contain our setting's text const settingsTextElem = document.createElement('div'); settingsTextElem.innerText = this._label; this._rootElement.appendChild(settingsTextElem); // create label element to wrap out input type const wrapperLabel = document.createElement('label'); wrapperLabel.classList.add('btn-overlay'); this._rootElement.appendChild(wrapperLabel); wrapperLabel.appendChild(this.button); } return this._rootElement; } } /***/ }), /***/ "./src/UI/LatencyTest.ts": /*!*******************************!*\ !*** ./src/UI/LatencyTest.ts ***! \*******************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "LatencyTest": () => (/* binding */ LatencyTest) /* harmony export */ }); /* harmony import */ var _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @epicgames-ps/lib-pixelstreamingfrontend-ue5.3 */ "@epicgames-ps/lib-pixelstreamingfrontend-ue5.3"); // Copyright Epic Games, Inc. All Rights Reserved. /** * Latency test UI elements and results handling. */ class LatencyTest { /** * Get the the button containing the stats icon. */ get rootElement() { if (!this._rootElement) { this._rootElement = document.createElement('section'); this._rootElement.classList.add('settingsContainer'); // make heading const heading = document.createElement('div'); heading.id = 'latencyTestHeader'; heading.classList.add('settings-text'); heading.classList.add('settingsHeader'); this._rootElement.appendChild(heading); const headingText = document.createElement('div'); headingText.innerHTML = 'Latency Test'; heading.appendChild(headingText); heading.appendChild(this.latencyTestButton); // make test results element const resultsParentElem = document.createElement('div'); resultsParentElem.id = 'latencyTestContainer'; resultsParentElem.classList.add('d-none'); this._rootElement.appendChild(resultsParentElem); resultsParentElem.appendChild(this.latencyTestResultsElement); } return this._rootElement; } get latencyTestResultsElement() { if (!this._latencyTestResultsElement) { this._latencyTestResultsElement = document.createElement('div'); this._latencyTestResultsElement.id = 'latencyStatsResults'; this._latencyTestResultsElement.classList.add('StatsResult'); } return this._latencyTestResultsElement; } get latencyTestButton() { if (!this._latencyTestButton) { this._latencyTestButton = document.createElement('input'); this._latencyTestButton.type = 'button'; this._latencyTestButton.value = 'Run Test'; this._latencyTestButton.id = 'btn-start-latency-test'; this._latencyTestButton.classList.add('streamTools-button'); this._latencyTestButton.classList.add('btn-flat'); } return this._latencyTestButton; } /** * Populate the UI based on the latency test's results. * @param latencyTimings The latency test results. */ handleTestResult(latencyTimings) { _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), latencyTimings.toString(), 6); let latencyStatsInnerHTML = ''; latencyStatsInnerHTML += '
Net latency RTT (ms): ' + latencyTimings.networkLatency + '
'; latencyStatsInnerHTML += '
UE Encode (ms): ' + latencyTimings.EncodeMs + '
'; latencyStatsInnerHTML += '
UE Capture (ms): ' + latencyTimings.CaptureToSendMs + '
'; latencyStatsInnerHTML += '
Browser send latency (ms): ' + latencyTimings.browserSendLatency + '
'; latencyStatsInnerHTML += latencyTimings.frameDisplayDeltaTimeMs && latencyTimings.browserReceiptTimeMs ? '
Browser receive latency (ms): ' + latencyTimings.frameDisplayDeltaTimeMs + '
' : ''; latencyStatsInnerHTML += '
Total latency (excluding browser) (ms): ' + latencyTimings.latencyExcludingDecode + '
'; latencyStatsInnerHTML += latencyTimings.endToEndLatency ? '
Total latency (ms): ' + latencyTimings.endToEndLatency + '
' : ''; this.latencyTestResultsElement.innerHTML = latencyStatsInnerHTML; } } /***/ }), /***/ "./src/UI/SettingsIcon.ts": /*!********************************!*\ !*** ./src/UI/SettingsIcon.ts ***! \********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SettingsIcon": () => (/* binding */ SettingsIcon) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Settings icon that can be clicked. */ class SettingsIcon { /** * Get the the button containing the settings icon. */ get rootElement() { if (!this._rootElement) { this._rootElement = document.createElement('button'); this._rootElement.type = 'button'; this._rootElement.classList.add('UiTool'); this._rootElement.id = 'settingsBtn'; this._rootElement.appendChild(this.settingsIcon); this._rootElement.appendChild(this.tooltipText); } return this._rootElement; } get tooltipText() { if (!this._tooltipText) { this._tooltipText = document.createElement('span'); this._tooltipText.classList.add('tooltiptext'); this._tooltipText.innerHTML = 'Settings'; } return this._tooltipText; } get settingsIcon() { if (!this._settingsIcon) { this._settingsIcon = document.createElementNS('http://www.w3.org/2000/svg', 'svg'); this._settingsIcon.setAttributeNS(null, 'id', 'settingsIcon'); this._settingsIcon.setAttributeNS(null, 'x', '0px'); this._settingsIcon.setAttributeNS(null, 'y', '0px'); this._settingsIcon.setAttributeNS(null, 'viewBox', '0 0 478.703 478.703'); // create svg group for the paths const svgGroup = document.createElementNS('http://www.w3.org/2000/svg', 'g'); svgGroup.classList.add('svgIcon'); this._settingsIcon.appendChild(svgGroup); // create paths for the icon itself, the inner and out path of a cog const path1 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path1.setAttributeNS(null, 'd', 'M454.2,189.101l-33.6-5.7c-3.5-11.3-8-22.2-13.5-32.6l19.8-27.7c8.4-11.8,7.1-27.9-3.2-38.1l-29.8-29.8\ c-5.6-5.6-13-8.7-20.9-8.7c-6.2,0-12.1,1.9-17.1,5.5l-27.8,19.8c-10.8-5.7-22.1-10.4-33.8-13.9l-5.6-33.2\ c-2.4-14.3-14.7-24.7-29.2-24.7h-42.1c-14.5,0-26.8,10.4-29.2,24.7l-5.8,34c-11.2,3.5-22.1,8.1-32.5,13.7l-27.5-19.8\ c-5-3.6-11-5.5-17.2-5.5c-7.9,0-15.4,3.1-20.9,8.7l-29.9,29.8c-10.2,10.2-11.6,26.3-3.2,38.1l20,28.1\ c-5.5,10.5-9.9,21.4-13.3,32.7l-33.2,5.6c-14.3,2.4-24.7,14.7-24.7,29.2v42.1c0,14.5,10.4,26.8,24.7,29.2l34,5.8\ c3.5,11.2,8.1,22.1,13.7,32.5l-19.7,27.4c-8.4,11.8-7.1,27.9,3.2,38.1l29.8,29.8c5.6,5.6,13,8.7,20.9,8.7c6.2,0,12.1-1.9,17.1-5.5\ l28.1-20c10.1,5.3,20.7,9.6,31.6,13l5.6,33.6c2.4,14.3,14.7,24.7,29.2,24.7h42.2c14.5,0,26.8-10.4,29.2-24.7l5.7-33.6\ c11.3-3.5,22.2-8,32.6-13.5l27.7,19.8c5,3.6,11,5.5,17.2,5.5l0,0c7.9,0,15.3-3.1,20.9-8.7l29.8-29.8c10.2-10.2,11.6-26.3,3.2-38.1\ l-19.8-27.8c5.5-10.5,10.1-21.4,13.5-32.6l33.6-5.6c14.3-2.4,24.7-14.7,24.7-29.2v-42.1\ C478.9,203.801,468.5,191.501,454.2,189.101z M451.9,260.401c0,1.3-0.9,2.4-2.2,2.6l-42,7c-5.3,0.9-9.5,4.8-10.8,9.9\ c-3.8,14.7-9.6,28.8-17.4,41.9c-2.7,4.6-2.5,10.3,0.6,14.7l24.7,34.8c0.7,1,0.6,2.5-0.3,3.4l-29.8,29.8c-0.7,0.7-1.4,0.8-1.9,0.8\ c-0.6,0-1.1-0.2-1.5-0.5l-34.7-24.7c-4.3-3.1-10.1-3.3-14.7-0.6c-13.1,7.8-27.2,13.6-41.9,17.4c-5.2,1.3-9.1,5.6-9.9,10.8l-7.1,42\ c-0.2,1.3-1.3,2.2-2.6,2.2h-42.1c-1.3,0-2.4-0.9-2.6-2.2l-7-42c-0.9-5.3-4.8-9.5-9.9-10.8c-14.3-3.7-28.1-9.4-41-16.8\ c-2.1-1.2-4.5-1.8-6.8-1.8c-2.7,0-5.5,0.8-7.8,2.5l-35,24.9c-0.5,0.3-1,0.5-1.5,0.5c-0.4,0-1.2-0.1-1.9-0.8l-29.8-29.8\ c-0.9-0.9-1-2.3-0.3-3.4l24.6-34.5c3.1-4.4,3.3-10.2,0.6-14.8c-7.8-13-13.8-27.1-17.6-41.8c-1.4-5.1-5.6-9-10.8-9.9l-42.3-7.2\ c-1.3-0.2-2.2-1.3-2.2-2.6v-42.1c0-1.3,0.9-2.4,2.2-2.6l41.7-7c5.3-0.9,9.6-4.8,10.9-10c3.7-14.7,9.4-28.9,17.1-42\ c2.7-4.6,2.4-10.3-0.7-14.6l-24.9-35c-0.7-1-0.6-2.5,0.3-3.4l29.8-29.8c0.7-0.7,1.4-0.8,1.9-0.8c0.6,0,1.1,0.2,1.5,0.5l34.5,24.6\ c4.4,3.1,10.2,3.3,14.8,0.6c13-7.8,27.1-13.8,41.8-17.6c5.1-1.4,9-5.6,9.9-10.8l7.2-42.3c0.2-1.3,1.3-2.2,2.6-2.2h42.1\ c1.3,0,2.4,0.9,2.6,2.2l7,41.7c0.9,5.3,4.8,9.6,10,10.9c15.1,3.8,29.5,9.7,42.9,17.6c4.6,2.7,10.3,2.5,14.7-0.6l34.5-24.8\ c0.5-0.3,1-0.5,1.5-0.5c0.4,0,1.2,0.1,1.9,0.8l29.8,29.8c0.9,0.9,1,2.3,0.3,3.4l-24.7,34.7c-3.1,4.3-3.3,10.1-0.6,14.7\ c7.8,13.1,13.6,27.2,17.4,41.9c1.3,5.2,5.6,9.1,10.8,9.9l42,7.1c1.3,0.2,2.2,1.3,2.2,2.6v42.1H451.9z'); const path2 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path2.setAttributeNS(null, 'd', 'M239.4,136.001c-57,0-103.3,46.3-103.3,103.3s46.3,103.3,103.3,103.3s103.3-46.3,103.3-103.3S296.4,136.001,239.4,136.001z M239.4,315.601c-42.1,0-76.3-34.2-76.3-76.3s34.2-76.3,76.3-76.3s76.3,34.2,76.3,76.3S281.5,315.601,239.4,315.601z'); svgGroup.appendChild(path1); svgGroup.appendChild(path2); } return this._settingsIcon; } } /***/ }), /***/ "./src/UI/SettingsPanel.ts": /*!*********************************!*\ !*** ./src/UI/SettingsPanel.ts ***! \*********************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "SettingsPanel": () => (/* binding */ SettingsPanel) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * A UI component containing all the settings for the application. */ class SettingsPanel { constructor() { this._rootElement = null; } /** * @returns Return or creates a HTML element that represents this setting in the DOM. */ get rootElement() { if (!this._rootElement) { this._rootElement = document.createElement('div'); this._rootElement.id = 'settings-panel'; this._rootElement.classList.add('panel-wrap'); const panelElem = document.createElement('div'); panelElem.classList.add('panel'); this._rootElement.appendChild(panelElem); const settingsHeading = document.createElement('div'); settingsHeading.id = 'settingsHeading'; settingsHeading.textContent = 'Settings'; panelElem.appendChild(settingsHeading); panelElem.appendChild(this.settingsCloseButton); panelElem.appendChild(this.settingsContentElement); } return this._rootElement; } get settingsContentElement() { if (!this._settingsContentElement) { this._settingsContentElement = document.createElement('div'); this._settingsContentElement.id = 'settingsContent'; } return this._settingsContentElement; } get settingsCloseButton() { if (!this._settingsCloseButton) { this._settingsCloseButton = document.createElement('div'); this._settingsCloseButton.id = 'settingsClose'; } return this._settingsCloseButton; } /** * Show settings panel. */ show() { if (!this.rootElement.classList.contains('panel-wrap-visible')) { this.rootElement.classList.add('panel-wrap-visible'); } } /** * Toggle the visibility of the settings panel. */ toggleVisibility() { this.rootElement.classList.toggle('panel-wrap-visible'); } /** * Hide settings panel. */ hide() { if (this.rootElement.classList.contains('panel-wrap-visible')) { this.rootElement.classList.remove('panel-wrap-visible'); } } } /***/ }), /***/ "./src/UI/StatsIcon.ts": /*!*****************************!*\ !*** ./src/UI/StatsIcon.ts ***! \*****************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "StatsIcon": () => (/* binding */ StatsIcon) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * Stats icon that can be clicked. */ class StatsIcon { /** * Get the the button containing the stats icon. */ get rootElement() { if (!this._rootElement) { this._rootElement = document.createElement('button'); this._rootElement.type = 'button'; this._rootElement.classList.add('UiTool'); this._rootElement.id = 'statsBtn'; this._rootElement.appendChild(this.statsIcon); this._rootElement.appendChild(this.tooltipText); } return this._rootElement; } get tooltipText() { if (!this._tooltipText) { this._tooltipText = document.createElement('span'); this._tooltipText.classList.add('tooltiptext'); this._tooltipText.innerHTML = 'Information'; } return this._tooltipText; } get statsIcon() { if (!this._statsIcon) { this._statsIcon = document.createElementNS('http://www.w3.org/2000/svg', 'svg'); this._statsIcon.setAttributeNS(null, 'id', 'statsIcon'); this._statsIcon.setAttributeNS(null, 'x', '0px'); this._statsIcon.setAttributeNS(null, 'y', '0px'); this._statsIcon.setAttributeNS(null, 'viewBox', '0 0 330 330'); // create svg group for the paths const svgGroup = document.createElementNS('http://www.w3.org/2000/svg', 'g'); svgGroup.classList.add('svgIcon'); this._statsIcon.appendChild(svgGroup); // create paths for the icon itself, the inner and out path of a cog const path1 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path1.setAttributeNS(null, 'd', 'M165,0.008C74.019,0.008,0,74.024,0,164.999c0,90.977,74.019,164.992,165,164.992s165-74.015,165-164.992C330,74.024,255.981,0.008,165,0.008z M165,299.992c-74.439,0-135-60.557-135-134.992S90.561,30.008,165,30.008s135,60.557,135,134.991C300,239.436,239.439,299.992,165,299.992z'); const path2 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path2.setAttributeNS(null, 'd', 'M165,130.008c-8.284,0-15,6.716-15,15v99.983c0,8.284,6.716,15,15,15s15-6.716,15-15v-99.983C180,136.725,173.284,130.008,165,130.008z'); const path3 = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path3.setAttributeNS(null, 'd', 'M165,70.011c-3.95,0-7.811,1.6-10.61,4.39c-2.79,2.79-4.39,6.66-4.39,10.61s1.6,7.81,4.39,10.61c2.79,2.79,6.66,4.39,10.61,4.39s7.81-1.6,10.609-4.39c2.79-2.8,4.391-6.66,4.391-10.61s-1.601-7.82-4.391-10.61C172.81,71.61,168.95,70.011,165,70.011z'); svgGroup.appendChild(path1); svgGroup.appendChild(path2); svgGroup.appendChild(path3); } return this._statsIcon; } } /***/ }), /***/ "./src/UI/StatsPanel.ts": /*!******************************!*\ !*** ./src/UI/StatsPanel.ts ***! \******************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "Stat": () => (/* binding */ Stat), /* harmony export */ "StatsPanel": () => (/* binding */ StatsPanel) /* harmony export */ }); /* harmony import */ var _LatencyTest__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./LatencyTest */ "./src/UI/LatencyTest.ts"); /* harmony import */ var _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! @epicgames-ps/lib-pixelstreamingfrontend-ue5.3 */ "@epicgames-ps/lib-pixelstreamingfrontend-ue5.3"); /* harmony import */ var _Util_MathUtils__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ../Util/MathUtils */ "./src/Util/MathUtils.ts"); /* harmony import */ var _DataChannelLatencyTest__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./DataChannelLatencyTest */ "./src/UI/DataChannelLatencyTest.ts"); // Copyright Epic Games, Inc. All Rights Reserved. /** * A stat structure, an id, the stat string, and the element where it is rendered. */ class Stat { } /** * A UI component containing all the stats for the application. */ class StatsPanel { constructor() { /* A map stats we are storing/rendering */ this.statsMap = new Map(); this.latencyTest = new _LatencyTest__WEBPACK_IMPORTED_MODULE_1__.LatencyTest(); this.dataChannelLatencyTest = new _DataChannelLatencyTest__WEBPACK_IMPORTED_MODULE_2__.DataChannelLatencyTest(); } /** * @returns Return or creates a HTML element that represents this setting in the DOM. */ get rootElement() { if (!this._rootElement) { this._rootElement = document.createElement('div'); this._rootElement.id = 'stats-panel'; this._rootElement.classList.add('panel-wrap'); const panelElem = document.createElement('div'); panelElem.classList.add('panel'); this._rootElement.appendChild(panelElem); const statsHeading = document.createElement('div'); statsHeading.id = 'statsHeading'; statsHeading.textContent = 'Information'; panelElem.appendChild(statsHeading); panelElem.appendChild(this.statsCloseButton); panelElem.appendChild(this.statsContentElement); } return this._rootElement; } get statsContentElement() { if (!this._statsContentElement) { this._statsContentElement = document.createElement('div'); this._statsContentElement.id = 'statsContent'; const streamToolStats = document.createElement('div'); streamToolStats.id = 'streamToolsStats'; streamToolStats.classList.add('container'); const controlStats = document.createElement('div'); controlStats.id = 'ControlStats'; controlStats.classList.add('row'); const statistics = document.createElement('section'); statistics.id = 'statistics'; statistics.classList.add('settingsContainer'); const statisticsHeader = document.createElement('div'); statisticsHeader.id = 'statisticsHeader'; statisticsHeader.classList.add('settings-text'); statisticsHeader.classList.add('settingsHeader'); const sessionStats = document.createElement('div'); sessionStats.innerHTML = 'Session Stats'; this._statsContentElement.appendChild(streamToolStats); streamToolStats.appendChild(controlStats); controlStats.appendChild(statistics); statistics.appendChild(statisticsHeader); statisticsHeader.appendChild(sessionStats); statistics.appendChild(this.statisticsContainer); controlStats.appendChild(this.latencyTest.rootElement); controlStats.appendChild(this.dataChannelLatencyTest.rootElement); } return this._statsContentElement; } get statisticsContainer() { if (!this._statisticsContainer) { this._statisticsContainer = document.createElement('div'); this._statisticsContainer.id = 'statisticsContainer'; this._statisticsContainer.classList.add('d-none'); this._statisticsContainer.appendChild(this.statsResult); } return this._statisticsContainer; } get statsResult() { if (!this._statsResult) { this._statsResult = document.createElement('div'); this._statsResult.id = 'statisticsResult'; this._statsResult.classList.add('StatsResult'); } return this._statsResult; } get statsCloseButton() { if (!this._statsCloseButton) { this._statsCloseButton = document.createElement('div'); this._statsCloseButton.id = 'statsClose'; } return this._statsCloseButton; } onDisconnect() { this.latencyTest.latencyTestButton.onclick = () => { // do nothing }; this.dataChannelLatencyTest.latencyTestButton.onclick = () => { //do nothing }; } onVideoInitialized(stream) { // starting a latency check this.latencyTest.latencyTestButton.onclick = () => { stream.requestLatencyTest(); }; this.dataChannelLatencyTest.latencyTestButton.onclick = () => { let started = stream.requestDataChannelLatencyTest({ duration: 1000, rps: 10, requestSize: 200, responseSize: 200 }); if (started) { this.dataChannelLatencyTest.handleTestStart(); } }; } configure(settings) { if (settings.DisableLatencyTest) { this.latencyTest.latencyTestButton.disabled = true; this.latencyTest.latencyTestButton.title = 'Disabled by -PixelStreamingDisableLatencyTester=true'; this.dataChannelLatencyTest.latencyTestButton.disabled = true; this.dataChannelLatencyTest.latencyTestButton.title = 'Disabled by -PixelStreamingDisableLatencyTester=true'; _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.Info(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), '-PixelStreamingDisableLatencyTester=true, requesting latency report from the the browser to UE is disabled.'); } } /** * Show stats panel. */ show() { if (!this.rootElement.classList.contains('panel-wrap-visible')) { this.rootElement.classList.add('panel-wrap-visible'); } } /** * Toggle the visibility of the stats panel. */ toggleVisibility() { this.rootElement.classList.toggle('panel-wrap-visible'); } /** * Hide the stats panel. */ hide() { if (this.rootElement.classList.contains('panel-wrap-visible')) { this.rootElement.classList.remove('panel-wrap-visible'); } } handlePlayerCount(playerCount) { this.addOrUpdateStat('PlayerCountStat', 'Players', playerCount.toString()); } /** * Handle stats coming in from browser/UE * @param stats the stats structure */ handleStats(stats) { var _a, _b, _c, _d, _e; // format numbering based on the browser language const numberFormat = new Intl.NumberFormat(window.navigator.language, { maximumFractionDigits: 0 }); // Inbound data const inboundData = _Util_MathUtils__WEBPACK_IMPORTED_MODULE_3__.MathUtils.formatBytes(stats.inboundVideoStats.bytesReceived, 2); this.addOrUpdateStat('InboundDataStat', 'Received', inboundData); // Packets lost const packetsLostStat = Object.prototype.hasOwnProperty.call(stats.inboundVideoStats, 'packetsLost') ? numberFormat.format(stats.inboundVideoStats.packetsLost) : 'Chrome only'; this.addOrUpdateStat('PacketsLostStat', 'Packets Lost', packetsLostStat); // Bitrate if (stats.inboundVideoStats.bitrate) { this.addOrUpdateStat('VideoBitrateStat', 'Video Bitrate (kbps)', stats.inboundVideoStats.bitrate.toString()); } if (stats.inboundAudioStats.bitrate) { this.addOrUpdateStat('AudioBitrateStat', 'Audio Bitrate (kbps)', stats.inboundAudioStats.bitrate.toString()); } // Video resolution const resStat = Object.prototype.hasOwnProperty.call(stats.inboundVideoStats, 'frameWidth') && stats.inboundVideoStats.frameWidth && Object.prototype.hasOwnProperty.call(stats.inboundVideoStats, 'frameHeight') && stats.inboundVideoStats.frameHeight ? stats.inboundVideoStats.frameWidth + 'x' + stats.inboundVideoStats.frameHeight : 'Chrome only'; this.addOrUpdateStat('VideoResStat', 'Video resolution', resStat); // Frames decoded const framesDecoded = Object.prototype.hasOwnProperty.call(stats.inboundVideoStats, 'framesDecoded') ? numberFormat.format(stats.inboundVideoStats.framesDecoded) : 'Chrome only'; this.addOrUpdateStat('FramesDecodedStat', 'Frames Decoded', framesDecoded); // Framerate if (stats.inboundVideoStats.framesPerSecond) { this.addOrUpdateStat('FramerateStat', 'Framerate', stats.inboundVideoStats.framesPerSecond.toString()); } // Frames dropped this.addOrUpdateStat('FramesDroppedStat', 'Frames dropped', (_a = stats.inboundVideoStats.framesDropped) === null || _a === void 0 ? void 0 : _a.toString()); if (stats.inboundVideoStats.codecId) { this.addOrUpdateStat('VideoCodecStat', 'Video codec', // Split the codec to remove the Fmtp line (_c = (_b = stats.codecs .get(stats.inboundVideoStats.codecId)) === null || _b === void 0 ? void 0 : _b.split(' ')[0]) !== null && _c !== void 0 ? _c : ''); } if (stats.inboundAudioStats.codecId) { this.addOrUpdateStat('AudioCodecStat', 'Audio codec', // Split the codec to remove the Fmtp line (_e = (_d = stats.codecs .get(stats.inboundAudioStats.codecId)) === null || _d === void 0 ? void 0 : _d.split(' ')[0]) !== null && _e !== void 0 ? _e : ''); } // RTT const netRTT = Object.prototype.hasOwnProperty.call(stats.candidatePair, 'currentRoundTripTime') && stats.isNumber(stats.candidatePair.currentRoundTripTime) ? numberFormat.format(stats.candidatePair.currentRoundTripTime * 1000) : "Can't calculate"; this.addOrUpdateStat('RTTStat', 'Net RTT (ms)', netRTT); this.addOrUpdateStat('DurationStat', 'Duration', stats.sessionStats.runTime); this.addOrUpdateStat('ControlsInputStat', 'Controls stream input', stats.sessionStats.controlsStreamInput); // QP this.addOrUpdateStat('QPStat', 'Video quantization parameter', stats.sessionStats.videoEncoderAvgQP.toString()); // todo: //statsText += `
Browser receive to composite (ms): ${stats.inboundVideoStats.receiveToCompositeMs}
`; _epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.Log(_epicgames_ps_lib_pixelstreamingfrontend_ue5_3__WEBPACK_IMPORTED_MODULE_0__.Logger.GetStackTrace(), `--------- Stats ---------\n ${stats}\n------------------------`, 6); } /** * Adds a new stat to the stats results in the DOM or updates an exiting stat. * @param id The id of the stat to add/update. * @param stat The contents of the stat. */ addOrUpdateStat(id, statLabel, stat) { const statHTML = `${statLabel}: ${stat}`; if (!this.statsMap.has(id)) { // create the stat const newStat = new Stat(); newStat.id = id; newStat.stat = stat; newStat.title = statLabel; newStat.element = document.createElement('div'); newStat.element.innerHTML = statHTML; // add the stat to the dom this.statsResult.appendChild(newStat.element); this.statsMap.set(id, newStat); } // update the existing stat else { const value = this.statsMap.get(id); if (value !== undefined) { value.element.innerHTML = statHTML; } } } } /***/ }), /***/ "./src/UI/UIConfigurationTypes.ts": /*!****************************************!*\ !*** ./src/UI/UIConfigurationTypes.ts ***! \****************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "UIElementCreationMode": () => (/* binding */ UIElementCreationMode), /* harmony export */ "isPanelEnabled": () => (/* binding */ isPanelEnabled) /* harmony export */ }); /** Whether a stream UI element is internally made, externally provided, or disabled. */ var UIElementCreationMode; (function (UIElementCreationMode) { UIElementCreationMode[UIElementCreationMode["CreateDefaultElement"] = 0] = "CreateDefaultElement"; UIElementCreationMode[UIElementCreationMode["UseCustomElement"] = 1] = "UseCustomElement"; UIElementCreationMode[UIElementCreationMode["Disable"] = 2] = "Disable"; })(UIElementCreationMode || (UIElementCreationMode = {})); function isPanelEnabled(config) { return !config || (!!config && config.isEnabled); } /***/ }), /***/ "./src/UI/VideoQpIndicator.ts": /*!************************************!*\ !*** ./src/UI/VideoQpIndicator.ts ***! \************************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "VideoQpIndicator": () => (/* binding */ VideoQpIndicator) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * A UI element showing the QP (quantization parameter) of the video stream at the last encoded frame (well, last transmitted QP really). * A blockier encoding will have a higher QP and this will make the indicator turn more red. * A non-blocky stream will have a lower QP and this will make the indicator turn more green. * The QP indicator is represented visually using a WiFi icon. */ class VideoQpIndicator { constructor() { this.videoEncoderAvgQP = -1; // non html elements this.statsText = ''; this.color = ''; // qp colors this.orangeQP = 26; this.redQP = 35; } /** * Get the root element of the QP indicator. */ get rootElement() { if (!this._rootElement) { // make the root element that contains the svg for the connection this._rootElement = document.createElement('div'); this._rootElement.id = 'connection'; this._rootElement.classList.add('UiTool'); // add svg icon for the connection strength this._rootElement.appendChild(this.qualityStatus); // add the text underneath the connection this._rootElement.appendChild(this.qualityText); // set colors to not connected initially this.updateQpTooltip(-1); } return this._rootElement; } /** * Get the text that displays under the icon. */ get qualityText() { if (!this._qualityText) { this._qualityText = document.createElement('span'); this._qualityText.id = 'qualityText'; this._qualityText.classList.add('tooltiptext'); } return this._qualityText; } /** * Get the icon. */ get qualityStatus() { if (!this._qualityStatus) { this._qualityStatus = document.createElementNS('http://www.w3.org/2000/svg', 'svg'); this._qualityStatus.setAttributeNS(null, 'id', 'connectionStrength'); this._qualityStatus.setAttributeNS(null, 'x', '0px'); this._qualityStatus.setAttributeNS(null, 'y', '0px'); this._qualityStatus.setAttributeNS(null, 'viewBox', '0 0 494.45 494.45'); // build wifi icon this.qualityStatus.appendChild(this.dot); this.qualityStatus.appendChild(this.middle); this.qualityStatus.appendChild(this.outer); this.qualityStatus.appendChild(this.inner); } return this._qualityStatus; } /** * Get the dot at the bottom of the wifi icon. */ get dot() { if (!this._dot) { this._dot = document.createElementNS('http://www.w3.org/2000/svg', 'circle'); this._dot.setAttributeNS(null, 'id', 'dot'); this._dot.setAttributeNS(null, 'cx', '247.125'); this._dot.setAttributeNS(null, 'cy', '398.925'); this._dot.setAttributeNS(null, 'r', '35.3'); } return this._dot; } /** * Get the outer arc of the wifi icon. */ get outer() { if (!this._outer) { this._outer = document.createElementNS('http://www.w3.org/2000/svg', 'path'); this._outer.setAttributeNS(null, 'id', 'outer'); this._outer.setAttributeNS(null, 'd', 'M467.925,204.625c-6.8,0-13.5-2.6-18.7-7.8c-111.5-111.4-292.7-111.4-404.1,0c-10.3,10.3-27.1,10.3-37.4,0s-10.3-27.1,0-37.4c64-64,149-99.2,239.5-99.2s175.5,35.2,239.5,99.2c10.3,10.3,10.3,27.1,0,37.4C481.425,202.025,474.625,204.625,467.925,204.625z'); } return this._outer; } /** * Get the middle arc of the wifi icon. */ get middle() { if (!this._middle) { this._middle = document.createElementNS('http://www.w3.org/2000/svg', 'path'); this._middle.setAttributeNS(null, 'id', 'middle'); this._middle.setAttributeNS(null, 'd', 'M395.225,277.325c-6.8,0-13.5-2.6-18.7-7.8c-71.4-71.3-187.4-71.3-258.8,0c-10.3,10.3-27.1,10.3-37.4,0s-10.3-27.1,0-37.4c92-92,241.6-92,333.6,0c10.3,10.3,10.3,27.1,0,37.4C408.725,274.725,401.925,277.325,395.225,277.325z'); } return this._middle; } /** * Get the inner arc of the wifi icon. */ get inner() { if (!this._inner) { this._inner = document.createElementNS('http://www.w3.org/2000/svg', 'path'); this._inner.setAttributeNS(null, 'id', 'inner'); this._inner.setAttributeNS(null, 'd', 'M323.625,348.825c-6.8,0-13.5-2.6-18.7-7.8c-15.4-15.4-36-23.9-57.8-23.9s-42.4,8.5-57.8,23.9c-10.3,10.3-27.1,10.3-37.4,0c-10.3-10.3-10.3-27.1,0-37.4c25.4-25.4,59.2-39.4,95.2-39.4s69.8,14,95.2,39.5c10.3,10.3,10.3,27.1,0,37.4C337.225,346.225,330.425,348.825,323.625,348.825z'); } return this._inner; } /** * Used to set the speed of the status light. * @param speed - Set the speed of the blink, higher numbers make the status light blink faster. */ blinkVideoQualityStatus(speed) { let iteration = speed; let opacity = 1; const tickID = setInterval(() => { opacity -= 0.1; this.qualityText.style.opacity = String(Math.abs((opacity - 0.5) * 2)); if (opacity <= 0.1) { if (--iteration == 0) { clearInterval(tickID); } else { opacity = 1; } } }, 100 / speed); } /** * updates the QP tooltip by converting the Video Encoder QP to a color light * @param QP - The video encoder QP number needed to find the average */ updateQpTooltip(QP) { this.videoEncoderAvgQP = QP; if (QP > this.redQP) { this.color = 'red'; this.blinkVideoQualityStatus(2); this.statsText = `
Poor encoding quality
`; this.outer.setAttributeNS(null, 'fill', '#3c3b40'); this.middle.setAttributeNS(null, 'fill', '#3c3b40'); this.inner.setAttributeNS(null, 'fill', this.color); this.dot.setAttributeNS(null, 'fill', this.color); } else if (QP > this.orangeQP) { this.color = 'orange'; this.blinkVideoQualityStatus(1); this.statsText = `
Blocky encoding quality
`; this.outer.setAttributeNS(null, 'fill', '#3c3b40'); this.middle.setAttributeNS(null, 'fill', this.color); this.inner.setAttributeNS(null, 'fill', this.color); this.dot.setAttributeNS(null, 'fill', this.color); } else if (QP <= 0) { this.color = '#b0b0b0'; this.outer.setAttributeNS(null, 'fill', '#3c3b40'); this.middle.setAttributeNS(null, 'fill', '#3c3b40'); this.inner.setAttributeNS(null, 'fill', '#3c3b40'); this.dot.setAttributeNS(null, 'fill', '#3c3b40'); this.statsText = `
Not connected
`; } else { this.color = 'lime'; this.qualityStatus.style.opacity = '1'; this.statsText = `
Clear encoding quality
`; this.outer.setAttributeNS(null, 'fill', this.color); this.middle.setAttributeNS(null, 'fill', this.color); this.inner.setAttributeNS(null, 'fill', this.color); this.dot.setAttributeNS(null, 'fill', this.color); } this.qualityText.innerHTML = this.statsText; } } /***/ }), /***/ "./src/UI/XRIcon.ts": /*!**************************!*\ !*** ./src/UI/XRIcon.ts ***! \**************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "XRIcon": () => (/* binding */ XRIcon) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. /** * XR icon that can be clicked. */ class XRIcon { /** * Get the the button containing the XR icon. */ get rootElement() { if (!this._rootElement) { this._rootElement = document.createElement('button'); this._rootElement.type = 'button'; this._rootElement.classList.add('UiTool'); this._rootElement.id = 'xrBtn'; this._rootElement.appendChild(this.xrIcon); this._rootElement.appendChild(this.tooltipText); } return this._rootElement; } get tooltipText() { if (!this._tooltipText) { this._tooltipText = document.createElement('span'); this._tooltipText.classList.add('tooltiptext'); this._tooltipText.innerHTML = 'XR'; } return this._tooltipText; } get xrIcon() { if (!this._xrIcon) { this._xrIcon = document.createElementNS('http://www.w3.org/2000/svg', 'svg'); this._xrIcon.setAttributeNS(null, 'id', 'xrIcon'); this._xrIcon.setAttributeNS(null, 'x', '0px'); this._xrIcon.setAttributeNS(null, 'y', '0px'); this._xrIcon.setAttributeNS(null, 'viewBox', '0 0 100 100'); // create svg group for the paths const svgGroup = document.createElementNS('http://www.w3.org/2000/svg', 'g'); svgGroup.classList.add('svgIcon'); this._xrIcon.appendChild(svgGroup); // create paths for the icon itself, the path of the xr headset const path = document.createElementNS('http://www.w3.org/2000/svg', 'path'); path.setAttributeNS(null, 'd', 'M29 41c-5 0-9 4-9 9s4 9 9 9 9-4 9-9-4-9-9-9zm0 14c-2.8 0-5-2.2-5-5s2.2-5 5-5 5 2.2 5 5-2.2 5-5 5zm42-14c-5 0-9 4-9 9s4 9 9 9 9-4 9-9-4-9-9-9zm0 14c-2.8 0-5-2.2-5-5s2.2-5 5-5 5 2.2 5 5-2.2 5-5 5zm12-31H17c-6.6 0-12 5.4-12 12v28c0 6.6 5.4 12 12 12h14.5c3.5 0 6.8-1.5 9-4.1l3.5-4c1.5-1.7 3.7-2.7 6-2.7s4.5 1 6 2.7l3.5 4c2.3 2.6 5.6 4.1 9 4.1H83c6.6 0 12-5.4 12-12V36c0-6.6-5.4-12-12-12zm8 40c0 4.4-3.6 8-8 8H68.5c-2.3 0-4.5-1-6-2.7l-3.5-4c-2.3-2.6-5.6-4.1-9-4.1-3.5 0-6.8 1.5-9 4.1l-3.5 4C36 71 33.8 72 31.5 72H17c-4.4 0-8-3.6-8-8V36c0-4.4 3.6-8 8-8h66c4.4 0 8 3.6 8 8v28z'); svgGroup.appendChild(path); } return this._xrIcon; } } /***/ }), /***/ "./src/Util/MathUtils.ts": /*!*******************************!*\ !*** ./src/Util/MathUtils.ts ***! \*******************************/ /***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "MathUtils": () => (/* binding */ MathUtils) /* harmony export */ }); // Copyright Epic Games, Inc. All Rights Reserved. class MathUtils { /** * formats Bytes coming in for video stats * @param bytes number to convert * @param decimals number of decimal places */ static formatBytes(bytes, decimals) { if (bytes === 0) { return '0'; } const factor = 1024; const dm = decimals < 0 ? 0 : decimals; const sizes = [ 'Bytes', 'KiB', 'MiB', 'GiB', 'TiB', 'PiB', 'EiB', 'ZiB', 'YiB' ]; const i = Math.floor(Math.log(bytes) / Math.log(factor)); return (parseFloat((bytes / Math.pow(factor, i)).toFixed(dm)) + ' ' + sizes[i]); } } /***/ }), /***/ "@epicgames-ps/lib-pixelstreamingfrontend-ue5.3": /*!*****************************************************************!*\ !*** external "@epicgames-ps/lib-pixelstreamingfrontend-ue5.3" ***! \*****************************************************************/ /***/ ((module) => { var x = y => { var x = {}; __webpack_require__.d(x, y); return x; } var y = x => () => x module.exports = __WEBPACK_EXTERNAL_MODULE__epicgames_ps_lib_pixelstreamingfrontend_ue5_3_512f3c9b__; /***/ }), /***/ "jss": /*!**********************!*\ !*** external "jss" ***! \**********************/ /***/ ((module) => { var x = y => { var x = {}; __webpack_require__.d(x, y); return x; } var y = x => () => x module.exports = __WEBPACK_EXTERNAL_MODULE_jss__; /***/ }), /***/ "jss-plugin-camel-case": /*!****************************************!*\ !*** external "jss-plugin-camel-case" ***! \****************************************/ /***/ ((module) => { var x = y => { var x = {}; __webpack_require__.d(x, y); return x; } var y = x => () => x module.exports = __WEBPACK_EXTERNAL_MODULE_jss_plugin_camel_case_de113355__; /***/ }), /***/ "jss-plugin-global": /*!************************************!*\ !*** external "jss-plugin-global" ***! \************************************/ /***/ ((module) => { var x = y => { var x = {}; __webpack_require__.d(x, y); return x; } var y = x => () => x module.exports = __WEBPACK_EXTERNAL_MODULE_jss_plugin_global_ef86f421__; /***/ }) /******/ }); /************************************************************************/ /******/ // 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/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-ui.ts ***! \******************************************/ __webpack_require__.r(__webpack_exports__); /* harmony export */ __webpack_require__.d(__webpack_exports__, { /* harmony export */ "AFKOverlay": () => (/* reexport safe */ _Overlay_AFKOverlay__WEBPACK_IMPORTED_MODULE_2__.AFKOverlay), /* harmony export */ "ActionOverlay": () => (/* reexport safe */ _Overlay_ActionOverlay__WEBPACK_IMPORTED_MODULE_3__.ActionOverlay), /* harmony export */ "Application": () => (/* reexport safe */ _Application_Application__WEBPACK_IMPORTED_MODULE_0__.Application), /* harmony export */ "ConfigUI": () => (/* reexport safe */ _Config_ConfigUI__WEBPACK_IMPORTED_MODULE_11__.ConfigUI), /* harmony export */ "ConnectOverlay": () => (/* reexport safe */ _Overlay_ConnectOverlay__WEBPACK_IMPORTED_MODULE_5__.ConnectOverlay), /* harmony export */ "DisconnectOverlay": () => (/* reexport safe */ _Overlay_DisconnectOverlay__WEBPACK_IMPORTED_MODULE_6__.DisconnectOverlay), /* harmony export */ "ErrorOverlay": () => (/* reexport safe */ _Overlay_ErrorOverlay__WEBPACK_IMPORTED_MODULE_7__.ErrorOverlay), /* harmony export */ "InfoOverlay": () => (/* reexport safe */ _Overlay_InfoOverlay__WEBPACK_IMPORTED_MODULE_8__.InfoOverlay), /* harmony export */ "OverlayBase": () => (/* reexport safe */ _Overlay_BaseOverlay__WEBPACK_IMPORTED_MODULE_4__.OverlayBase), /* harmony export */ "PixelStreamingApplicationStyle": () => (/* reexport safe */ _Styles_PixelStreamingApplicationStyles__WEBPACK_IMPORTED_MODULE_1__.PixelStreamingApplicationStyle), /* harmony export */ "PlayOverlay": () => (/* reexport safe */ _Overlay_PlayOverlay__WEBPACK_IMPORTED_MODULE_9__.PlayOverlay), /* harmony export */ "SettingUIBase": () => (/* reexport safe */ _Config_SettingUIBase__WEBPACK_IMPORTED_MODULE_12__.SettingUIBase), /* harmony export */ "SettingUIFlag": () => (/* reexport safe */ _Config_SettingUIFlag__WEBPACK_IMPORTED_MODULE_13__.SettingUIFlag), /* harmony export */ "SettingUINumber": () => (/* reexport safe */ _Config_SettingUINumber__WEBPACK_IMPORTED_MODULE_14__.SettingUINumber), /* harmony export */ "SettingUIOption": () => (/* reexport safe */ _Config_SettingUIOption__WEBPACK_IMPORTED_MODULE_15__.SettingUIOption), /* harmony export */ "SettingUIText": () => (/* reexport safe */ _Config_SettingUIText__WEBPACK_IMPORTED_MODULE_16__.SettingUIText), /* harmony export */ "TextOverlay": () => (/* reexport safe */ _Overlay_TextOverlay__WEBPACK_IMPORTED_MODULE_10__.TextOverlay), /* harmony export */ "UIElementCreationMode": () => (/* reexport safe */ _UI_UIConfigurationTypes__WEBPACK_IMPORTED_MODULE_17__.UIElementCreationMode) /* harmony export */ }); /* harmony import */ var _Application_Application__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./Application/Application */ "./src/Application/Application.ts"); /* harmony import */ var _Styles_PixelStreamingApplicationStyles__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./Styles/PixelStreamingApplicationStyles */ "./src/Styles/PixelStreamingApplicationStyles.ts"); /* harmony import */ var _Overlay_AFKOverlay__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./Overlay/AFKOverlay */ "./src/Overlay/AFKOverlay.ts"); /* harmony import */ var _Overlay_ActionOverlay__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./Overlay/ActionOverlay */ "./src/Overlay/ActionOverlay.ts"); /* harmony import */ var _Overlay_BaseOverlay__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! ./Overlay/BaseOverlay */ "./src/Overlay/BaseOverlay.ts"); /* harmony import */ var _Overlay_ConnectOverlay__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! ./Overlay/ConnectOverlay */ "./src/Overlay/ConnectOverlay.ts"); /* harmony import */ var _Overlay_DisconnectOverlay__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! ./Overlay/DisconnectOverlay */ "./src/Overlay/DisconnectOverlay.ts"); /* harmony import */ var _Overlay_ErrorOverlay__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! ./Overlay/ErrorOverlay */ "./src/Overlay/ErrorOverlay.ts"); /* harmony import */ var _Overlay_InfoOverlay__WEBPACK_IMPORTED_MODULE_8__ = __webpack_require__(/*! ./Overlay/InfoOverlay */ "./src/Overlay/InfoOverlay.ts"); /* harmony import */ var _Overlay_PlayOverlay__WEBPACK_IMPORTED_MODULE_9__ = __webpack_require__(/*! ./Overlay/PlayOverlay */ "./src/Overlay/PlayOverlay.ts"); /* harmony import */ var _Overlay_TextOverlay__WEBPACK_IMPORTED_MODULE_10__ = __webpack_require__(/*! ./Overlay/TextOverlay */ "./src/Overlay/TextOverlay.ts"); /* harmony import */ var _Config_ConfigUI__WEBPACK_IMPORTED_MODULE_11__ = __webpack_require__(/*! ./Config/ConfigUI */ "./src/Config/ConfigUI.ts"); /* harmony import */ var _Config_SettingUIBase__WEBPACK_IMPORTED_MODULE_12__ = __webpack_require__(/*! ./Config/SettingUIBase */ "./src/Config/SettingUIBase.ts"); /* harmony import */ var _Config_SettingUIFlag__WEBPACK_IMPORTED_MODULE_13__ = __webpack_require__(/*! ./Config/SettingUIFlag */ "./src/Config/SettingUIFlag.ts"); /* harmony import */ var _Config_SettingUINumber__WEBPACK_IMPORTED_MODULE_14__ = __webpack_require__(/*! ./Config/SettingUINumber */ "./src/Config/SettingUINumber.ts"); /* harmony import */ var _Config_SettingUIOption__WEBPACK_IMPORTED_MODULE_15__ = __webpack_require__(/*! ./Config/SettingUIOption */ "./src/Config/SettingUIOption.ts"); /* harmony import */ var _Config_SettingUIText__WEBPACK_IMPORTED_MODULE_16__ = __webpack_require__(/*! ./Config/SettingUIText */ "./src/Config/SettingUIText.ts"); /* harmony import */ var _UI_UIConfigurationTypes__WEBPACK_IMPORTED_MODULE_17__ = __webpack_require__(/*! ./UI/UIConfigurationTypes */ "./src/UI/UIConfigurationTypes.ts"); // Copyright Epic Games, Inc. All Rights Reserved. })(); var __webpack_exports__AFKOverlay = __webpack_exports__.AFKOverlay; var __webpack_exports__ActionOverlay = __webpack_exports__.ActionOverlay; var __webpack_exports__Application = __webpack_exports__.Application; var __webpack_exports__ConfigUI = __webpack_exports__.ConfigUI; var __webpack_exports__ConnectOverlay = __webpack_exports__.ConnectOverlay; var __webpack_exports__DisconnectOverlay = __webpack_exports__.DisconnectOverlay; var __webpack_exports__ErrorOverlay = __webpack_exports__.ErrorOverlay; var __webpack_exports__InfoOverlay = __webpack_exports__.InfoOverlay; var __webpack_exports__OverlayBase = __webpack_exports__.OverlayBase; var __webpack_exports__PixelStreamingApplicationStyle = __webpack_exports__.PixelStreamingApplicationStyle; var __webpack_exports__PlayOverlay = __webpack_exports__.PlayOverlay; var __webpack_exports__SettingUIBase = __webpack_exports__.SettingUIBase; var __webpack_exports__SettingUIFlag = __webpack_exports__.SettingUIFlag; var __webpack_exports__SettingUINumber = __webpack_exports__.SettingUINumber; var __webpack_exports__SettingUIOption = __webpack_exports__.SettingUIOption; var __webpack_exports__SettingUIText = __webpack_exports__.SettingUIText; var __webpack_exports__TextOverlay = __webpack_exports__.TextOverlay; var __webpack_exports__UIElementCreationMode = __webpack_exports__.UIElementCreationMode; export { __webpack_exports__AFKOverlay as AFKOverlay, __webpack_exports__ActionOverlay as ActionOverlay, __webpack_exports__Application as Application, __webpack_exports__ConfigUI as ConfigUI, __webpack_exports__ConnectOverlay as ConnectOverlay, __webpack_exports__DisconnectOverlay as DisconnectOverlay, __webpack_exports__ErrorOverlay as ErrorOverlay, __webpack_exports__InfoOverlay as InfoOverlay, __webpack_exports__OverlayBase as OverlayBase, __webpack_exports__PixelStreamingApplicationStyle as PixelStreamingApplicationStyle, __webpack_exports__PlayOverlay as PlayOverlay, __webpack_exports__SettingUIBase as SettingUIBase, __webpack_exports__SettingUIFlag as SettingUIFlag, __webpack_exports__SettingUINumber as SettingUINumber, __webpack_exports__SettingUIOption as SettingUIOption, __webpack_exports__SettingUIText as SettingUIText, __webpack_exports__TextOverlay as TextOverlay, __webpack_exports__UIElementCreationMode as UIElementCreationMode }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"lib-pixelstreamingfrontend-ui.esm.js","mappings":"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","sources":["webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Application/Application.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Config/ConfigUI.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Config/SettingUIBase.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Config/SettingUIFlag.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Config/SettingUINumber.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Config/SettingUIOption.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Config/SettingUIText.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Overlay/AFKOverlay.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Overlay/ActionOverlay.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Overlay/BaseOverlay.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Overlay/ConnectOverlay.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Overlay/DisconnectOverlay.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Overlay/ErrorOverlay.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Overlay/InfoOverlay.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Overlay/PlayOverlay.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Overlay/TextOverlay.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Styles/PixelStreamingApplicationStyles.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/UI/Controls.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/UI/DataChannelLatencyTest.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/UI/FullscreenIcon.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/UI/LabelledButton.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/UI/LatencyTest.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/UI/SettingsIcon.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/UI/SettingsPanel.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/UI/StatsIcon.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/UI/StatsPanel.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/UI/UIConfigurationTypes.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/UI/VideoQpIndicator.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/UI/XRIcon.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/Util/MathUtils.ts","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/external module \"@epicgames-ps/lib-pixelstreamingfrontend-ue5.3\"","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/external module \"jss\"","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/external module \"jss-plugin-camel-case\"","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/external module \"jss-plugin-global\"","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/webpack/bootstrap","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/webpack/runtime/define property getters","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/webpack/runtime/hasOwnProperty shorthand","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/webpack/runtime/make namespace object","webpack://@epicgames-ps/lib-pixelstreamingfrontend-ui-ue5.3/./src/pixelstreamingfrontend-ui.ts"],"sourcesContent":["// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport {\n    PixelStreaming,\n    Flags,\n    Logger,\n    AggregatedStats,\n    LatencyTestResults,\n    InitialSettings,\n    MessageStreamerList\n} from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\nimport { OverlayBase } from '../Overlay/BaseOverlay';\nimport { ActionOverlay } from '../Overlay/ActionOverlay';\nimport { TextOverlay } from '../Overlay/TextOverlay';\nimport { ConnectOverlay } from '../Overlay/ConnectOverlay';\nimport { DisconnectOverlay } from '../Overlay/DisconnectOverlay';\nimport { PlayOverlay } from '../Overlay/PlayOverlay';\nimport { InfoOverlay } from '../Overlay/InfoOverlay';\nimport { ErrorOverlay } from '../Overlay/ErrorOverlay';\nimport { AFKOverlay } from '../Overlay/AFKOverlay';\nimport { Controls, ControlsUIConfiguration } from '../UI/Controls';\nimport { LabelledButton } from '../UI/LabelledButton';\nimport { SettingsPanel } from '../UI/SettingsPanel';\nimport { StatsPanel } from '../UI/StatsPanel';\nimport { VideoQpIndicator } from '../UI/VideoQpIndicator';\nimport { ConfigUI, LightMode } from '../Config/ConfigUI';\nimport { \n    UIElementCreationMode, \n    PanelConfiguration, \n    isPanelEnabled,\n    UIElementConfig\n} from '../UI/UIConfigurationTypes'\nimport { FullScreenIconBase, FullScreenIconExternal } from '../UI/FullscreenIcon';\nimport {\n    DataChannelLatencyTestResult\n} from \"@epicgames-ps/lib-pixelstreamingfrontend-ue5.3/types/DataChannel/DataChannelLatencyTestResults\";\n\n\n/** \n * Configuration of the internal video QP indicator element.\n * By default, one will be made, but if needed this can be disabled.\n * \n * Note: For custom UI elements to react to the QP being changed, use a PixelStreaming \n * object's addEventListener('videoEncoderAvgQP', ...) or removeEventListener(...).\n */\nexport type VideoQPIndicatorConfig = {\n    disableIndicator?: boolean\n}\n\n/**\n * UI Options can be provided when creating an Application, to configure it's internal\n * and external behaviour, enable/disable features, and connect to external UI.\n */\nexport interface UIOptions {\n    stream: PixelStreaming;\n    onColorModeChanged?: (isLightMode: boolean) => void;\n    /** By default, a settings panel and associate visibility toggle button will be made.\n      * If needed, this behaviour can be configured. */\n    settingsPanelConfig?: PanelConfiguration;\n    /** By default, a stats panel and associate visibility toggle button will be made.\n      * If needed, this behaviour can be configured. */\n    statsPanelConfig?: PanelConfiguration;\n    /** If needed, the full screen button can be external or disabled. */\n    fullScreenControlsConfig? : UIElementConfig,\n    /** If needed, XR button can be external or disabled. */\n    xrControlsConfig? : UIElementConfig,\n    /** Configuration of the video QP indicator. */\n    videoQpIndicatorConfig? : VideoQPIndicatorConfig\n}\n\n/**\n * An Application is a combination of UI elements to display and manage a WebRTC Pixel Streaming\n * connection. It includes features for controlling a stream with mouse and keyboard, \n * managing connection endpoints, as well as displaying stats and other information about it.\n */\nexport class Application {\n    stream: PixelStreaming;\n\n    _rootElement: HTMLElement;\n    _uiFeatureElement: HTMLElement;\n\n    // set the overlay placeholders\n    currentOverlay: OverlayBase | null;\n    disconnectOverlay: ActionOverlay;\n    connectOverlay: ActionOverlay;\n    playOverlay: ActionOverlay;\n    infoOverlay: TextOverlay;\n    errorOverlay: TextOverlay;\n    afkOverlay: AFKOverlay;\n\n    controls: Controls;\n\n    settingsPanel: SettingsPanel;\n    statsPanel: StatsPanel;\n    videoQpIndicator: VideoQpIndicator;\n\n    configUI: ConfigUI;\n\n    onColorModeChanged: UIOptions[\"onColorModeChanged\"];\n\n    protected _options : UIOptions;\n\n    /**\n     * @param options - Initialization options\n     */\n    constructor(options: UIOptions) {\n        this._options = options;\n        \n        this.stream = options.stream;\n        this.onColorModeChanged = options.onColorModeChanged;\n        this.configUI = new ConfigUI(this.stream.config);\n\n        this.createOverlays();\n\n        if (isPanelEnabled(options.statsPanelConfig)) {\n            // Add stats panel\n            this.statsPanel = new StatsPanel();\n            this.uiFeaturesElement.appendChild(this.statsPanel.rootElement);\n        }\n        \n        if (isPanelEnabled(options.settingsPanelConfig)) {\n            // Add settings panel\n            this.settingsPanel = new SettingsPanel();\n            this.uiFeaturesElement.appendChild(this.settingsPanel.rootElement);\n            this.configureSettings();\n        }\n        \n        if (!options.videoQpIndicatorConfig || !options.videoQpIndicatorConfig.disableIndicator) {\n            // Add the video stream QP indicator\n            this.videoQpIndicator = new VideoQpIndicator();\n            this.uiFeaturesElement.appendChild(this.videoQpIndicator.rootElement);\n        }\n\n        this.createButtons();\n\n        this.registerCallbacks();\n\n        this.showConnectOrAutoConnectOverlays();\n\n        this.setColorMode(this.configUI.isCustomFlagEnabled(LightMode));\n    }\n\n    public createOverlays(): void {\n        // build all of the overlays\n        this.disconnectOverlay = new DisconnectOverlay(\n            this.stream.videoElementParent\n        );\n        this.connectOverlay = new ConnectOverlay(\n            this.stream.videoElementParent\n        );\n        this.playOverlay = new PlayOverlay(\n            this.stream.videoElementParent\n        );\n        this.infoOverlay = new InfoOverlay(\n            this.stream.videoElementParent\n        );\n        this.errorOverlay = new ErrorOverlay(\n            this.stream.videoElementParent\n        );\n        this.afkOverlay = new AFKOverlay(\n            this.stream.videoElementParent\n        );\n\n        this.disconnectOverlay.onAction(() => this.stream.reconnect());\n\n        // Build the webRtc connect overlay Event Listener and show the connect overlay\n        this.connectOverlay.onAction(() => this.stream.connect());\n\n        // set up the play overlays action\n        this.playOverlay.onAction(() => this.stream.play());\n    }\n\n    /**\n     * Set up button click functions and button functionality\n     */\n    public createButtons() {\n        const controlsUIConfig : ControlsUIConfiguration = {\n            statsButtonType : !!this._options.statsPanelConfig\n                ? this._options.statsPanelConfig.visibilityButtonConfig\n                : undefined,\n            settingsButtonType: !!this._options.settingsPanelConfig\n                ? this._options.settingsPanelConfig.visibilityButtonConfig\n                : undefined,\n            fullscreenButtonType: this._options.fullScreenControlsConfig,\n            xrIconType: this._options.xrControlsConfig\n        }\n        // Setup controls\n        const controls = new Controls(controlsUIConfig);\n        this.uiFeaturesElement.appendChild(controls.rootElement);\n\n        // When we fullscreen we want this element to be the root\n        const fullScreenButton : FullScreenIconBase | undefined = \n            // Depending on if we're creating an internal button, or using an external one\n            (!!this._options.fullScreenControlsConfig \n                && this._options.fullScreenControlsConfig.creationMode === UIElementCreationMode.UseCustomElement)\n            // Either create a fullscreen class based on the external button\n            ? new FullScreenIconExternal(this._options.fullScreenControlsConfig.customElement)\n            // Or use the one created by the Controls initializer earlier\n            : controls.fullscreenIcon;\n        if (fullScreenButton) {\n            fullScreenButton.fullscreenElement = /iPad|iPhone|iPod/.test(navigator.userAgent) ? this.stream.videoElementParent.getElementsByTagName(\"video\")[0] : this.rootElement;\n        }\n\n        // Add settings button to controls\n        const settingsButton : HTMLElement | undefined = \n            !!controls.settingsIcon ? controls.settingsIcon.rootElement : \n            this._options.settingsPanelConfig.visibilityButtonConfig.customElement;\n        if (!!settingsButton) settingsButton.onclick = () =>\n            this.settingsClicked();\n        if (!!this.settingsPanel) this.settingsPanel.settingsCloseButton.onclick = () =>\n            this.settingsClicked();\n\n        // Add WebXR button to controls\n        const xrButton : HTMLElement | undefined = \n            !!controls.xrIcon ? controls.xrIcon.rootElement : \n            this._options.xrControlsConfig.creationMode === UIElementCreationMode.UseCustomElement ?\n            this._options.xrControlsConfig.customElement : undefined;\n        if (!!xrButton) xrButton.onclick = () =>\n            this.stream.toggleXR();\n\n        // setup the stats/info button\n        const statsButton : HTMLElement | undefined = \n            !!controls.statsIcon ? controls.statsIcon.rootElement : \n            this._options.statsPanelConfig.visibilityButtonConfig.customElement;\n        if (!!statsButton) statsButton.onclick = () => this.statsClicked()\n\n        if (!!this.statsPanel) {\n            this.statsPanel.statsCloseButton.onclick = () => this.statsClicked();\n        }\n\n        // Add command buttons (if we have somewhere to add them to)\n        if (!!this.settingsPanel) {\n            // Add button for toggle fps\n            const showFPSButton = new LabelledButton('Show FPS', 'Toggle');\n            showFPSButton.addOnClickListener(() => {\n                this.stream.requestShowFps();\n            });\n\n            // Add button for restart stream\n            const restartStreamButton = new LabelledButton(\n                'Restart Stream',\n                'Restart'\n            );\n            restartStreamButton.addOnClickListener(() => {\n                this.stream.reconnect();\n            });\n\n            // Add button for request keyframe\n            const requestKeyframeButton = new LabelledButton(\n                'Request keyframe',\n                'Request'\n            );\n            requestKeyframeButton.addOnClickListener(() => {\n                this.stream.requestIframe();\n            });\n\n            const commandsSectionElem = this.configUI.buildSectionWithHeading(\n                this.settingsPanel.settingsContentElement,\n                'Commands'\n            );\n            commandsSectionElem.appendChild(showFPSButton.rootElement);\n            commandsSectionElem.appendChild(requestKeyframeButton.rootElement);\n            commandsSectionElem.appendChild(restartStreamButton.rootElement);\n        }\n    }\n\n    /**\n     * Configure the settings with on change listeners and any additional per experience settings.\n     */\n    configureSettings(): void {\n        // This builds all the settings sections and flags under this `settingsContent` element.\n        this.configUI.populateSettingsElement(\n            this.settingsPanel.settingsContentElement\n        );\n\n        this.configUI.addCustomFlagOnSettingChangedListener(\n            LightMode,\n            (isLightMode: boolean) => {\n                this.configUI.setCustomFlagLabel(\n                    LightMode,\n                    `Color Scheme: ${isLightMode ? 'Light' : 'Dark'} Mode`\n                );\n                this.setColorMode(isLightMode);\n            }\n        );\n    }\n\n    registerCallbacks() {\n        this.stream.addEventListener(\n            'afkWarningActivate',\n            ({ data: { countDown, dismissAfk } }) =>\n                this.showAfkOverlay(countDown, dismissAfk)\n        );\n        this.stream.addEventListener(\n            'afkWarningUpdate',\n            ({ data: { countDown } }) =>\n                this.afkOverlay.updateCountdown(countDown)\n        );\n        this.stream.addEventListener(\n            'afkWarningDeactivate',\n            () => this.afkOverlay.hide()\n        );\n        this.stream.addEventListener('afkTimedOut', () =>\n            this.afkOverlay.hide()\n        );\n        this.stream.addEventListener(\n            'videoEncoderAvgQP',\n            ({ data: { avgQP } }) => this.onVideoEncoderAvgQP(avgQP)\n        );\n        this.stream.addEventListener('webRtcSdp', () =>\n            this.onWebRtcSdp()\n        );\n        this.stream.addEventListener('webRtcAutoConnect', () =>\n            this.onWebRtcAutoConnect()\n        );\n        this.stream.addEventListener('webRtcConnecting', () =>\n            this.onWebRtcConnecting()\n        );\n        this.stream.addEventListener('webRtcConnected', () =>\n            this.onWebRtcConnected()\n        );\n        this.stream.addEventListener('webRtcFailed', () =>\n            this.onWebRtcFailed()\n        );\n        this.stream.addEventListener(\n            'webRtcDisconnected',\n            ({ data: { eventString, allowClickToReconnect } }) =>\n                this.onDisconnect(eventString, allowClickToReconnect)\n        );\n        this.stream.addEventListener('videoInitialized', () =>\n            this.onVideoInitialized()\n        );\n        this.stream.addEventListener('streamLoading', () =>\n            this.onStreamLoading()\n        );\n        this.stream.addEventListener(\n            'playStreamError',\n            ({ data: { message } }) => this.onPlayStreamError(message)\n        );\n        this.stream.addEventListener('playStream', () =>\n            this.onPlayStream()\n        );\n        this.stream.addEventListener(\n            'playStreamRejected',\n            ({ data: { reason } }) => this.onPlayStreamRejected(reason)\n        );\n        this.stream.addEventListener(\n            'loadFreezeFrame',\n            ({ data: { shouldShowPlayOverlay } }) =>\n                this.onLoadFreezeFrame(shouldShowPlayOverlay)\n        );\n        this.stream.addEventListener(\n            'statsReceived',\n            ({ data: { aggregatedStats } }) =>\n                this.onStatsReceived(aggregatedStats)\n        );\n        this.stream.addEventListener(\n            'latencyTestResult',\n            ({ data: { latencyTimings } }) =>\n                this.onLatencyTestResults(latencyTimings)\n        );\n        this.stream.addEventListener(\n            'dataChannelLatencyTestResult',\n            ({data: { result } }) =>\n                this.onDataChannelLatencyTestResults(result)\n        )\n        this.stream.addEventListener(\n            'streamerListMessage',\n            ({ data: { messageStreamerList, autoSelectedStreamerId, wantedStreamerId } }) =>\n                this.handleStreamerListMessage(messageStreamerList, autoSelectedStreamerId, wantedStreamerId)\n        );\n        this.stream.addEventListener(\n            'settingsChanged',\n            (event) => this.configUI.onSettingsChanged(event)\n        );\n        this.stream.addEventListener(\n            'playerCount', \n            ({ data: { count }}) => \n                this.onPlayerCount(count)\n        );\n    }\n\n    /**\n     * Gets the rootElement of the application, video stream and all UI are children of this element.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            this._rootElement = document.createElement('div');\n            this._rootElement.id = 'playerUI';\n            this._rootElement.classList.add('noselect');\n            this._rootElement.appendChild(\n                this.stream.videoElementParent\n            );\n            this._rootElement.appendChild(this.uiFeaturesElement);\n        }\n        return this._rootElement;\n    }\n\n    /**\n     * Gets the element that contains all the UI features, like the stats and settings panels.\n     */\n    public get uiFeaturesElement(): HTMLElement {\n        if (!this._uiFeatureElement) {\n            this._uiFeatureElement = document.createElement('div');\n            this._uiFeatureElement.id = 'uiFeatures';\n        }\n        return this._uiFeatureElement;\n    }\n\n    /**\n     * Shows the disconnect overlay\n     * @param updateText - the text that will be displayed in the overlay\n     */\n    showDisconnectOverlay(updateText: string) {\n        this.hideCurrentOverlay();\n        this.updateDisconnectOverlay(updateText);\n        this.disconnectOverlay.show();\n        this.currentOverlay = this.disconnectOverlay;\n    }\n\n    /**\n     * Update the disconnect overlays span text\n     * @param updateText - the new countdown number\n     */\n    updateDisconnectOverlay(updateText: string) {\n        this.disconnectOverlay.update(updateText);\n    }\n\n    /**\n     * Activates the disconnect overlays action\n     */\n    onDisconnectionAction() {\n        this.disconnectOverlay.activate();\n    }\n\n    /**\n     * Hides the current overlay\n     */\n    hideCurrentOverlay() {\n        if (this.currentOverlay != null) {\n            this.currentOverlay.hide();\n            this.currentOverlay = null;\n        }\n    }\n\n    /**\n     * Shows the connect overlay\n     */\n    showConnectOverlay() {\n        this.hideCurrentOverlay();\n        this.connectOverlay.show();\n        this.currentOverlay = this.connectOverlay;\n    }\n\n    /**\n     * Shows the play overlay\n     */\n    showPlayOverlay() {\n        this.hideCurrentOverlay();\n        this.playOverlay.show();\n        this.currentOverlay = this.playOverlay;\n    }\n\n    /**\n     * Shows the text overlay\n     * @param text - the text that will be shown in the overlay\n     */\n    showTextOverlay(text: string) {\n        this.hideCurrentOverlay();\n        this.infoOverlay.update(text);\n        this.infoOverlay.show();\n        this.currentOverlay = this.infoOverlay;\n    }\n\n    /**\n     * Shows the error overlay\n     * @param text - the text that will be shown in the overlay\n     */\n    showErrorOverlay(text: string) {\n        this.hideCurrentOverlay();\n        this.errorOverlay.update(text);\n        this.errorOverlay.show();\n        this.currentOverlay = this.errorOverlay;\n    }\n\n    /**\n     * Shows or hides the settings panel if clicked\n     */\n    settingsClicked() {\n        this.statsPanel?.hide();\n        this.settingsPanel.toggleVisibility();\n    }\n\n    /**\n     * Shows or hides the stats panel if clicked\n     */\n    statsClicked() {\n        this.settingsPanel?.hide();\n        this.statsPanel.toggleVisibility();\n    }\n\n    /**\n     * Activates the connect overlays action\n     */\n    onConnectAction() {\n        this.connectOverlay.activate();\n    }\n\n    /**\n     * Activates the play overlays action\n     */\n    onPlayAction() {\n        this.playOverlay.activate();\n    }\n\n    /**\n     * Shows the afk overlay\n     * @param countDown - the countdown number for the afk countdown\n     */\n    showAfkOverlay(countDown: number, dismissAfk: () => void) {\n        this.hideCurrentOverlay();\n        this.afkOverlay.updateCountdown(countDown);\n        this.afkOverlay.onAction(() => dismissAfk());\n        this.afkOverlay.show();\n        this.currentOverlay = this.afkOverlay;\n    }\n\n    /**\n     * Show the Connect Overlay or auto connect\n     */\n    showConnectOrAutoConnectOverlays() {\n        // set up if the auto play will be used or regular click to start\n        if (!this.stream.config.isFlagEnabled(Flags.AutoConnect)) {\n            this.showConnectOverlay();\n        }\n    }\n\n    /**\n     * Show the webRtcAutoConnect Overlay and connect\n     */\n    onWebRtcAutoConnect() {\n        this.showTextOverlay('Auto Connecting Now');\n    }\n\n    /**\n     * Set up functionality to happen when receiving a webRTC answer\n     */\n    onWebRtcSdp() {\n        this.showTextOverlay('WebRTC Connection Negotiated');\n    }\n\n    /**\n     * Shows a text overlay to alert the user the stream is currently loading\n     */\n    onStreamLoading() {\n        // build the spinner span\n        const spinnerSpan: HTMLSpanElement = document.createElement('span');\n        spinnerSpan.className = 'visually-hidden';\n        spinnerSpan.innerHTML = 'Loading...';\n\n        // build the spinner div\n        const spinnerDiv: HTMLDivElement = document.createElement('div');\n        spinnerDiv.id = 'loading-spinner';\n        spinnerDiv.className = 'spinner-border ms-2';\n        spinnerDiv.setAttribute('role', 'status');\n\n        // append the spinner to the element\n        spinnerDiv.appendChild(spinnerSpan);\n\n        this.showTextOverlay('Loading Stream ' + spinnerDiv.outerHTML);\n    }\n\n    /**\n     * Event fired when the video is disconnected - displays the error overlay and resets the buttons stream tools upon disconnect\n     * @param eventString - the event text that will be shown in the overlay\n     * @param allowClickToReconnect - true if we want to allow the user to click to reconnect. Otherwise it's just a message.\n     */\n    onDisconnect(eventString: string, allowClickToReconnect: boolean) {\n        const overlayMessage = 'Disconnected' + (eventString ? `: ${eventString}` : '');\n        if (allowClickToReconnect) {\n            this.showDisconnectOverlay(`${overlayMessage} Click To Restart.`);\n        } else {\n            this.showErrorOverlay(overlayMessage);\n        }\n        // disable starting a latency checks\n        this.statsPanel?.onDisconnect();\n    }\n\n    /**\n     * Handles when Web Rtc is connecting\n     */\n    onWebRtcConnecting() {\n        this.showTextOverlay('Starting connection to server, please wait');\n    }\n\n    /**\n     * Handles when Web Rtc has connected\n     */\n    onWebRtcConnected() {\n        this.showTextOverlay('WebRTC connected, waiting for video');\n    }\n\n    /**\n     * Handles when Web Rtc fails to connect\n     */\n    onWebRtcFailed() {\n        this.showErrorOverlay('Unable to setup video');\n    }\n\n    onLoadFreezeFrame(shouldShowPlayOverlay: boolean) {\n        if (shouldShowPlayOverlay === true) {\n            Logger.Log(Logger.GetStackTrace(), 'showing play overlay');\n            this.showPlayOverlay();\n        }\n    }\n\n    onPlayStream() {\n        this.hideCurrentOverlay();\n    }\n\n    onPlayStreamError(message: string) {\n        this.showErrorOverlay(message);\n    }\n\n    onPlayStreamRejected(onRejectedReason: unknown) {\n        this.showPlayOverlay();\n    }\n\n    onVideoInitialized() {\n        if (!this.stream.config.isFlagEnabled(Flags.AutoPlayVideo)) {\n            this.showPlayOverlay();\n        }\n        this.statsPanel?.onVideoInitialized(this.stream);\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        // Update internal QP indicator if one is present\n        if (!!this.videoQpIndicator) {\n            this.videoQpIndicator.updateQpTooltip(QP);\n        }\n    }\n\n    onInitialSettings(settings: InitialSettings) {\n        if (settings.PixelStreamingSettings) {\n            this.statsPanel?.configure(settings.PixelStreamingSettings);\n        }\n    }\n\n    onStatsReceived(aggregatedStats: AggregatedStats) {\n        // Grab all stats we can off the aggregated stats\n        this.statsPanel?.handleStats(aggregatedStats);\n    }\n\n    onLatencyTestResults(latencyTimings: LatencyTestResults) {\n        this.statsPanel?.latencyTest.handleTestResult(latencyTimings);\n    }\n\n    onDataChannelLatencyTestResults(result: DataChannelLatencyTestResult) {\n        this.statsPanel?.dataChannelLatencyTest.handleTestResult(result);\n    }\n\n    onPlayerCount(playerCount: number) {\n        this.statsPanel?.handlePlayerCount(playerCount);\n    }\n\n    handleStreamerListMessage(messageStreamingList: MessageStreamerList, autoSelectedStreamerId: string, wantedStreamerId: string) {\n        const waitForStreamer = this.stream.config.isFlagEnabled(Flags.WaitForStreamer);\n        const isReconnecting = this.stream.isReconnecting();\n        let message: string = null;\n        let allowRestart: boolean = true;\n\n        if (!autoSelectedStreamerId) {\n            if (waitForStreamer && wantedStreamerId) {\n                if (isReconnecting) {\n                    message = `Waiting for ${wantedStreamerId} to become available.`;\n                    allowRestart = false;\n                } else {\n                    message = `Gave up waiting for ${wantedStreamerId} to become available. Click to try again`;\n                    if (messageStreamingList.ids.length > 0) {\n                        message += ` or select a streamer from the settings menu.`;\n                    }\n                    allowRestart = true;\n                }\n            } else if (messageStreamingList.ids.length == 0) {\n                if (isReconnecting) {\n                    message = `Waiting for a streamer to become available.`;\n                    allowRestart = false;\n                } else {\n                    message = `No streamers available. Click to try again.`;\n                    allowRestart = true;\n                }\n            } else {\n                message = `Multiple streamers available. Select one from the settings menu.`;\n                allowRestart = false;\n            }\n\n            if (allowRestart) {\n                this.showDisconnectOverlay(message);\n            } else {\n                this.showTextOverlay(message);\n            }\n        }\n    }\n\n    /**\n     * Set light/dark color mode\n     * @param isLightMode - should we use a light or dark color scheme\n     */\n    setColorMode(isLightMode: boolean) {\n        if (this.onColorModeChanged) {\n            this.onColorModeChanged(isLightMode);\n        }\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport {\n    Config,\n    FlagsIds,\n    NumericParametersIds,\n    OptionParametersIds,\n    TextParametersIds,\n    TextParameters,\n    OptionParameters,\n    Flags,\n    NumericParameters,\n    SettingsChangedEvent,\n    SettingFlag,\n    SettingNumber,\n    SettingText,\n    SettingOption,\n    Logger,\n    SettingBase\n} from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\nimport { SettingUIFlag } from './SettingUIFlag';\nimport { SettingUINumber } from './SettingUINumber';\nimport { SettingUIText } from './SettingUIText';\nimport { SettingUIOption } from './SettingUIOption';\n\nexport const LightMode = 'LightMode' as const;\ntype ExtraFlags = typeof LightMode;\nexport type FlagsIdsExtended = FlagsIds | ExtraFlags;\n\nexport class ConfigUI {\n    private customFlags = new Map<\n        FlagsIdsExtended,\n        SettingFlag<FlagsIdsExtended>\n    >();\n\n    /* A map of flags that can be toggled - options that can be set in the application - e.g. Use Mic? */\n    private flagsUi = new Map<\n        FlagsIdsExtended,\n        SettingUIFlag<FlagsIdsExtended>\n    >();\n\n    /* A map of numerical settings - options that can be in the application - e.g. MinBitrate */\n    private numericParametersUi = new Map<\n        NumericParametersIds,\n        SettingUINumber\n    >();\n\n    /* A map of text settings - e.g. signalling server url */\n    private textParametersUi = new Map<TextParametersIds, SettingUIText>();\n\n    /* A map of enum based settings - e.g. preferred codec */\n    private optionParametersUi = new Map<\n        OptionParametersIds,\n        SettingUIOption\n    >();\n\n    // ------------ Settings -----------------\n\n    constructor(config: Config) {\n        this.createCustomUISettings(config.useUrlParams);\n        this.registerSettingsUIComponents(config);\n    }\n\n    /**\n     * Create custom UI settings that are not provided by the Pixel Streaming library.\n     */\n    createCustomUISettings(useUrlParams: boolean) {\n        this.customFlags.set(\n            LightMode,\n            new SettingFlag<FlagsIdsExtended>(\n                LightMode,\n                'Color Scheme: Dark Mode',\n                'Page styling will be either light or dark',\n                false /*if want to use system pref: (window.matchMedia && window.matchMedia('(prefers-color-scheme: light)').matches)*/,\n                useUrlParams,\n                (isLightMode: boolean, setting: SettingBase) => {\n                    setting.label = `Color Scheme: ${isLightMode ? 'Light' : 'Dark'} Mode`;\n                }\n            )\n        );\n    }\n\n    /**\n     * Creates UI wrapper components for each setting element in config.\n     * @param config\n     */\n    registerSettingsUIComponents(config: Config) {\n        for (const setting of config.getFlags()) {\n            this.flagsUi.set(setting.id, new SettingUIFlag(setting));\n        }\n        for (const setting of Array.from(this.customFlags.values())) {\n            this.flagsUi.set(\n                setting.id,\n                new SettingUIFlag<FlagsIdsExtended>(setting)\n            );\n        }\n        for (const setting of config.getTextSettings()) {\n            this.textParametersUi.set(setting.id, new SettingUIText(setting));\n        }\n        for (const setting of config.getNumericSettings()) {\n            this.numericParametersUi.set(\n                setting.id,\n                new SettingUINumber(setting)\n            );\n        }\n        for (const setting of config.getOptionSettings()) {\n            this.optionParametersUi.set(\n                setting.id,\n                new SettingUIOption(setting)\n            );\n        }\n    }\n\n    /**\n     * Make DOM elements for a settings section with a heading.\n     * @param settingsElem The parent container for our DOM elements.\n     * @param sectionHeading The heading element to go into the section.\n     * @returns The constructed DOM element for the section.\n     */\n    buildSectionWithHeading(settingsElem: HTMLElement, sectionHeading: string) {\n        // make section element\n        const sectionElem = document.createElement('section');\n        sectionElem.classList.add('settingsContainer');\n\n        // make section heading\n        const psSettingsHeader = document.createElement('div');\n        psSettingsHeader.classList.add('settingsHeader');\n        psSettingsHeader.classList.add('settings-text');\n        psSettingsHeader.textContent = sectionHeading;\n\n        // add section and heading to parent settings element\n        sectionElem.appendChild(psSettingsHeader);\n        settingsElem.appendChild(sectionElem);\n        return sectionElem;\n    }\n\n    /**\n     * Setup flags with their default values and add them to the `Config.flags` map.\n     * @param settingsElem - The element that contains all the individual settings sections, flags, and so on.\n     */\n    populateSettingsElement(settingsElem: HTMLElement): void {\n        /* Setup all Pixel Streaming specific settings */\n        const psSettingsSection = this.buildSectionWithHeading(\n            settingsElem,\n            'Pixel Streaming'\n        );\n\n        // make settings show up in DOM\n        this.addSettingText(\n            psSettingsSection,\n            this.textParametersUi.get(TextParameters.SignallingServerUrl)\n        );\n        this.addSettingOption(\n            psSettingsSection,\n            this.optionParametersUi.get(OptionParameters.StreamerId)\n        );\n        this.addSettingFlag(\n            psSettingsSection,\n            this.flagsUi.get(Flags.AutoConnect)\n        );\n        this.addSettingFlag(\n            psSettingsSection,\n            this.flagsUi.get(Flags.AutoPlayVideo)\n        );\n        this.addSettingFlag(\n            psSettingsSection,\n            this.flagsUi.get(Flags.BrowserSendOffer)\n        );\n        this.addSettingFlag(\n            psSettingsSection, \n            this.flagsUi.get(Flags.UseMic)\n        );\n        this.addSettingFlag(\n            psSettingsSection,\n            this.flagsUi.get(Flags.StartVideoMuted)\n        );\n        this.addSettingFlag(\n            psSettingsSection,\n            this.flagsUi.get(Flags.IsQualityController)\n        );\n        this.addSettingFlag(\n            psSettingsSection,\n            this.flagsUi.get(Flags.ForceMonoAudio)\n        );\n        this.addSettingFlag(\n            psSettingsSection,\n            this.flagsUi.get(Flags.ForceTURN)\n        );\n        this.addSettingFlag(\n            psSettingsSection,\n            this.flagsUi.get(Flags.SuppressBrowserKeys)\n        );\n        this.addSettingFlag(\n            psSettingsSection,\n            this.flagsUi.get(Flags.AFKDetection)\n        );\n        this.addSettingFlag(\n            psSettingsSection,\n            this.flagsUi.get(Flags.WaitForStreamer)\n        );\n        this.addSettingNumeric(\n            psSettingsSection,\n            this.numericParametersUi.get(NumericParameters.AFKTimeoutSecs)\n        );\n        this.addSettingNumeric(\n            psSettingsSection,\n            this.numericParametersUi.get(NumericParameters.MaxReconnectAttempts)\n        );\n        this.addSettingNumeric(\n            psSettingsSection,\n            this.numericParametersUi.get(NumericParameters.StreamerAutoJoinInterval)\n        );\n\n        /* Setup all view/ui related settings under this section */\n        const viewSettingsSection = this.buildSectionWithHeading(\n            settingsElem,\n            'UI'\n        );\n        this.addSettingFlag(\n            viewSettingsSection,\n            this.flagsUi.get(Flags.MatchViewportResolution)\n        );\n\n        this.addSettingFlag(\n            viewSettingsSection,\n            this.flagsUi.get(Flags.HoveringMouseMode)\n        );\n\n        this.addSettingFlag(viewSettingsSection, this.flagsUi.get(LightMode));\n\n        /* Setup all encoder related settings under this section */\n        const inputSettingsSection = this.buildSectionWithHeading(\n            settingsElem,\n            'Input'\n        );\n        \n        this.addSettingFlag(\n            inputSettingsSection,\n            this.flagsUi.get(Flags.KeyboardInput)\n        );\n\n        this.addSettingFlag(\n            inputSettingsSection,\n            this.flagsUi.get(Flags.MouseInput)\n        );\n\n        this.addSettingFlag(\n            inputSettingsSection,\n            this.flagsUi.get(Flags.TouchInput)\n        );\n\n        this.addSettingFlag(\n            inputSettingsSection,\n            this.flagsUi.get(Flags.GamepadInput)\n        );\n\n        this.addSettingFlag(\n            inputSettingsSection,\n            this.flagsUi.get(Flags.XRControllerInput)\n        );\n\n        /* Setup all encoder related settings under this section */\n        const encoderSettingsSection = this.buildSectionWithHeading(\n            settingsElem,\n            'Encoder'\n        );\n\n        this.addSettingNumeric(\n            encoderSettingsSection,\n            this.numericParametersUi.get(NumericParameters.MinQP)\n        );\n        this.addSettingNumeric(\n            encoderSettingsSection,\n            this.numericParametersUi.get(NumericParameters.MaxQP)\n        );\n\n        const preferredCodecOption = this.optionParametersUi.get(\n            OptionParameters.PreferredCodec\n        );\n        this.addSettingOption(\n            encoderSettingsSection,\n            this.optionParametersUi.get(OptionParameters.PreferredCodec)\n        );\n        if (\n            preferredCodecOption &&\n            [...preferredCodecOption.selector.options]\n                .map((o) => o.value)\n                .includes('Only available on Chrome')\n        ) {\n            preferredCodecOption.disable();\n        }\n\n        /* Setup all webrtc related settings under this section */\n        const webrtcSettingsSection = this.buildSectionWithHeading(\n            settingsElem,\n            'WebRTC'\n        );\n\n        this.addSettingNumeric(\n            webrtcSettingsSection,\n            this.numericParametersUi.get(NumericParameters.WebRTCFPS)\n        );\n        this.addSettingNumeric(\n            webrtcSettingsSection,\n            this.numericParametersUi.get(NumericParameters.WebRTCMinBitrate)\n        );\n        this.addSettingNumeric(\n            webrtcSettingsSection,\n            this.numericParametersUi.get(NumericParameters.WebRTCMaxBitrate)\n        );\n    }\n\n    /**\n     * Add a SettingText element to a particular settings section in the DOM and registers that text in the text settings map.\n     * @param settingsSection The settings section HTML element.\n     * @param settingText The textual settings object.\n     */\n    addSettingText(\n        settingsSection: HTMLElement,\n        settingText?: SettingUIText\n    ): void {\n        if (settingText) {\n            settingsSection.appendChild(settingText.rootElement);\n            this.textParametersUi.set(settingText.setting.id, settingText);\n        }\n    }\n\n    /**\n     * Add a SettingFlag element to a particular settings section in the DOM and registers that flag in the Config.flag map.\n     * @param settingsSection The settings section HTML element.\n     * @param settingFlag The settings flag object.\n     */\n    addSettingFlag(\n        settingsSection: HTMLElement,\n        settingFlag?: SettingUIFlag<FlagsIdsExtended>\n    ): void {\n        if (settingFlag) {\n            settingsSection.appendChild(settingFlag.rootElement);\n            this.flagsUi.set(settingFlag.setting.id, settingFlag);\n        }\n    }\n\n    /**\n     * Add a numeric setting element to a particular settings section in the DOM and registers that flag in the Config.numericParameters map.\n     * @param settingsSection The settings section HTML element.\n     * @param settingFlag The settings flag object.\n     */\n    addSettingNumeric(\n        settingsSection: HTMLElement,\n        setting?: SettingUINumber\n    ): void {\n        if (setting) {\n            settingsSection.appendChild(setting.rootElement);\n            this.numericParametersUi.set(setting.setting.id, setting);\n        }\n    }\n\n    /**\n     * Add an enum based settings element to a particular settings section in the DOM and registers that flag in the Config.enumParameters map.\n     * @param settingsSection The settings section HTML element.\n     * @param settingFlag The settings flag object.\n     */\n    addSettingOption(\n        settingsSection: HTMLElement,\n        setting?: SettingUIOption\n    ): void {\n        if (setting) {\n            settingsSection.appendChild(setting.rootElement);\n            this.optionParametersUi.set(setting.setting.id, setting);\n        }\n    }\n\n    onSettingsChanged({ data: { id, target, type } }: SettingsChangedEvent) {\n        if (type === 'flag') {\n            const _id = id as FlagsIds;\n            const _target = target as SettingFlag;\n            const setting = this.flagsUi.get(_id);\n            if (setting) {\n                if (setting.flag !== _target.flag) {\n                    setting.flag = _target.flag;\n                }\n                if (setting.label !== _target.label) {\n                    setting.label = _target.label;\n                }\n            }\n        } else if (type === 'number') {\n            const _id = id as NumericParametersIds;\n            const _target = target as SettingNumber;\n            const setting = this.numericParametersUi.get(_id);\n            if (setting) {\n                if (setting.number !== _target.number) {\n                    setting.number = _target.number;\n                }\n                if (setting.label !== _target.label) {\n                    setting.label = _target.label;\n                }\n            }\n        } else if (type === 'text') {\n            const _id = id as TextParametersIds;\n            const _target = target as SettingText;\n            const setting = this.textParametersUi.get(_id);\n            if (setting) {\n                if (setting.text !== _target.text) {\n                    setting.text = _target.text;\n                }\n                if (setting.label !== _target.label) {\n                    setting.label = _target.label;\n                }\n            }\n        } else if (type === 'option') {\n            const _id = id as OptionParametersIds;\n            const _target = target as SettingOption;\n            const setting = this.optionParametersUi.get(_id);\n            if (setting) {\n                const uiOptions = setting.options;\n                const targetOptions = _target.options;\n                if (\n                    uiOptions.length !== targetOptions.length ||\n                    !uiOptions.every((value) => targetOptions.includes(value))\n                ) {\n                    setting.options = _target.options;\n                }\n                if (setting.selected !== _target.selected) {\n                    setting.selected = _target.selected;\n                }\n                if (setting.label !== _target.label) {\n                    setting.label = _target.label;\n                }\n            }\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    addCustomFlagOnSettingChangedListener(\n        id: ExtraFlags,\n        onChangeListener: (newFlagValue: boolean) => void\n    ): void {\n        if (this.customFlags.has(id)) {\n            this.customFlags.get(id).onChange = onChangeListener;\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    setCustomFlagLabel(id: ExtraFlags, label: string) {\n        if (!this.customFlags.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.customFlags.get(id).label = label;\n            this.flagsUi.get(id).label = label;\n        }\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    isCustomFlagEnabled(id: ExtraFlags): boolean {\n        return this.customFlags.get(id).flag as boolean;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { SettingBase } from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\n\n/**\n * Base class for a setting that has a text label, an arbitrary setting value it stores, an a HTML element that represents this setting.\n */\nexport class SettingUIBase {\n    _setting: SettingBase;\n    _rootElement: HTMLElement;\n\n    constructor(setting: SettingBase) {\n        this._setting = setting;\n    }\n\n    /**\n     * @returns The setting component.\n     */\n    public get setting(): SettingBase {\n        return this._setting;\n    }\n\n    /**\n     * @returns Return or creates a HTML element that represents this setting in the DOM.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            this._rootElement = document.createElement('div');\n        }\n        return this._rootElement;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport type {\n    FlagsIds,\n    SettingFlag\n} from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\nimport { SettingUIBase } from './SettingUIBase';\n\nexport class SettingUIFlag<\n    CustomIds extends string = FlagsIds\n> extends SettingUIBase {\n    /* We toggle this checkbox to reflect the value of our setting's boolean flag. */\n    _checkbox: HTMLInputElement; // input type=\"checkbox\"\n\n    /* This element contains a text node that reflects the setting's text label. */\n    _settingsTextElem: HTMLElement;\n\n    onChangeEmit: (changedValue: boolean) => void;\n\n    constructor(setting: SettingFlag<CustomIds>) {\n        super(setting);\n\n        this.label = setting.label;\n        this.flag = setting.flag;\n    }\n\n    /**\n     * @returns The setting component.\n     */\n    public get setting(): SettingFlag<CustomIds> {\n        return this._setting as SettingFlag<CustomIds>;\n    }\n\n    public get settingsTextElem(): HTMLElement {\n        if (!this._settingsTextElem) {\n            this._settingsTextElem = document.createElement('div');\n            this._settingsTextElem.innerText = this.setting._label;\n            this._settingsTextElem.title = this.setting.description;\n        }\n        return this._settingsTextElem;\n    }\n\n    public get checkbox(): HTMLInputElement {\n        if (!this._checkbox) {\n            this._checkbox = document.createElement('input');\n            this._checkbox.type = 'checkbox';\n        }\n        return this._checkbox;\n    }\n\n    /**\n     * @returns Return or creates a HTML element that represents this setting in the DOM.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            // create root div with \"setting\" css class\n            this._rootElement = document.createElement('div');\n            this._rootElement.id = this.setting.id;\n            this._rootElement.classList.add('setting');\n\n            // create div element to contain our setting's text\n            this._rootElement.appendChild(this.settingsTextElem);\n\n            // create label element to wrap out input type\n            const wrapperLabel = document.createElement('label');\n            wrapperLabel.classList.add('tgl-switch');\n            this._rootElement.appendChild(wrapperLabel);\n\n            // create input type=checkbox\n            this.checkbox.title = this.setting.description;\n            this.checkbox.classList.add('tgl');\n            this.checkbox.classList.add('tgl-flat');\n            const slider = document.createElement('div');\n            slider.classList.add('tgl-slider');\n            wrapperLabel.appendChild(this.checkbox);\n            wrapperLabel.appendChild(slider);\n\n            // setup on change from checkbox\n            this.checkbox.addEventListener('change', () => {\n                if (this.setting.flag !== this.checkbox.checked) {\n                    this.setting.flag = this.checkbox.checked;\n                    this.setting.updateURLParams();\n                }\n            });\n        }\n        return this._rootElement;\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.checkbox.checked = inValue;\n    }\n\n    /**\n     * Get value\n     */\n    public get flag() {\n        return this.checkbox.checked;\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.settingsTextElem.innerText = inLabel;\n    }\n\n    /**\n     * Get label\n     */\n    public get label() {\n        return this.settingsTextElem.innerText;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport type {\n    NumericParametersIds,\n    SettingNumber\n} from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\nimport { Logger } from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\nimport { SettingUIBase } from './SettingUIBase';\n\n/**\n * A number spinner with a text label beside it.\n */\nexport class SettingUINumber<\n    CustomIds extends string = NumericParametersIds\n> extends SettingUIBase {\n    _spinner: HTMLInputElement;\n\n    /* This element contains a text node that reflects the setting's text label. */\n    _settingsTextElem: HTMLElement;\n\n    constructor(setting: SettingNumber<CustomIds>) {\n        super(setting);\n\n        this.label = this.setting.label;\n        this.number = this.setting.number;\n    }\n\n    /**\n     * @returns The setting component.\n     */\n    public get setting(): SettingNumber<CustomIds> {\n        return this._setting as SettingNumber<CustomIds>;\n    }\n\n    public get settingsTextElem(): HTMLElement {\n        if (!this._settingsTextElem) {\n            this._settingsTextElem = document.createElement('label');\n            this._settingsTextElem.innerText = this.setting.label;\n            this._settingsTextElem.title = this.setting.description;\n        }\n        return this._settingsTextElem;\n    }\n\n    /**\n     * Get the HTMLInputElement for the button.\n     */\n    public get spinner(): HTMLInputElement {\n        if (!this._spinner) {\n            this._spinner = document.createElement('input');\n            this._spinner.type = 'number';\n            this._spinner.min = this.setting.min.toString();\n            this._spinner.max = this.setting.max.toString();\n            this._spinner.value = this.setting.number.toString();\n            this._spinner.title = this.setting.description;\n            this._spinner.classList.add('form-control');\n        }\n        return this._spinner;\n    }\n\n    /**\n     * @returns Return or creates a HTML element that represents this setting in the DOM.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            // create root div with \"setting\" css class\n            this._rootElement = document.createElement('div');\n            this._rootElement.classList.add('setting');\n            this._rootElement.classList.add('form-group');\n\n            // create div element to contain our setting's text\n            this._rootElement.appendChild(this.settingsTextElem);\n\n            // create label element to wrap out input type\n            this._rootElement.appendChild(this.spinner);\n\n            // setup onchange\n            this.spinner.onchange = (event: Event) => {\n                const inputElem = event.target as HTMLInputElement;\n\n                const parsedValue = Number.parseInt(inputElem.value);\n\n                if (Number.isNaN(parsedValue)) {\n                    Logger.Warning(\n                        Logger.GetStackTrace(),\n                        `Could not parse value change into a valid number - value was ${inputElem.value}, resetting value to ${this.setting.min}`\n                    );\n                    if (this.setting.number !== this.setting.min) {\n                        this.setting.number = this.setting.min;\n                    }\n                } else {\n                    if (this.setting.number !== parsedValue) {\n                        this.setting.number = parsedValue;\n                        this.setting.updateURLParams();\n                    }\n                }\n            };\n        }\n        return this._rootElement;\n    }\n\n    /**\n     * Set the number in the spinner (will be clamped within range).\n     */\n    public set number(newNumber: number) {\n        this.spinner.value = this.setting.clamp(newNumber).toString();\n    }\n\n    /**\n     * Get value\n     */\n    public get number() {\n        return +this.spinner.value;\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.settingsTextElem.innerText = inLabel;\n    }\n\n    /**\n     * Get label\n     */\n    public get label() {\n        return this.settingsTextElem.innerText;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport type {\n    OptionParametersIds,\n    SettingOption\n} from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\nimport { SettingUIBase } from './SettingUIBase';\n\nexport class SettingUIOption<\n    CustomIds extends string = OptionParametersIds\n> extends SettingUIBase {\n    /* A select element that reflects the value of this setting. */\n    _selector: HTMLSelectElement; // <select></select>\n\n    /* This element contains a text node that reflects the setting's text label. */\n    _settingsTextElem: HTMLElement;\n\n    constructor(setting: SettingOption<CustomIds>) {\n        super(setting);\n\n        this.label = this.setting.label;\n        this.options = this.setting.options;\n        this.selected = this.setting.selected;\n    }\n\n    /**\n     * @returns The setting component.\n     */\n    public get setting(): SettingOption<CustomIds> {\n        return this._setting as SettingOption<CustomIds>;\n    }\n\n    public get selector(): HTMLSelectElement {\n        if (!this._selector) {\n            this._selector = document.createElement('select');\n            this._selector.classList.add('form-control');\n            this._selector.classList.add('settings-option');\n        }\n        return this._selector;\n    }\n\n    public get settingsTextElem(): HTMLElement {\n        if (!this._settingsTextElem) {\n            this._settingsTextElem = document.createElement('div');\n            this._settingsTextElem.innerText = this.setting.label;\n            this._settingsTextElem.title = this.setting.description;\n        }\n        return this._settingsTextElem;\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.settingsTextElem.innerText = inLabel;\n    }\n\n    /**\n     * Get label\n     */\n    public get label() {\n        return this.settingsTextElem.innerText;\n    }\n\n    /**\n     * @returns Return or creates a HTML element that represents this setting in the DOM.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            // create root div with \"setting\" css class\n            this._rootElement = document.createElement('div');\n            this._rootElement.id = this.setting.id;\n            this._rootElement.classList.add('setting');\n            this._rootElement.classList.add('form-group');\n\n            // create div element to contain our setting's text\n            this._rootElement.appendChild(this.settingsTextElem);\n\n            // create label element to wrap out input type\n            const wrapperLabel = document.createElement('label');\n            this._rootElement.appendChild(wrapperLabel);\n\n            // create select element\n            this.selector.title = this.setting.description;\n            wrapperLabel.appendChild(this.selector);\n\n            // setup on change from selector\n            this.selector.onchange = () => {\n                if (this.setting.selected !== this.selector.value) {\n                    this.setting.selected = this.selector.value;\n                    this.setting.updateURLParams();\n                }\n            };\n        }\n        return this._rootElement;\n    }\n\n    public set options(values: Array<string>) {\n        for (let i = this.selector.options.length - 1; i >= 0; i--) {\n            this.selector.remove(i);\n        }\n\n        values.forEach((value: string) => {\n            const opt = document.createElement('option');\n            opt.value = value;\n            opt.innerHTML = value;\n            this.selector.appendChild(opt);\n        });\n    }\n\n    public get options() {\n        return [...this.selector.options].map((o) => o.value);\n    }\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        const filteredList = this.options.filter(\n            (option: string) => option.indexOf(value) !== -1\n        );\n        if (filteredList.length) {\n            this.selector.value = filteredList[0];\n        }\n    }\n\n    public get selected() {\n        return this.selector.value;\n    }\n\n    public disable() {\n        this.selector.disabled = true;\n    }\n\n    public enable() {\n        this.selector.disabled = false;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport type {\n    SettingText,\n    TextParametersIds\n} from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\nimport { SettingUIBase } from './SettingUIBase';\n\nexport class SettingUIText<\n    CustomIds extends string = TextParametersIds\n> extends SettingUIBase {\n    /* A text box that reflects the value of this setting. */\n    _textbox: HTMLInputElement; // input type=\"text\"\n\n    /* This element contains a text node that reflects the setting's text label. */\n    _settingsTextElem: HTMLElement;\n\n    constructor(setting: SettingText<CustomIds>) {\n        super(setting);\n\n        this.label = this.setting.label;\n        this.text = this.setting.text;\n    }\n\n    /**\n     * @returns The setting component.\n     */\n    public get setting(): SettingText<CustomIds> {\n        return this._setting as SettingText<CustomIds>;\n    }\n\n    public get settingsTextElem(): HTMLElement {\n        if (!this._settingsTextElem) {\n            this._settingsTextElem = document.createElement('div');\n            this._settingsTextElem.innerText = this.setting.label;\n            this._settingsTextElem.title = this.setting.description;\n        }\n        return this._settingsTextElem;\n    }\n\n    public get textbox(): HTMLInputElement {\n        if (!this._textbox) {\n            this._textbox = document.createElement('input');\n            this._textbox.classList.add('form-control');\n            this._textbox.type = 'textbox';\n        }\n        return this._textbox;\n    }\n\n    /**\n     * @returns Return or creates a HTML element that represents this setting in the DOM.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            // create root div with \"setting\" css class\n            this._rootElement = document.createElement('div');\n            this._rootElement.id = this.setting.id;\n            this._rootElement.classList.add('setting');\n\n            // create div element to contain our setting's text\n            this._rootElement.appendChild(this.settingsTextElem);\n\n            // create label element to wrap out input type\n            const wrapperLabel = document.createElement('label');\n            this._rootElement.appendChild(wrapperLabel);\n\n            // create input type=checkbox\n            this.textbox.title = this.setting.description;\n            wrapperLabel.appendChild(this.textbox);\n\n            // setup on change from checkbox\n            this.textbox.addEventListener('input', () => {\n                if (this.setting.text !== this.textbox.value) {\n                    this.setting.text = this.textbox.value;\n                    this.setting.updateURLParams();\n                }\n            });\n        }\n        return this._rootElement;\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.textbox.value = inValue;\n    }\n\n    /**\n     * Get value\n     */\n    public get text() {\n        return this.textbox.value;\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.settingsTextElem.innerText = inLabel;\n    }\n\n    /**\n     * Get label\n     */\n    public get label() {\n        return this.settingsTextElem.innerText;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { ActionOverlay } from './ActionOverlay';\n\n/**\n * Show an overlay for when the session is unattended, it begins a countdown timer, which when elapsed will disconnect the stream.\n */\nexport class AFKOverlay extends ActionOverlay {\n    /**\n     * @returns The created root element of this overlay.\n     */\n    public static createRootElement(): HTMLElement {\n        const afkOverlayHtml = document.createElement('div');\n        afkOverlayHtml.id = 'afkOverlay';\n        afkOverlayHtml.className = 'clickableState';\n        return afkOverlayHtml;\n    }\n\n    /**\n     * @returns The created content element of this overlay, which contain some text for an afk count down.\n     */\n    public static createContentElement(): HTMLElement {\n        const afkOverlayHtmlInner = document.createElement('div');\n        afkOverlayHtmlInner.id = 'afkOverlayInner';\n        afkOverlayHtmlInner.innerHTML =\n            '<center>No activity detected<br>Disconnecting in <span id=\"afkCountDownNumber\"></span> seconds<br>Click to continue<br></center>';\n        return afkOverlayHtmlInner;\n    }\n\n    /**\n     * Construct an Afk overlay\n     * @param parentElement the element this overlay will be inserted into\n     */\n    public constructor(rootDiv: HTMLElement) {\n        super(\n            rootDiv,\n            AFKOverlay.createRootElement(),\n            AFKOverlay.createContentElement()\n        );\n\n        this.rootElement.addEventListener('click', () => {\n            this.activate();\n        });\n    }\n\n    /**\n     * Update the count down spans number for the overlay\n     * @param countdown the count down number to be inserted into the span for updating\n     */\n    public updateCountdown(countdown: number): void {\n        this.textElement.innerHTML = `<center>No activity detected<br>Disconnecting in <span id=\"afkCountDownNumber\">${countdown}</span> seconds<br>Click to continue<br></center>`;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\n\nimport { OverlayBase } from './BaseOverlay';\n\n/**\n * Class for the base action overlay structure\n */\nexport class ActionOverlay extends OverlayBase {\n    onActionCallback: (...args: []) => void;\n\n    /**\n     * Construct an action overlay\n     * @param rootDiv the root element this overlay will be inserted into\n     * @param rootElement the root element that is the overlay\n     * @param contentElement an element that contains text for the action overlay\n     */\n    public constructor(\n        rootDiv: HTMLElement,\n        rootElement: HTMLElement,\n        contentElement: HTMLElement\n    ) {\n        super(rootDiv, rootElement, contentElement);\n        this.onActionCallback = () => {\n            /* do nothing */ Logger.Info(\n                Logger.GetStackTrace(),\n                'Did you forget to set the onAction callback in your overlay?'\n            );\n        };\n    }\n\n    /**\n     * Update the text overlays inner text\n     * @param text the update text to be inserted into the overlay\n     */\n    public update(text: string): void {\n        if (text != null || text != undefined) {\n            this.textElement.innerHTML = text;\n        }\n    }\n\n    /**\n     * Set a method as an event emitter callback\n     * @param callBack the method that is to be called when the event is emitted\n     */\n    onAction(callBack: (...args: []) => void) {\n        this.onActionCallback = callBack;\n    }\n\n    /**\n     * Activate an event that is attached to the event emitter\n     */\n    activate() {\n        this.onActionCallback();\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Class for the base overlay structure\n */\nexport class OverlayBase {\n    protected rootElement: HTMLElement;\n    protected rootDiv: HTMLElement;\n    public textElement: HTMLElement;\n\n    /**\n     * Construct an overlay\n     * @param rootDiv the root element this overlay will be inserted into\n     * @param rootElement the root element that is the overlay\n     */\n    protected constructor(\n        rootDiv: HTMLElement,\n        rootElement: HTMLElement,\n        textElement: HTMLElement\n    ) {\n        this.rootDiv = rootDiv;\n        this.rootElement = rootElement;\n        this.textElement = textElement;\n        this.rootElement.appendChild(this.textElement);\n        this.hide();\n        this.rootDiv.appendChild(this.rootElement);\n    }\n\n    /**\n     * Show the overlay\n     */\n    public show(): void {\n        this.rootElement.classList.remove('hiddenState');\n    }\n\n    /**\n     * Hide the overlay\n     */\n    public hide(): void {\n        this.rootElement.classList.add('hiddenState');\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { ActionOverlay } from './ActionOverlay';\n\n/**\n * Overlay shown during connection, has a button that can be clicked to initiate a connection.\n */\nexport class ConnectOverlay extends ActionOverlay {\n    /**\n     * @returns The created root element of this overlay.\n     */\n    public static createRootElement(): HTMLElement {\n        const connectElem = document.createElement('div');\n        connectElem.id = 'connectOverlay';\n        connectElem.className = 'clickableState';\n        return connectElem;\n    }\n\n    /**\n     * @returns The created content element of this overlay, which contain whatever content this element contains, like text or a button.\n     */\n    public static createContentElement(): HTMLElement {\n        const connectContentElem = document.createElement('div');\n        connectContentElem.id = 'connectButton';\n        connectContentElem.innerHTML = 'Click to start';\n        return connectContentElem;\n    }\n\n    /**\n     * Construct a connect overlay with a connection button.\n     * @param parentElem the parent element this overlay will be inserted into.\n     */\n    public constructor(parentElem: HTMLElement) {\n        super(\n            parentElem,\n            ConnectOverlay.createRootElement(),\n            ConnectOverlay.createContentElement()\n        );\n\n        // add the new event listener\n        this.rootElement.addEventListener('click', () => {\n            this.activate();\n        });\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { ActionOverlay } from './ActionOverlay';\n\n/**\n * Overlay shown during disconnection, has a reconnection element that can be clicked to reconnect.\n */\nexport class DisconnectOverlay extends ActionOverlay {\n    /**\n     * @returns The created root element of this overlay.\n     */\n    public static createRootElement(): HTMLElement {\n        const disconnectOverlayHtml = document.createElement('div');\n        disconnectOverlayHtml.id = 'disconnectOverlay';\n        disconnectOverlayHtml.className = 'clickableState';\n        return disconnectOverlayHtml;\n    }\n\n    /**\n     * @returns The created content element of this overlay, which contain whatever content this element contains, like text or a button.\n     */\n    public static createContentElement(): HTMLElement {\n        // build the inner html container\n        const disconnectOverlayHtmlContainer = document.createElement('div');\n        disconnectOverlayHtmlContainer.id = 'disconnectButton';\n        disconnectOverlayHtmlContainer.innerHTML = 'Click To Restart';\n\n        return disconnectOverlayHtmlContainer;\n    }\n\n    /**\n     * Construct a disconnect overlay with a retry connection icon.\n     * @param parentElem the parent element this overlay will be inserted into.\n     */\n    public constructor(parentElem: HTMLElement) {\n        super(\n            parentElem,\n            DisconnectOverlay.createRootElement(),\n            DisconnectOverlay.createContentElement()\n        );\n\n        // add the new event listener\n        this.rootElement.addEventListener('click', () => {\n            this.activate();\n        });\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { TextOverlay } from './TextOverlay';\n\n/**\n * Generic overlay used to show textual error info to the user.\n */\nexport class ErrorOverlay extends TextOverlay {\n    /**\n     * @returns The created root element of this overlay.\n     */\n    public static createRootElement(): HTMLElement {\n        const errorOverlayHtml = document.createElement('div');\n        errorOverlayHtml.id = 'errorOverlay';\n        errorOverlayHtml.className = 'textDisplayState';\n        return errorOverlayHtml;\n    }\n\n    /**\n     * @returns The created content element of this overlay, which contain whatever content this element contains, like text or a button.\n     */\n    public static createContentElement(): HTMLElement {\n        const errorOverlayHtmlInner = document.createElement('div');\n        errorOverlayHtmlInner.id = 'errorOverlayInner';\n        return errorOverlayHtmlInner;\n    }\n\n    /**\n     * Construct a connect overlay with a connection button.\n     * @param parentElem the parent element this overlay will be inserted into.\n     */\n    public constructor(parentElem: HTMLElement) {\n        super(\n            parentElem,\n            ErrorOverlay.createRootElement(),\n            ErrorOverlay.createContentElement()\n        );\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { TextOverlay } from './TextOverlay';\n\n/**\n * Generic overlay used to show textual info to the user.\n */\nexport class InfoOverlay extends TextOverlay {\n    /**\n     * @returns The created root element of this overlay.\n     */\n    public static createRootElement(): HTMLElement {\n        const infoOverlayHtml = document.createElement('div');\n        infoOverlayHtml.id = 'infoOverlay';\n        infoOverlayHtml.className = 'textDisplayState';\n        return infoOverlayHtml;\n    }\n\n    /**\n     * @returns The created content element of this overlay, which contain whatever content this element contains, like text or a button.\n     */\n    public static createContentElement(): HTMLElement {\n        const infoOverlayHtmlInner = document.createElement('div');\n        infoOverlayHtmlInner.id = 'messageOverlayInner';\n        return infoOverlayHtmlInner;\n    }\n\n    /**\n     * Construct a connect overlay with a connection button.\n     * @param parentElem the parent element this overlay will be inserted into.\n     */\n    public constructor(parentElem: HTMLElement) {\n        super(\n            parentElem,\n            InfoOverlay.createRootElement(),\n            InfoOverlay.createContentElement()\n        );\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { ActionOverlay } from './ActionOverlay';\n\n/**\n * Overlay shown when stream is ready to play.\n */\nexport class PlayOverlay extends ActionOverlay {\n    /**\n     * @returns The created root element of this overlay.\n     */\n    public static createRootElement(): HTMLElement {\n        const playElem = document.createElement('div');\n        playElem.id = 'playOverlay';\n        playElem.className = 'clickableState';\n        return playElem;\n    }\n\n    /**\n     * @returns The created content element of this overlay, which contain whatever content this element contains, like text or a button.\n     */\n    public static createContentElement(): HTMLElement {\n        // todo: change this to an svg\n        const playOverlayHtmlInner = document.createElement('img');\n        playOverlayHtmlInner.id = 'playButton';\n        playOverlayHtmlInner.src =\n            'data:image/png;base64,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';\n        playOverlayHtmlInner.alt = 'Start Streaming';\n        return playOverlayHtmlInner;\n    }\n\n    /**\n     * Construct a connect overlay with a connection button.\n     * @param parentElem the parent element this overlay will be inserted into.\n     */\n    public constructor(parentElem: HTMLElement) {\n        super(\n            parentElem,\n            PlayOverlay.createRootElement(),\n            PlayOverlay.createContentElement()\n        );\n\n        // add the new event listener\n        this.rootElement.addEventListener('click', () => {\n            this.activate();\n        });\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { OverlayBase } from './BaseOverlay';\n\n/**\n * Class for the text overlay base\n */\nexport class TextOverlay extends OverlayBase {\n    /**\n     * Construct a text overlay\n     * @param rootDiv the root element this overlay will be inserted into\n     * @param rootElement the root element that is the overlay\n     * @param textElement an element that contains text for the action overlay\n     */\n    public constructor(\n        rootDiv: HTMLElement,\n        rootElement: HTMLElement,\n        textElement: HTMLElement\n    ) {\n        super(rootDiv, rootElement, textElement);\n    }\n\n    /**\n     * Update the text overlays inner text\n     * @param text the update text to be inserted into the overlay\n     */\n    public update(text: string): void {\n        if (text != null || text != undefined) {\n            this.textElement.innerHTML = text;\n        }\n    }\n}\n","/* Copyright Epic Games, Inc. All Rights Reserved. */\n\nimport jss, { Styles } from 'jss';\nimport global from 'jss-plugin-global';\nimport camelCase from 'jss-plugin-camel-case';\n\nexport interface ColorPalette {\n    '--color0': string;\n    '--color1': string;\n    '--color2': string;\n    '--color3': string;\n    '--color4': string;\n    '--color5': string;\n    '--color6': string;\n    '--color7': string;\n}\n\nexport class PixelStreamingApplicationStyle {\n    defaultLightModePalette: ColorPalette = {\n        '--color0': '#e2e0dd80',\n        '--color1': '#FFFFFF',\n        '--color2': '#000000',\n        '--color3': '#0585fe',\n        '--color4': '#35b350',\n        '--color5': '#ffab00',\n        '--color6': '#e1e2dd',\n        '--color7': '#c3c4bf'\n    };\n\n    defaultDarkModePalette: ColorPalette = {\n        '--color0': '#1D1F2280',\n        '--color1': '#000000',\n        '--color2': '#FFFFFF',\n        '--color3': '#0585fe',\n        '--color4': '#35b350',\n        '--color5': '#ffab00',\n        '--color6': '#1e1d22',\n        '--color7': '#3c3b40'\n    };\n\n    defaultStyles = {\n        ':root': {\n            '--color0': '#1D1F2280',\n            '--color1': '#000000',\n            '--color2': '#FFFFFF',\n            '--color3': '#0585fe',\n            '--color4': '#35b350',\n            '--color5': '#ffab00',\n            '--color6': '#1e1d22',\n            '--color7': '#3c3b40',\n            '--color8': '#41008c',\n            '--color9': '#3e0070',\n            '--color10': '#2e0052',\n            '--color11': 'rgba(65,0,139,1)'\n        },\n        '.noselect': {\n            userSelect: 'none'\n        },\n        '#playerUI': {\n            width: '100%',\n            height: '100%',\n            position: 'relative'\n        },\n        '#videoElementParent': {\n            width: '100%',\n            height: '100%',\n            position: 'absolute',\n            backgroundColor: 'var(--color1)'\n        },\n        '#uiFeatures': {\n            width: '100%',\n            height: '100%',\n            zIndex: '30',\n            position: 'relative',\n            color: 'var(--color2)',\n            pointerEvents: 'none',\n            overflow: 'hidden'\n        },\n        '.UiTool .tooltiptext': {\n            visibility: 'hidden',\n            width: 'auto',\n            color: 'var(--color2)',\n            textAlign: 'center',\n            borderRadius: '15px',\n            padding: '0px 10px',\n            fontFamily: \"'Montserrat', sans-serif\",\n            fontSize: '0.75rem',\n            letterSpacing: '0.75px',\n            position: 'absolute',\n            top: '0',\n            transform: 'translateY(25%)',\n            left: '125%',\n            zIndex: '20'\n        },\n        '.UiTool:hover .tooltiptext': {\n            visibility: 'visible',\n            backgroundColor: 'var(--color7)'\n        },\n        '#connection .tooltiptext': {\n            top: '125%',\n            transform: 'translateX(-25%)',\n            left: '0',\n            zIndex: '20',\n            padding: '5px 10px'\n        },\n        '#connection': {\n            position: 'absolute',\n            bottom: '8%',\n            left: '5%',\n            fontFamily: \"'Michroma', sans-serif\",\n            height: '3rem',\n            width: '3rem',\n            pointerEvents: 'all'\n        },\n        '#settings-panel .tooltiptext': {\n            display: 'block',\n            top: '125%',\n            transform: 'translateX(-50%)',\n            left: '0',\n            zIndex: '20',\n            padding: '5px 10px',\n            border: '3px solid var(--color3)',\n            width: 'max-content',\n            fallbacks: [\n                {\n                    width: 'max-content'\n                },\n                {\n                    border: '3px solid var(--color3)'\n                },\n                {\n                    padding: '5px 10px'\n                },\n                {\n                    zIndex: '20'\n                },\n                {\n                    left: '0'\n                },\n                {\n                    transform: 'translateX(-50%)'\n                },\n                {\n                    top: '125%'\n                },\n                {\n                    display: 'block'\n                }\n            ]\n        },\n        '#controls': {\n            position: 'absolute',\n            top: '3%',\n            left: '2%',\n            fontFamily: \"'Michroma', sans-serif\",\n            pointerEvents: 'all',\n            display: 'block'\n        },\n        '#controls>*': {\n            marginBottom: '0.5rem',\n            borderRadius: '50%',\n            display: 'block',\n            height: '2rem',\n            lineHeight: '1.75rem',\n            padding: '0.5rem'\n        },\n        '#controls #additionalinfo': {\n            textAlign: 'center',\n            fontFamily: \"'Montserrat', sans-serif\"\n        },\n        '#fullscreen-btn': {\n            padding: '0.6rem !important'\n        },\n        '#minimizeIcon': {\n            display: 'none'\n        },\n        '#settingsBtn, #statsBtn': {\n            cursor: 'pointer'\n        },\n        '#uiFeatures button': {\n            backgroundColor: 'var(--color7)',\n            border: '1px solid var(--color7)',\n            color: 'var(--color2)',\n            position: 'relative',\n            width: '3rem',\n            height: '3rem',\n            padding: '0.5rem',\n            textAlign: 'center'\n        },\n        '#uiFeatures button:hover': {\n            backgroundColor: 'var(--color3)',\n            border: '3px solid var(--color3)',\n            transition: '0.25s ease',\n            paddingLeft: '0.55rem',\n            paddingTop: '0.55rem'\n        },\n        '#uiFeatures button:active': {\n            border: '3px solid var(--color3)',\n            backgroundColor: 'var(--color7)',\n            paddingLeft: '0.55rem',\n            paddingTop: '0.55rem'\n        },\n        '.btn-flat': {\n            backgroundColor: 'transparent',\n            color: 'var(--color2)',\n            fontFamily: \"'Montserrat'\",\n            fontWeight: 'bold',\n            border: '3px solid var(--color3)',\n            borderRadius: '1rem',\n            fontSize: '0.75rem',\n            paddingLeft: '0.5rem',\n            paddingRight: '0.5rem',\n            cursor: 'pointer',\n            textAlign: 'center'\n        },\n        '.btn-flat:hover': {\n            backgroundColor: 'var(--color3)',\n            transition: 'ease 0.3s'\n        },\n        '.btn-flat:disabled': {\n            background: 'var(--color7)',\n            borderColor: 'var(--color3)',\n            color: 'var(--color3)',\n            cursor: 'default'\n        },\n        '.btn-flat:active': {\n            backgroundColor: 'transparent'\n        },\n        '.btn-flat:focus': {\n            outline: 'none'\n        },\n        '#uiFeatures img': {\n            width: '100%',\n            height: '100%'\n        },\n        '.panel-wrap': {\n            position: 'absolute',\n            top: '0',\n            bottom: '0',\n            right: '0',\n            height: '100%',\n            minWidth: '20vw',\n            maxWidth: '90vw',\n            transform: 'translateX(100%)',\n            transition: '.3s ease-out',\n            pointerEvents: 'all',\n            backdropFilter: 'blur(10px)',\n            '-webkit-backdrop-filter': 'blur(10px)',\n            overflowY: 'auto',\n            overflowX: 'hidden',\n            backgroundColor: 'var(--color0)'\n        },\n        '.panel-wrap-visible': {\n            transform: 'translateX(0%)'\n        },\n        '.panel': {\n            overflowY: 'auto',\n            padding: '1em'\n        },\n        '#settingsHeading, #statsHeading': {\n            display: 'inline-block',\n            fontSize: '2em',\n            marginBlockStart: '0.67em',\n            marginBlockEnd: '0.67em',\n            marginInlineStart: '0px',\n            marginInlineEnd: '0px',\n            position: 'relative',\n            padding: '0 0 0 2rem'\n        },\n        '#settingsClose, #statsClose': {\n            margin: '0.5rem',\n            paddingTop: '0.5rem',\n            paddingBottom: '0.5rem',\n            paddingRight: '0.5rem',\n            fontSize: '2em',\n            float: 'right'\n        },\n        '#settingsClose:after, #statsClose:after': {\n            paddingLeft: '0.5rem',\n            display: 'inline-block',\n            content: '\"\\\\00d7\"'\n        },\n        '#settingsClose:hover, #statsClose:hover': {\n            color: 'var(--color3)',\n            transition: 'ease 0.3s'\n        },\n        '#settingsContent, #statsContent': {\n            marginLeft: '2rem',\n            marginRight: '2rem'\n        },\n        '.setting': {\n            display: 'flex',\n            flexDirection: 'row',\n            justifyContent: 'space-between',\n            padding: '0.15rem 10px 0.15rem 10px'\n        },\n        '.settings-text': {\n            color: 'var(--color2)',\n            verticalAlign: 'middle',\n            fontWeight: 'normal'\n        },\n        '.settings-option': {\n            width: '100%',\n            textOverflow: 'ellipsis',\n            whiteSpace: 'nowrap'\n        },\n        '#connectOverlay, #playOverlay, #infoOverlay, #errorOverlay, #afkOverlay, #disconnectOverlay':\n            {\n                zIndex: '30',\n                position: 'absolute',\n                color: 'var(--color2)',\n                fontSize: '1.8em',\n                width: '100%',\n                height: '100%',\n                backgroundColor: 'var(--color1)',\n                alignItems: 'center',\n                justifyContent: 'center',\n                textTransform: 'uppercase'\n            },\n        '.clickableState': {\n            alignItems: 'center',\n            justifyContent: 'center',\n            display: 'flex',\n            cursor: 'pointer'\n        },\n        '.textDisplayState': {\n            display: 'flex'\n        },\n        '.hiddenState': {\n            display: 'none'\n        },\n        '#playButton, #connectButton': {\n            display: 'inline-block',\n            height: 'auto',\n            zIndex: '30'\n        },\n        'img#playButton': {\n            maxWidth: '241px',\n            width: '10%'\n        },\n        '#uiInteraction': {\n            position: 'fixed'\n        },\n        '#UIInteractionButtonBoundary': {\n            padding: '2px'\n        },\n        '#UIInteractionButton': {\n            cursor: 'pointer'\n        },\n        '#hiddenInput': {\n            position: 'absolute',\n            left: '-10%',\n            width: '0px',\n            opacity: '0'\n        },\n        '#editTextButton': {\n            position: 'absolute',\n            height: '40px',\n            width: '40px'\n        },\n        '.btn-overlay': {\n            verticalAlign: 'middle',\n            display: 'inline-block'\n        },\n        '.tgl-switch': {\n            verticalAlign: 'middle',\n            display: 'inline-block'\n        },\n        '.tgl-switch .tgl': {\n            display: 'none'\n        },\n        '.tgl, .tgl:after, .tgl:before, .tgl *, .tgl *:after, .tgl *:before, .tgl+.tgl-slider':\n            {\n                '-webkit-box-sizing': 'border-box',\n                boxSizing: 'border-box'\n            },\n        '.tgl::-moz-selection, .tgl:after::-moz-selection, .tgl:before::-moz-selection, .tgl *::-moz-selection, .tgl *:after::-moz-selection, .tgl *:before::-moz-selection, .tgl+.tgl-slider::-moz-selection':\n            {\n                background: 'none'\n            },\n        '.tgl::selection, .tgl:after::selection, .tgl:before::selection, .tgl *::selection, .tgl *:after::selection, .tgl *:before::selection, .tgl+.tgl-slider::selection':\n            {\n                background: 'none'\n            },\n        '.tgl-slider': {},\n        '.tgl+.tgl-slider': {\n            outline: '0',\n            display: 'block',\n            width: '40px',\n            height: '18px',\n            position: 'relative',\n            cursor: 'pointer',\n            userSelect: 'none'\n        },\n        '.tgl+.tgl-slider:after, .tgl+.tgl-slider:before': {\n            position: 'relative',\n            display: 'block',\n            content: '\"\"',\n            width: '50%',\n            height: '100%'\n        },\n        '.tgl+.tgl-slider:after': {\n            left: '0'\n        },\n        '.tgl+.tgl-slider:before': {\n            display: 'none'\n        },\n        '.tgl-flat+.tgl-slider': {\n            padding: '2px',\n            '-webkit-transition': 'all .2s ease',\n            transition: 'all .2s ease',\n            background: 'var(--color6)',\n            border: '3px solid var(--color7)',\n            borderRadius: '2em'\n        },\n        '.tgl-flat+.tgl-slider:after': {\n            '-webkit-transition': 'all .2s ease',\n            transition: 'all .2s ease',\n            background: 'var(--color7)',\n            content: '\"\"',\n            borderRadius: '1em'\n        },\n        '.tgl-flat:checked+.tgl-slider': {\n            border: '3px solid var(--color3)'\n        },\n        '.tgl-flat:checked+.tgl-slider:after': {\n            left: '50%',\n            background: 'var(--color3)'\n        },\n        '.btn-apply': {\n            display: 'block !important',\n            marginLeft: 'auto',\n            marginRight: 'auto',\n            width: '40%'\n        },\n        '.form-control': {\n            backgroundColor: 'var(--color7)',\n            border: '2px solid var(--color7)',\n            borderRadius: '4px',\n            color: 'var(--color2)',\n            textAlign: 'right',\n            fontFamily: 'inherit'\n        },\n        '.form-control:hover': {\n            borderColor: 'var(--color7)'\n        },\n        '.form-group': {\n            paddingTop: '4px',\n            display: 'grid',\n            gridTemplateColumns: '80% 20%',\n            rowGap: '4px',\n            paddingRight: '10px',\n            paddingLeft: '10px'\n        },\n        '.form-group label': {\n            verticalAlign: 'middle',\n            fontWeight: 'normal'\n        },\n        '.settingsContainer': {\n            display: 'flex',\n            flexDirection: 'column',\n            borderBottom: '1px solid var(--color7)',\n            paddingTop: '10px',\n            paddingBottom: '10px'\n        },\n        '.settingsContainer> :first-child': {\n            marginTop: '4px',\n            marginBottom: '4px',\n            fontWeight: 'bold',\n            justifyContent: 'space-between',\n            display: 'flex',\n            flexDirection: 'row',\n            alignItems: 'baseline'\n        },\n        '.collapse': {\n            paddingLeft: '5%'\n        },\n        '#streamTools': {\n            borderBottomRightRadius: '5px',\n            borderBottomLeftRadius: '5px',\n            userSelect: 'none',\n            position: 'absolute',\n            top: '0',\n            right: '2%',\n            zIndex: '100',\n            border: '4px solid var(--colour8)',\n            borderTopWidth: '0px'\n        },\n        '.settingsHeader': {\n            fontStyle: 'italic'\n        },\n        '#streamToolsHeader': {\n            display: 'flex',\n            flexDirection: 'row',\n            justifyContent: 'space-between',\n            borderBottom: '1px solid var(--colour8)',\n            backgroundColor: 'var(--color7)'\n        },\n        '.streamTools': {\n            backgroundColor: 'var(--color2)',\n            fontFamily: 'var(--buttonFont)',\n            fontWeight: 'lighter',\n            color: 'var(--color7)'\n        },\n        '.streamTools-shown>#streamToolsSettings, .streamTools-shown>#streamToolsStats':\n            {\n                display: 'block'\n            },\n        '#streamToolsToggle': {\n            width: '100%'\n        },\n        '#qualityStatus': {\n            fontSize: '37px',\n            paddingRight: '4px'\n        },\n        '.svgIcon': {\n            fill: 'var(--color2)'\n        }\n    };\n\n    customStyles?: Partial<Styles>;\n    lightModePalette: ColorPalette;\n    darkModePalette: ColorPalette;\n\n    constructor(options?: {\n        customStyles?: Partial<Styles>;\n        lightModePalette?: ColorPalette;\n        darkModePalette?: ColorPalette;\n        jssInsertionPoint?: string | HTMLElement;\n    }) {\n        const { customStyles, lightModePalette, darkModePalette, jssInsertionPoint } =\n            options ?? {};\n        // One time setup with default plugins and settings.\n        const jssOptions = {\n            // JSS has many interesting plugins we may wish to turn on\n            //plugins: [functions(), template(), global(), extend(), nested(), compose(), camelCase(), defaultUnit(options.defaultUnit), expand(), vendorPrefixer(), propsSort()]\n            plugins: [global(), camelCase()],\n            insertionPoint: jssInsertionPoint\n        };\n\n        jss.setup(jssOptions);\n\n        this.customStyles = customStyles;\n        this.lightModePalette =\n            lightModePalette ?? this.defaultLightModePalette;\n        this.darkModePalette = darkModePalette ?? this.defaultDarkModePalette;\n    }\n\n    applyStyleSheet() {\n        // Todo: refactor codebase to use jss at a component level, classes can be grabbed like so:\n        //const {pixelStreamingClasses} = jss.createStyleSheet(styles).attach();\n\n        // attach generated style sheet to page\n        jss.createStyleSheet({\n            '@global': {\n                ...this.defaultStyles,\n                ...this.customStyles\n            }\n        }).attach();\n    }\n\n    applyPalette(palette: ColorPalette) {\n        const rootElement = document.querySelector(':root') as HTMLElement;\n        rootElement.style.setProperty('--color0', palette['--color0']);\n        rootElement.style.setProperty('--color1', palette['--color1']);\n        rootElement.style.setProperty('--color2', palette['--color2']);\n        rootElement.style.setProperty('--color3', palette['--color3']);\n        rootElement.style.setProperty('--color4', palette['--color4']);\n        rootElement.style.setProperty('--color5', palette['--color5']);\n        rootElement.style.setProperty('--color6', palette['--color6']);\n        rootElement.style.setProperty('--color7', palette['--color7']);\n    }\n\n    /**\n     * Update the players color variables\n     * @param isLightMode - should we use a light or dark color scheme\n     */\n    setColorMode(isLightMode: boolean) {\n        if (isLightMode) {\n            this.applyPalette(this.lightModePalette);\n        } else {\n            this.applyPalette(this.darkModePalette);\n        }\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { FullScreenIcon } from './FullscreenIcon';\nimport { SettingsIcon } from './SettingsIcon';\nimport { StatsIcon } from './StatsIcon';\nimport { XRIcon } from './XRIcon';\nimport { WebXRController } from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\nimport { UIElementConfig, UIElementCreationMode } from '../UI/UIConfigurationTypes'\n\n/**\n * Configures how UI elements to control the stream are created. \n * By default, a button will be created for each control. That can be overriden per-control\n * to use an externally provided element, or to disable the element entirely.\n */\nexport type ControlsUIConfiguration = {\n    //[Property in keyof Controls as `${Property}Type`]? : UIElementType;\n    statsButtonType? : UIElementConfig,\n    fullscreenButtonType? : UIElementConfig,\n    settingsButtonType? : UIElementConfig,\n    xrIconType? : UIElementConfig\n}\n\n// If there isn't a type provided, default behaviour is to create the element.\nfunction shouldCreateButton(type : UIElementConfig | undefined) : boolean {\n    return (type == undefined) ? true : (type.creationMode === UIElementCreationMode.CreateDefaultElement);\n}\n\n/**\n * Element containing various controls like stats, settings, fullscreen.\n */\nexport class Controls {\n    statsIcon: StatsIcon;\n    fullscreenIcon: FullScreenIcon;\n    settingsIcon: SettingsIcon;\n    xrIcon: XRIcon;\n\n    _rootElement: HTMLElement;\n\n    /**\n     * Construct the controls\n     */\n    constructor(config? : ControlsUIConfiguration) {\n        if (!config || shouldCreateButton(config.statsButtonType)) {\n            this.statsIcon = new StatsIcon();\n        }\n        if (!config || shouldCreateButton(config.settingsButtonType)){\n            this.settingsIcon = new SettingsIcon();\n        }\n        if (!config || shouldCreateButton(config.fullscreenButtonType)) {\n            this.fullscreenIcon = new FullScreenIcon();\n        }\n        if (!config || shouldCreateButton(config.xrIconType)){\n            this.xrIcon = new XRIcon();\n        }\n    }\n\n    /**\n     * Get the element containing the controls.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            this._rootElement = document.createElement('div');\n            this._rootElement.id = 'controls';\n            if (!!this.fullscreenIcon) {\n                this._rootElement.appendChild(this.fullscreenIcon.rootElement);\n            }\n            if (!!this.settingsIcon) {\n                this._rootElement.appendChild(this.settingsIcon.rootElement);\n            }\n            if (!!this.statsIcon) {\n                this._rootElement.appendChild(this.statsIcon.rootElement);\n            }\n            if (!!this.xrIcon) {\n                WebXRController.isSessionSupported('immersive-vr').then(\n                (supported: boolean) => {\n                    if (supported) {\n                        this._rootElement.appendChild(this.xrIcon.rootElement);\n                    }\n                });\n            };\n        }\n        return this._rootElement;\n    }\n}","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { Logger } from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\nimport {\n    DataChannelLatencyTestResult\n} from \"@epicgames-ps/lib-pixelstreamingfrontend-ue5.3/types/DataChannel/DataChannelLatencyTestResults\";\n\n/**\n * DataChannel Latency test UI elements and results handling.\n */\nexport class DataChannelLatencyTest {\n    _rootElement: HTMLElement;\n    _latencyTestButton: HTMLInputElement;\n    _latencyTestResultsElement: HTMLElement;\n\n    /**\n     * Get the button containing the stats icon.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            this._rootElement = document.createElement('section');\n            this._rootElement.classList.add('settingsContainer');\n\n            // make heading\n            const heading = document.createElement('div');\n            heading.id = 'dataChannelLatencyTestHeader';\n            heading.classList.add('settings-text');\n            heading.classList.add('settingsHeader');\n            this._rootElement.appendChild(heading);\n\n            const headingText = document.createElement('div');\n            headingText.innerHTML = 'Data Channel Latency Test';\n            heading.appendChild(headingText);\n            heading.appendChild(this.latencyTestButton);\n\n            // make test results element\n            const resultsParentElem = document.createElement('div');\n            resultsParentElem.id = 'dataChannelLatencyTestContainer';\n            resultsParentElem.classList.add('d-none');\n            this._rootElement.appendChild(resultsParentElem);\n\n            resultsParentElem.appendChild(this.latencyTestResultsElement);\n        }\n        return this._rootElement;\n    }\n\n    public get latencyTestResultsElement(): HTMLElement {\n        if (!this._latencyTestResultsElement) {\n            this._latencyTestResultsElement = document.createElement('div');\n            this._latencyTestResultsElement.id = 'dataChannelLatencyStatsResults';\n            this._latencyTestResultsElement.classList.add('StatsResult');\n        }\n        return this._latencyTestResultsElement;\n    }\n\n    public get latencyTestButton(): HTMLInputElement {\n        if (!this._latencyTestButton) {\n            this._latencyTestButton = document.createElement('input');\n            this._latencyTestButton.type = 'button';\n            this._latencyTestButton.value = 'Run Test';\n            this._latencyTestButton.id = 'btn-start-data-channel-latency-test';\n            this._latencyTestButton.classList.add('streamTools-button');\n            this._latencyTestButton.classList.add('btn-flat');\n        }\n        return this._latencyTestButton;\n    }\n\n    /**\n     * Populate the UI based on the latency test's results.\n     * @param result The latency test results.\n     */\n    public handleTestResult(result: DataChannelLatencyTestResult) {\n        Logger.Log(\n            Logger.GetStackTrace(),\n            result.toString(),\n            6\n        );\n        /**\n         * Check we have results, NaN would mean that UE version we talk to doesn't support our test\n         */\n        if (isNaN(result.dataChannelRtt)) {\n            this.latencyTestResultsElement.innerHTML = '<div>Not supported</div>';\n            return;\n        }\n        let latencyStatsInnerHTML = '';\n        latencyStatsInnerHTML +=\n            '<div>Data channel RTT (ms): ' +\n            result.dataChannelRtt +\n            '</div>';\n        /**\n         * Separate path time discovery works only when UE and Player clocks have been synchronized.\n         */\n        if (result.playerToStreamerTime >= 0 && result.streamerToPlayerTime >= 0) {\n            latencyStatsInnerHTML +=\n                '<div>Player to Streamer path (ms): ' + result.playerToStreamerTime + '</div>';\n            latencyStatsInnerHTML +=\n                '<div>Streamer to Player path (ms): ' +\n                result.streamerToPlayerTime +\n                '</div>';\n        }\n        this.latencyTestResultsElement.innerHTML = latencyStatsInnerHTML;\n        //setup button to download the detailed results\n        let downloadButton: HTMLInputElement = document.createElement('input');\n        downloadButton.type = 'button';\n        downloadButton.value = 'Download';\n        downloadButton.classList.add('streamTools-button');\n        downloadButton.classList.add('btn-flat');\n        downloadButton.onclick = () => {\n            let file = new Blob([result.exportLatencyAsCSV()], {type: 'text/plain'});\n            let a = document.createElement(\"a\"),\n                url = URL.createObjectURL(file);\n            a.href = url;\n            a.download = \"data_channel_latency_test_results.csv\";\n            document.body.appendChild(a);\n            a.click();\n            setTimeout(function() {\n                document.body.removeChild(a);\n                window.URL.revokeObjectURL(url);\n            }, 0);\n        }\n        this.latencyTestResultsElement.appendChild(downloadButton);\n    }\n\n    public handleTestStart() {\n        this.latencyTestResultsElement.innerHTML =\n            '<div>Test in progress</div>';\n    }\n\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Declare additions to base types for cross browser fullscreen functionality.\n */\ndeclare global {\n    interface Document {\n        webkitIsFullScreen?: boolean;\n        mozFullScreen?: boolean;\n        webkitFullscreenEnabled?: boolean;\n        mozCancelFullScreen?: () => Promise<void>;\n        msExitFullscreen?: () => Promise<void>;\n        webkitExitFullscreen?: () => Promise<void>;\n        mozFullScreenElement?: Element;\n        msFullscreenElement?: Element;\n        webkitFullscreenElement?: Element;\n    }\n\n    interface HTMLElement {\n        msRequestFullscreen?: () => Promise<void>;\n        mozRequestFullscreen?: () => Promise<void>;\n        webkitRequestFullscreen?: () => Promise<void>;\n        webkitEnterFullscreen?: () => void;\n    }\n}\n\n/**\n * Base class for an element (i.e. button) that, when clicked, will toggle fullscreen of a given element.\n * Can be initialized with any HTMLElement, if it is set as rootElement in the constructor.\n */\nexport class FullScreenIconBase {\n    isFullscreen = false;\n    fullscreenElement: HTMLElement | HTMLVideoElement;\n\n    _rootElement: HTMLElement;\n\n    public get rootElement() {\n        return this._rootElement;\n    }\n\n    public set rootElement(element) {\n        element.onclick = () => this.toggleFullscreen();\n        this._rootElement = element;\n    }\n\n    constructor() {       \n        // set up the full screen events\n        document.addEventListener(\n            'webkitfullscreenchange',\n            () => this.onFullscreenChange(),\n            false\n        );\n        document.addEventListener(\n            'mozfullscreenchange',\n            () => this.onFullscreenChange(),\n            false\n        );\n        document.addEventListener(\n            'fullscreenchange',\n            () => this.onFullscreenChange(),\n            false\n        );\n        document.addEventListener(\n            'MSFullscreenChange',\n            () => this.onFullscreenChange(),\n            false\n        );\n    }\n\n    /**\n     * Makes the document or fullscreenElement fullscreen.\n     */\n    toggleFullscreen() {\n        // if already full screen; exit\n        // else go fullscreen\n        if (\n            document.fullscreenElement ||\n            document.webkitFullscreenElement ||\n            document.mozFullScreenElement ||\n            document.msFullscreenElement\n        ) {\n            if (document.exitFullscreen) {\n                document.exitFullscreen();\n            } else if (document.mozCancelFullScreen) {\n                document.mozCancelFullScreen();\n            } else if (document.webkitExitFullscreen) {\n                document.webkitExitFullscreen();\n            } else if (document.msExitFullscreen) {\n                document.msExitFullscreen();\n            }\n        } else {\n            const element = this.fullscreenElement;\n\n            if (!element) {\n                return;\n            }\n            if (element.requestFullscreen) {\n                element.requestFullscreen();\n            } else if (element.mozRequestFullscreen) {\n                element.mozRequestFullscreen();\n            } else if (element.webkitRequestFullscreen) {\n                element.webkitRequestFullscreen();\n            } else if (element.msRequestFullscreen) {\n                element.msRequestFullscreen();\n            } else if (element.webkitEnterFullscreen) {\n                element.webkitEnterFullscreen(); //for iphone this code worked\n            }\n        }\n        this.onFullscreenChange();\n    }\n\n    /**\n     * Handles the fullscreen button on change\n     */\n    onFullscreenChange() {\n        this.isFullscreen =\n            document.webkitIsFullScreen ||\n            document.mozFullScreen ||\n            (document.msFullscreenElement &&\n                document.msFullscreenElement !== null) ||\n            (document.fullscreenElement && document.fullscreenElement !== null);\n    }\n}\n\n/**\n * An implementation of FullScreenIconBase that uses an externally\n * provided HTMLElement for toggling full screen.\n */\nexport class FullScreenIconExternal extends FullScreenIconBase {\n\n    constructor(externalButton : HTMLElement) {\n        super();\n        this.rootElement = externalButton;\n    }\n\n}\n\n/**\n * The default fullscreen icon that contains a button and svgs for each state.\n */\nexport class FullScreenIcon extends FullScreenIconBase {\n    _maximizeIcon: SVGElement;\n    _minimizeIcon: SVGElement;\n    _tooltipText: HTMLElement;\n\n    constructor() {\n        super();\n        \n        const createdButton : HTMLButtonElement = document.createElement('button');\n        createdButton.type = 'button';\n        createdButton.classList.add('UiTool');\n        createdButton.id = 'fullscreen-btn';\n        createdButton.appendChild(this.maximizeIcon);\n        createdButton.appendChild(this.minimizeIcon);\n        createdButton.appendChild(this.tooltipText);\n\n        this.rootElement = createdButton;\n    }\n\n    public get tooltipText(): HTMLElement {\n        if (!this._tooltipText) {\n            this._tooltipText = document.createElement('span');\n            this._tooltipText.classList.add('tooltiptext');\n            this._tooltipText.innerHTML = 'Fullscreen';\n        }\n        return this._tooltipText;\n    }\n\n    public get maximizeIcon(): SVGElement {\n        if (!this._maximizeIcon) {\n            this._maximizeIcon = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'svg'\n            );\n            this._maximizeIcon.setAttributeNS(null, 'id', 'maximizeIcon');\n            this._maximizeIcon.setAttributeNS(null, 'x', '0px');\n            this._maximizeIcon.setAttributeNS(null, 'y', '0px');\n            this._maximizeIcon.setAttributeNS(\n                null,\n                'viewBox',\n                '0 0 384.97 384.97'\n            );\n\n            // create svg group for the paths\n            const svgGroup = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'g'\n            );\n            svgGroup.classList.add('svgIcon');\n            this._maximizeIcon.appendChild(svgGroup);\n\n            // create paths for the icon itself, one for each corner\n            const path1 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path1.setAttributeNS(\n                null,\n                'd',\n                'M384.97,12.03c0-6.713-5.317-12.03-12.03-12.03H264.847c-6.833,0-11.922,5.39-11.934,12.223c0,6.821,5.101,11.838,11.934,11.838h96.062l-0.193,96.519c0,6.833,5.197,12.03,12.03,12.03c6.833-0.012,12.03-5.197,12.03-12.03l0.193-108.369c0-0.036-0.012-0.06-0.012-0.084C384.958,12.09,384.97,12.066,384.97,12.03z'\n            );\n\n            const path2 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path2.setAttributeNS(\n                null,\n                'd',\n                'M120.496,0H12.403c-0.036,0-0.06,0.012-0.096,0.012C12.283,0.012,12.247,0,12.223,0C5.51,0,0.192,5.317,0.192,12.03L0,120.399c0,6.833,5.39,11.934,12.223,11.934c6.821,0,11.838-5.101,11.838-11.934l0.192-96.339h96.242c6.833,0,12.03-5.197,12.03-12.03C132.514,5.197,127.317,0,120.496,0z'\n            );\n\n            const path3 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path3.setAttributeNS(\n                null,\n                'd',\n                'M120.123,360.909H24.061v-96.242c0-6.833-5.197-12.03-12.03-12.03S0,257.833,0,264.667v108.092c0,0.036,0.012,0.06,0.012,0.084c0,0.036-0.012,0.06-0.012,0.096c0,6.713,5.317,12.03,12.03,12.03h108.092c6.833,0,11.922-5.39,11.934-12.223C132.057,365.926,126.956,360.909,120.123,360.909z'\n            );\n\n            const path4 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path4.setAttributeNS(\n                null,\n                'd',\n                'M372.747,252.913c-6.833,0-11.85,5.101-11.838,11.934v96.062h-96.242c-6.833,0-12.03,5.197-12.03,12.03s5.197,12.03,12.03,12.03h108.092c0.036,0,0.06-0.012,0.084-0.012c0.036-0.012,0.06,0.012,0.096,0.012c6.713,0,12.03-5.317,12.03-12.03V264.847C384.97,258.014,379.58,252.913,372.747,252.913z'\n            );\n\n            svgGroup.appendChild(path1);\n            svgGroup.appendChild(path2);\n            svgGroup.appendChild(path3);\n            svgGroup.appendChild(path4);\n        }\n        return this._maximizeIcon;\n    }\n\n    public get minimizeIcon(): SVGElement {\n        if (!this._minimizeIcon) {\n            this._minimizeIcon = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'svg'\n            );\n            this._minimizeIcon.setAttributeNS(null, 'id', 'minimizeIcon');\n            this._minimizeIcon.setAttributeNS(null, 'x', '0px');\n            this._minimizeIcon.setAttributeNS(null, 'y', '0px');\n            this._minimizeIcon.setAttributeNS(\n                null,\n                'viewBox',\n                '0 0 385.331 385.331'\n            );\n\n            // create svg group for the paths\n            const svgGroup = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'g'\n            );\n            svgGroup.classList.add('svgIcon');\n            this._minimizeIcon.appendChild(svgGroup);\n\n            // create paths for the icon itself, one for each corner\n            const path1 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path1.setAttributeNS(\n                null,\n                'd',\n                'M264.943,156.665h108.273c6.833,0,11.934-5.39,11.934-12.211c0-6.833-5.101-11.85-11.934-11.838h-96.242V36.181c0-6.833-5.197-12.03-12.03-12.03s-12.03,5.197-12.03,12.03v108.273c0,0.036,0.012,0.06,0.012,0.084c0,0.036-0.012,0.06-0.012,0.096C252.913,151.347,258.23,156.677,264.943,156.665z'\n            );\n\n            const path2 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path2.setAttributeNS(\n                null,\n                'd',\n                'M120.291,24.247c-6.821,0-11.838,5.113-11.838,11.934v96.242H12.03c-6.833,0-12.03,5.197-12.03,12.03c0,6.833,5.197,12.03,12.03,12.03h108.273c0.036,0,0.06-0.012,0.084-0.012c0.036,0,0.06,0.012,0.096,0.012c6.713,0,12.03-5.317,12.03-12.03V36.181C132.514,29.36,127.124,24.259,120.291,24.247z'\n            );\n\n            const path3 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path3.setAttributeNS(\n                null,\n                'd',\n                'M120.387,228.666H12.115c-6.833,0.012-11.934,5.39-11.934,12.223c0,6.833,5.101,11.85,11.934,11.838h96.242v96.423c0,6.833,5.197,12.03,12.03,12.03c6.833,0,12.03-5.197,12.03-12.03V240.877c0-0.036-0.012-0.06-0.012-0.084c0-0.036,0.012-0.06,0.012-0.096C132.418,233.983,127.1,228.666,120.387,228.666z'\n            );\n\n            const path4 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path4.setAttributeNS(\n                null,\n                'd',\n                'M373.3,228.666H265.028c-0.036,0-0.06,0.012-0.084,0.012c-0.036,0-0.06-0.012-0.096-0.012c-6.713,0-12.03,5.317-12.03,12.03v108.273c0,6.833,5.39,11.922,12.223,11.934c6.821,0.012,11.838-5.101,11.838-11.922v-96.242H373.3c6.833,0,12.03-5.197,12.03-12.03S380.134,228.678,373.3,228.666z'\n            );\n\n            svgGroup.appendChild(path1);\n            svgGroup.appendChild(path2);\n            svgGroup.appendChild(path3);\n            svgGroup.appendChild(path4);\n        }\n        return this._minimizeIcon;\n    }\n\n    onFullscreenChange() {\n        super.onFullscreenChange();\n\n        const minimize = this.minimizeIcon;\n        const maximize = this.maximizeIcon;\n\n        if (this.isFullscreen) {\n            minimize.style.display = 'inline';\n            //ios disappearing svg fix\n            minimize.style.transform = 'translate(0, 0)';\n            maximize.style.display = 'none';\n        } else {\n            minimize.style.display = 'none';\n            maximize.style.display = 'inline';\n            //ios disappearing svg fix\n            maximize.style.transform = 'translate(0, 0)';\n        }\n    }\n\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * A button with a text label beside it.\n */\nexport class LabelledButton {\n    _label: string;\n    _buttonText: string;\n    _rootElement: HTMLElement;\n    _button: HTMLInputElement;\n\n    constructor(label: string, buttonText: string) {\n        this._label = label;\n        this._buttonText = buttonText;\n    }\n\n    /**\n     * Add a click listener to the button element.\n     */\n    public addOnClickListener(onClickFunc: () => void) {\n        this.button.addEventListener('click', onClickFunc);\n    }\n\n    /**\n     * Get the HTMLInputElement for the button.\n     */\n    public get button(): HTMLInputElement {\n        if (!this._button) {\n            this._button = document.createElement('input');\n            this._button.type = 'button';\n            this._button.value = this._buttonText;\n            this._button.classList.add('overlay-button');\n            this._button.classList.add('btn-flat');\n        }\n        return this._button;\n    }\n\n    /**\n     * @returns Return or creates a HTML element that represents this setting in the DOM.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            // create root div with \"setting\" css class\n            this._rootElement = document.createElement('div');\n            this._rootElement.classList.add('setting');\n\n            // create div element to contain our setting's text\n            const settingsTextElem = document.createElement('div');\n            settingsTextElem.innerText = this._label;\n            this._rootElement.appendChild(settingsTextElem);\n\n            // create label element to wrap out input type\n            const wrapperLabel = document.createElement('label');\n            wrapperLabel.classList.add('btn-overlay');\n            this._rootElement.appendChild(wrapperLabel);\n\n            wrapperLabel.appendChild(this.button);\n        }\n        return this._rootElement;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { LatencyTestResults } from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\nimport { Logger } from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\n\n/**\n * Latency test UI elements and results handling.\n */\nexport class LatencyTest {\n    _rootElement: HTMLElement;\n    _latencyTestButton: HTMLInputElement;\n    _latencyTestResultsElement: HTMLElement;\n\n    /**\n     * Get the the button containing the stats icon.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            this._rootElement = document.createElement('section');\n            this._rootElement.classList.add('settingsContainer');\n\n            // make heading\n            const heading = document.createElement('div');\n            heading.id = 'latencyTestHeader';\n            heading.classList.add('settings-text');\n            heading.classList.add('settingsHeader');\n            this._rootElement.appendChild(heading);\n\n            const headingText = document.createElement('div');\n            headingText.innerHTML = 'Latency Test';\n            heading.appendChild(headingText);\n            heading.appendChild(this.latencyTestButton);\n\n            // make test results element\n            const resultsParentElem = document.createElement('div');\n            resultsParentElem.id = 'latencyTestContainer';\n            resultsParentElem.classList.add('d-none');\n            this._rootElement.appendChild(resultsParentElem);\n\n            resultsParentElem.appendChild(this.latencyTestResultsElement);\n        }\n        return this._rootElement;\n    }\n\n    public get latencyTestResultsElement(): HTMLElement {\n        if (!this._latencyTestResultsElement) {\n            this._latencyTestResultsElement = document.createElement('div');\n            this._latencyTestResultsElement.id = 'latencyStatsResults';\n            this._latencyTestResultsElement.classList.add('StatsResult');\n        }\n        return this._latencyTestResultsElement;\n    }\n\n    public get latencyTestButton(): HTMLInputElement {\n        if (!this._latencyTestButton) {\n            this._latencyTestButton = document.createElement('input');\n            this._latencyTestButton.type = 'button';\n            this._latencyTestButton.value = 'Run Test';\n            this._latencyTestButton.id = 'btn-start-latency-test';\n            this._latencyTestButton.classList.add('streamTools-button');\n            this._latencyTestButton.classList.add('btn-flat');\n        }\n        return this._latencyTestButton;\n    }\n\n    /**\n     * Populate the UI based on the latency test's results.\n     * @param latencyTimings The latency test results.\n     */\n    public handleTestResult(latencyTimings: LatencyTestResults) {\n        Logger.Log(Logger.GetStackTrace(), latencyTimings.toString(), 6);\n        let latencyStatsInnerHTML = '';\n        latencyStatsInnerHTML +=\n            '<div>Net latency RTT (ms): ' +\n            latencyTimings.networkLatency +\n            '</div>';\n        latencyStatsInnerHTML +=\n            '<div>UE Encode (ms): ' + latencyTimings.EncodeMs + '</div>';\n        latencyStatsInnerHTML +=\n            '<div>UE Capture (ms): ' +\n            latencyTimings.CaptureToSendMs +\n            '</div>';\n        latencyStatsInnerHTML +=\n            '<div>Browser send latency (ms): ' +\n            latencyTimings.browserSendLatency +\n            '</div>';\n        latencyStatsInnerHTML +=\n            latencyTimings.frameDisplayDeltaTimeMs &&\n            latencyTimings.browserReceiptTimeMs\n                ? '<div>Browser receive latency (ms): ' +\n                  latencyTimings.frameDisplayDeltaTimeMs +\n                  '</div>'\n                : '';\n        latencyStatsInnerHTML +=\n            '<div>Total latency (excluding browser) (ms): ' +\n            latencyTimings.latencyExcludingDecode +\n            '</div>';\n        latencyStatsInnerHTML += latencyTimings.endToEndLatency\n            ? '<div>Total latency (ms): ' +\n              latencyTimings.endToEndLatency +\n              '</div>'\n            : '';\n        this.latencyTestResultsElement.innerHTML = latencyStatsInnerHTML;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Settings icon that can be clicked.\n */\nexport class SettingsIcon {\n    _rootElement: HTMLButtonElement;\n    _settingsIcon: SVGElement;\n    _tooltipText: HTMLElement;\n\n    /**\n     * Get the the button containing the settings icon.\n     */\n    public get rootElement(): HTMLButtonElement {\n        if (!this._rootElement) {\n            this._rootElement = document.createElement('button');\n            this._rootElement.type = 'button';\n            this._rootElement.classList.add('UiTool');\n            this._rootElement.id = 'settingsBtn';\n            this._rootElement.appendChild(this.settingsIcon);\n            this._rootElement.appendChild(this.tooltipText);\n        }\n        return this._rootElement;\n    }\n\n    public get tooltipText(): HTMLElement {\n        if (!this._tooltipText) {\n            this._tooltipText = document.createElement('span');\n            this._tooltipText.classList.add('tooltiptext');\n            this._tooltipText.innerHTML = 'Settings';\n        }\n        return this._tooltipText;\n    }\n\n    public get settingsIcon(): SVGElement {\n        if (!this._settingsIcon) {\n            this._settingsIcon = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'svg'\n            );\n            this._settingsIcon.setAttributeNS(null, 'id', 'settingsIcon');\n            this._settingsIcon.setAttributeNS(null, 'x', '0px');\n            this._settingsIcon.setAttributeNS(null, 'y', '0px');\n            this._settingsIcon.setAttributeNS(\n                null,\n                'viewBox',\n                '0 0 478.703 478.703'\n            );\n\n            // create svg group for the paths\n            const svgGroup = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'g'\n            );\n            svgGroup.classList.add('svgIcon');\n            this._settingsIcon.appendChild(svgGroup);\n\n            // create paths for the icon itself, the inner and out path of a cog\n            const path1 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path1.setAttributeNS(\n                null,\n                'd',\n                'M454.2,189.101l-33.6-5.7c-3.5-11.3-8-22.2-13.5-32.6l19.8-27.7c8.4-11.8,7.1-27.9-3.2-38.1l-29.8-29.8\\\n\t\t\tc-5.6-5.6-13-8.7-20.9-8.7c-6.2,0-12.1,1.9-17.1,5.5l-27.8,19.8c-10.8-5.7-22.1-10.4-33.8-13.9l-5.6-33.2\\\n\t\t\tc-2.4-14.3-14.7-24.7-29.2-24.7h-42.1c-14.5,0-26.8,10.4-29.2,24.7l-5.8,34c-11.2,3.5-22.1,8.1-32.5,13.7l-27.5-19.8\\\n\t\t\tc-5-3.6-11-5.5-17.2-5.5c-7.9,0-15.4,3.1-20.9,8.7l-29.9,29.8c-10.2,10.2-11.6,26.3-3.2,38.1l20,28.1\\\n\t\t\tc-5.5,10.5-9.9,21.4-13.3,32.7l-33.2,5.6c-14.3,2.4-24.7,14.7-24.7,29.2v42.1c0,14.5,10.4,26.8,24.7,29.2l34,5.8\\\n\t\t\tc3.5,11.2,8.1,22.1,13.7,32.5l-19.7,27.4c-8.4,11.8-7.1,27.9,3.2,38.1l29.8,29.8c5.6,5.6,13,8.7,20.9,8.7c6.2,0,12.1-1.9,17.1-5.5\\\n\t\t\tl28.1-20c10.1,5.3,20.7,9.6,31.6,13l5.6,33.6c2.4,14.3,14.7,24.7,29.2,24.7h42.2c14.5,0,26.8-10.4,29.2-24.7l5.7-33.6\\\n\t\t\tc11.3-3.5,22.2-8,32.6-13.5l27.7,19.8c5,3.6,11,5.5,17.2,5.5l0,0c7.9,0,15.3-3.1,20.9-8.7l29.8-29.8c10.2-10.2,11.6-26.3,3.2-38.1\\\n\t\t\tl-19.8-27.8c5.5-10.5,10.1-21.4,13.5-32.6l33.6-5.6c14.3-2.4,24.7-14.7,24.7-29.2v-42.1\\\n\t\t\tC478.9,203.801,468.5,191.501,454.2,189.101z M451.9,260.401c0,1.3-0.9,2.4-2.2,2.6l-42,7c-5.3,0.9-9.5,4.8-10.8,9.9\\\n\t\t\tc-3.8,14.7-9.6,28.8-17.4,41.9c-2.7,4.6-2.5,10.3,0.6,14.7l24.7,34.8c0.7,1,0.6,2.5-0.3,3.4l-29.8,29.8c-0.7,0.7-1.4,0.8-1.9,0.8\\\n\t\t\tc-0.6,0-1.1-0.2-1.5-0.5l-34.7-24.7c-4.3-3.1-10.1-3.3-14.7-0.6c-13.1,7.8-27.2,13.6-41.9,17.4c-5.2,1.3-9.1,5.6-9.9,10.8l-7.1,42\\\n\t\t\tc-0.2,1.3-1.3,2.2-2.6,2.2h-42.1c-1.3,0-2.4-0.9-2.6-2.2l-7-42c-0.9-5.3-4.8-9.5-9.9-10.8c-14.3-3.7-28.1-9.4-41-16.8\\\n\t\t\tc-2.1-1.2-4.5-1.8-6.8-1.8c-2.7,0-5.5,0.8-7.8,2.5l-35,24.9c-0.5,0.3-1,0.5-1.5,0.5c-0.4,0-1.2-0.1-1.9-0.8l-29.8-29.8\\\n\t\t\tc-0.9-0.9-1-2.3-0.3-3.4l24.6-34.5c3.1-4.4,3.3-10.2,0.6-14.8c-7.8-13-13.8-27.1-17.6-41.8c-1.4-5.1-5.6-9-10.8-9.9l-42.3-7.2\\\n\t\t\tc-1.3-0.2-2.2-1.3-2.2-2.6v-42.1c0-1.3,0.9-2.4,2.2-2.6l41.7-7c5.3-0.9,9.6-4.8,10.9-10c3.7-14.7,9.4-28.9,17.1-42\\\n\t\t\tc2.7-4.6,2.4-10.3-0.7-14.6l-24.9-35c-0.7-1-0.6-2.5,0.3-3.4l29.8-29.8c0.7-0.7,1.4-0.8,1.9-0.8c0.6,0,1.1,0.2,1.5,0.5l34.5,24.6\\\n\t\t\tc4.4,3.1,10.2,3.3,14.8,0.6c13-7.8,27.1-13.8,41.8-17.6c5.1-1.4,9-5.6,9.9-10.8l7.2-42.3c0.2-1.3,1.3-2.2,2.6-2.2h42.1\\\n\t\t\tc1.3,0,2.4,0.9,2.6,2.2l7,41.7c0.9,5.3,4.8,9.6,10,10.9c15.1,3.8,29.5,9.7,42.9,17.6c4.6,2.7,10.3,2.5,14.7-0.6l34.5-24.8\\\n\t\t\tc0.5-0.3,1-0.5,1.5-0.5c0.4,0,1.2,0.1,1.9,0.8l29.8,29.8c0.9,0.9,1,2.3,0.3,3.4l-24.7,34.7c-3.1,4.3-3.3,10.1-0.6,14.7\\\n\t\t\tc7.8,13.1,13.6,27.2,17.4,41.9c1.3,5.2,5.6,9.1,10.8,9.9l42,7.1c1.3,0.2,2.2,1.3,2.2,2.6v42.1H451.9z'\n            );\n\n            const path2 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path2.setAttributeNS(\n                null,\n                'd',\n                'M239.4,136.001c-57,0-103.3,46.3-103.3,103.3s46.3,103.3,103.3,103.3s103.3-46.3,103.3-103.3S296.4,136.001,239.4,136.001z M239.4,315.601c-42.1,0-76.3-34.2-76.3-76.3s34.2-76.3,76.3-76.3s76.3,34.2,76.3,76.3S281.5,315.601,239.4,315.601z'\n            );\n\n            svgGroup.appendChild(path1);\n            svgGroup.appendChild(path2);\n        }\n        return this._settingsIcon;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * A UI component containing all the settings for the application.\n */\nexport class SettingsPanel {\n    _rootElement: HTMLElement;\n    _settingsCloseButton: HTMLElement;\n    _settingsContentElement: HTMLElement;\n\n    constructor() {\n        this._rootElement = null;\n    }\n\n    /**\n     * @returns Return or creates a HTML element that represents this setting in the DOM.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            this._rootElement = document.createElement('div');\n            this._rootElement.id = 'settings-panel';\n            this._rootElement.classList.add('panel-wrap');\n\n            const panelElem = document.createElement('div');\n            panelElem.classList.add('panel');\n            this._rootElement.appendChild(panelElem);\n\n            const settingsHeading = document.createElement('div');\n            settingsHeading.id = 'settingsHeading';\n            settingsHeading.textContent = 'Settings';\n            panelElem.appendChild(settingsHeading);\n\n            panelElem.appendChild(this.settingsCloseButton);\n            panelElem.appendChild(this.settingsContentElement);\n        }\n        return this._rootElement;\n    }\n\n    public get settingsContentElement(): HTMLElement {\n        if (!this._settingsContentElement) {\n            this._settingsContentElement = document.createElement('div');\n            this._settingsContentElement.id = 'settingsContent';\n        }\n        return this._settingsContentElement;\n    }\n\n    public get settingsCloseButton(): HTMLElement {\n        if (!this._settingsCloseButton) {\n            this._settingsCloseButton = document.createElement('div');\n            this._settingsCloseButton.id = 'settingsClose';\n        }\n        return this._settingsCloseButton;\n    }\n\n    /**\n     * Show settings panel.\n     */\n    public show(): void {\n        if (!this.rootElement.classList.contains('panel-wrap-visible')) {\n            this.rootElement.classList.add('panel-wrap-visible');\n        }\n    }\n\n    /**\n     * Toggle the visibility of the settings panel.\n     */\n    public toggleVisibility(): void {\n        this.rootElement.classList.toggle('panel-wrap-visible');\n    }\n\n    /**\n     * Hide settings panel.\n     */\n    public hide(): void {\n        if (this.rootElement.classList.contains('panel-wrap-visible')) {\n            this.rootElement.classList.remove('panel-wrap-visible');\n        }\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * Stats icon that can be clicked.\n */\nexport class StatsIcon {\n    _rootElement: HTMLButtonElement;\n    _statsIcon: SVGElement;\n    _tooltipText: HTMLElement;\n\n    /**\n     * Get the the button containing the stats icon.\n     */\n    public get rootElement(): HTMLButtonElement {\n        if (!this._rootElement) {\n            this._rootElement = document.createElement('button');\n            this._rootElement.type = 'button';\n            this._rootElement.classList.add('UiTool');\n            this._rootElement.id = 'statsBtn';\n            this._rootElement.appendChild(this.statsIcon);\n            this._rootElement.appendChild(this.tooltipText);\n        }\n        return this._rootElement;\n    }\n\n    public get tooltipText(): HTMLElement {\n        if (!this._tooltipText) {\n            this._tooltipText = document.createElement('span');\n            this._tooltipText.classList.add('tooltiptext');\n            this._tooltipText.innerHTML = 'Information';\n        }\n        return this._tooltipText;\n    }\n\n    public get statsIcon(): SVGElement {\n        if (!this._statsIcon) {\n            this._statsIcon = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'svg'\n            );\n            this._statsIcon.setAttributeNS(null, 'id', 'statsIcon');\n            this._statsIcon.setAttributeNS(null, 'x', '0px');\n            this._statsIcon.setAttributeNS(null, 'y', '0px');\n            this._statsIcon.setAttributeNS(null, 'viewBox', '0 0 330 330');\n\n            // create svg group for the paths\n            const svgGroup = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'g'\n            );\n            svgGroup.classList.add('svgIcon');\n            this._statsIcon.appendChild(svgGroup);\n\n            // create paths for the icon itself, the inner and out path of a cog\n            const path1 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path1.setAttributeNS(\n                null,\n                'd',\n                'M165,0.008C74.019,0.008,0,74.024,0,164.999c0,90.977,74.019,164.992,165,164.992s165-74.015,165-164.992C330,74.024,255.981,0.008,165,0.008z M165,299.992c-74.439,0-135-60.557-135-134.992S90.561,30.008,165,30.008s135,60.557,135,134.991C300,239.436,239.439,299.992,165,299.992z'\n            );\n\n            const path2 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path2.setAttributeNS(\n                null,\n                'd',\n                'M165,130.008c-8.284,0-15,6.716-15,15v99.983c0,8.284,6.716,15,15,15s15-6.716,15-15v-99.983C180,136.725,173.284,130.008,165,130.008z'\n            );\n\n            const path3 = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            path3.setAttributeNS(\n                null,\n                'd',\n                'M165,70.011c-3.95,0-7.811,1.6-10.61,4.39c-2.79,2.79-4.39,6.66-4.39,10.61s1.6,7.81,4.39,10.61c2.79,2.79,6.66,4.39,10.61,4.39s7.81-1.6,10.609-4.39c2.79-2.8,4.391-6.66,4.391-10.61s-1.601-7.82-4.391-10.61C172.81,71.61,168.95,70.011,165,70.011z'\n            );\n\n            svgGroup.appendChild(path1);\n            svgGroup.appendChild(path2);\n            svgGroup.appendChild(path3);\n        }\n        return this._statsIcon;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nimport { LatencyTest } from './LatencyTest';\nimport {InitialSettings, Logger, PixelStreaming} from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\nimport { AggregatedStats } from '@epicgames-ps/lib-pixelstreamingfrontend-ue5.3';\nimport { MathUtils } from '../Util/MathUtils';\nimport {DataChannelLatencyTest} from \"./DataChannelLatencyTest\";\nimport {PixelStreamingSettings} from \"@epicgames-ps/lib-pixelstreamingfrontend-ue5.3/types/DataChannel/InitialSettings\";\n\n/**\n * A stat structure, an id, the stat string, and the element where it is rendered.\n */\nexport class Stat {\n    id: string;\n    title: string;\n    stat: string;\n    element: HTMLElement;\n}\n\n/**\n * A UI component containing all the stats for the application.\n */\nexport class StatsPanel {\n    _rootElement: HTMLElement;\n    _statsCloseButton: HTMLElement;\n    _statsContentElement: HTMLElement;\n    _statisticsContainer: HTMLElement;\n    _statsResult: HTMLElement;\n\n    latencyTest: LatencyTest;\n    dataChannelLatencyTest: DataChannelLatencyTest;\n\n    /* A map stats we are storing/rendering */\n    statsMap = new Map<string, Stat>();\n\n    constructor() {\n        this.latencyTest = new LatencyTest();\n        this.dataChannelLatencyTest = new DataChannelLatencyTest();\n    }\n\n    /**\n     * @returns Return or creates a HTML element that represents this setting in the DOM.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            this._rootElement = document.createElement('div');\n            this._rootElement.id = 'stats-panel';\n            this._rootElement.classList.add('panel-wrap');\n\n            const panelElem = document.createElement('div');\n            panelElem.classList.add('panel');\n            this._rootElement.appendChild(panelElem);\n\n            const statsHeading = document.createElement('div');\n            statsHeading.id = 'statsHeading';\n            statsHeading.textContent = 'Information';\n            panelElem.appendChild(statsHeading);\n\n            panelElem.appendChild(this.statsCloseButton);\n            panelElem.appendChild(this.statsContentElement);\n        }\n        return this._rootElement;\n    }\n\n    public get statsContentElement(): HTMLElement {\n        if (!this._statsContentElement) {\n            this._statsContentElement = document.createElement('div');\n            this._statsContentElement.id = 'statsContent';\n\n            const streamToolStats = document.createElement('div');\n            streamToolStats.id = 'streamToolsStats';\n            streamToolStats.classList.add('container');\n\n            const controlStats = document.createElement('div');\n            controlStats.id = 'ControlStats';\n            controlStats.classList.add('row');\n\n            const statistics = document.createElement('section');\n            statistics.id = 'statistics';\n            statistics.classList.add('settingsContainer');\n\n            const statisticsHeader = document.createElement('div');\n            statisticsHeader.id = 'statisticsHeader';\n            statisticsHeader.classList.add('settings-text');\n            statisticsHeader.classList.add('settingsHeader');\n\n            const sessionStats = document.createElement('div');\n            sessionStats.innerHTML = 'Session Stats';\n\n            this._statsContentElement.appendChild(streamToolStats);\n            streamToolStats.appendChild(controlStats);\n            controlStats.appendChild(statistics);\n            statistics.appendChild(statisticsHeader);\n            statisticsHeader.appendChild(sessionStats);\n            statistics.appendChild(this.statisticsContainer);\n\n            controlStats.appendChild(this.latencyTest.rootElement);\n            controlStats.appendChild(this.dataChannelLatencyTest.rootElement);\n        }\n        return this._statsContentElement;\n    }\n\n    public get statisticsContainer(): HTMLElement {\n        if (!this._statisticsContainer) {\n            this._statisticsContainer = document.createElement('div');\n            this._statisticsContainer.id = 'statisticsContainer';\n            this._statisticsContainer.classList.add('d-none');\n            this._statisticsContainer.appendChild(this.statsResult);\n        }\n        return this._statisticsContainer;\n    }\n\n    public get statsResult(): HTMLElement {\n        if (!this._statsResult) {\n            this._statsResult = document.createElement('div');\n            this._statsResult.id = 'statisticsResult';\n            this._statsResult.classList.add('StatsResult');\n        }\n        return this._statsResult;\n    }\n\n    public get statsCloseButton(): HTMLElement {\n        if (!this._statsCloseButton) {\n            this._statsCloseButton = document.createElement('div');\n            this._statsCloseButton.id = 'statsClose';\n        }\n        return this._statsCloseButton;\n    }\n\n    public onDisconnect(): void {\n        this.latencyTest.latencyTestButton.onclick = () => {\n            // do nothing\n        }\n        this.dataChannelLatencyTest.latencyTestButton.onclick = () => {\n            //do nothing\n        }\n    }\n\n    public onVideoInitialized(stream: PixelStreaming): void {\n        // starting a latency check\n        this.latencyTest.latencyTestButton.onclick = () => {\n            stream.requestLatencyTest();\n        };\n        this.dataChannelLatencyTest.latencyTestButton.onclick = () => {\n            let started = stream.requestDataChannelLatencyTest({\n                duration: 1000,\n                rps: 10,\n                requestSize: 200,\n                responseSize: 200\n            });\n            if (started) {\n                this.dataChannelLatencyTest.handleTestStart();\n            }\n        };\n    }\n\n    public configure(settings: PixelStreamingSettings): void {\n        if (settings.DisableLatencyTest) {\n            this.latencyTest.latencyTestButton.disabled = true;\n            this.latencyTest.latencyTestButton.title =\n                'Disabled by -PixelStreamingDisableLatencyTester=true';\n            this.dataChannelLatencyTest.latencyTestButton.disabled = true;\n            this.dataChannelLatencyTest.latencyTestButton.title =\n                'Disabled by -PixelStreamingDisableLatencyTester=true';\n            Logger.Info(\n                Logger.GetStackTrace(),\n                '-PixelStreamingDisableLatencyTester=true, requesting latency report from the the browser to UE is disabled.'\n            );\n        }\n    }\n\n    /**\n     * Show stats panel.\n     */\n    public show(): void {\n        if (!this.rootElement.classList.contains('panel-wrap-visible')) {\n            this.rootElement.classList.add('panel-wrap-visible');\n        }\n    }\n\n    /**\n     * Toggle the visibility of the stats panel.\n     */\n    public toggleVisibility(): void {\n        this.rootElement.classList.toggle('panel-wrap-visible');\n    }\n\n    /**\n     * Hide the stats panel.\n     */\n    public hide(): void {\n        if (this.rootElement.classList.contains('panel-wrap-visible')) {\n            this.rootElement.classList.remove('panel-wrap-visible');\n        }\n    }\n\n    public handlePlayerCount(playerCount: number) {\n        this.addOrUpdateStat(\n            'PlayerCountStat',\n            'Players',\n            playerCount.toString()\n        );\n    }\n\n    /**\n     * Handle stats coming in from browser/UE\n     * @param stats the stats structure\n     */\n    public handleStats(stats: AggregatedStats) {\n        // format numbering based on the browser language\n        const numberFormat = new Intl.NumberFormat(window.navigator.language, {\n            maximumFractionDigits: 0\n        });\n\n        // Inbound data\n        const inboundData = MathUtils.formatBytes(\n            stats.inboundVideoStats.bytesReceived,\n            2\n        );\n        this.addOrUpdateStat('InboundDataStat', 'Received', inboundData);\n\n        // Packets lost\n        const packetsLostStat = Object.prototype.hasOwnProperty.call(\n            stats.inboundVideoStats,\n            'packetsLost'\n        )\n            ? numberFormat.format(stats.inboundVideoStats.packetsLost)\n            : 'Chrome only';\n        this.addOrUpdateStat(\n            'PacketsLostStat',\n            'Packets Lost',\n            packetsLostStat\n        );\n\n        // Bitrate\n        if (stats.inboundVideoStats.bitrate) {\n            this.addOrUpdateStat(\n                'VideoBitrateStat',\n                'Video Bitrate (kbps)',\n                stats.inboundVideoStats.bitrate.toString()\n            );\n        }\n\n        if (stats.inboundAudioStats.bitrate) {\n            this.addOrUpdateStat(\n                'AudioBitrateStat',\n                'Audio Bitrate (kbps)',\n                stats.inboundAudioStats.bitrate.toString()\n            );\n        }\n\n        // Video resolution\n        const resStat =\n            Object.prototype.hasOwnProperty.call(\n                stats.inboundVideoStats,\n                'frameWidth'\n            ) &&\n            stats.inboundVideoStats.frameWidth &&\n            Object.prototype.hasOwnProperty.call(\n                stats.inboundVideoStats,\n                'frameHeight'\n            ) &&\n            stats.inboundVideoStats.frameHeight\n                ? stats.inboundVideoStats.frameWidth +\n                  'x' +\n                  stats.inboundVideoStats.frameHeight\n                : 'Chrome only';\n        this.addOrUpdateStat('VideoResStat', 'Video resolution', resStat);\n\n        // Frames decoded\n        const framesDecoded = Object.prototype.hasOwnProperty.call(\n            stats.inboundVideoStats,\n            'framesDecoded'\n        )\n            ? numberFormat.format(stats.inboundVideoStats.framesDecoded)\n            : 'Chrome only';\n        this.addOrUpdateStat(\n            'FramesDecodedStat',\n            'Frames Decoded',\n            framesDecoded\n        );\n\n        // Framerate\n        if (stats.inboundVideoStats.framesPerSecond) {\n            this.addOrUpdateStat(\n                'FramerateStat',\n                'Framerate',\n                stats.inboundVideoStats.framesPerSecond.toString()\n            );\n        }\n\n        // Frames dropped\n        this.addOrUpdateStat(\n            'FramesDroppedStat',\n            'Frames dropped',\n            stats.inboundVideoStats.framesDropped?.toString()\n        );\n\n        if (stats.inboundVideoStats.codecId) {\n            this.addOrUpdateStat(\n                'VideoCodecStat',\n                'Video codec',\n                // Split the codec to remove the Fmtp line\n                stats.codecs\n                    .get(stats.inboundVideoStats.codecId)\n                    ?.split(' ')[0] ?? ''\n            );\n        }\n\n        if (stats.inboundAudioStats.codecId) {\n            this.addOrUpdateStat(\n                'AudioCodecStat',\n                'Audio codec',\n                // Split the codec to remove the Fmtp line\n                stats.codecs\n                    .get(stats.inboundAudioStats.codecId)\n                    ?.split(' ')[0] ?? ''\n            );\n        }\n\n        // RTT\n        const netRTT =\n            Object.prototype.hasOwnProperty.call(\n                stats.candidatePair,\n                'currentRoundTripTime'\n            ) && stats.isNumber(stats.candidatePair.currentRoundTripTime)\n                ? numberFormat.format(\n                      stats.candidatePair.currentRoundTripTime * 1000\n                  )\n                : \"Can't calculate\";\n        this.addOrUpdateStat('RTTStat', 'Net RTT (ms)', netRTT);\n\n        this.addOrUpdateStat(\n            'DurationStat',\n            'Duration',\n            stats.sessionStats.runTime\n        );\n\n        this.addOrUpdateStat(\n            'ControlsInputStat',\n            'Controls stream input',\n            stats.sessionStats.controlsStreamInput\n        );\n\n        // QP\n        this.addOrUpdateStat(\n            'QPStat',\n            'Video quantization parameter',\n            stats.sessionStats.videoEncoderAvgQP.toString()\n        );\n\n        // todo:\n        //statsText += `<div>Browser receive to composite (ms): ${stats.inboundVideoStats.receiveToCompositeMs}</div>`;\n\n        Logger.Log(\n            Logger.GetStackTrace(),\n            `--------- Stats ---------\\n ${stats}\\n------------------------`,\n            6\n        );\n    }\n\n    /**\n     * Adds a new stat to the stats results in the DOM or updates an exiting stat.\n     * @param id The id of the stat to add/update.\n     * @param stat The contents of the stat.\n     */\n    public addOrUpdateStat(id: string, statLabel: string, stat: string) {\n        const statHTML = `${statLabel}: ${stat}`;\n\n        if (!this.statsMap.has(id)) {\n            // create the stat\n            const newStat = new Stat();\n            newStat.id = id;\n            newStat.stat = stat;\n            newStat.title = statLabel;\n            newStat.element = document.createElement('div');\n            newStat.element.innerHTML = statHTML;\n            // add the stat to the dom\n            this.statsResult.appendChild(newStat.element);\n            this.statsMap.set(id, newStat);\n        }\n        // update the existing stat\n        else {\n            const value = this.statsMap.get(id);\n            if (value !== undefined) {\n                value.element.innerHTML = statHTML;\n            }\n        }\n    }\n}\n","/** Whether a stream UI element is internally made, externally provided, or disabled. */\nexport enum UIElementCreationMode {\n    CreateDefaultElement,\n    UseCustomElement,\n    Disable\n}\n\n/** A configuration for different UI elements which control/display info related to the stream. */\nexport type UIElementConfig = {\n    // In which way is this element created?\n    creationMode : UIElementCreationMode,\n    // (Only relevant if when mode is CreateCustomElement) Visualizing element\n    customElement? : HTMLElement\n}\n\n/**\n * Configures a general stream-related UI panel. \n * For example: is it created, and if it is, what kind of button is used to show/hide it.\n * This configuration is used for the settings panel and stats panel by default.\n * \n * Note: For cases where the panel needs to be created, but a button isn't needed, \n * the panel element can be placed anywhere in the DOM as needed (see Application class). \n */\nexport type PanelConfiguration = {\n    // If panel is enabled, HTML elements for it will be created, and funtionality bound\n    isEnabled : boolean,\n    // (Only relevant if isEnabled) The type of the button to show/hide this panel\n    visibilityButtonConfig? : UIElementConfig\n}\n\nexport function isPanelEnabled(config : PanelConfiguration | undefined) : boolean {\n    return !config || (!!config && config.isEnabled);\n}","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * A UI element showing the QP (quantization parameter) of the video stream at the last encoded frame (well, last transmitted QP really).\n * A blockier encoding will have a higher QP and this will make the indicator turn more red.\n * A non-blocky stream will have a lower QP and this will make the indicator turn more green.\n * The QP indicator is represented visually using a WiFi icon.\n */\nexport class VideoQpIndicator {\n    videoEncoderAvgQP = -1;\n\n    // non html elements\n    statsText = '';\n    color = '';\n\n    // qp colors\n    readonly orangeQP = 26;\n    readonly redQP = 35;\n\n    _rootElement: HTMLElement;\n    _qualityText: HTMLElement;\n    _qualityStatus: SVGElement;\n    _dot: SVGElement;\n    _outer: SVGElement;\n    _middle: SVGElement;\n    _inner: SVGElement;\n\n    /**\n     * Get the root element of the QP indicator.\n     */\n    public get rootElement(): HTMLElement {\n        if (!this._rootElement) {\n            // make the root element that contains the svg for the connection\n            this._rootElement = document.createElement('div');\n            this._rootElement.id = 'connection';\n            this._rootElement.classList.add('UiTool');\n\n            // add svg icon for the connection strength\n            this._rootElement.appendChild(this.qualityStatus);\n\n            // add the text underneath the connection\n            this._rootElement.appendChild(this.qualityText);\n\n            // set colors to not connected initially\n            this.updateQpTooltip(-1);\n        }\n        return this._rootElement;\n    }\n\n    /**\n     * Get the text that displays under the icon.\n     */\n    public get qualityText(): HTMLElement {\n        if (!this._qualityText) {\n            this._qualityText = document.createElement('span');\n            this._qualityText.id = 'qualityText';\n            this._qualityText.classList.add('tooltiptext');\n        }\n        return this._qualityText;\n    }\n\n    /**\n     * Get the icon.\n     */\n    public get qualityStatus(): SVGElement {\n        if (!this._qualityStatus) {\n            this._qualityStatus = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'svg'\n            );\n            this._qualityStatus.setAttributeNS(\n                null,\n                'id',\n                'connectionStrength'\n            );\n            this._qualityStatus.setAttributeNS(null, 'x', '0px');\n            this._qualityStatus.setAttributeNS(null, 'y', '0px');\n            this._qualityStatus.setAttributeNS(\n                null,\n                'viewBox',\n                '0 0 494.45 494.45'\n            );\n\n            // build wifi icon\n            this.qualityStatus.appendChild(this.dot);\n            this.qualityStatus.appendChild(this.middle);\n            this.qualityStatus.appendChild(this.outer);\n            this.qualityStatus.appendChild(this.inner);\n        }\n        return this._qualityStatus;\n    }\n\n    /**\n     * Get the dot at the bottom of the wifi icon.\n     */\n    public get dot(): SVGElement {\n        if (!this._dot) {\n            this._dot = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'circle'\n            );\n            this._dot.setAttributeNS(null, 'id', 'dot');\n            this._dot.setAttributeNS(null, 'cx', '247.125');\n            this._dot.setAttributeNS(null, 'cy', '398.925');\n            this._dot.setAttributeNS(null, 'r', '35.3');\n        }\n        return this._dot;\n    }\n\n    /**\n     * Get the outer arc of the wifi icon.\n     */\n    public get outer(): SVGElement {\n        if (!this._outer) {\n            this._outer = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            this._outer.setAttributeNS(null, 'id', 'outer');\n            this._outer.setAttributeNS(\n                null,\n                'd',\n                'M467.925,204.625c-6.8,0-13.5-2.6-18.7-7.8c-111.5-111.4-292.7-111.4-404.1,0c-10.3,10.3-27.1,10.3-37.4,0s-10.3-27.1,0-37.4c64-64,149-99.2,239.5-99.2s175.5,35.2,239.5,99.2c10.3,10.3,10.3,27.1,0,37.4C481.425,202.025,474.625,204.625,467.925,204.625z'\n            );\n        }\n        return this._outer;\n    }\n\n    /**\n     * Get the middle arc of the wifi icon.\n     */\n    public get middle(): SVGElement {\n        if (!this._middle) {\n            this._middle = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            this._middle.setAttributeNS(null, 'id', 'middle');\n            this._middle.setAttributeNS(\n                null,\n                'd',\n                'M395.225,277.325c-6.8,0-13.5-2.6-18.7-7.8c-71.4-71.3-187.4-71.3-258.8,0c-10.3,10.3-27.1,10.3-37.4,0s-10.3-27.1,0-37.4c92-92,241.6-92,333.6,0c10.3,10.3,10.3,27.1,0,37.4C408.725,274.725,401.925,277.325,395.225,277.325z'\n            );\n        }\n        return this._middle;\n    }\n\n    /**\n     * Get the inner arc of the wifi icon.\n     */\n    public get inner(): SVGElement {\n        if (!this._inner) {\n            this._inner = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n            this._inner.setAttributeNS(null, 'id', 'inner');\n            this._inner.setAttributeNS(\n                null,\n                'd',\n                'M323.625,348.825c-6.8,0-13.5-2.6-18.7-7.8c-15.4-15.4-36-23.9-57.8-23.9s-42.4,8.5-57.8,23.9c-10.3,10.3-27.1,10.3-37.4,0c-10.3-10.3-10.3-27.1,0-37.4c25.4-25.4,59.2-39.4,95.2-39.4s69.8,14,95.2,39.5c10.3,10.3,10.3,27.1,0,37.4C337.225,346.225,330.425,348.825,323.625,348.825z'\n            );\n        }\n        return this._inner;\n    }\n\n    /**\n     * Used to set the speed of the status light.\n     * @param speed - Set the speed of the blink, higher numbers make the status light blink faster.\n     */\n    blinkVideoQualityStatus(speed: number) {\n        let iteration = speed;\n        let opacity = 1;\n        const tickID = setInterval(() => {\n            opacity -= 0.1;\n            this.qualityText.style.opacity = String(\n                Math.abs((opacity - 0.5) * 2)\n            );\n            if (opacity <= 0.1) {\n                if (--iteration == 0) {\n                    clearInterval(tickID);\n                } else {\n                    opacity = 1;\n                }\n            }\n        }, 100 / speed);\n    }\n\n    /**\n     * updates the QP tooltip by converting the Video Encoder QP to a color light\n     * @param QP - The video encoder QP number needed to find the average\n     */\n    updateQpTooltip(QP: number) {\n        this.videoEncoderAvgQP = QP;\n        if (QP > this.redQP) {\n            this.color = 'red';\n            this.blinkVideoQualityStatus(2);\n            this.statsText = `<div style=\"color: ${this.color}\">Poor encoding quality</div>`;\n            this.outer.setAttributeNS(null, 'fill', '#3c3b40');\n            this.middle.setAttributeNS(null, 'fill', '#3c3b40');\n            this.inner.setAttributeNS(null, 'fill', this.color);\n            this.dot.setAttributeNS(null, 'fill', this.color);\n        } else if (QP > this.orangeQP) {\n            this.color = 'orange';\n            this.blinkVideoQualityStatus(1);\n            this.statsText = `<div style=\"color: ${this.color}\">Blocky encoding quality</div>`;\n            this.outer.setAttributeNS(null, 'fill', '#3c3b40');\n            this.middle.setAttributeNS(null, 'fill', this.color);\n            this.inner.setAttributeNS(null, 'fill', this.color);\n            this.dot.setAttributeNS(null, 'fill', this.color);\n        } else if (QP <= 0) {\n            this.color = '#b0b0b0';\n            this.outer.setAttributeNS(null, 'fill', '#3c3b40');\n            this.middle.setAttributeNS(null, 'fill', '#3c3b40');\n            this.inner.setAttributeNS(null, 'fill', '#3c3b40');\n            this.dot.setAttributeNS(null, 'fill', '#3c3b40');\n            this.statsText = `<div style=\"color: ${this.color}\">Not connected</div>`;\n        } else {\n            this.color = 'lime';\n            this.qualityStatus.style.opacity = '1';\n            this.statsText = `<div style=\"color: ${this.color}\">Clear encoding quality</div>`;\n            this.outer.setAttributeNS(null, 'fill', this.color);\n            this.middle.setAttributeNS(null, 'fill', this.color);\n            this.inner.setAttributeNS(null, 'fill', this.color);\n            this.dot.setAttributeNS(null, 'fill', this.color);\n        }\n        this.qualityText.innerHTML = this.statsText;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\n/**\n * XR icon that can be clicked.\n */\nexport class XRIcon {\n    _rootElement: HTMLButtonElement;\n    _xrIcon: SVGElement;\n    _tooltipText: HTMLElement;\n\n    /**\n     * Get the the button containing the XR icon.\n     */\n    public get rootElement(): HTMLButtonElement {\n        if (!this._rootElement) {\n            this._rootElement = document.createElement('button');\n            this._rootElement.type = 'button';\n            this._rootElement.classList.add('UiTool');\n            this._rootElement.id = 'xrBtn';\n            this._rootElement.appendChild(this.xrIcon);\n            this._rootElement.appendChild(this.tooltipText);\n        }\n        return this._rootElement;\n    }\n\n    public get tooltipText(): HTMLElement {\n        if (!this._tooltipText) {\n            this._tooltipText = document.createElement('span');\n            this._tooltipText.classList.add('tooltiptext');\n            this._tooltipText.innerHTML = 'XR';\n        }\n        return this._tooltipText;\n    }\n\n    public get xrIcon(): SVGElement {\n        if (!this._xrIcon) {\n            this._xrIcon = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'svg'\n            );\n            this._xrIcon.setAttributeNS(null, 'id', 'xrIcon');\n            this._xrIcon.setAttributeNS(null, 'x', '0px');\n            this._xrIcon.setAttributeNS(null, 'y', '0px');\n            this._xrIcon.setAttributeNS(null, 'viewBox', '0 0 100 100');\n\n            // create svg group for the paths\n            const svgGroup = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'g'\n            );\n            svgGroup.classList.add('svgIcon');\n            this._xrIcon.appendChild(svgGroup);\n\n            // create paths for the icon itself, the path of the xr headset\n            const path = document.createElementNS(\n                'http://www.w3.org/2000/svg',\n                'path'\n            );\n\n            path.setAttributeNS(\n                null,\n                'd',\n                'M29 41c-5 0-9 4-9 9s4 9 9 9 9-4 9-9-4-9-9-9zm0 14c-2.8 0-5-2.2-5-5s2.2-5 5-5 5 2.2 5 5-2.2 5-5 5zm42-14c-5 0-9 4-9 9s4 9 9 9 9-4 9-9-4-9-9-9zm0 14c-2.8 0-5-2.2-5-5s2.2-5 5-5 5 2.2 5 5-2.2 5-5 5zm12-31H17c-6.6 0-12 5.4-12 12v28c0 6.6 5.4 12 12 12h14.5c3.5 0 6.8-1.5 9-4.1l3.5-4c1.5-1.7 3.7-2.7 6-2.7s4.5 1 6 2.7l3.5 4c2.3 2.6 5.6 4.1 9 4.1H83c6.6 0 12-5.4 12-12V36c0-6.6-5.4-12-12-12zm8 40c0 4.4-3.6 8-8 8H68.5c-2.3 0-4.5-1-6-2.7l-3.5-4c-2.3-2.6-5.6-4.1-9-4.1-3.5 0-6.8 1.5-9 4.1l-3.5 4C36 71 33.8 72 31.5 72H17c-4.4 0-8-3.6-8-8V36c0-4.4 3.6-8 8-8h66c4.4 0 8 3.6 8 8v28z'\n            );\n\n            svgGroup.appendChild(path);\n        }\n        return this._xrIcon;\n    }\n}\n","// Copyright Epic Games, Inc. All Rights Reserved.\n\nexport class MathUtils {\n    /**\n     * formats Bytes coming in for video stats\n     * @param bytes number to convert\n     * @param decimals number of decimal places\n     */\n    static formatBytes(bytes: number, decimals: number): string {\n        if (bytes === 0) {\n            return '0';\n        }\n\n        const factor = 1024;\n        const dm = decimals < 0 ? 0 : decimals;\n        const sizes = [\n            'Bytes',\n            'KiB',\n            'MiB',\n            'GiB',\n            'TiB',\n            'PiB',\n            'EiB',\n            'ZiB',\n            'YiB'\n        ];\n\n        const i = Math.floor(Math.log(bytes) / Math.log(factor));\n\n        return (\n            parseFloat((bytes / Math.pow(factor, i)).toFixed(dm)) +\n            ' ' +\n            sizes[i]\n        );\n    }\n}\n","var x = y => { var x = {}; __webpack_require__.d(x, y); return x; }\nvar y = x => () => x\nmodule.exports = __WEBPACK_EXTERNAL_MODULE__epicgames_ps_lib_pixelstreamingfrontend_ue5_3_512f3c9b__;","var x = y => { var x = {}; __webpack_require__.d(x, y); return x; }\nvar y = x => () => x\nmodule.exports = __WEBPACK_EXTERNAL_MODULE_jss__;","var x = y => { var x = {}; __webpack_require__.d(x, y); return x; }\nvar y = x => () => x\nmodule.exports = __WEBPACK_EXTERNAL_MODULE_jss_plugin_camel_case_de113355__;","var x = y => { var x = {}; __webpack_require__.d(x, y); return x; }\nvar y = x => () => x\nmodule.exports = __WEBPACK_EXTERNAL_MODULE_jss_plugin_global_ef86f421__;","// 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","// 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 { Application, UIOptions, VideoQPIndicatorConfig } from './Application/Application';\n\nexport { PixelStreamingApplicationStyle } from './Styles/PixelStreamingApplicationStyles';\n\nexport { AFKOverlay } from './Overlay/AFKOverlay';\nexport { ActionOverlay } from './Overlay/ActionOverlay';\nexport { OverlayBase } from './Overlay/BaseOverlay';\nexport { ConnectOverlay } from './Overlay/ConnectOverlay';\nexport { DisconnectOverlay } from './Overlay/DisconnectOverlay';\nexport { ErrorOverlay } from './Overlay/ErrorOverlay';\nexport { InfoOverlay } from './Overlay/InfoOverlay';\nexport { PlayOverlay } from './Overlay/PlayOverlay';\nexport { TextOverlay } from './Overlay/TextOverlay';\nexport { ConfigUI } from './Config/ConfigUI';\nexport { SettingUIBase } from './Config/SettingUIBase';\nexport { SettingUIFlag } from './Config/SettingUIFlag';\nexport { SettingUINumber } from './Config/SettingUINumber';\nexport { SettingUIOption } from './Config/SettingUIOption';\nexport { SettingUIText } from './Config/SettingUIText';\nexport { PanelConfiguration, UIElementConfig, UIElementCreationMode } from './UI/UIConfigurationTypes'\n"],"names":[],"sourceRoot":""}