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":""}