mirror of
https://github.com/redphx/better-xcloud.git
synced 2025-06-30 19:31:44 +02:00
Compare commits
59 Commits
Author | SHA1 | Date | |
---|---|---|---|
3a91210ba7 | |||
14f2d8a741 | |||
c24d1620b6 | |||
63f30111cb | |||
d30a628fb1 | |||
5b80170c8b | |||
203346c0a1 | |||
9719454ea1 | |||
59a178bb16 | |||
fd1494ebfa | |||
8e6dec4b70 | |||
6e905621f6 | |||
76b205a65a | |||
af41dc7c5e | |||
d0f43db1fd | |||
eed0aa9d9e | |||
9007663a3a | |||
8f6bc5cb1b | |||
12d8d766dc | |||
aeffccaf67 | |||
b2736d574d | |||
98cf893956 | |||
086afafedf | |||
bd58355ef5 | |||
109cd63a7b | |||
8ea6b7f81a | |||
e7c10d43f5 | |||
2f7a57e084 | |||
c99e38b097 | |||
f6ec6d7c9b | |||
e69fa19ef3 | |||
cc422b31a4 | |||
9609d0ae7b | |||
506fd71433 | |||
f40b8cb0b2 | |||
49a6c036a3 | |||
f5a5a79a82 | |||
7ec449160a | |||
fecc5411da | |||
f704452171 | |||
135193813c | |||
bb57f72e64 | |||
69d7cbfffb | |||
92e6828cb2 | |||
12ad81e9c7 | |||
102e0bd318 | |||
9308963bc2 | |||
c90e013dc1 | |||
037927b9be | |||
dabab9acb1 | |||
a4a52c6bc3 | |||
eebd7434ea | |||
ec1805f832 | |||
34f959d5ae | |||
784a31ce43 | |||
df266d32fc | |||
a6ccd6666e | |||
fe609034d6 | |||
97ec29faa0 |
16
build.sh
Executable file
16
build.sh
Executable file
@ -0,0 +1,16 @@
|
|||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
build_all () {
|
||||||
|
# Clear screen
|
||||||
|
printf "\033c"
|
||||||
|
|
||||||
|
# Build all variants
|
||||||
|
bun build.ts --version $1 --variant full
|
||||||
|
bun build.ts --version $1 --variant lite
|
||||||
|
|
||||||
|
# Wait for key
|
||||||
|
read -p ">> Press Enter to build again..."
|
||||||
|
build_all $1
|
||||||
|
}
|
||||||
|
|
||||||
|
build_all $1
|
91
build.ts
91
build.ts
@ -5,6 +5,8 @@ import { sys } from "typescript";
|
|||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
import txtScriptHeader from "./src/assets/header_script.txt" with { type: "text" };
|
import txtScriptHeader from "./src/assets/header_script.txt" with { type: "text" };
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
|
import txtScriptHeaderLite from "./src/assets/header_script.lite.txt" with { type: "text" };
|
||||||
|
// @ts-ignore
|
||||||
import txtMetaHeader from "./src/assets/header_meta.txt" with { type: "text" };
|
import txtMetaHeader from "./src/assets/header_meta.txt" with { type: "text" };
|
||||||
import { assert } from "node:console";
|
import { assert } from "node:console";
|
||||||
import { ESLint } from "eslint";
|
import { ESLint } from "eslint";
|
||||||
@ -16,6 +18,8 @@ enum BuildTarget {
|
|||||||
WEBOS = 'webos',
|
WEBOS = 'webos',
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type BuildVariant = 'full' | 'lite';
|
||||||
|
|
||||||
const postProcess = (str: string): string => {
|
const postProcess = (str: string): string => {
|
||||||
// Unescape unicode charaters
|
// Unescape unicode charaters
|
||||||
str = unescape((str.replace(/\\u/g, '%u')));
|
str = unescape((str.replace(/\\u/g, '%u')));
|
||||||
@ -35,12 +39,42 @@ const postProcess = (str: string): string => {
|
|||||||
// Add ADDITIONAL CODE block
|
// Add ADDITIONAL CODE block
|
||||||
str = str.replace('var DEFAULT_FLAGS', '\n/* ADDITIONAL CODE */\n\nvar DEFAULT_FLAGS');
|
str = str.replace('var DEFAULT_FLAGS', '\n/* ADDITIONAL CODE */\n\nvar DEFAULT_FLAGS');
|
||||||
|
|
||||||
// Minify SVG
|
str = str.replaceAll('(e) => `', 'e => `');
|
||||||
str = str.replaceAll(/= "(<svg.*)";/g, function(match) {
|
|
||||||
match = match.replaceAll(/\\n*\s*/g, '');
|
// Simplify object definitions
|
||||||
return match;
|
// {[1]: "a"} => {1: "a"}
|
||||||
|
str = str.replaceAll(/\[(\d+)\]: /g, '$1: ');
|
||||||
|
// {["a"]: 1, ["b-c"]: 2} => {a: 1, "b-c": 2}
|
||||||
|
str = str.replaceAll(/\["([^"]+)"\]: /g, function(match, p1) {
|
||||||
|
if (p1.includes('-') || p1.match(/^\d/)) {
|
||||||
|
p1 = `"${p1}"`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return p1 + ': ';
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Minify SVG import code
|
||||||
|
const svgMap = {}
|
||||||
|
str = str.replaceAll(/var ([\w_]+) = ("<svg.*?");\n\n/g, function(match, p1, p2) {
|
||||||
|
// Remove new lines in SVG
|
||||||
|
p2 = p2.replaceAll(/\\n*\s*/g, '');
|
||||||
|
|
||||||
|
svgMap[p1] = p2;
|
||||||
|
return '';
|
||||||
|
});
|
||||||
|
|
||||||
|
for (const name in svgMap) {
|
||||||
|
str = str.replace(`: ${name}`, `: ${svgMap[name]}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Collapse empty brackets
|
||||||
|
str = str.replaceAll(/\{[\s\n]+\}/g, '{}');
|
||||||
|
|
||||||
|
// Collapse if/else blocks without curly braces
|
||||||
|
str = str.replaceAll(/((if \(.*?\)|else)\n\s+)/g, '$2 ');
|
||||||
|
|
||||||
|
// Remove blank lines
|
||||||
|
str = str.replaceAll(/\n([\s]*)\n/g, "\n");
|
||||||
|
|
||||||
assert(str.includes('/* ADDITIONAL CODE */'));
|
assert(str.includes('/* ADDITIONAL CODE */'));
|
||||||
assert(str.includes('window.BX_EXPOSED = BxExposed'));
|
assert(str.includes('window.BX_EXPOSED = BxExposed'));
|
||||||
@ -50,7 +84,7 @@ const postProcess = (str: string): string => {
|
|||||||
return str;
|
return str;
|
||||||
}
|
}
|
||||||
|
|
||||||
const build = async (target: BuildTarget, version: string, config: any={}) => {
|
const build = async (target: BuildTarget, version: string, variant: BuildVariant, config: any={}) => {
|
||||||
console.log('-- Target:', target);
|
console.log('-- Target:', target);
|
||||||
const startTime = performance.now();
|
const startTime = performance.now();
|
||||||
|
|
||||||
@ -58,6 +92,11 @@ const build = async (target: BuildTarget, version: string, config: any={}) => {
|
|||||||
if (target !== BuildTarget.ALL) {
|
if (target !== BuildTarget.ALL) {
|
||||||
outputScriptName += `.${target}`;
|
outputScriptName += `.${target}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (variant !== 'full') {
|
||||||
|
outputScriptName += `.${variant}`;
|
||||||
|
}
|
||||||
|
|
||||||
let outputMetaName = outputScriptName;
|
let outputMetaName = outputScriptName;
|
||||||
outputScriptName += '.user.js';
|
outputScriptName += '.user.js';
|
||||||
outputMetaName += '.meta.js';
|
outputMetaName += '.meta.js';
|
||||||
@ -73,6 +112,7 @@ const build = async (target: BuildTarget, version: string, config: any={}) => {
|
|||||||
},
|
},
|
||||||
define: {
|
define: {
|
||||||
'Bun.env.BUILD_TARGET': JSON.stringify(target),
|
'Bun.env.BUILD_TARGET': JSON.stringify(target),
|
||||||
|
'Bun.env.BUILD_VARIANT': JSON.stringify(variant),
|
||||||
'Bun.env.SCRIPT_VERSION': JSON.stringify(version),
|
'Bun.env.SCRIPT_VERSION': JSON.stringify(version),
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
@ -87,7 +127,13 @@ const build = async (target: BuildTarget, version: string, config: any={}) => {
|
|||||||
let result = postProcess(await readFile(path, 'utf-8'));
|
let result = postProcess(await readFile(path, 'utf-8'));
|
||||||
|
|
||||||
// Replace [[VERSION]] with real value
|
// Replace [[VERSION]] with real value
|
||||||
const scriptHeader = txtScriptHeader.replace('[[VERSION]]', version);
|
let scriptHeader: string;
|
||||||
|
if (variant === 'full') {
|
||||||
|
scriptHeader = txtScriptHeader;
|
||||||
|
} else {
|
||||||
|
scriptHeader = txtScriptHeaderLite;
|
||||||
|
}
|
||||||
|
scriptHeader = scriptHeader.replace('[[VERSION]]', version);
|
||||||
|
|
||||||
// Save to script
|
// Save to script
|
||||||
await Bun.write(path, scriptHeader + result);
|
await Bun.write(path, scriptHeader + result);
|
||||||
@ -118,28 +164,44 @@ const buildTargets = [
|
|||||||
const { values, positionals } = parseArgs({
|
const { values, positionals } = parseArgs({
|
||||||
args: Bun.argv,
|
args: Bun.argv,
|
||||||
options: {
|
options: {
|
||||||
version: {
|
version: {
|
||||||
type: 'string',
|
type: 'string',
|
||||||
|
},
|
||||||
|
|
||||||
},
|
variant: {
|
||||||
|
type: 'string',
|
||||||
|
default: 'full',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
strict: true,
|
strict: true,
|
||||||
allowPositionals: true,
|
allowPositionals: true,
|
||||||
});
|
}) as {
|
||||||
|
values: {
|
||||||
|
version: string,
|
||||||
|
variant: BuildVariant,
|
||||||
|
},
|
||||||
|
positionals: string[],
|
||||||
|
};
|
||||||
|
|
||||||
if (!values['version']) {
|
if (!values['version']) {
|
||||||
console.log('Missing --version param');
|
console.log('Missing --version param');
|
||||||
sys.exit(-1);
|
sys.exit(-1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (values['variant'] !== 'full' && values['variant'] !== 'lite') {
|
||||||
|
console.log('--variant param must be either "full" or "lite"');
|
||||||
|
sys.exit(-1);
|
||||||
|
}
|
||||||
|
|
||||||
async function main() {
|
async function main() {
|
||||||
const config = {};
|
const config = {};
|
||||||
console.log('Building: ', values['version']);
|
console.log(`Building: VERSION=${values['version']}, VARIANT=${values['variant']}`);
|
||||||
for (const target of buildTargets) {
|
for (const target of buildTargets) {
|
||||||
await build(target, values['version']!!, config);
|
await build(target, values['version']!!, values['variant'], config);
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log('\n** Press Enter to build or Esc to exit');
|
console.log('')
|
||||||
|
// console.log('\n** Press Enter to build or Esc to exit');
|
||||||
}
|
}
|
||||||
|
|
||||||
function onKeyPress(data: any) {
|
function onKeyPress(data: any) {
|
||||||
@ -152,6 +214,9 @@ function onKeyPress(data: any) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
main();
|
main();
|
||||||
|
|
||||||
|
/*
|
||||||
process.stdin.setRawMode(true);
|
process.stdin.setRawMode(true);
|
||||||
process.stdin.resume();
|
process.stdin.resume();
|
||||||
process.stdin.on('data', onKeyPress);
|
process.stdin.on('data', onKeyPress);
|
||||||
|
*/
|
||||||
|
5
dist/better-xcloud.lite.meta.js
vendored
Normal file
5
dist/better-xcloud.lite.meta.js
vendored
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
// ==UserScript==
|
||||||
|
// @name Better xCloud
|
||||||
|
// @namespace https://github.com/redphx
|
||||||
|
// @version 5.8.0
|
||||||
|
// ==/UserScript==
|
5637
dist/better-xcloud.lite.user.js
vendored
Normal file
5637
dist/better-xcloud.lite.user.js
vendored
Normal file
File diff suppressed because one or more lines are too long
2
dist/better-xcloud.meta.js
vendored
2
dist/better-xcloud.meta.js
vendored
@ -1,5 +1,5 @@
|
|||||||
// ==UserScript==
|
// ==UserScript==
|
||||||
// @name Better xCloud
|
// @name Better xCloud
|
||||||
// @namespace https://github.com/redphx
|
// @namespace https://github.com/redphx
|
||||||
// @version 5.7.3
|
// @version 5.8.0
|
||||||
// ==/UserScript==
|
// ==/UserScript==
|
||||||
|
5559
dist/better-xcloud.user.js
vendored
5559
dist/better-xcloud.user.js
vendored
File diff suppressed because one or more lines are too long
13
package.json
13
package.json
@ -2,6 +2,7 @@
|
|||||||
"name": "better-xcloud",
|
"name": "better-xcloud",
|
||||||
"module": "src/index.ts",
|
"module": "src/index.ts",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
"sideEffects": false,
|
||||||
"browserslist": [
|
"browserslist": [
|
||||||
"Chrome >= 80"
|
"Chrome >= 80"
|
||||||
],
|
],
|
||||||
@ -9,14 +10,14 @@
|
|||||||
"build": "build.ts"
|
"build": "build.ts"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/bun": "^1.1.8",
|
"@types/bun": "^1.1.10",
|
||||||
"@types/node": "^22.5.2",
|
"@types/node": "^22.7.4",
|
||||||
"@types/stylus": "^0.48.42",
|
"@types/stylus": "^0.48.43",
|
||||||
"eslint": "^9.9.1",
|
"eslint": "^9.12.0",
|
||||||
"eslint-plugin-compat": "^6.0.0",
|
"eslint-plugin-compat": "^6.0.1",
|
||||||
"stylus": "^0.63.0"
|
"stylus": "^0.63.0"
|
||||||
},
|
},
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"typescript": "^5.5.4"
|
"typescript": "^5.6.2"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -179,15 +179,3 @@ button.bx-inactive {
|
|||||||
opacity: 0.2;
|
opacity: 0.2;
|
||||||
background: transparent !important;
|
background: transparent !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.bx-button-shortcut {
|
|
||||||
max-width: max-content;
|
|
||||||
margin: 10px 0 0 0;
|
|
||||||
flex: 1 0 auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (min-width: 568px) and (max-height: 480px) {
|
|
||||||
.bx-button-shortcut {
|
|
||||||
margin: 8px 0 0 10px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
21
src/assets/css/misc.styl
Normal file
21
src/assets/css/misc.styl
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
.bx-product-details-buttons {
|
||||||
|
display: flex;
|
||||||
|
gap: 10px;
|
||||||
|
flex-direction: row;
|
||||||
|
|
||||||
|
button {
|
||||||
|
max-width: max-content;
|
||||||
|
margin: 10px 0 0 0;
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 568px) and (max-height: 480px) {
|
||||||
|
.bx-product-details-buttons {
|
||||||
|
flex-direction: column;
|
||||||
|
|
||||||
|
button {
|
||||||
|
margin: 8px 0 0 10px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,6 +1,11 @@
|
|||||||
.bx-navigation-dialog {
|
.bx-navigation-dialog {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
z-index: var(--bx-navigation-dialog-z-index);
|
z-index: var(--bx-navigation-dialog-z-index);
|
||||||
|
font-family: var(--bx-title-font);
|
||||||
|
|
||||||
|
*:focus {
|
||||||
|
outline: none !important;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.bx-navigation-dialog-overlay {
|
.bx-navigation-dialog-overlay {
|
||||||
|
@ -1,36 +1,16 @@
|
|||||||
.bx-remote-play-popup {
|
|
||||||
width: 100%;
|
|
||||||
max-width: 1920px;
|
|
||||||
margin: auto;
|
|
||||||
position: relative;
|
|
||||||
height: 0.1px;
|
|
||||||
overflow: visible;
|
|
||||||
z-index: var(--bx-remote-play-popup-z-index);
|
|
||||||
}
|
|
||||||
|
|
||||||
.bx-remote-play-container {
|
.bx-remote-play-container {
|
||||||
position: absolute;
|
position: fixed;
|
||||||
right: 10px;
|
top: 50%;
|
||||||
top: 0;
|
left: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
|
||||||
|
color: white;
|
||||||
background: #1a1b1e;
|
background: #1a1b1e;
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
width: 420px;
|
width: 420px;
|
||||||
max-width: calc(100vw - 20px);
|
max-width: calc(100vw - 20px);
|
||||||
margin: 0 0 0 auto;
|
margin: 0 0 0 auto;
|
||||||
padding: 20px;
|
padding: 20px;
|
||||||
box-shadow: #00000080 0px 0px 12px 0px;
|
|
||||||
|
|
||||||
@media (min-width:480px) and (min-height:calc(480px + 1px)) {
|
|
||||||
right: calc(env(safe-area-inset-right, 0px) + 32px);
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (min-width:768px) and (min-height:calc(480px + 1px)) {
|
|
||||||
right: calc(env(safe-area-inset-right, 0px) + 48px);
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (min-width:1920px) and (min-height:calc(480px + 1px)) {
|
|
||||||
right: calc(env(safe-area-inset-right, 0px) + 80px);
|
|
||||||
}
|
|
||||||
|
|
||||||
> .bx-button {
|
> .bx-button {
|
||||||
display: table;
|
display: table;
|
||||||
@ -57,14 +37,6 @@
|
|||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
span {
|
|
||||||
font-weight: bold;
|
|
||||||
font-size: 18px;
|
|
||||||
display: block;
|
|
||||||
margin-bottom: 8px;
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.bx-remote-play-resolution {
|
.bx-remote-play-resolution {
|
||||||
@ -114,10 +86,15 @@
|
|||||||
|
|
||||||
.bx-remote-play-power-state {
|
.bx-remote-play-power-state {
|
||||||
color: #888;
|
color: #888;
|
||||||
font-size: 14px;
|
font-size: 12px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.bx-remote-play-connect-button {
|
.bx-remote-play-connect-button {
|
||||||
min-height: 100%;
|
min-height: 100%;
|
||||||
margin: 4px 0;
|
margin: 4px 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.bx-remote-play-buttons {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
@ -37,8 +37,6 @@ button_color(name, normal, hover, active, disabled)
|
|||||||
--bx-navigation-dialog-z-index: 30100;
|
--bx-navigation-dialog-z-index: 30100;
|
||||||
--bx-navigation-dialog-overlay-z-index: 30000;
|
--bx-navigation-dialog-overlay-z-index: 30000;
|
||||||
|
|
||||||
--bx-remote-play-popup-z-index: 20000;
|
|
||||||
|
|
||||||
--bx-game-bar-z-index: 10000;
|
--bx-game-bar-z-index: 10000;
|
||||||
--bx-screenshot-animation-z-index: 9000;
|
--bx-screenshot-animation-z-index: 9000;
|
||||||
--bx-wait-time-box-z-index: 1000;
|
--bx-wait-time-box-z-index: 1000;
|
||||||
|
@ -130,7 +130,6 @@
|
|||||||
|
|
||||||
&:focus {
|
&:focus {
|
||||||
border-color: #fff;
|
border-color: #fff;
|
||||||
outline: none;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
&[data-group=global] {
|
&[data-group=global] {
|
||||||
@ -234,11 +233,6 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
&:focus,
|
|
||||||
*:focus {
|
|
||||||
outline: none !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.bx-top-buttons {
|
.bx-top-buttons {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
@ -306,6 +300,7 @@
|
|||||||
text-align: left;
|
text-align: left;
|
||||||
align-self: center;
|
align-self: center;
|
||||||
margin-bottom: 0 !important;
|
margin-bottom: 0 !important;
|
||||||
|
flex: 1;
|
||||||
|
|
||||||
+ * {
|
+ * {
|
||||||
margin: 0 0 0 auto;
|
margin: 0 0 0 auto;
|
||||||
|
@ -71,7 +71,9 @@ div[class^=StreamMenu-module__container] .bx-badges {
|
|||||||
|
|
||||||
/* STATS BAR */
|
/* STATS BAR */
|
||||||
.bx-stats-bar {
|
.bx-stats-bar {
|
||||||
display: block;
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
gap: 8px;
|
||||||
user-select: none;
|
user-select: none;
|
||||||
-webkit-user-select: none;
|
-webkit-user-select: none;
|
||||||
position: fixed;
|
position: fixed;
|
||||||
@ -84,22 +86,32 @@ div[class^=StreamMenu-module__container] .bx-badges {
|
|||||||
z-index: var(--bx-stats-bar-z-index);
|
z-index: var(--bx-stats-bar-z-index);
|
||||||
text-wrap: nowrap;
|
text-wrap: nowrap;
|
||||||
|
|
||||||
|
&[data-stats*="[time]"] > .bx-stat-time,
|
||||||
|
&[data-stats*="[play]"] > .bx-stat-play,
|
||||||
|
&[data-stats*="[batt]"] > .bx-stat-batt,
|
||||||
&[data-stats*="[fps]"] > .bx-stat-fps,
|
&[data-stats*="[fps]"] > .bx-stat-fps,
|
||||||
&[data-stats*="[ping]"] > .bx-stat-ping,
|
&[data-stats*="[ping]"] > .bx-stat-ping,
|
||||||
&[data-stats*="[btr]"] > .bx-stat-btr,
|
&[data-stats*="[btr]"] > .bx-stat-btr,
|
||||||
&[data-stats*="[dt]"] > .bx-stat-dt,
|
&[data-stats*="[dt]"] > .bx-stat-dt,
|
||||||
&[data-stats*="[pl]"] > .bx-stat-pl,
|
&[data-stats*="[pl]"] > .bx-stat-pl,
|
||||||
&[data-stats*="[fl]"] > .bx-stat-fl {
|
&[data-stats*="[fl]"] > .bx-stat-fl,
|
||||||
display: inline-block;
|
&[data-stats*="[dl]"] > .bx-stat-dl,
|
||||||
|
&[data-stats*="[ul]"] > .bx-stat-ul {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: baseline;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&[data-stats$="[time]"] > .bx-stat-time,
|
||||||
|
&[data-stats$="[play]"] > .bx-stat-play,
|
||||||
|
&[data-stats$="[batt]"] > .bx-stat-batt,
|
||||||
&[data-stats$="[fps]"] > .bx-stat-fps,
|
&[data-stats$="[fps]"] > .bx-stat-fps,
|
||||||
&[data-stats$="[ping]"] > .bx-stat-ping,
|
&[data-stats$="[ping]"] > .bx-stat-ping,
|
||||||
&[data-stats$="[btr]"] > .bx-stat-btr,
|
&[data-stats$="[btr]"] > .bx-stat-btr,
|
||||||
&[data-stats$="[dt]"] > .bx-stat-dt,
|
&[data-stats$="[dt]"] > .bx-stat-dt,
|
||||||
&[data-stats$="[pl]"] > .bx-stat-pl,
|
&[data-stats$="[pl]"] > .bx-stat-pl,
|
||||||
&[data-stats$="[fl]"] > .bx-stat-fl {
|
&[data-stats$="[fl]"] > .bx-stat-fl,
|
||||||
margin-right: 0;
|
&[data-stats$="[dl]"] > .bx-stat-dl,
|
||||||
|
&[data-stats$="[ul]"] > .bx-stat-ul {
|
||||||
border-right: none;
|
border-right: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -137,7 +149,6 @@ div[class^=StreamMenu-module__container] .bx-badges {
|
|||||||
|
|
||||||
> div {
|
> div {
|
||||||
display: none;
|
display: none;
|
||||||
margin-right: 8px;
|
|
||||||
border-right: 1px solid #fff;
|
border-right: 1px solid #fff;
|
||||||
padding-right: 8px;
|
padding-right: 8px;
|
||||||
}
|
}
|
||||||
@ -145,7 +156,7 @@ div[class^=StreamMenu-module__container] .bx-badges {
|
|||||||
label {
|
label {
|
||||||
margin: 0 8px 0 0;
|
margin: 0 8px 0 0;
|
||||||
font-family: var(--bx-title-font);
|
font-family: var(--bx-title-font);
|
||||||
font-size: inherit;
|
font-size: 70%;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
cursor: help;
|
cursor: help;
|
||||||
|
@ -16,3 +16,4 @@
|
|||||||
@import 'game-bar.styl';
|
@import 'game-bar.styl';
|
||||||
@import 'stream-stats.styl';
|
@import 'stream-stats.styl';
|
||||||
@import 'mkb.styl';
|
@import 'mkb.styl';
|
||||||
|
@import 'misc.styl';
|
||||||
|
@ -4,12 +4,16 @@
|
|||||||
flex: 0 1 auto;
|
flex: 0 1 auto;
|
||||||
|
|
||||||
select {
|
select {
|
||||||
display: none !important;
|
// Render offscreen instead of "display: none" so we could get its size
|
||||||
|
position: absolute !important;
|
||||||
|
top: -9999px !important;
|
||||||
|
left: -9999px !important;
|
||||||
|
visibility: hidden !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
> div, button.bx-select-value {
|
> div, button.bx-select-value {
|
||||||
min-width: 110px;
|
min-width: 120px;
|
||||||
text-align: center;
|
text-align: left;
|
||||||
margin: 0 8px;
|
margin: 0 8px;
|
||||||
line-height: 24px;
|
line-height: 24px;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
@ -53,13 +57,14 @@
|
|||||||
|
|
||||||
span {
|
span {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
text-align: center;
|
text-align: left;
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
}
|
}
|
||||||
|
|
||||||
input {
|
input {
|
||||||
margin: 0 4px;
|
margin: 0 4px;
|
||||||
accent-color: var(--bx-primary-button-color);
|
accent-color: var(--bx-primary-button-color);
|
||||||
|
pointer-events: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
&:hover,
|
&:hover,
|
||||||
|
13
src/assets/header_script.lite.txt
Normal file
13
src/assets/header_script.lite.txt
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
// ==UserScript==
|
||||||
|
// @name Better xCloud (Lite)
|
||||||
|
// @namespace https://github.com/redphx
|
||||||
|
// @version [[VERSION]]
|
||||||
|
// @description Improve Xbox Cloud Gaming (xCloud) experience
|
||||||
|
// @author redphx
|
||||||
|
// @license MIT
|
||||||
|
// @match https://www.xbox.com/*/play*
|
||||||
|
// @match https://www.xbox.com/*/auth/msa?*loggedIn*
|
||||||
|
// @run-at document-end
|
||||||
|
// @grant none
|
||||||
|
// ==/UserScript==
|
||||||
|
"use strict";
|
121
src/index.ts
121
src/index.ts
@ -1,3 +1,5 @@
|
|||||||
|
import { compressCss, isFullVersion } from "@macros/build" with {type: "macro"};
|
||||||
|
|
||||||
import "@utils/global";
|
import "@utils/global";
|
||||||
import { BxEvent } from "@utils/bx-event";
|
import { BxEvent } from "@utils/bx-event";
|
||||||
import { BX_FLAGS } from "@utils/bx-flags";
|
import { BX_FLAGS } from "@utils/bx-flags";
|
||||||
@ -16,7 +18,7 @@ import { MouseCursorHider } from "@modules/mkb/mouse-cursor-hider";
|
|||||||
import { TouchController } from "@modules/touch-controller";
|
import { TouchController } from "@modules/touch-controller";
|
||||||
import { checkForUpdate, disablePwa, productTitleToSlug } from "@utils/utils";
|
import { checkForUpdate, disablePwa, productTitleToSlug } from "@utils/utils";
|
||||||
import { Patcher } from "@modules/patcher";
|
import { Patcher } from "@modules/patcher";
|
||||||
import { RemotePlay } from "@modules/remote-play";
|
import { RemotePlayManager } from "@/modules/remote-play-manager";
|
||||||
import { onHistoryChanged, patchHistoryMethod } from "@utils/history";
|
import { onHistoryChanged, patchHistoryMethod } from "@utils/history";
|
||||||
import { VibrationManager } from "@modules/vibration-manager";
|
import { VibrationManager } from "@modules/vibration-manager";
|
||||||
import { overridePreloadState } from "@utils/preload-state";
|
import { overridePreloadState } from "@utils/preload-state";
|
||||||
@ -35,12 +37,11 @@ import { ProductDetailsPage } from "./modules/ui/product-details";
|
|||||||
import { NavigationDialogManager } from "./modules/ui/dialog/navigation-dialog";
|
import { NavigationDialogManager } from "./modules/ui/dialog/navigation-dialog";
|
||||||
import { PrefKey } from "./enums/pref-keys";
|
import { PrefKey } from "./enums/pref-keys";
|
||||||
import { getPref, StreamTouchController } from "./utils/settings-storages/global-settings-storage";
|
import { getPref, StreamTouchController } from "./utils/settings-storages/global-settings-storage";
|
||||||
import { compressCss } from "@macros/build" with {type: "macro"};
|
|
||||||
import { SettingsNavigationDialog } from "./modules/ui/dialog/settings-dialog";
|
import { SettingsNavigationDialog } from "./modules/ui/dialog/settings-dialog";
|
||||||
import { StreamUiHandler } from "./modules/stream/stream-ui";
|
import { StreamUiHandler } from "./modules/stream/stream-ui";
|
||||||
import { UserAgent } from "./utils/user-agent";
|
import { UserAgent } from "./utils/user-agent";
|
||||||
import { XboxApi } from "./utils/xbox-api";
|
import { XboxApi } from "./utils/xbox-api";
|
||||||
|
import { StreamStatsCollector } from "./utils/stream-stats-collector";
|
||||||
|
|
||||||
// Handle login page
|
// Handle login page
|
||||||
if (window.location.pathname.includes('/auth/msa')) {
|
if (window.location.pathname.includes('/auth/msa')) {
|
||||||
@ -63,12 +64,14 @@ if (window.location.pathname.includes('/auth/msa')) {
|
|||||||
|
|
||||||
BxLogger.info('readyState', document.readyState);
|
BxLogger.info('readyState', document.readyState);
|
||||||
|
|
||||||
if (BX_FLAGS.SafariWorkaround && document.readyState !== 'loading') {
|
if (isFullVersion() && BX_FLAGS.SafariWorkaround && document.readyState !== 'loading') {
|
||||||
// Stop loading
|
// Stop loading
|
||||||
window.stop();
|
window.stop();
|
||||||
|
|
||||||
// Show the reloading overlay
|
// We need to set it to an empty string first to work around Bun's bug
|
||||||
const css = compressCss(`
|
// https://github.com/oven-sh/bun/issues/12067
|
||||||
|
let css = '';
|
||||||
|
css += compressCss(`
|
||||||
.bx-reload-overlay {
|
.bx-reload-overlay {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
top: 0;
|
top: 0;
|
||||||
@ -115,6 +118,7 @@ if (BX_FLAGS.SafariWorkaround && document.readyState !== 'loading') {
|
|||||||
}, '🤓 ' + t('how-to-fix'));
|
}, '🤓 ' + t('how-to-fix'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Show the reloading overlay
|
||||||
const $fragment = document.createDocumentFragment();
|
const $fragment = document.createDocumentFragment();
|
||||||
$fragment.appendChild(CE('style', {}, css));
|
$fragment.appendChild(CE('style', {}, css));
|
||||||
$fragment.appendChild(CE('div',{
|
$fragment.appendChild(CE('div',{
|
||||||
@ -157,7 +161,7 @@ document.addEventListener('readystatechange', e => {
|
|||||||
|
|
||||||
if (STATES.isSignedIn) {
|
if (STATES.isSignedIn) {
|
||||||
// Preload Remote Play
|
// Preload Remote Play
|
||||||
getPref(PrefKey.REMOTE_PLAY_ENABLED) && RemotePlay.preload();
|
getPref(PrefKey.REMOTE_PLAY_ENABLED) && RemotePlayManager.getInstance().initialize();
|
||||||
} else {
|
} else {
|
||||||
// Show Settings button in the header when not signed in
|
// Show Settings button in the header when not signed in
|
||||||
window.setTimeout(HeaderSection.watchHeader, 2000);
|
window.setTimeout(HeaderSection.watchHeader, 2000);
|
||||||
@ -189,8 +193,11 @@ window.addEventListener(BxEvent.XCLOUD_SERVERS_UNAVAILABLE, e => {
|
|||||||
window.setTimeout(HeaderSection.watchHeader, 2000);
|
window.setTimeout(HeaderSection.watchHeader, 2000);
|
||||||
|
|
||||||
// Open Settings dialog on Unsupported page
|
// Open Settings dialog on Unsupported page
|
||||||
SettingsNavigationDialog.getInstance().show();
|
const $unsupportedPage = document.querySelector('div[class^=UnsupportedMarketPage-module__container]') as HTMLElement;
|
||||||
});
|
if ($unsupportedPage) {
|
||||||
|
SettingsNavigationDialog.getInstance().show();
|
||||||
|
}
|
||||||
|
}, {once: true});
|
||||||
|
|
||||||
window.addEventListener(BxEvent.XCLOUD_SERVERS_READY, e => {
|
window.addEventListener(BxEvent.XCLOUD_SERVERS_READY, e => {
|
||||||
STATES.isSignedIn = true;
|
STATES.isSignedIn = true;
|
||||||
@ -224,15 +231,17 @@ window.addEventListener(BxEvent.STREAM_PLAYING, e => {
|
|||||||
STATES.isPlaying = true;
|
STATES.isPlaying = true;
|
||||||
StreamUiHandler.observe();
|
StreamUiHandler.observe();
|
||||||
|
|
||||||
if (getPref(PrefKey.GAME_BAR_POSITION) !== 'off') {
|
if (isFullVersion() && getPref(PrefKey.GAME_BAR_POSITION) !== 'off') {
|
||||||
const gameBar = GameBar.getInstance();
|
const gameBar = GameBar.getInstance();
|
||||||
gameBar.reset();
|
gameBar.reset();
|
||||||
gameBar.enable();
|
gameBar.enable();
|
||||||
gameBar.showBar();
|
gameBar.showBar();
|
||||||
}
|
}
|
||||||
|
|
||||||
const $video = (e as any).$video as HTMLVideoElement;
|
if (isFullVersion()) {
|
||||||
Screenshot.updateCanvasSize($video.videoWidth, $video.videoHeight);
|
const $video = (e as any).$video as HTMLVideoElement;
|
||||||
|
Screenshot.updateCanvasSize($video.videoWidth, $video.videoHeight);
|
||||||
|
}
|
||||||
|
|
||||||
updateVideoPlayer();
|
updateVideoPlayer();
|
||||||
});
|
});
|
||||||
@ -285,9 +294,11 @@ function unload() {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Stop MKB listeners
|
if (isFullVersion()) {
|
||||||
EmulatedMkbHandler.getInstance().destroy();
|
// Stop MKB listeners
|
||||||
NativeMkbHandler.getInstance().destroy();
|
EmulatedMkbHandler.getInstance().destroy();
|
||||||
|
NativeMkbHandler.getInstance().destroy();
|
||||||
|
}
|
||||||
|
|
||||||
// Destroy StreamPlayer
|
// Destroy StreamPlayer
|
||||||
STATES.currentStream.streamPlayer?.destroy();
|
STATES.currentStream.streamPlayer?.destroy();
|
||||||
@ -300,9 +311,11 @@ function unload() {
|
|||||||
NavigationDialogManager.getInstance().hide();
|
NavigationDialogManager.getInstance().hide();
|
||||||
StreamStats.getInstance().onStoppedPlaying();
|
StreamStats.getInstance().onStoppedPlaying();
|
||||||
|
|
||||||
MouseCursorHider.stop();
|
if (isFullVersion()) {
|
||||||
TouchController.reset();
|
MouseCursorHider.stop();
|
||||||
GameBar.getInstance().disable();
|
TouchController.reset();
|
||||||
|
GameBar.getInstance().disable();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
window.addEventListener(BxEvent.STREAM_STOPPED, unload);
|
window.addEventListener(BxEvent.STREAM_STOPPED, unload);
|
||||||
@ -310,7 +323,7 @@ window.addEventListener('pagehide', e => {
|
|||||||
BxEvent.dispatch(window, BxEvent.STREAM_STOPPED);
|
BxEvent.dispatch(window, BxEvent.STREAM_STOPPED);
|
||||||
});
|
});
|
||||||
|
|
||||||
window.addEventListener(BxEvent.CAPTURE_SCREENSHOT, e => {
|
isFullVersion() && window.addEventListener(BxEvent.CAPTURE_SCREENSHOT, e => {
|
||||||
Screenshot.takeScreenshot();
|
Screenshot.takeScreenshot();
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -365,15 +378,13 @@ function waitForRootDialog() {
|
|||||||
|
|
||||||
|
|
||||||
function main() {
|
function main() {
|
||||||
waitForRootDialog();
|
|
||||||
|
|
||||||
// Monkey patches
|
// Monkey patches
|
||||||
patchRtcPeerConnection();
|
patchRtcPeerConnection();
|
||||||
patchRtcCodecs();
|
patchRtcCodecs();
|
||||||
interceptHttpRequests();
|
interceptHttpRequests();
|
||||||
patchVideoApi();
|
patchVideoApi();
|
||||||
patchCanvasContext();
|
patchCanvasContext();
|
||||||
AppInterface && patchPointerLockApi();
|
isFullVersion() && AppInterface && patchPointerLockApi();
|
||||||
|
|
||||||
getPref(PrefKey.AUDIO_ENABLE_VOLUME_CONTROL) && patchAudioContext();
|
getPref(PrefKey.AUDIO_ENABLE_VOLUME_CONTROL) && patchAudioContext();
|
||||||
|
|
||||||
@ -382,52 +393,58 @@ function main() {
|
|||||||
disableAdobeAudienceManager();
|
disableAdobeAudienceManager();
|
||||||
}
|
}
|
||||||
|
|
||||||
STATES.userAgent.capabilities.touch && TouchController.updateCustomList();
|
waitForRootDialog();
|
||||||
overridePreloadState();
|
|
||||||
|
|
||||||
VibrationManager.initialSetup();
|
|
||||||
|
|
||||||
// Check for Update
|
|
||||||
BX_FLAGS.CheckForUpdate && checkForUpdate();
|
|
||||||
|
|
||||||
// Setup UI
|
// Setup UI
|
||||||
addCss();
|
addCss();
|
||||||
Toast.setup();
|
Toast.setup();
|
||||||
(getPref(PrefKey.GAME_BAR_POSITION) !== 'off') && GameBar.getInstance();
|
|
||||||
Screenshot.setup();
|
|
||||||
|
|
||||||
GuideMenu.addEventListeners();
|
GuideMenu.addEventListeners();
|
||||||
|
StreamStatsCollector.setupEvents();
|
||||||
StreamBadges.setupEvents();
|
StreamBadges.setupEvents();
|
||||||
StreamStats.setupEvents();
|
StreamStats.setupEvents();
|
||||||
EmulatedMkbHandler.setupEvents();
|
|
||||||
|
|
||||||
Patcher.init();
|
if (isFullVersion()) {
|
||||||
|
(getPref(PrefKey.GAME_BAR_POSITION) !== 'off') && GameBar.getInstance();
|
||||||
|
Screenshot.setup();
|
||||||
|
|
||||||
disablePwa();
|
STATES.userAgent.capabilities.touch && TouchController.updateCustomList();
|
||||||
|
overridePreloadState();
|
||||||
|
|
||||||
|
VibrationManager.initialSetup();
|
||||||
|
|
||||||
|
// Check for Update
|
||||||
|
BX_FLAGS.CheckForUpdate && checkForUpdate();
|
||||||
|
|
||||||
|
Patcher.init();
|
||||||
|
disablePwa();
|
||||||
|
|
||||||
|
// Preload Remote Play
|
||||||
|
if (getPref(PrefKey.REMOTE_PLAY_ENABLED)) {
|
||||||
|
RemotePlayManager.detect();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === StreamTouchController.ALL) {
|
||||||
|
TouchController.setup();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start PointerProviderServer
|
||||||
|
if (getPref(PrefKey.MKB_ENABLED) && AppInterface) {
|
||||||
|
STATES.pointerServerPort = AppInterface.startPointerServer() || 9269;
|
||||||
|
BxLogger.info('startPointerServer', 'Port', STATES.pointerServerPort.toString());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Show wait time in game card
|
||||||
|
getPref(PrefKey.UI_GAME_CARD_SHOW_WAIT_TIME) && GameTile.setup();
|
||||||
|
|
||||||
|
EmulatedMkbHandler.setupEvents();
|
||||||
|
}
|
||||||
|
|
||||||
// Show a toast when connecting/disconecting controller
|
// Show a toast when connecting/disconecting controller
|
||||||
if (getPref(PrefKey.CONTROLLER_SHOW_CONNECTION_STATUS)) {
|
if (getPref(PrefKey.CONTROLLER_SHOW_CONNECTION_STATUS)) {
|
||||||
window.addEventListener('gamepadconnected', e => showGamepadToast(e.gamepad));
|
window.addEventListener('gamepadconnected', e => showGamepadToast(e.gamepad));
|
||||||
window.addEventListener('gamepaddisconnected', e => showGamepadToast(e.gamepad));
|
window.addEventListener('gamepaddisconnected', e => showGamepadToast(e.gamepad));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Preload Remote Play
|
|
||||||
if (getPref(PrefKey.REMOTE_PLAY_ENABLED)) {
|
|
||||||
RemotePlay.detect();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === StreamTouchController.ALL) {
|
|
||||||
TouchController.setup();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Start PointerProviderServer
|
|
||||||
if (getPref(PrefKey.MKB_ENABLED) && AppInterface) {
|
|
||||||
STATES.pointerServerPort = AppInterface.startPointerServer() || 9269;
|
|
||||||
BxLogger.info('startPointerServer', 'Port', STATES.pointerServerPort.toString());
|
|
||||||
}
|
|
||||||
|
|
||||||
// Show wait time in game card
|
|
||||||
getPref(PrefKey.UI_GAME_CARD_SHOW_WAIT_TIME) && GameTile.setup();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
main();
|
main();
|
||||||
|
@ -1,5 +1,13 @@
|
|||||||
import stylus from 'stylus';
|
import stylus from 'stylus';
|
||||||
|
|
||||||
|
export const isFullVersion = () => {
|
||||||
|
return Bun.env.BUILD_VARIANT === 'full';
|
||||||
|
};
|
||||||
|
|
||||||
|
export const isLiteVersion = () => {
|
||||||
|
return Bun.env.BUILD_VARIANT === 'lite';
|
||||||
|
};
|
||||||
|
|
||||||
export const renderStylus = async () => {
|
export const renderStylus = async () => {
|
||||||
const file = Bun.file('./src/assets/css/styles.styl');
|
const file = Bun.file('./src/assets/css/styles.styl');
|
||||||
const cssStr = await file.text();
|
const cssStr = await file.text();
|
||||||
|
@ -3,7 +3,6 @@ import { GamepadKey } from "@enums/mkb";
|
|||||||
import { PrompFont } from "@enums/prompt-font";
|
import { PrompFont } from "@enums/prompt-font";
|
||||||
import { CE, removeChildElements } from "@utils/html";
|
import { CE, removeChildElements } from "@utils/html";
|
||||||
import { t } from "@utils/translation";
|
import { t } from "@utils/translation";
|
||||||
import { EmulatedMkbHandler } from "./mkb/mkb-handler";
|
|
||||||
import { StreamStats } from "./stream/stream-stats";
|
import { StreamStats } from "./stream/stream-stats";
|
||||||
import { MicrophoneShortcut } from "./shortcuts/shortcut-microphone";
|
import { MicrophoneShortcut } from "./shortcuts/shortcut-microphone";
|
||||||
import { StreamUiShortcut } from "./shortcuts/shortcut-stream-ui";
|
import { StreamUiShortcut } from "./shortcuts/shortcut-stream-ui";
|
||||||
@ -15,6 +14,7 @@ import { setNearby } from "@/utils/navigation-utils";
|
|||||||
import { PrefKey } from "@/enums/pref-keys";
|
import { PrefKey } from "@/enums/pref-keys";
|
||||||
import { getPref } from "@/utils/settings-storages/global-settings-storage";
|
import { getPref } from "@/utils/settings-storages/global-settings-storage";
|
||||||
import { SettingsNavigationDialog } from "./ui/dialog/settings-dialog";
|
import { SettingsNavigationDialog } from "./ui/dialog/settings-dialog";
|
||||||
|
import { VIRTUAL_GAMEPAD_ID } from "./mkb/mkb-handler";
|
||||||
|
|
||||||
const enum ShortcutAction {
|
const enum ShortcutAction {
|
||||||
BETTER_XCLOUD_SETTINGS_SHOW = 'bx-settings-show',
|
BETTER_XCLOUD_SETTINGS_SHOW = 'bx-settings-show',
|
||||||
@ -185,7 +185,7 @@ export class ControllerShortcut {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Ignore emulated gamepad
|
// Ignore emulated gamepad
|
||||||
if (gamepad.id === EmulatedMkbHandler.VIRTUAL_GAMEPAD_ID) {
|
if (gamepad.id === VIRTUAL_GAMEPAD_ID) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -314,6 +314,7 @@ export class ControllerShortcut {
|
|||||||
const $selectProfile = CE<HTMLSelectElement>('select', {class: 'bx-shortcut-profile', autocomplete: 'off'});
|
const $selectProfile = CE<HTMLSelectElement>('select', {class: 'bx-shortcut-profile', autocomplete: 'off'});
|
||||||
|
|
||||||
const $profile = PREF_CONTROLLER_FRIENDLY_UI ? BxSelectElement.wrap($selectProfile) : $selectProfile;
|
const $profile = PREF_CONTROLLER_FRIENDLY_UI ? BxSelectElement.wrap($selectProfile) : $selectProfile;
|
||||||
|
$profile.classList.add('bx-full-width');
|
||||||
|
|
||||||
const $container = CE('div', {
|
const $container = CE('div', {
|
||||||
'data-has-gamepad': 'false',
|
'data-has-gamepad': 'false',
|
||||||
@ -390,6 +391,8 @@ export class ControllerShortcut {
|
|||||||
|
|
||||||
if (PREF_CONTROLLER_FRIENDLY_UI) {
|
if (PREF_CONTROLLER_FRIENDLY_UI) {
|
||||||
const $bxSelect = BxSelectElement.wrap($select);
|
const $bxSelect = BxSelectElement.wrap($select);
|
||||||
|
$bxSelect.classList.add('bx-full-width');
|
||||||
|
|
||||||
$div.appendChild($bxSelect);
|
$div.appendChild($bxSelect);
|
||||||
setNearby($row, {
|
setNearby($row, {
|
||||||
focus: $bxSelect,
|
focus: $bxSelect,
|
||||||
|
@ -44,7 +44,7 @@ export class LoadingScreen {
|
|||||||
static #hideRocket() {
|
static #hideRocket() {
|
||||||
let $bgStyle = LoadingScreen.#$bgStyle;
|
let $bgStyle = LoadingScreen.#$bgStyle;
|
||||||
|
|
||||||
const css = compressCss(`
|
$bgStyle.textContent! += compressCss(`
|
||||||
#game-stream div[class*=RocketAnimation-module__container] > svg {
|
#game-stream div[class*=RocketAnimation-module__container] > svg {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
@ -53,7 +53,6 @@ export class LoadingScreen {
|
|||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
`);
|
`);
|
||||||
$bgStyle.textContent! += css;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static #setBackground(imageUrl: string) {
|
static #setBackground(imageUrl: string) {
|
||||||
@ -63,7 +62,7 @@ export class LoadingScreen {
|
|||||||
// Limit max width to reduce image size
|
// Limit max width to reduce image size
|
||||||
imageUrl = imageUrl + '?w=1920';
|
imageUrl = imageUrl + '?w=1920';
|
||||||
|
|
||||||
const css = compressCss(`
|
$bgStyle.textContent! += compressCss(`
|
||||||
#game-stream {
|
#game-stream {
|
||||||
background-color: transparent !important;
|
background-color: transparent !important;
|
||||||
background-position: center center !important;
|
background-position: center center !important;
|
||||||
@ -75,7 +74,6 @@ export class LoadingScreen {
|
|||||||
transition: opacity 0.3s ease-in-out !important;
|
transition: opacity 0.3s ease-in-out !important;
|
||||||
}
|
}
|
||||||
`) + `#game-stream {background-image: linear-gradient(#00000033, #000000e6), url(${imageUrl}) !important;}`;
|
`) + `#game-stream {background-image: linear-gradient(#00000033, #000000e6), url(${imageUrl}) !important;}`;
|
||||||
$bgStyle.textContent! += css;
|
|
||||||
|
|
||||||
const bg = new Image();
|
const bg = new Image();
|
||||||
bg.onload = e => {
|
bg.onload = e => {
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
import { isFullVersion } from "@macros/build" with {type: "macro"};
|
||||||
|
|
||||||
import { MkbPreset } from "./mkb-preset";
|
import { MkbPreset } from "./mkb-preset";
|
||||||
import { GamepadKey, MkbPresetKey, GamepadStick, MouseMapTo, WheelCode } from "@enums/mkb";
|
import { GamepadKey, MkbPresetKey, GamepadStick, MouseMapTo, WheelCode } from "@enums/mkb";
|
||||||
import { createButton, ButtonStyle, CE } from "@utils/html";
|
import { createButton, ButtonStyle, CE } from "@utils/html";
|
||||||
@ -26,6 +28,7 @@ const PointerToMouseButton = {
|
|||||||
4: 1,
|
4: 1,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const VIRTUAL_GAMEPAD_ID = 'Xbox 360 Controller';
|
||||||
|
|
||||||
class WebSocketMouseDataProvider extends MouseDataProvider {
|
class WebSocketMouseDataProvider extends MouseDataProvider {
|
||||||
#pointerClient: PointerClient | undefined
|
#pointerClient: PointerClient | undefined
|
||||||
@ -136,10 +139,8 @@ export class EmulatedMkbHandler extends MkbHandler {
|
|||||||
static readonly DEFAULT_DEADZONE_COUNTERWEIGHT = 0.01;
|
static readonly DEFAULT_DEADZONE_COUNTERWEIGHT = 0.01;
|
||||||
static readonly MAXIMUM_STICK_RANGE = 1.1;
|
static readonly MAXIMUM_STICK_RANGE = 1.1;
|
||||||
|
|
||||||
static VIRTUAL_GAMEPAD_ID = 'Xbox 360 Controller';
|
|
||||||
|
|
||||||
#VIRTUAL_GAMEPAD = {
|
#VIRTUAL_GAMEPAD = {
|
||||||
id: EmulatedMkbHandler.VIRTUAL_GAMEPAD_ID,
|
id: VIRTUAL_GAMEPAD_ID,
|
||||||
index: 3,
|
index: 3,
|
||||||
connected: false,
|
connected: false,
|
||||||
hapticActuators: null,
|
hapticActuators: null,
|
||||||
@ -678,7 +679,7 @@ export class EmulatedMkbHandler extends MkbHandler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static setupEvents() {
|
static setupEvents() {
|
||||||
window.addEventListener(BxEvent.STREAM_PLAYING, () => {
|
isFullVersion() && window.addEventListener(BxEvent.STREAM_PLAYING, () => {
|
||||||
if (STATES.currentStream.titleInfo?.details.hasMkbSupport) {
|
if (STATES.currentStream.titleInfo?.details.hasMkbSupport) {
|
||||||
// Enable native MKB in Android app
|
// Enable native MKB in Android app
|
||||||
if (AppInterface && getPref(PrefKey.NATIVE_MKB_ENABLED) === 'on') {
|
if (AppInterface && getPref(PrefKey.NATIVE_MKB_ENABLED) === 'on') {
|
||||||
|
@ -54,7 +54,7 @@ const LOG_TAG = 'Patcher';
|
|||||||
const PATCHES = {
|
const PATCHES = {
|
||||||
// Disable ApplicationInsights.track() function
|
// Disable ApplicationInsights.track() function
|
||||||
disableAiTrack(str: string) {
|
disableAiTrack(str: string) {
|
||||||
const text = '.track=function(';
|
let text = '.track=function(';
|
||||||
const index = str.indexOf(text);
|
const index = str.indexOf(text);
|
||||||
if (index < 0) {
|
if (index < 0) {
|
||||||
return false;
|
return false;
|
||||||
@ -69,7 +69,7 @@ const PATCHES = {
|
|||||||
|
|
||||||
// Set disableTelemetry() to true
|
// Set disableTelemetry() to true
|
||||||
disableTelemetry(str: string) {
|
disableTelemetry(str: string) {
|
||||||
const text = '.disableTelemetry=function(){return!1}';
|
let text = '.disableTelemetry=function(){return!1}';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -78,7 +78,7 @@ const PATCHES = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
disableTelemetryProvider(str: string) {
|
disableTelemetryProvider(str: string) {
|
||||||
const text = 'this.enableLightweightTelemetry=!';
|
let text = 'this.enableLightweightTelemetry=!';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -99,7 +99,7 @@ const PATCHES = {
|
|||||||
|
|
||||||
// Disable IndexDB logging
|
// Disable IndexDB logging
|
||||||
disableIndexDbLogging(str: string) {
|
disableIndexDbLogging(str: string) {
|
||||||
const text = ',this.logsDb=new';
|
let text = ',this.logsDb=new';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -111,7 +111,7 @@ const PATCHES = {
|
|||||||
|
|
||||||
// Set custom website layout
|
// Set custom website layout
|
||||||
websiteLayout(str: string) {
|
websiteLayout(str: string) {
|
||||||
const text = '?"tv":"default"';
|
let text = '?"tv":"default"';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -131,7 +131,7 @@ const PATCHES = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
remotePlayKeepAlive(str: string) {
|
remotePlayKeepAlive(str: string) {
|
||||||
const text = 'onServerDisconnectMessage(e){';
|
let text = 'onServerDisconnectMessage(e){';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -143,7 +143,7 @@ const PATCHES = {
|
|||||||
|
|
||||||
// Enable Remote Play feature
|
// Enable Remote Play feature
|
||||||
remotePlayConnectMode(str: string) {
|
remotePlayConnectMode(str: string) {
|
||||||
const text = 'connectMode:"cloud-connect",';
|
let text = 'connectMode:"cloud-connect",';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -151,25 +151,44 @@ const PATCHES = {
|
|||||||
return str.replace(text, codeRemotePlayEnable);
|
return str.replace(text, codeRemotePlayEnable);
|
||||||
},
|
},
|
||||||
|
|
||||||
// Disable achievement toast in Remote Play
|
// Remote Play: Disable achievement toast
|
||||||
remotePlayDisableAchievementToast(str: string) {
|
remotePlayDisableAchievementToast(str: string) {
|
||||||
const text = '.AchievementUnlock:{';
|
let text = '.AchievementUnlock:{';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
const newCode = `
|
const newCode = `if (!!window.BX_REMOTE_PLAY_CONFIG) return;`;
|
||||||
if (!!window.BX_REMOTE_PLAY_CONFIG) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
return str.replace(text, text + newCode);
|
return str.replace(text, text + newCode);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
// Remote Play: Prevent adding "Fortnite" to the "Jump back in" list
|
||||||
|
remotePlayRecentlyUsedTitleIds(str: string) {
|
||||||
|
let text = '(e.data.recentlyUsedTitleIds)){';
|
||||||
|
if (!str.includes(text)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const newCode = `if (window.BX_REMOTE_PLAY_CONFIG) return;`;
|
||||||
|
return str.replace(text, text + newCode);
|
||||||
|
},
|
||||||
|
|
||||||
|
// Remote Play: change web page's title
|
||||||
|
/*
|
||||||
|
remotePlayWebTitle(str: string) {
|
||||||
|
let text = '"undefined"!==typeof e&&document.title!==e';
|
||||||
|
if (!str.includes(text)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const newCode = `if (window.BX_REMOTE_PLAY_CONFIG) { e = "${t('remote-play')} - ${t('better-xcloud')}"; }`;
|
||||||
|
return str.replace(text, newCode + text);
|
||||||
|
},
|
||||||
|
*/
|
||||||
|
|
||||||
// Block WebRTC stats collector
|
// Block WebRTC stats collector
|
||||||
blockWebRtcStatsCollector(str: string) {
|
blockWebRtcStatsCollector(str: string) {
|
||||||
const text = 'this.shouldCollectStats=!0';
|
let text = 'this.shouldCollectStats=!0';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -210,7 +229,7 @@ if (!!window.BX_REMOTE_PLAY_CONFIG) {
|
|||||||
},
|
},
|
||||||
|
|
||||||
enableXcloudLogger(str: string) {
|
enableXcloudLogger(str: string) {
|
||||||
const text = 'this.telemetryProvider=e}log(e,t,r){';
|
let text = 'this.telemetryProvider=e}log(e,t,r){';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -226,7 +245,7 @@ logFunc(logTag, '//', logMessage);
|
|||||||
},
|
},
|
||||||
|
|
||||||
enableConsoleLogging(str: string) {
|
enableConsoleLogging(str: string) {
|
||||||
const text = 'static isConsoleLoggingAllowed(){';
|
let text = 'static isConsoleLoggingAllowed(){';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -237,7 +256,7 @@ logFunc(logTag, '//', logMessage);
|
|||||||
|
|
||||||
// Control controller vibration
|
// Control controller vibration
|
||||||
playVibration(str: string) {
|
playVibration(str: string) {
|
||||||
const text = '}playVibration(e){';
|
let text = '}playVibration(e){';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -278,7 +297,7 @@ logFunc(logTag, '//', logMessage);
|
|||||||
},
|
},
|
||||||
|
|
||||||
patchUpdateInputConfigurationAsync(str: string) {
|
patchUpdateInputConfigurationAsync(str: string) {
|
||||||
const text = 'async updateInputConfigurationAsync(e){';
|
let text = 'async updateInputConfigurationAsync(e){';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -291,7 +310,7 @@ logFunc(logTag, '//', logMessage);
|
|||||||
|
|
||||||
// Add patches that are only needed when start playing
|
// Add patches that are only needed when start playing
|
||||||
loadingEndingChunks(str: string) {
|
loadingEndingChunks(str: string) {
|
||||||
const text = '"FamilySagaManager"';
|
let text = '"FamilySagaManager"';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -316,7 +335,7 @@ logFunc(logTag, '//', logMessage);
|
|||||||
},
|
},
|
||||||
|
|
||||||
exposeTouchLayoutManager(str: string) {
|
exposeTouchLayoutManager(str: string) {
|
||||||
const text = 'this._perScopeLayoutsStream=new';
|
let text = 'this._perScopeLayoutsStream=new';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -363,7 +382,7 @@ if (window.BX_EXPOSED.stopTakRendering) {
|
|||||||
},
|
},
|
||||||
|
|
||||||
supportLocalCoOp(str: string) {
|
supportLocalCoOp(str: string) {
|
||||||
const text = 'this.gamepadMappingsToSend=[],';
|
let text = 'this.gamepadMappingsToSend=[],';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -375,7 +394,7 @@ if (window.BX_EXPOSED.stopTakRendering) {
|
|||||||
},
|
},
|
||||||
|
|
||||||
forceFortniteConsole(str: string) {
|
forceFortniteConsole(str: string) {
|
||||||
const text = 'sendTouchInputEnabledMessage(e){';
|
let text = 'sendTouchInputEnabledMessage(e){';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -387,7 +406,7 @@ if (window.BX_EXPOSED.stopTakRendering) {
|
|||||||
},
|
},
|
||||||
|
|
||||||
disableTakRenderer(str: string) {
|
disableTakRenderer(str: string) {
|
||||||
const text = 'const{TakRenderer:';
|
let text = 'const{TakRenderer:';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -427,7 +446,7 @@ if (titleInfo && !titleInfo.details.hasTouchSupport && !titleInfo.details.hasFak
|
|||||||
},
|
},
|
||||||
|
|
||||||
streamCombineSources(str: string) {
|
streamCombineSources(str: string) {
|
||||||
const text = 'this.useCombinedAudioVideoStream=!!this.deviceInformation.isTizen';
|
let text = 'this.useCombinedAudioVideoStream=!!this.deviceInformation.isTizen';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -437,7 +456,7 @@ if (titleInfo && !titleInfo.details.hasTouchSupport && !titleInfo.details.hasFak
|
|||||||
},
|
},
|
||||||
|
|
||||||
patchStreamHud(str: string) {
|
patchStreamHud(str: string) {
|
||||||
const text = 'let{onCollapse';
|
let text = 'let{onCollapse';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -459,7 +478,7 @@ e.guideUI = null;
|
|||||||
},
|
},
|
||||||
|
|
||||||
broadcastPollingMode(str: string) {
|
broadcastPollingMode(str: string) {
|
||||||
const text = '.setPollingMode=e=>{';
|
let text = '.setPollingMode=e=>{';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -483,7 +502,7 @@ BxEvent.dispatch(window, BxEvent.XCLOUD_POLLING_MODE_CHANGED, {mode: e.toLowerCa
|
|||||||
},
|
},
|
||||||
|
|
||||||
patchXcloudTitleInfo(str: string) {
|
patchXcloudTitleInfo(str: string) {
|
||||||
const text = 'async cloudConnect';
|
let text = 'async cloudConnect';
|
||||||
let index = str.indexOf(text);
|
let index = str.indexOf(text);
|
||||||
if (index < 0) {
|
if (index < 0) {
|
||||||
return false;
|
return false;
|
||||||
@ -505,7 +524,7 @@ BxLogger.info('patchXcloudTitleInfo', ${titleInfoVar});
|
|||||||
},
|
},
|
||||||
|
|
||||||
patchRemotePlayMkb(str: string) {
|
patchRemotePlayMkb(str: string) {
|
||||||
const text = 'async homeConsoleConnect';
|
let text = 'async homeConsoleConnect';
|
||||||
let index = str.indexOf(text);
|
let index = str.indexOf(text);
|
||||||
if (index < 0) {
|
if (index < 0) {
|
||||||
return false;
|
return false;
|
||||||
@ -533,7 +552,7 @@ BxLogger.info('patchRemotePlayMkb', ${configsVar});
|
|||||||
},
|
},
|
||||||
|
|
||||||
patchAudioMediaStream(str: string) {
|
patchAudioMediaStream(str: string) {
|
||||||
const text = '.srcObject=this.audioMediaStream,';
|
let text = '.srcObject=this.audioMediaStream,';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -545,7 +564,7 @@ BxLogger.info('patchRemotePlayMkb', ${configsVar});
|
|||||||
},
|
},
|
||||||
|
|
||||||
patchCombinedAudioVideoMediaStream(str: string) {
|
patchCombinedAudioVideoMediaStream(str: string) {
|
||||||
const text = '.srcObject=this.combinedAudioVideoStream';
|
let text = '.srcObject=this.combinedAudioVideoStream';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -556,7 +575,7 @@ BxLogger.info('patchRemotePlayMkb', ${configsVar});
|
|||||||
},
|
},
|
||||||
|
|
||||||
patchTouchControlDefaultOpacity(str: string) {
|
patchTouchControlDefaultOpacity(str: string) {
|
||||||
const text = 'opacityMultiplier:1';
|
let text = 'opacityMultiplier:1';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -568,7 +587,7 @@ BxLogger.info('patchRemotePlayMkb', ${configsVar});
|
|||||||
},
|
},
|
||||||
|
|
||||||
patchShowSensorControls(str: string) {
|
patchShowSensorControls(str: string) {
|
||||||
const text = '{shouldShowSensorControls:';
|
let text = '{shouldShowSensorControls:';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -581,7 +600,7 @@ BxLogger.info('patchRemotePlayMkb', ${configsVar});
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
exposeEventTarget(str: string) {
|
exposeEventTarget(str: string) {
|
||||||
const text ='this._eventTarget=new EventTarget';
|
let text ='this._eventTarget=new EventTarget';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -598,7 +617,7 @@ window.dispatchEvent(new Event('${BxEvent.STREAM_EVENT_TARGET_READY}'))
|
|||||||
|
|
||||||
// Class with: connectAsync(), doConnectAsync(), setPlayClient()
|
// Class with: connectAsync(), doConnectAsync(), setPlayClient()
|
||||||
exposeStreamSession(str: string) {
|
exposeStreamSession(str: string) {
|
||||||
const text =',this._connectionType=';
|
let text =',this._connectionType=';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -612,7 +631,7 @@ true` + text;
|
|||||||
},
|
},
|
||||||
|
|
||||||
skipFeedbackDialog(str: string) {
|
skipFeedbackDialog(str: string) {
|
||||||
const text = '&&this.shouldTransitionToFeedback(';
|
let text = '&&this.shouldTransitionToFeedback(';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -622,7 +641,7 @@ true` + text;
|
|||||||
},
|
},
|
||||||
|
|
||||||
enableNativeMkb(str: string) {
|
enableNativeMkb(str: string) {
|
||||||
const text = 'e.mouseSupported&&e.keyboardSupported&&e.fullscreenSupported;';
|
let text = 'e.mouseSupported&&e.keyboardSupported&&e.fullscreenSupported;';
|
||||||
if ((!str.includes(text))) {
|
if ((!str.includes(text))) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -632,7 +651,7 @@ true` + text;
|
|||||||
},
|
},
|
||||||
|
|
||||||
patchMouseAndKeyboardEnabled(str: string) {
|
patchMouseAndKeyboardEnabled(str: string) {
|
||||||
const text = 'get mouseAndKeyboardEnabled(){';
|
let text = 'get mouseAndKeyboardEnabled(){';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -642,7 +661,7 @@ true` + text;
|
|||||||
},
|
},
|
||||||
|
|
||||||
exposeInputSink(str: string) {
|
exposeInputSink(str: string) {
|
||||||
const text = 'this.controlChannel=null,this.inputChannel=null';
|
let text = 'this.controlChannel=null,this.inputChannel=null';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -654,7 +673,7 @@ true` + text;
|
|||||||
},
|
},
|
||||||
|
|
||||||
disableNativeRequestPointerLock(str: string) {
|
disableNativeRequestPointerLock(str: string) {
|
||||||
const text = 'async requestPointerLock(){';
|
let text = 'async requestPointerLock(){';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -665,7 +684,7 @@ true` + text;
|
|||||||
|
|
||||||
// Fix crashing when RequestInfo.origin is empty
|
// Fix crashing when RequestInfo.origin is empty
|
||||||
patchRequestInfoCrash(str: string) {
|
patchRequestInfoCrash(str: string) {
|
||||||
const text = 'if(!e)throw new Error("RequestInfo.origin is falsy");';
|
let text = 'if(!e)throw new Error("RequestInfo.origin is falsy");';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -675,7 +694,7 @@ true` + text;
|
|||||||
},
|
},
|
||||||
|
|
||||||
exposeDialogRoutes(str: string) {
|
exposeDialogRoutes(str: string) {
|
||||||
const text = 'return{goBack:function(){';
|
let text = 'return{goBack:function(){';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -830,7 +849,7 @@ if (e && e.id) {
|
|||||||
|
|
||||||
// Override Storage.getSettings()
|
// Override Storage.getSettings()
|
||||||
overrideStorageGetSettings(str: string) {
|
overrideStorageGetSettings(str: string) {
|
||||||
const text = '}getSetting(e){';
|
let text = '}getSetting(e){';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -894,7 +913,7 @@ if (this.baseStorageKey in window.BX_EXPOSED.overrideSettings) {
|
|||||||
},
|
},
|
||||||
|
|
||||||
detectBrowserRouterReady(str: string) {
|
detectBrowserRouterReady(str: string) {
|
||||||
const text = 'BrowserRouter:()=>';
|
let text = 'BrowserRouter:()=>';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -984,6 +1003,7 @@ let PATCH_ORDERS: PatchArray = [
|
|||||||
'remotePlayKeepAlive',
|
'remotePlayKeepAlive',
|
||||||
'remotePlayDirectConnectUrl',
|
'remotePlayDirectConnectUrl',
|
||||||
'remotePlayDisableAchievementToast',
|
'remotePlayDisableAchievementToast',
|
||||||
|
'remotePlayRecentlyUsedTitleIds',
|
||||||
STATES.userAgent.capabilities.touch && 'patchUpdateInputConfigurationAsync',
|
STATES.userAgent.capabilities.touch && 'patchUpdateInputConfigurationAsync',
|
||||||
] : []),
|
] : []),
|
||||||
|
|
||||||
|
232
src/modules/remote-play-manager.ts
Normal file
232
src/modules/remote-play-manager.ts
Normal file
@ -0,0 +1,232 @@
|
|||||||
|
import { STATES, AppInterface } from "@utils/global";
|
||||||
|
import { Toast } from "@utils/toast";
|
||||||
|
import { BxEvent } from "@utils/bx-event";
|
||||||
|
import { t } from "@utils/translation";
|
||||||
|
import { localRedirect } from "@modules/ui/ui";
|
||||||
|
import { BxLogger } from "@utils/bx-logger";
|
||||||
|
import { HeaderSection } from "./ui/header";
|
||||||
|
import { PrefKey } from "@/enums/pref-keys";
|
||||||
|
import { getPref, setPref } from "@/utils/settings-storages/global-settings-storage";
|
||||||
|
import { RemotePlayNavigationDialog } from "./ui/dialog/remote-play-dialog";
|
||||||
|
|
||||||
|
const LOG_TAG = 'RemotePlay';
|
||||||
|
|
||||||
|
export const enum RemotePlayConsoleState {
|
||||||
|
ON = 'On',
|
||||||
|
OFF = 'Off',
|
||||||
|
STANDBY = 'ConnectedStandby',
|
||||||
|
UNKNOWN = 'Unknown',
|
||||||
|
}
|
||||||
|
|
||||||
|
type RemotePlayRegion = {
|
||||||
|
name: string;
|
||||||
|
baseUri: string;
|
||||||
|
isDefault: boolean;
|
||||||
|
};
|
||||||
|
|
||||||
|
type RemotePlayConsole = {
|
||||||
|
deviceName: string;
|
||||||
|
serverId: string;
|
||||||
|
powerState: RemotePlayConsoleState;
|
||||||
|
consoleType: string;
|
||||||
|
// playPath: string;
|
||||||
|
// outOfHomeWarning: string;
|
||||||
|
// wirelessWarning: string;
|
||||||
|
// isDevKit: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export class RemotePlayManager {
|
||||||
|
private static instance: RemotePlayManager;
|
||||||
|
public static getInstance(): RemotePlayManager {
|
||||||
|
if (!this.instance) {
|
||||||
|
this.instance = new RemotePlayManager();
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
private isInitialized = false;
|
||||||
|
|
||||||
|
private XCLOUD_TOKEN!: string;
|
||||||
|
private XHOME_TOKEN!: string;
|
||||||
|
|
||||||
|
private consoles!: Array<RemotePlayConsole>;
|
||||||
|
private regions: Array<RemotePlayRegion> = [];
|
||||||
|
|
||||||
|
initialize() {
|
||||||
|
if (this.isInitialized) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.isInitialized = true;
|
||||||
|
|
||||||
|
this.getXhomeToken(() => {
|
||||||
|
this.getConsolesList(() => {
|
||||||
|
BxLogger.info(LOG_TAG, 'Consoles', this.consoles);
|
||||||
|
|
||||||
|
STATES.supportedRegion && HeaderSection.showRemotePlayButton();
|
||||||
|
BxEvent.dispatch(window, BxEvent.REMOTE_PLAY_READY);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
get xcloudToken() {
|
||||||
|
return this.XCLOUD_TOKEN;
|
||||||
|
}
|
||||||
|
|
||||||
|
set xcloudToken(token: string) {
|
||||||
|
this.XCLOUD_TOKEN = token;
|
||||||
|
}
|
||||||
|
|
||||||
|
get xhomeToken() {
|
||||||
|
return this.XHOME_TOKEN;
|
||||||
|
}
|
||||||
|
|
||||||
|
getConsoles() {
|
||||||
|
return this.consoles;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private getXhomeToken(callback: any) {
|
||||||
|
if (this.XHOME_TOKEN) {
|
||||||
|
callback();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let GSSV_TOKEN;
|
||||||
|
try {
|
||||||
|
GSSV_TOKEN = JSON.parse(localStorage.getItem('xboxcom_xbl_user_info')!).tokens['http://gssv.xboxlive.com/'].token;
|
||||||
|
} catch (e) {
|
||||||
|
for (let i = 0; i < localStorage.length; i++){
|
||||||
|
const key = localStorage.key(i)!;
|
||||||
|
if (!key.startsWith('Auth.User.')) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const json = JSON.parse(localStorage.getItem(key)!);
|
||||||
|
for (const token of json.tokens) {
|
||||||
|
if (!token.relyingParty.includes('gssv.xboxlive.com')) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
GSSV_TOKEN = token.tokenData.token;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const request = new Request('https://xhome.gssv-play-prod.xboxlive.com/v2/login/user', {
|
||||||
|
method: 'POST',
|
||||||
|
body: JSON.stringify({
|
||||||
|
offeringId: 'xhome',
|
||||||
|
token: GSSV_TOKEN,
|
||||||
|
}),
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json; charset=utf-8',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
fetch(request).then(resp => resp.json())
|
||||||
|
.then(json => {
|
||||||
|
this.regions = json.offeringSettings.regions;
|
||||||
|
this.XHOME_TOKEN = json.gsToken;
|
||||||
|
callback();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async getConsolesList(callback: any) {
|
||||||
|
if (this.consoles) {
|
||||||
|
callback();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const options = {
|
||||||
|
method: 'GET',
|
||||||
|
headers: {
|
||||||
|
'Authorization': `Bearer ${this.XHOME_TOKEN}`,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
// Test servers one by one
|
||||||
|
for (const region of this.regions) {
|
||||||
|
try {
|
||||||
|
const request = new Request(`${region.baseUri}/v6/servers/home?mr=50`, options);
|
||||||
|
const resp = await fetch(request);
|
||||||
|
|
||||||
|
const json = await resp.json();
|
||||||
|
if (json.results.length === 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.consoles = json.results;
|
||||||
|
|
||||||
|
// Store working server
|
||||||
|
STATES.remotePlay.server = region.baseUri;
|
||||||
|
|
||||||
|
break;
|
||||||
|
} catch (e) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
// None of the servers worked
|
||||||
|
if (!STATES.remotePlay.server) {
|
||||||
|
this.consoles = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
|
||||||
|
play(serverId: string, resolution?: string) {
|
||||||
|
if (resolution) {
|
||||||
|
setPref(PrefKey.REMOTE_PLAY_RESOLUTION, resolution);
|
||||||
|
}
|
||||||
|
|
||||||
|
STATES.remotePlay.config = {
|
||||||
|
serverId: serverId,
|
||||||
|
};
|
||||||
|
window.BX_REMOTE_PLAY_CONFIG = STATES.remotePlay.config;
|
||||||
|
|
||||||
|
localRedirect('/launch/fortnite/BT5P2X999VH2#remote-play');
|
||||||
|
}
|
||||||
|
|
||||||
|
togglePopup(force = null) {
|
||||||
|
if (!this.isReady()) {
|
||||||
|
Toast.show(t('getting-consoles-list'));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.consoles.length === 0) {
|
||||||
|
Toast.show(t('no-consoles-found'), '', {instant: true});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Show native dialog in Android app
|
||||||
|
if (AppInterface && AppInterface.showRemotePlayDialog) {
|
||||||
|
AppInterface.showRemotePlayDialog(JSON.stringify(this.consoles));
|
||||||
|
(document.activeElement as HTMLElement).blur();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
RemotePlayNavigationDialog.getInstance().show();
|
||||||
|
}
|
||||||
|
|
||||||
|
static detect() {
|
||||||
|
if (!getPref(PrefKey.REMOTE_PLAY_ENABLED)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
STATES.remotePlay.isPlaying = window.location.pathname.includes('/launch/') && window.location.hash.startsWith('#remote-play');
|
||||||
|
if (STATES.remotePlay?.isPlaying) {
|
||||||
|
window.BX_REMOTE_PLAY_CONFIG = STATES.remotePlay.config;
|
||||||
|
// Remove /launch/... from URL
|
||||||
|
window.history.replaceState({origin: 'better-xcloud'}, '', 'https://www.xbox.com/' + location.pathname.substring(1, 6) + '/play');
|
||||||
|
} else {
|
||||||
|
window.BX_REMOTE_PLAY_CONFIG = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
isReady() {
|
||||||
|
return this.consoles !== null;
|
||||||
|
}
|
||||||
|
}
|
@ -1,364 +0,0 @@
|
|||||||
import { STATES, AppInterface } from "@utils/global";
|
|
||||||
import { CE, createButton, ButtonStyle } from "@utils/html";
|
|
||||||
import { BxIcon } from "@utils/bx-icon";
|
|
||||||
import { Toast } from "@utils/toast";
|
|
||||||
import { BxEvent } from "@utils/bx-event";
|
|
||||||
import { t } from "@utils/translation";
|
|
||||||
import { localRedirect } from "@modules/ui/ui";
|
|
||||||
import { BxLogger } from "@utils/bx-logger";
|
|
||||||
import { HeaderSection } from "./ui/header";
|
|
||||||
import { PrefKey } from "@/enums/pref-keys";
|
|
||||||
import { getPref, setPref } from "@/utils/settings-storages/global-settings-storage";
|
|
||||||
|
|
||||||
const LOG_TAG = 'RemotePlay';
|
|
||||||
|
|
||||||
const enum RemotePlayConsoleState {
|
|
||||||
ON = 'On',
|
|
||||||
OFF = 'Off',
|
|
||||||
STANDBY = 'ConnectedStandby',
|
|
||||||
UNKNOWN = 'Unknown',
|
|
||||||
}
|
|
||||||
|
|
||||||
type RemotePlayRegion = {
|
|
||||||
name: string;
|
|
||||||
baseUri: string;
|
|
||||||
isDefault: boolean;
|
|
||||||
};
|
|
||||||
|
|
||||||
type RemotePlayConsole = {
|
|
||||||
deviceName: string;
|
|
||||||
serverId: string;
|
|
||||||
powerState: RemotePlayConsoleState;
|
|
||||||
consoleType: string;
|
|
||||||
// playPath: string;
|
|
||||||
// outOfHomeWarning: string;
|
|
||||||
// wirelessWarning: string;
|
|
||||||
// isDevKit: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
export class RemotePlay {
|
|
||||||
static XCLOUD_TOKEN: string;
|
|
||||||
static XHOME_TOKEN: string;
|
|
||||||
static #CONSOLES: Array<RemotePlayConsole>;
|
|
||||||
static #REGIONS: Array<RemotePlayRegion>;
|
|
||||||
|
|
||||||
static readonly #STATE_LABELS: {[key in RemotePlayConsoleState]: string} = {
|
|
||||||
[RemotePlayConsoleState.ON]: t('powered-on'),
|
|
||||||
[RemotePlayConsoleState.OFF]: t('powered-off'),
|
|
||||||
[RemotePlayConsoleState.STANDBY]: t('standby'),
|
|
||||||
[RemotePlayConsoleState.UNKNOWN]: t('unknown'),
|
|
||||||
};
|
|
||||||
|
|
||||||
static readonly BASE_DEVICE_INFO = {
|
|
||||||
appInfo: {
|
|
||||||
env: {
|
|
||||||
clientAppId: window.location.host,
|
|
||||||
clientAppType: 'browser',
|
|
||||||
clientAppVersion: '24.17.36',
|
|
||||||
clientSdkVersion: '10.1.14',
|
|
||||||
httpEnvironment: 'prod',
|
|
||||||
sdkInstallId: '',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
dev: {
|
|
||||||
displayInfo: {
|
|
||||||
dimensions: {
|
|
||||||
widthInPixels: 1920,
|
|
||||||
heightInPixels: 1080,
|
|
||||||
},
|
|
||||||
pixelDensity: {
|
|
||||||
dpiX: 1,
|
|
||||||
dpiY: 1,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
hw: {
|
|
||||||
make: 'Microsoft',
|
|
||||||
model: 'unknown',
|
|
||||||
sdktype: 'web',
|
|
||||||
},
|
|
||||||
os: {
|
|
||||||
name: 'windows',
|
|
||||||
ver: '22631.2715',
|
|
||||||
platform: 'desktop',
|
|
||||||
},
|
|
||||||
browser: {
|
|
||||||
browserName: 'chrome',
|
|
||||||
browserVersion: '125.0',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
static #$content: HTMLElement;
|
|
||||||
|
|
||||||
static #initialize() {
|
|
||||||
if (RemotePlay.#$content) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
RemotePlay.#$content = CE('div', {}, t('getting-consoles-list'));
|
|
||||||
RemotePlay.#getXhomeToken(() => {
|
|
||||||
RemotePlay.#getConsolesList(() => {
|
|
||||||
BxLogger.info(LOG_TAG, 'Consoles', RemotePlay.#CONSOLES);
|
|
||||||
if (RemotePlay.#CONSOLES && RemotePlay.#CONSOLES.length > 0) {
|
|
||||||
STATES.supportedRegion && HeaderSection.showRemotePlayButton();
|
|
||||||
}
|
|
||||||
|
|
||||||
RemotePlay.#renderConsoles();
|
|
||||||
BxEvent.dispatch(window, BxEvent.REMOTE_PLAY_READY);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
static #renderConsoles() {
|
|
||||||
const $fragment = CE('div', {'class': 'bx-remote-play-container'});
|
|
||||||
|
|
||||||
if (!RemotePlay.#CONSOLES || RemotePlay.#CONSOLES.length === 0) {
|
|
||||||
$fragment.appendChild(CE('span', {}, t('no-consoles-found')));
|
|
||||||
RemotePlay.#$content = CE('div', {}, $fragment);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const $settingNote = CE('p', {});
|
|
||||||
|
|
||||||
const resolutions = [1080, 720];
|
|
||||||
const currentResolution = getPref(PrefKey.REMOTE_PLAY_RESOLUTION);
|
|
||||||
const $resolutionGroup = CE('div', {});
|
|
||||||
for (const resolution of resolutions) {
|
|
||||||
const value = `${resolution}p`;
|
|
||||||
const id = `bx_radio_xhome_resolution_${resolution}`;
|
|
||||||
|
|
||||||
const $radio = CE<HTMLInputElement>('input', {
|
|
||||||
'type': 'radio',
|
|
||||||
'value': value,
|
|
||||||
'id': id,
|
|
||||||
'name': 'bx_radio_xhome_resolution',
|
|
||||||
}, value);
|
|
||||||
|
|
||||||
$radio.addEventListener('change', e => {
|
|
||||||
const value = (e.target as HTMLInputElement).value;
|
|
||||||
|
|
||||||
$settingNote.textContent = value === '1080p' ? '✅ ' + t('can-stream-xbox-360-games') : '❌ ' + t('cant-stream-xbox-360-games');
|
|
||||||
setPref(PrefKey.REMOTE_PLAY_RESOLUTION, value);
|
|
||||||
});
|
|
||||||
|
|
||||||
const $label = CE('label', {
|
|
||||||
'for': id,
|
|
||||||
'class': 'bx-remote-play-resolution',
|
|
||||||
}, $radio, `${resolution}p`);
|
|
||||||
|
|
||||||
$resolutionGroup.appendChild($label);
|
|
||||||
|
|
||||||
if (currentResolution === value) {
|
|
||||||
$radio.checked = true;
|
|
||||||
$radio.dispatchEvent(new Event('change'));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const $qualitySettings = CE('div', {'class': 'bx-remote-play-settings'},
|
|
||||||
CE('div', {},
|
|
||||||
CE('label', {}, t('target-resolution'), $settingNote),
|
|
||||||
$resolutionGroup,
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
$fragment.appendChild($qualitySettings);
|
|
||||||
|
|
||||||
// Render concoles list
|
|
||||||
for (let con of RemotePlay.#CONSOLES) {
|
|
||||||
const $child = CE('div', {'class': 'bx-remote-play-device-wrapper'},
|
|
||||||
CE('div', {'class': 'bx-remote-play-device-info'},
|
|
||||||
CE('div', {},
|
|
||||||
CE('span', {'class': 'bx-remote-play-device-name'}, con.deviceName),
|
|
||||||
CE('span', {'class': 'bx-remote-play-console-type'}, con.consoleType.replace('Xbox', ''))
|
|
||||||
),
|
|
||||||
CE('div', {'class': 'bx-remote-play-power-state'}, RemotePlay.#STATE_LABELS[con.powerState]),
|
|
||||||
),
|
|
||||||
|
|
||||||
// Connect button
|
|
||||||
createButton({
|
|
||||||
classes: ['bx-remote-play-connect-button'],
|
|
||||||
label: t('console-connect'),
|
|
||||||
style: ButtonStyle.PRIMARY | ButtonStyle.FOCUSABLE,
|
|
||||||
onClick: e => {
|
|
||||||
RemotePlay.play(con.serverId);
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
);
|
|
||||||
|
|
||||||
$fragment.appendChild($child);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add Help button
|
|
||||||
$fragment.appendChild(createButton({
|
|
||||||
icon: BxIcon.QUESTION,
|
|
||||||
style: ButtonStyle.GHOST | ButtonStyle.FOCUSABLE,
|
|
||||||
url: 'https://better-xcloud.github.io/remote-play',
|
|
||||||
label: t('help'),
|
|
||||||
}));
|
|
||||||
|
|
||||||
RemotePlay.#$content = CE('div', {}, $fragment);
|
|
||||||
}
|
|
||||||
|
|
||||||
static #getXhomeToken(callback: any) {
|
|
||||||
if (RemotePlay.XHOME_TOKEN) {
|
|
||||||
callback();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
let GSSV_TOKEN;
|
|
||||||
try {
|
|
||||||
GSSV_TOKEN = JSON.parse(localStorage.getItem('xboxcom_xbl_user_info')!).tokens['http://gssv.xboxlive.com/'].token;
|
|
||||||
} catch (e) {
|
|
||||||
for (let i = 0; i < localStorage.length; i++){
|
|
||||||
const key = localStorage.key(i)!;
|
|
||||||
if (!key.startsWith('Auth.User.')) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
const json = JSON.parse(localStorage.getItem(key)!);
|
|
||||||
for (const token of json.tokens) {
|
|
||||||
if (!token.relyingParty.includes('gssv.xboxlive.com')) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
GSSV_TOKEN = token.tokenData.token;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const request = new Request('https://xhome.gssv-play-prod.xboxlive.com/v2/login/user', {
|
|
||||||
method: 'POST',
|
|
||||||
body: JSON.stringify({
|
|
||||||
offeringId: 'xhome',
|
|
||||||
token: GSSV_TOKEN,
|
|
||||||
}),
|
|
||||||
headers: {
|
|
||||||
'Content-Type': 'application/json; charset=utf-8',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
fetch(request).then(resp => resp.json())
|
|
||||||
.then(json => {
|
|
||||||
RemotePlay.#REGIONS = json.offeringSettings.regions;
|
|
||||||
RemotePlay.XHOME_TOKEN = json.gsToken;
|
|
||||||
callback();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
static async #getConsolesList(callback: any) {
|
|
||||||
if (RemotePlay.#CONSOLES) {
|
|
||||||
callback();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const options = {
|
|
||||||
method: 'GET',
|
|
||||||
headers: {
|
|
||||||
'Authorization': `Bearer ${RemotePlay.XHOME_TOKEN}`,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
// Test servers one by one
|
|
||||||
for (const region of RemotePlay.#REGIONS) {
|
|
||||||
try {
|
|
||||||
const request = new Request(`${region.baseUri}/v6/servers/home?mr=50`, options);
|
|
||||||
const resp = await fetch(request);
|
|
||||||
|
|
||||||
const json = await resp.json();
|
|
||||||
RemotePlay.#CONSOLES = json.results;
|
|
||||||
|
|
||||||
// Store working server
|
|
||||||
STATES.remotePlay.server = region.baseUri;
|
|
||||||
|
|
||||||
callback();
|
|
||||||
} catch (e) {}
|
|
||||||
|
|
||||||
if (RemotePlay.#CONSOLES) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// None of the servers worked
|
|
||||||
if (!STATES.remotePlay.server) {
|
|
||||||
RemotePlay.#CONSOLES = [];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static play(serverId: string, resolution?: string) {
|
|
||||||
if (resolution) {
|
|
||||||
setPref(PrefKey.REMOTE_PLAY_RESOLUTION, resolution);
|
|
||||||
}
|
|
||||||
|
|
||||||
STATES.remotePlay.config = {
|
|
||||||
serverId: serverId,
|
|
||||||
};
|
|
||||||
window.BX_REMOTE_PLAY_CONFIG = STATES.remotePlay.config;
|
|
||||||
|
|
||||||
localRedirect('/launch/fortnite/BT5P2X999VH2#remote-play');
|
|
||||||
RemotePlay.detachPopup();
|
|
||||||
}
|
|
||||||
|
|
||||||
static preload() {
|
|
||||||
RemotePlay.#initialize();
|
|
||||||
}
|
|
||||||
|
|
||||||
static detachPopup() {
|
|
||||||
// Detach popup from body
|
|
||||||
const $popup = document.querySelector('.bx-remote-play-popup');
|
|
||||||
$popup && $popup.remove();
|
|
||||||
}
|
|
||||||
|
|
||||||
static togglePopup(force = null) {
|
|
||||||
if (!getPref(PrefKey.REMOTE_PLAY_ENABLED) || !RemotePlay.isReady()) {
|
|
||||||
Toast.show(t('getting-consoles-list'));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
RemotePlay.#initialize();
|
|
||||||
|
|
||||||
if (AppInterface && AppInterface.showRemotePlayDialog) {
|
|
||||||
AppInterface.showRemotePlayDialog(JSON.stringify(RemotePlay.#CONSOLES));
|
|
||||||
(document.activeElement as HTMLElement).blur();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (document.querySelector('.bx-remote-play-popup')) {
|
|
||||||
if (force === false) {
|
|
||||||
RemotePlay.#$content.classList.add('bx-gone');
|
|
||||||
} else {
|
|
||||||
RemotePlay.#$content.classList.toggle('bx-gone');
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const $header = document.querySelector('#gamepass-root header')!;
|
|
||||||
|
|
||||||
const group = $header.firstElementChild!.getAttribute('data-group')!;
|
|
||||||
RemotePlay.#$content.setAttribute('data-group', group);
|
|
||||||
RemotePlay.#$content.classList.add('bx-remote-play-popup');
|
|
||||||
RemotePlay.#$content.classList.remove('bx-gone');
|
|
||||||
|
|
||||||
$header.insertAdjacentElement('afterend', RemotePlay.#$content);
|
|
||||||
}
|
|
||||||
|
|
||||||
static detect() {
|
|
||||||
if (!getPref(PrefKey.REMOTE_PLAY_ENABLED)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
STATES.remotePlay.isPlaying = window.location.pathname.includes('/launch/') && window.location.hash.startsWith('#remote-play');
|
|
||||||
if (STATES.remotePlay?.isPlaying) {
|
|
||||||
window.BX_REMOTE_PLAY_CONFIG = STATES.remotePlay.config;
|
|
||||||
// Remove /launch/... from URL
|
|
||||||
window.history.replaceState({origin: 'better-xcloud'}, '', 'https://www.xbox.com/' + location.pathname.substring(1, 6) + '/play');
|
|
||||||
} else {
|
|
||||||
window.BX_REMOTE_PLAY_CONFIG = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static isReady() {
|
|
||||||
return RemotePlay.#CONSOLES !== null && RemotePlay.#CONSOLES.length > 0;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,3 +1,5 @@
|
|||||||
|
import { isFullVersion } from "@macros/build" with {type: "macro"};
|
||||||
|
|
||||||
import { CE } from "@/utils/html";
|
import { CE } from "@/utils/html";
|
||||||
import { WebGL2Player } from "./player/webgl2-player";
|
import { WebGL2Player } from "./player/webgl2-player";
|
||||||
import { Screenshot } from "@/utils/screenshot";
|
import { Screenshot } from "@/utils/screenshot";
|
||||||
@ -232,7 +234,7 @@ export class StreamPlayer {
|
|||||||
webGL2Player.setFilter(2);
|
webGL2Player.setFilter(2);
|
||||||
}
|
}
|
||||||
|
|
||||||
Screenshot.updateCanvasFilters('none');
|
isFullVersion() && Screenshot.updateCanvasFilters('none');
|
||||||
|
|
||||||
webGL2Player.setSharpness(options.sharpness || 0);
|
webGL2Player.setSharpness(options.sharpness || 0);
|
||||||
webGL2Player.setSaturation(options.saturation || 100);
|
webGL2Player.setSaturation(options.saturation || 100);
|
||||||
@ -246,7 +248,7 @@ export class StreamPlayer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Apply video filters to screenshots
|
// Apply video filters to screenshots
|
||||||
if (getPref(PrefKey.SCREENSHOT_APPLY_FILTERS)) {
|
if (isFullVersion() && getPref(PrefKey.SCREENSHOT_APPLY_FILTERS)) {
|
||||||
Screenshot.updateCanvasFilters(filters);
|
Screenshot.updateCanvasFilters(filters);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,30 +1,52 @@
|
|||||||
|
import { isLiteVersion } from "@macros/build" with {type: "macro"};
|
||||||
|
|
||||||
import { t } from "@utils/translation";
|
import { t } from "@utils/translation";
|
||||||
import { BxEvent } from "@utils/bx-event";
|
import { BxEvent } from "@utils/bx-event";
|
||||||
import { CE, createSvgIcon } from "@utils/html";
|
import { CE, createSvgIcon, humanFileSize } from "@utils/html";
|
||||||
import { STATES } from "@utils/global";
|
import { STATES } from "@utils/global";
|
||||||
import { BxLogger } from "@/utils/bx-logger";
|
import { BxLogger } from "@/utils/bx-logger";
|
||||||
import { BxIcon } from "@/utils/bx-icon";
|
import { BxIcon } from "@/utils/bx-icon";
|
||||||
|
import { GuideMenuTab } from "../ui/guide-menu";
|
||||||
|
import { StreamStat, StreamStatsCollector } from "@/utils/stream-stats-collector";
|
||||||
|
|
||||||
|
|
||||||
|
type StreamBadgeInfo = {
|
||||||
|
name: string,
|
||||||
|
$element?: HTMLElement,
|
||||||
|
icon: typeof BxIcon,
|
||||||
|
color: string,
|
||||||
|
};
|
||||||
|
|
||||||
|
type StreamServerInfo = {
|
||||||
|
server?: {
|
||||||
|
ipv6: boolean,
|
||||||
|
region?: string,
|
||||||
|
},
|
||||||
|
|
||||||
|
video?: {
|
||||||
|
width: number,
|
||||||
|
height: number,
|
||||||
|
codec: string,
|
||||||
|
profile?: string,
|
||||||
|
},
|
||||||
|
|
||||||
|
audio?: {
|
||||||
|
codec: string,
|
||||||
|
bitrate: number,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
enum StreamBadge {
|
enum StreamBadge {
|
||||||
PLAYTIME = 'playtime',
|
PLAYTIME = 'playtime',
|
||||||
BATTERY = 'battery',
|
BATTERY = 'battery',
|
||||||
DOWNLOAD = 'in',
|
DOWNLOAD = 'download',
|
||||||
UPLOAD = 'out',
|
UPLOAD = 'upload',
|
||||||
|
|
||||||
SERVER = 'server',
|
SERVER = 'server',
|
||||||
VIDEO = 'video',
|
VIDEO = 'video',
|
||||||
AUDIO = 'audio',
|
AUDIO = 'audio',
|
||||||
}
|
}
|
||||||
|
|
||||||
const StreamBadgeIcon: Partial<{[key in StreamBadge]: any}> = {
|
|
||||||
[StreamBadge.PLAYTIME]: BxIcon.PLAYTIME,
|
|
||||||
[StreamBadge.VIDEO]: BxIcon.DISPLAY,
|
|
||||||
[StreamBadge.BATTERY]: BxIcon.BATTERY,
|
|
||||||
[StreamBadge.DOWNLOAD]: BxIcon.DOWNLOAD,
|
|
||||||
[StreamBadge.UPLOAD]: BxIcon.UPLOAD,
|
|
||||||
[StreamBadge.SERVER]: BxIcon.SERVER,
|
|
||||||
[StreamBadge.AUDIO]: BxIcon.AUDIO,
|
|
||||||
}
|
|
||||||
|
|
||||||
export class StreamBadges {
|
export class StreamBadges {
|
||||||
private static instance: StreamBadges;
|
private static instance: StreamBadges;
|
||||||
@ -36,91 +58,100 @@ export class StreamBadges {
|
|||||||
return StreamBadges.instance;
|
return StreamBadges.instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ipv6 = false;
|
private serverInfo: StreamServerInfo = {};
|
||||||
#resolution?: {width: number, height: number} | null = null;
|
|
||||||
#video?: {codec: string, profile?: string | null} | null = null;
|
|
||||||
#audio?: {codec: string, bitrate: number} | null = null;
|
|
||||||
#region = '';
|
|
||||||
|
|
||||||
startBatteryLevel = 100;
|
private badges: Record<StreamBadge, StreamBadgeInfo> = {
|
||||||
startTimestamp = 0;
|
[StreamBadge.PLAYTIME]: {
|
||||||
|
name: t('playtime'),
|
||||||
|
icon: BxIcon.PLAYTIME,
|
||||||
|
color: '#ff004d',
|
||||||
|
},
|
||||||
|
[StreamBadge.BATTERY]: {
|
||||||
|
name: t('battery'),
|
||||||
|
icon: BxIcon.BATTERY,
|
||||||
|
color: '#00b543',
|
||||||
|
},
|
||||||
|
[StreamBadge.DOWNLOAD]: {
|
||||||
|
name: t('download'),
|
||||||
|
icon: BxIcon.DOWNLOAD,
|
||||||
|
color: '#29adff',
|
||||||
|
},
|
||||||
|
[StreamBadge.UPLOAD]: {
|
||||||
|
name: t('upload'),
|
||||||
|
icon: BxIcon.UPLOAD,
|
||||||
|
color: '#ff77a8',
|
||||||
|
},
|
||||||
|
[StreamBadge.SERVER]: {
|
||||||
|
name: t('server'),
|
||||||
|
icon: BxIcon.SERVER,
|
||||||
|
color: '#ff6c24',
|
||||||
|
},
|
||||||
|
[StreamBadge.VIDEO]: {
|
||||||
|
name: t('video'),
|
||||||
|
icon: BxIcon.DISPLAY,
|
||||||
|
color: '#742f29',
|
||||||
|
},
|
||||||
|
[StreamBadge.AUDIO]: {
|
||||||
|
name: t('audio'),
|
||||||
|
icon: BxIcon.AUDIO,
|
||||||
|
color: '#5f574f',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
#$container: HTMLElement | undefined;
|
private $container: HTMLElement | undefined;
|
||||||
#cachedDoms: Partial<{[key in StreamBadge]: HTMLElement}> = {};
|
|
||||||
|
|
||||||
#interval?: number | null;
|
private intervalId?: number | null;
|
||||||
readonly #REFRESH_INTERVAL = 3000;
|
private readonly REFRESH_INTERVAL = 3 * 1000;
|
||||||
|
|
||||||
setRegion(region: string) {
|
setRegion(region: string) {
|
||||||
this.#region = region;
|
this.serverInfo.server = {
|
||||||
|
region: region,
|
||||||
|
ipv6: false,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
#renderBadge(name: StreamBadge, value: string, color: string) {
|
renderBadge(name: StreamBadge, value: string) {
|
||||||
|
const badgeInfo = this.badges[name];
|
||||||
|
|
||||||
let $badge;
|
let $badge;
|
||||||
if (this.#cachedDoms[name]) {
|
if (badgeInfo.$element) {
|
||||||
$badge = this.#cachedDoms[name]!;
|
$badge = badgeInfo.$element;
|
||||||
$badge.lastElementChild!.textContent = value;
|
$badge.lastElementChild!.textContent = value;
|
||||||
return $badge;
|
return $badge;
|
||||||
}
|
}
|
||||||
|
|
||||||
$badge = CE('div', {'class': 'bx-badge', 'title': t(`badge-${name}`)},
|
$badge = CE('div', {class: 'bx-badge', title: badgeInfo.name},
|
||||||
CE('span', {'class': 'bx-badge-name'}, createSvgIcon(StreamBadgeIcon[name])),
|
CE('span', {class: 'bx-badge-name'}, createSvgIcon(badgeInfo.icon)),
|
||||||
CE('span', {'class': 'bx-badge-value', 'style': `background-color: ${color}`}, value),
|
CE('span', {class: 'bx-badge-value', style: `background-color: ${badgeInfo.color}`}, value),
|
||||||
);
|
);
|
||||||
|
|
||||||
if (name === StreamBadge.BATTERY) {
|
if (name === StreamBadge.BATTERY) {
|
||||||
$badge.classList.add('bx-badge-battery');
|
$badge.classList.add('bx-badge-battery');
|
||||||
}
|
}
|
||||||
|
|
||||||
this.#cachedDoms[name] = $badge;
|
this.badges[name].$element = $badge;
|
||||||
return $badge;
|
return $badge;
|
||||||
}
|
}
|
||||||
|
|
||||||
async #updateBadges(forceUpdate = false) {
|
private async updateBadges(forceUpdate = false) {
|
||||||
if (!this.#$container || (!forceUpdate && !this.#$container.isConnected)) {
|
if (!this.$container || (!forceUpdate && !this.$container.isConnected)) {
|
||||||
this.#stop();
|
this.stop();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Playtime
|
const statsCollector = StreamStatsCollector.getInstance();
|
||||||
let now = +new Date;
|
await statsCollector.collect();
|
||||||
const diffSeconds = Math.ceil((now - this.startTimestamp) / 1000);
|
|
||||||
const playtime = this.#secondsToHm(diffSeconds);
|
|
||||||
|
|
||||||
// Battery
|
const play = statsCollector.getStat(StreamStat.PLAYTIME);
|
||||||
let batteryLevel = '100%';
|
const batt = statsCollector.getStat(StreamStat.BATTERY);
|
||||||
let batteryLevelInt = 100;
|
const dl = statsCollector.getStat(StreamStat.DOWNLOAD);
|
||||||
let isCharging = false;
|
const ul = statsCollector.getStat(StreamStat.UPLOAD);
|
||||||
if (STATES.browser.capabilities.batteryApi) {
|
|
||||||
try {
|
|
||||||
const bm = await (navigator as NavigatorBattery).getBattery();
|
|
||||||
isCharging = bm.charging;
|
|
||||||
batteryLevelInt = Math.round(bm.level * 100);
|
|
||||||
batteryLevel = `${batteryLevelInt}%`;
|
|
||||||
|
|
||||||
if (batteryLevelInt != this.startBatteryLevel) {
|
|
||||||
const diffLevel = Math.round(batteryLevelInt - this.startBatteryLevel);
|
|
||||||
const sign = diffLevel > 0 ? '+' : '';
|
|
||||||
batteryLevel += ` (${sign}${diffLevel}%)`;
|
|
||||||
}
|
|
||||||
} catch(e) {}
|
|
||||||
}
|
|
||||||
|
|
||||||
const stats = await STATES.currentStream.peerConnection?.getStats()!;
|
|
||||||
let totalIn = 0;
|
|
||||||
let totalOut = 0;
|
|
||||||
stats.forEach(stat => {
|
|
||||||
if (stat.type === 'candidate-pair' && stat.packetsReceived > 0 && stat.state === 'succeeded') {
|
|
||||||
totalIn += stat.bytesReceived;
|
|
||||||
totalOut += stat.bytesSent;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
const badges = {
|
const badges = {
|
||||||
[StreamBadge.DOWNLOAD]: totalIn ? this.#humanFileSize(totalIn) : null,
|
[StreamBadge.DOWNLOAD]: dl.toString(),
|
||||||
[StreamBadge.UPLOAD]: totalOut ? this.#humanFileSize(totalOut) : null,
|
[StreamBadge.UPLOAD]: ul.toString(),
|
||||||
[StreamBadge.PLAYTIME]: playtime,
|
[StreamBadge.PLAYTIME]: play.toString(),
|
||||||
[StreamBadge.BATTERY]: batteryLevel,
|
[StreamBadge.BATTERY]: batt.toString(),
|
||||||
};
|
};
|
||||||
|
|
||||||
let name: keyof typeof badges;
|
let name: keyof typeof badges;
|
||||||
@ -130,97 +161,44 @@ export class StreamBadges {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
const $elm = this.#cachedDoms[name]!;
|
const $elm = this.badges[name].$element;
|
||||||
$elm && ($elm.lastElementChild!.textContent = value);
|
if (!$elm) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$elm.lastElementChild!.textContent = value;
|
||||||
|
|
||||||
if (name === StreamBadge.BATTERY) {
|
if (name === StreamBadge.BATTERY) {
|
||||||
if (this.startBatteryLevel === 100 && batteryLevelInt === 100) {
|
if (batt.current === 100 && batt.start === 100) {
|
||||||
// Hide battery badge when the battery is 100%
|
// Hide battery badge when the battery is 100%
|
||||||
$elm.classList.add('bx-gone');
|
$elm.classList.add('bx-gone');
|
||||||
} else {
|
} else {
|
||||||
// Show charging status
|
// Show charging status
|
||||||
$elm.dataset.charging = isCharging.toString()
|
$elm.dataset.charging = batt.isCharging.toString();
|
||||||
$elm.classList.remove('bx-gone');
|
$elm.classList.remove('bx-gone');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async #start() {
|
private async start() {
|
||||||
await this.#updateBadges(true);
|
await this.updateBadges(true);
|
||||||
this.#stop();
|
this.stop();
|
||||||
this.#interval = window.setInterval(this.#updateBadges.bind(this), this.#REFRESH_INTERVAL);
|
this.intervalId = window.setInterval(this.updateBadges.bind(this), this.REFRESH_INTERVAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
#stop() {
|
private stop() {
|
||||||
this.#interval && clearInterval(this.#interval);
|
this.intervalId && clearInterval(this.intervalId);
|
||||||
this.#interval = null;
|
this.intervalId = null;
|
||||||
}
|
|
||||||
|
|
||||||
#secondsToHm(seconds: number) {
|
|
||||||
let h = Math.floor(seconds / 3600);
|
|
||||||
let m = Math.floor(seconds % 3600 / 60) + 1;
|
|
||||||
|
|
||||||
if (m === 60) {
|
|
||||||
h += 1;
|
|
||||||
m = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
const output = [];
|
|
||||||
h > 0 && output.push(`${h}h`);
|
|
||||||
m > 0 && output.push(`${m}m`);
|
|
||||||
|
|
||||||
return output.join(' ');
|
|
||||||
}
|
|
||||||
|
|
||||||
// https://stackoverflow.com/a/20732091
|
|
||||||
#humanFileSize(size: number) {
|
|
||||||
const units = ['B', 'KB', 'MB', 'GB', 'TB'];
|
|
||||||
|
|
||||||
const i = size == 0 ? 0 : Math.floor(Math.log(size) / Math.log(1024));
|
|
||||||
return (size / Math.pow(1024, i)).toFixed(2) + ' ' + units[i];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async render() {
|
async render() {
|
||||||
if (this.#$container) {
|
if (this.$container) {
|
||||||
this.#start();
|
this.start();
|
||||||
return this.#$container;
|
return this.$container;
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.#getServerStats();
|
await this.getServerStats();
|
||||||
|
|
||||||
// Video
|
|
||||||
let video = '';
|
|
||||||
if (this.#resolution) {
|
|
||||||
video = `${this.#resolution.height}p`;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (this.#video) {
|
|
||||||
video && (video += '/');
|
|
||||||
video += this.#video.codec;
|
|
||||||
if (this.#video.profile) {
|
|
||||||
const profile = this.#video.profile;
|
|
||||||
|
|
||||||
let quality = profile;
|
|
||||||
if (profile.startsWith('4d')) {
|
|
||||||
quality = t('visual-quality-high');
|
|
||||||
} else if (profile.startsWith('42e')) {
|
|
||||||
quality = t('visual-quality-normal');
|
|
||||||
} else if (profile.startsWith('420')) {
|
|
||||||
quality = t('visual-quality-low');
|
|
||||||
}
|
|
||||||
|
|
||||||
video += ` (${quality})`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Audio
|
|
||||||
let audio;
|
|
||||||
if (this.#audio) {
|
|
||||||
audio = this.#audio.codec;
|
|
||||||
const bitrate = this.#audio.bitrate / 1000;
|
|
||||||
audio += ` (${bitrate} kHz)`;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Battery
|
// Battery
|
||||||
let batteryLevel = '';
|
let batteryLevel = '';
|
||||||
@ -228,46 +206,50 @@ export class StreamBadges {
|
|||||||
batteryLevel = '100%';
|
batteryLevel = '100%';
|
||||||
}
|
}
|
||||||
|
|
||||||
// Server + Region
|
|
||||||
let server = this.#region;
|
|
||||||
server += '@' + (this.#ipv6 ? 'IPv6' : 'IPv4');
|
|
||||||
|
|
||||||
const BADGES = [
|
const BADGES = [
|
||||||
[StreamBadge.PLAYTIME, '1m', '#ff004d'],
|
[StreamBadge.PLAYTIME, '1m'],
|
||||||
[StreamBadge.BATTERY, batteryLevel, '#00b543'],
|
[StreamBadge.BATTERY, batteryLevel],
|
||||||
[StreamBadge.DOWNLOAD, this.#humanFileSize(0), '#29adff'],
|
[StreamBadge.DOWNLOAD, humanFileSize(0)],
|
||||||
[StreamBadge.UPLOAD, this.#humanFileSize(0), '#ff77a8'],
|
[StreamBadge.UPLOAD, humanFileSize(0)],
|
||||||
[StreamBadge.SERVER, server, '#ff6c24'],
|
this.serverInfo.server ? this.badges.server.$element : [StreamBadge.SERVER, '?'],
|
||||||
video ? [StreamBadge.VIDEO, video, '#742f29'] : null,
|
this.serverInfo.video ? this.badges.video.$element : [StreamBadge.VIDEO, '?'],
|
||||||
audio ? [StreamBadge.AUDIO, audio, '#5f574f'] : null,
|
this.serverInfo.audio ? this.badges.audio.$element : [StreamBadge.AUDIO, '?'],
|
||||||
];
|
];
|
||||||
|
|
||||||
const $container = CE('div', {'class': 'bx-badges'});
|
const $container = CE('div', {class: 'bx-badges'});
|
||||||
BADGES.forEach(item => {
|
BADGES.forEach(item => {
|
||||||
if (!item) {
|
if (!item) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const $badge = this.#renderBadge(...(item as [StreamBadge, string, string]));
|
let $badge: HTMLElement;
|
||||||
|
if (!(item instanceof HTMLElement)) {
|
||||||
|
$badge = this.renderBadge(...(item as [StreamBadge, string]));
|
||||||
|
} else {
|
||||||
|
$badge = item;
|
||||||
|
}
|
||||||
|
|
||||||
$container.appendChild($badge);
|
$container.appendChild($badge);
|
||||||
});
|
});
|
||||||
|
|
||||||
this.#$container = $container;
|
this.$container = $container;
|
||||||
await this.#start();
|
await this.start();
|
||||||
|
|
||||||
return $container;
|
return $container;
|
||||||
}
|
}
|
||||||
|
|
||||||
async #getServerStats() {
|
private async getServerStats() {
|
||||||
const stats = await STATES.currentStream.peerConnection!.getStats();
|
const stats = await STATES.currentStream.peerConnection!.getStats();
|
||||||
|
|
||||||
const allVideoCodecs: {[index: string]: RTCBasicStat} = {};
|
const allVideoCodecs: Record<string, RTCBasicStat> = {};
|
||||||
let videoCodecId;
|
let videoCodecId;
|
||||||
|
let videoWidth = 0;
|
||||||
|
let videoHeight = 0;
|
||||||
|
|
||||||
const allAudioCodecs: {[index: string]: RTCBasicStat} = {};
|
const allAudioCodecs: Record<string, RTCBasicStat> = {};
|
||||||
let audioCodecId;
|
let audioCodecId;
|
||||||
|
|
||||||
const allCandidates: {[index: string]: string} = {};
|
const allCandidates: Record<string, string> = {};
|
||||||
let candidateId;
|
let candidateId;
|
||||||
|
|
||||||
stats.forEach((stat: RTCBasicStat) => {
|
stats.forEach((stat: RTCBasicStat) => {
|
||||||
@ -284,6 +266,8 @@ export class StreamBadges {
|
|||||||
// Get the codecId of the video/audio track currently being used
|
// Get the codecId of the video/audio track currently being used
|
||||||
if (stat.kind === 'video') {
|
if (stat.kind === 'video') {
|
||||||
videoCodecId = stat.codecId;
|
videoCodecId = stat.codecId;
|
||||||
|
videoWidth = stat.frameWidth;
|
||||||
|
videoHeight = stat.frameHeight;
|
||||||
} else if (stat.kind === 'audio') {
|
} else if (stat.kind === 'audio') {
|
||||||
audioCodecId = stat.codecId;
|
audioCodecId = stat.codecId;
|
||||||
}
|
}
|
||||||
@ -297,71 +281,91 @@ export class StreamBadges {
|
|||||||
// Get video codec from codecId
|
// Get video codec from codecId
|
||||||
if (videoCodecId) {
|
if (videoCodecId) {
|
||||||
const videoStat = allVideoCodecs[videoCodecId];
|
const videoStat = allVideoCodecs[videoCodecId];
|
||||||
const video: any = {
|
const video: StreamServerInfo['video'] = {
|
||||||
|
width: videoWidth,
|
||||||
|
height: videoHeight,
|
||||||
codec: videoStat.mimeType.substring(6),
|
codec: videoStat.mimeType.substring(6),
|
||||||
};
|
};
|
||||||
|
|
||||||
if (video.codec === 'H264') {
|
if (video.codec === 'H264') {
|
||||||
const match = /profile-level-id=([0-9a-f]{6})/.exec(videoStat.sdpFmtpLine);
|
const match = /profile-level-id=([0-9a-f]{6})/.exec(videoStat.sdpFmtpLine);
|
||||||
video.profile = match ? match[1] : null;
|
match && (video.profile = match[1]);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.#video = video;
|
let text = videoHeight + 'p';
|
||||||
|
text && (text += '/');
|
||||||
|
text += video.codec;
|
||||||
|
if (video.profile) {
|
||||||
|
const profile = video.profile;
|
||||||
|
|
||||||
|
let quality = profile;
|
||||||
|
if (profile.startsWith('4d')) {
|
||||||
|
quality = t('visual-quality-high');
|
||||||
|
} else if (profile.startsWith('42e')) {
|
||||||
|
quality = t('visual-quality-normal');
|
||||||
|
} else if (profile.startsWith('420')) {
|
||||||
|
quality = t('visual-quality-low');
|
||||||
|
}
|
||||||
|
|
||||||
|
text += ` (${quality})`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Render badge
|
||||||
|
this.badges.video.$element = this.renderBadge(StreamBadge.VIDEO, text);
|
||||||
|
|
||||||
|
this.serverInfo.video = video;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get audio codec from codecId
|
// Get audio codec from codecId
|
||||||
if (audioCodecId) {
|
if (audioCodecId) {
|
||||||
const audioStat = allAudioCodecs[audioCodecId];
|
const audioStat = allAudioCodecs[audioCodecId];
|
||||||
this.#audio = {
|
const audio: StreamServerInfo['audio'] = {
|
||||||
codec: audioStat.mimeType.substring(6),
|
codec: audioStat.mimeType.substring(6),
|
||||||
bitrate: audioStat.clockRate,
|
bitrate: audioStat.clockRate,
|
||||||
}
|
};
|
||||||
|
|
||||||
|
const bitrate = audio.bitrate / 1000;
|
||||||
|
const text = `${audio.codec} (${bitrate} kHz)`;
|
||||||
|
this.badges.audio.$element = this.renderBadge(StreamBadge.AUDIO, text);
|
||||||
|
|
||||||
|
this.serverInfo.audio = audio;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get server type
|
// Get server type
|
||||||
if (candidateId) {
|
if (candidateId) {
|
||||||
BxLogger.info('candidate', candidateId, allCandidates);
|
BxLogger.info('candidate', candidateId, allCandidates);
|
||||||
this.#ipv6 = allCandidates[candidateId].includes(':');
|
|
||||||
|
// Server + Region
|
||||||
|
const server = this.serverInfo.server;
|
||||||
|
if (server) {
|
||||||
|
server.ipv6 = allCandidates[candidateId].includes(':');
|
||||||
|
|
||||||
|
let text = '';
|
||||||
|
if (server.region) {
|
||||||
|
text += server.region;
|
||||||
|
}
|
||||||
|
|
||||||
|
text += '@' + (server ? 'IPv6' : 'IPv4');
|
||||||
|
this.badges.server.$element = this.renderBadge(StreamBadge.SERVER, text);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static setupEvents() {
|
static setupEvents() {
|
||||||
window.addEventListener(BxEvent.STREAM_PLAYING, e => {
|
// Since the Lite version doesn't have the "..." button on System menu
|
||||||
const $video = (e as any).$video;
|
// we need to display Stream badges in the Guide menu instead
|
||||||
const streamBadges = StreamBadges.getInstance();
|
isLiteVersion() && window.addEventListener(BxEvent.XCLOUD_GUIDE_MENU_SHOWN, async e => {
|
||||||
|
const where = (e as any).where as GuideMenuTab;
|
||||||
|
|
||||||
streamBadges.#resolution = {
|
if (where !== GuideMenuTab.HOME || !STATES.isPlaying) {
|
||||||
width: $video.videoWidth,
|
|
||||||
height: $video.videoHeight,
|
|
||||||
};
|
|
||||||
streamBadges.startTimestamp = +new Date;
|
|
||||||
|
|
||||||
// Get battery level
|
|
||||||
try {
|
|
||||||
STATES.browser.capabilities.batteryApi && (navigator as NavigatorBattery).getBattery().then(bm => {
|
|
||||||
streamBadges.startBatteryLevel = Math.round(bm.level * 100);
|
|
||||||
});
|
|
||||||
} catch(e) {}
|
|
||||||
});
|
|
||||||
|
|
||||||
/*
|
|
||||||
Don't do this until xCloud remove the Stream Menu page
|
|
||||||
|
|
||||||
window.addEventListener(BxEvent.XCLOUD_GUIDE_SHOWN, async e => {
|
|
||||||
const where = (e as any).where as XcloudGuideWhere;
|
|
||||||
|
|
||||||
if (where !== XcloudGuideWhere.HOME || !STATES.isPlaying) {
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const $btnQuit = document.querySelector('#gamepass-dialog-root a[class*=QuitGameButton]');
|
const $btnQuit = document.querySelector('#gamepass-dialog-root a[class*=QuitGameButton]');
|
||||||
if (!$btnQuit) {
|
if ($btnQuit) {
|
||||||
return;
|
// Add badges
|
||||||
|
$btnQuit.insertAdjacentElement('beforebegin', await StreamBadges.getInstance().render());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add badges
|
|
||||||
$btnQuit.insertAdjacentElement('beforebegin', await StreamBadges.getInstance().render());
|
|
||||||
});
|
});
|
||||||
*/
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -4,15 +4,8 @@ import { t } from "@utils/translation"
|
|||||||
import { STATES } from "@utils/global"
|
import { STATES } from "@utils/global"
|
||||||
import { PrefKey } from "@/enums/pref-keys"
|
import { PrefKey } from "@/enums/pref-keys"
|
||||||
import { getPref } from "@/utils/settings-storages/global-settings-storage"
|
import { getPref } from "@/utils/settings-storages/global-settings-storage"
|
||||||
|
import { StreamStat, StreamStatsCollector, type StreamStatGrade } from "@/utils/stream-stats-collector"
|
||||||
|
|
||||||
export enum StreamStat {
|
|
||||||
PING = 'ping',
|
|
||||||
FPS = 'fps',
|
|
||||||
BITRATE = 'btr',
|
|
||||||
DECODE_TIME = 'dt',
|
|
||||||
PACKETS_LOST = 'pl',
|
|
||||||
FRAMES_LOST = 'fl',
|
|
||||||
};
|
|
||||||
|
|
||||||
export class StreamStats {
|
export class StreamStats {
|
||||||
private static instance: StreamStats;
|
private static instance: StreamStats;
|
||||||
@ -24,58 +17,95 @@ export class StreamStats {
|
|||||||
return StreamStats.instance;
|
return StreamStats.instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
#timeoutId?: number | null;
|
private intervalId?: number | null;
|
||||||
readonly #updateInterval = 1000;
|
private readonly REFRESH_INTERVAL = 1 * 1000;
|
||||||
|
|
||||||
#$container: HTMLElement | undefined;
|
private stats = {
|
||||||
#$fps: HTMLElement | undefined;
|
[StreamStat.CLOCK]: {
|
||||||
#$ping: HTMLElement | undefined;
|
name: t('clock'),
|
||||||
#$dt: HTMLElement | undefined;
|
$element: CE('span'),
|
||||||
#$pl: HTMLElement | undefined;
|
},
|
||||||
#$fl: HTMLElement | undefined;
|
[StreamStat.PLAYTIME]: {
|
||||||
#$br: HTMLElement | undefined;
|
name: t('playtime'),
|
||||||
|
$element: CE('span'),
|
||||||
|
},
|
||||||
|
[StreamStat.BATTERY]: {
|
||||||
|
name: t('battery'),
|
||||||
|
$element: CE('span'),
|
||||||
|
},
|
||||||
|
[StreamStat.PING]: {
|
||||||
|
name: t('stat-ping'),
|
||||||
|
$element: CE('span'),
|
||||||
|
},
|
||||||
|
[StreamStat.FPS]: {
|
||||||
|
name: t('stat-fps'),
|
||||||
|
$element: CE('span'),
|
||||||
|
},
|
||||||
|
[StreamStat.BITRATE]: {
|
||||||
|
name: t('stat-bitrate'),
|
||||||
|
$element: CE('span'),
|
||||||
|
},
|
||||||
|
[StreamStat.DECODE_TIME]: {
|
||||||
|
name: t('stat-decode-time'),
|
||||||
|
$element: CE('span'),
|
||||||
|
},
|
||||||
|
[StreamStat.PACKETS_LOST]: {
|
||||||
|
name: t('stat-packets-lost'),
|
||||||
|
$element: CE('span'),
|
||||||
|
},
|
||||||
|
[StreamStat.FRAMES_LOST]: {
|
||||||
|
name: t('stat-frames-lost'),
|
||||||
|
$element: CE('span'),
|
||||||
|
},
|
||||||
|
[StreamStat.DOWNLOAD]: {
|
||||||
|
name: t('download'),
|
||||||
|
$element: CE('span'),
|
||||||
|
},
|
||||||
|
[StreamStat.UPLOAD]: {
|
||||||
|
name: t('upload'),
|
||||||
|
$element: CE('span'),
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
#lastVideoStat?: RTCBasicStat | null;
|
private $container!: HTMLElement;
|
||||||
|
|
||||||
#quickGlanceObserver?: MutationObserver | null;
|
quickGlanceObserver?: MutationObserver | null;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
this.#render();
|
this.render();
|
||||||
}
|
}
|
||||||
|
|
||||||
start(glancing=false) {
|
async start(glancing=false) {
|
||||||
if (!this.isHidden() || (glancing && this.isGlancing())) {
|
if (!this.isHidden() || (glancing && this.isGlancing())) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.#$container) {
|
this.intervalId && clearInterval(this.intervalId);
|
||||||
this.#$container.classList.remove('bx-gone');
|
await this.update(true);
|
||||||
this.#$container.dataset.display = glancing ? 'glancing' : 'fixed';
|
|
||||||
}
|
|
||||||
|
|
||||||
this.#timeoutId = window.setTimeout(this.#update.bind(this), this.#updateInterval);
|
this.$container.classList.remove('bx-gone');
|
||||||
|
this.$container.dataset.display = glancing ? 'glancing' : 'fixed';
|
||||||
|
|
||||||
|
this.intervalId = window.setInterval(this.update.bind(this), this.REFRESH_INTERVAL);
|
||||||
}
|
}
|
||||||
|
|
||||||
stop(glancing=false) {
|
async stop(glancing=false) {
|
||||||
if (glancing && !this.isGlancing()) {
|
if (glancing && !this.isGlancing()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.#timeoutId && clearTimeout(this.#timeoutId);
|
this.intervalId && clearInterval(this.intervalId);
|
||||||
this.#timeoutId = null;
|
this.intervalId = null;
|
||||||
this.#lastVideoStat = null;
|
|
||||||
|
|
||||||
if (this.#$container) {
|
this.$container.removeAttribute('data-display');
|
||||||
this.#$container.removeAttribute('data-display');
|
this.$container.classList.add('bx-gone');
|
||||||
this.#$container.classList.add('bx-gone');
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
toggle() {
|
async toggle() {
|
||||||
if (this.isGlancing()) {
|
if (this.isGlancing()) {
|
||||||
this.#$container && (this.#$container.dataset.display = 'fixed');
|
this.$container && (this.$container.dataset.display = 'fixed');
|
||||||
} else {
|
} else {
|
||||||
this.isHidden() ? this.start() : this.stop();
|
this.isHidden() ? await this.start() : await this.stop();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -85,11 +115,11 @@ export class StreamStats {
|
|||||||
this.hideSettingsUi();
|
this.hideSettingsUi();
|
||||||
}
|
}
|
||||||
|
|
||||||
isHidden = () => this.#$container && this.#$container.classList.contains('bx-gone');
|
isHidden = () => this.$container.classList.contains('bx-gone');
|
||||||
isGlancing = () => this.#$container && this.#$container.dataset.display === 'glancing';
|
isGlancing = () => this.$container.dataset.display === 'glancing';
|
||||||
|
|
||||||
quickGlanceSetup() {
|
quickGlanceSetup() {
|
||||||
if (!STATES.isPlaying || this.#quickGlanceObserver) {
|
if (!STATES.isPlaying || this.quickGlanceObserver) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -98,20 +128,23 @@ export class StreamStats {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.#quickGlanceObserver = new MutationObserver((mutationList, observer) => {
|
this.quickGlanceObserver = new MutationObserver((mutationList, observer) => {
|
||||||
for (let record of mutationList) {
|
for (const record of mutationList) {
|
||||||
if (record.attributeName && record.attributeName === 'aria-expanded') {
|
const $target = record.target as HTMLElement;
|
||||||
const expanded = (record.target as HTMLElement).ariaExpanded;
|
if (!$target.className || !$target.className.startsWith('GripHandle')) {
|
||||||
if (expanded === 'true') {
|
continue;
|
||||||
this.isHidden() && this.start(true);
|
}
|
||||||
} else {
|
|
||||||
this.stop(true);
|
const expanded = (record.target as HTMLElement).ariaExpanded;
|
||||||
}
|
if (expanded === 'true') {
|
||||||
|
this.isHidden() && this.start(true);
|
||||||
|
} else {
|
||||||
|
this.stop(true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
this.#quickGlanceObserver.observe($uiContainer, {
|
this.quickGlanceObserver.observe($uiContainer, {
|
||||||
attributes: true,
|
attributes: true,
|
||||||
attributeFilter: ['aria-expanded'],
|
attributeFilter: ['aria-expanded'],
|
||||||
subtree: true,
|
subtree: true,
|
||||||
@ -119,98 +152,54 @@ export class StreamStats {
|
|||||||
}
|
}
|
||||||
|
|
||||||
quickGlanceStop() {
|
quickGlanceStop() {
|
||||||
this.#quickGlanceObserver && this.#quickGlanceObserver.disconnect();
|
this.quickGlanceObserver && this.quickGlanceObserver.disconnect();
|
||||||
this.#quickGlanceObserver = null;
|
this.quickGlanceObserver = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
async #update() {
|
private async update(forceUpdate=false) {
|
||||||
if (this.isHidden() || !STATES.currentStream.peerConnection) {
|
if ((!forceUpdate && this.isHidden()) || !STATES.currentStream.peerConnection) {
|
||||||
this.onStoppedPlaying();
|
this.onStoppedPlaying();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.#timeoutId = null;
|
|
||||||
const startTime = performance.now();
|
|
||||||
|
|
||||||
const PREF_STATS_CONDITIONAL_FORMATTING = getPref(PrefKey.STATS_CONDITIONAL_FORMATTING);
|
const PREF_STATS_CONDITIONAL_FORMATTING = getPref(PrefKey.STATS_CONDITIONAL_FORMATTING);
|
||||||
|
let grade: StreamStatGrade = '';
|
||||||
|
|
||||||
const stats = await STATES.currentStream.peerConnection.getStats();
|
// Collect stats
|
||||||
let grade = '';
|
const statsCollector = StreamStatsCollector.getInstance();
|
||||||
|
await statsCollector.collect();
|
||||||
|
|
||||||
stats.forEach(stat => {
|
let statKey: keyof typeof this.stats;
|
||||||
if (stat.type === 'inbound-rtp' && stat.kind === 'video') {
|
for (statKey in this.stats) {
|
||||||
// FPS
|
grade = '';
|
||||||
this.#$fps!.textContent = stat.framesPerSecond || 0;
|
|
||||||
|
|
||||||
// Packets Lost
|
const stat = this.stats[statKey];
|
||||||
const packetsLost = Math.max(0, stat.packetsLost); // packetsLost can be negative, but we don't care about that
|
const value = statsCollector.getStat(statKey);
|
||||||
const packetsReceived = stat.packetsReceived;
|
const $element = stat.$element;
|
||||||
const packetsLostPercentage = (packetsLost * 100 / ((packetsLost + packetsReceived) || 1)).toFixed(2);
|
$element.textContent = value.toString();
|
||||||
this.#$pl!.textContent = packetsLostPercentage === '0.00' ? packetsLost.toString() : `${packetsLost} (${packetsLostPercentage}%)`;
|
|
||||||
|
|
||||||
// Frames dropped
|
// Get stat's grade
|
||||||
const framesDropped = stat.framesDropped;
|
if (PREF_STATS_CONDITIONAL_FORMATTING) {
|
||||||
const framesReceived = stat.framesReceived;
|
if (statKey === StreamStat.PING || statKey === StreamStat.DECODE_TIME) {
|
||||||
const framesDroppedPercentage = (framesDropped * 100 / ((framesDropped + framesReceived) || 1)).toFixed(2);
|
grade = (value as any).calculateGrade();
|
||||||
this.#$fl!.textContent = framesDroppedPercentage === '0.00' ? framesDropped : `${framesDropped} (${framesDroppedPercentage}%)`;
|
|
||||||
|
|
||||||
if (!this.#lastVideoStat) {
|
|
||||||
this.#lastVideoStat = stat;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const lastStat = this.#lastVideoStat;
|
|
||||||
// Bitrate
|
|
||||||
const timeDiff = stat.timestamp - lastStat.timestamp;
|
|
||||||
const bitrate = 8 * (stat.bytesReceived - lastStat.bytesReceived) / timeDiff / 1000;
|
|
||||||
this.#$br!.textContent = `${bitrate.toFixed(2)} Mbps`;
|
|
||||||
|
|
||||||
// Decode time
|
|
||||||
const totalDecodeTimeDiff = stat.totalDecodeTime - lastStat.totalDecodeTime;
|
|
||||||
const framesDecodedDiff = stat.framesDecoded - lastStat.framesDecoded;
|
|
||||||
const currentDecodeTime = totalDecodeTimeDiff / framesDecodedDiff * 1000;
|
|
||||||
|
|
||||||
if (isNaN(currentDecodeTime)) {
|
|
||||||
this.#$dt!.textContent = '??ms';
|
|
||||||
} else {
|
|
||||||
this.#$dt!.textContent = `${currentDecodeTime.toFixed(2)}ms`;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (PREF_STATS_CONDITIONAL_FORMATTING) {
|
|
||||||
grade = (currentDecodeTime > 12) ? 'bad' : (currentDecodeTime > 9) ? 'ok' : (currentDecodeTime > 6) ? 'good' : '';
|
|
||||||
this.#$dt!.dataset.grade = grade;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.#lastVideoStat = stat;
|
|
||||||
} else if (stat.type === 'candidate-pair' && stat.packetsReceived > 0 && stat.state === 'succeeded') {
|
|
||||||
// Round Trip Time
|
|
||||||
const roundTripTime = !!stat.currentRoundTripTime ? stat.currentRoundTripTime * 1000 : -1;
|
|
||||||
this.#$ping!.textContent = roundTripTime === -1 ? '???' : roundTripTime.toString();
|
|
||||||
|
|
||||||
if (PREF_STATS_CONDITIONAL_FORMATTING) {
|
|
||||||
grade = (roundTripTime > 100) ? 'bad' : (roundTripTime > 75) ? 'ok' : (roundTripTime > 40) ? 'good' : '';
|
|
||||||
this.#$ping!.dataset.grade = grade;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
|
||||||
|
|
||||||
const lapsedTime = performance.now() - startTime;
|
if ($element.dataset.grade !== grade) {
|
||||||
this.#timeoutId = window.setTimeout(this.#update.bind(this), this.#updateInterval - lapsedTime);
|
$element.dataset.grade = grade;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
refreshStyles() {
|
refreshStyles() {
|
||||||
const PREF_ITEMS = getPref(PrefKey.STATS_ITEMS);
|
const PREF_ITEMS = getPref(PrefKey.STATS_ITEMS);
|
||||||
const PREF_POSITION = getPref(PrefKey.STATS_POSITION);
|
|
||||||
const PREF_TRANSPARENT = getPref(PrefKey.STATS_TRANSPARENT);
|
|
||||||
const PREF_OPACITY = getPref(PrefKey.STATS_OPACITY);
|
|
||||||
const PREF_TEXT_SIZE = getPref(PrefKey.STATS_TEXT_SIZE);
|
|
||||||
|
|
||||||
const $container = this.#$container!;
|
const $container = this.$container;
|
||||||
$container.dataset.stats = '[' + PREF_ITEMS.join('][') + ']';
|
$container.dataset.stats = '[' + PREF_ITEMS.join('][') + ']';
|
||||||
$container.dataset.position = PREF_POSITION;
|
$container.dataset.position = getPref(PrefKey.STATS_POSITION);
|
||||||
$container.dataset.transparent = PREF_TRANSPARENT;
|
$container.dataset.transparent = getPref(PrefKey.STATS_TRANSPARENT);
|
||||||
$container.style.opacity = PREF_OPACITY + '%';
|
$container.style.opacity = getPref(PrefKey.STATS_OPACITY) + '%';
|
||||||
$container.style.fontSize = PREF_TEXT_SIZE;
|
$container.style.fontSize = getPref(PrefKey.STATS_TEXT_SIZE);
|
||||||
}
|
}
|
||||||
|
|
||||||
hideSettingsUi() {
|
hideSettingsUi() {
|
||||||
@ -219,34 +208,25 @@ export class StreamStats {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#render() {
|
private async render() {
|
||||||
const stats = {
|
this.$container = CE('div', {class: 'bx-stats-bar bx-gone'});
|
||||||
[StreamStat.PING]: [t('stat-ping'), this.#$ping = CE('span', {}, '0')],
|
|
||||||
[StreamStat.FPS]: [t('stat-fps'), this.#$fps = CE('span', {}, '0')],
|
|
||||||
[StreamStat.BITRATE]: [t('stat-bitrate'), this.#$br = CE('span', {}, '0 Mbps')],
|
|
||||||
[StreamStat.DECODE_TIME]: [t('stat-decode-time'), this.#$dt = CE('span', {}, '0ms')],
|
|
||||||
[StreamStat.PACKETS_LOST]: [t('stat-packets-lost'), this.#$pl = CE('span', {}, '0')],
|
|
||||||
[StreamStat.FRAMES_LOST]: [t('stat-frames-lost'), this.#$fl = CE('span', {}, '0')],
|
|
||||||
};
|
|
||||||
|
|
||||||
const $barFragment = document.createDocumentFragment();
|
let statKey: keyof typeof this.stats;
|
||||||
let statKey: keyof typeof stats;
|
for (statKey in this.stats) {
|
||||||
for (statKey in stats) {
|
const stat = this.stats[statKey];
|
||||||
const $div = CE('div', {
|
const $div = CE('div', {
|
||||||
'class': `bx-stat-${statKey}`,
|
class: `bx-stat-${statKey}`,
|
||||||
title: stats[statKey][0]
|
title: stat.name,
|
||||||
},
|
},
|
||||||
CE('label', {}, statKey.toUpperCase()),
|
CE('label', {}, statKey.toUpperCase()),
|
||||||
stats[statKey][1],
|
stat.$element,
|
||||||
);
|
);
|
||||||
|
|
||||||
$barFragment.appendChild($div);
|
this.$container.appendChild($div);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.#$container = CE('div', {'class': 'bx-stats-bar bx-gone'}, $barFragment);
|
|
||||||
this.refreshStyles();
|
this.refreshStyles();
|
||||||
|
document.documentElement.appendChild(this.$container);
|
||||||
document.documentElement.appendChild(this.#$container!);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static setupEvents() {
|
static setupEvents() {
|
||||||
@ -255,8 +235,8 @@ export class StreamStats {
|
|||||||
const PREF_STATS_SHOW_WHEN_PLAYING = getPref(PrefKey.STATS_SHOW_WHEN_PLAYING);
|
const PREF_STATS_SHOW_WHEN_PLAYING = getPref(PrefKey.STATS_SHOW_WHEN_PLAYING);
|
||||||
|
|
||||||
const streamStats = StreamStats.getInstance();
|
const streamStats = StreamStats.getInstance();
|
||||||
// Setup Stat's Quick Glance mode
|
|
||||||
|
|
||||||
|
// Setup Stat's Quick Glance mode
|
||||||
if (PREF_STATS_SHOW_WHEN_PLAYING) {
|
if (PREF_STATS_SHOW_WHEN_PLAYING) {
|
||||||
streamStats.start();
|
streamStats.start();
|
||||||
} else if (PREF_STATS_QUICK_GLANCE) {
|
} else if (PREF_STATS_QUICK_GLANCE) {
|
||||||
|
@ -135,12 +135,6 @@ export class StreamUiHandler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private static handleSystemMenu($streamHud: HTMLElement) {
|
private static handleSystemMenu($streamHud: HTMLElement) {
|
||||||
// Grip handle
|
|
||||||
const $gripHandle = $streamHud.querySelector('button[class^=GripHandle]') as HTMLElement;
|
|
||||||
if (!$gripHandle) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get the last button
|
// Get the last button
|
||||||
const $orgButton = $streamHud.querySelector('div[class^=HUDButton]') as HTMLElement;
|
const $orgButton = $streamHud.querySelector('div[class^=HUDButton]') as HTMLElement;
|
||||||
if (!$orgButton) {
|
if (!$orgButton) {
|
||||||
@ -148,14 +142,14 @@ export class StreamUiHandler {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const hideGripHandle = () => {
|
const hideGripHandle = () => {
|
||||||
if (!$gripHandle) {
|
// Grip handle
|
||||||
return;
|
const $gripHandle = document.querySelector('#StreamHud button[class^=GripHandle]') as HTMLElement;
|
||||||
|
if ($gripHandle && $gripHandle.ariaExpanded === 'true') {
|
||||||
|
$gripHandle.dispatchEvent(new PointerEvent('pointerdown'));
|
||||||
|
$gripHandle.click();
|
||||||
|
$gripHandle.dispatchEvent(new PointerEvent('pointerdown'));
|
||||||
|
$gripHandle.click();
|
||||||
}
|
}
|
||||||
|
|
||||||
$gripHandle.dispatchEvent(new PointerEvent('pointerdown'));
|
|
||||||
$gripHandle.click();
|
|
||||||
$gripHandle.dispatchEvent(new PointerEvent('pointerdown'));
|
|
||||||
$gripHandle.click();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create Stream Settings button
|
// Create Stream Settings button
|
||||||
@ -178,12 +172,12 @@ export class StreamUiHandler {
|
|||||||
let $btnStreamStats = StreamUiHandler.$btnStreamStats;
|
let $btnStreamStats = StreamUiHandler.$btnStreamStats;
|
||||||
if (typeof $btnStreamStats === 'undefined') {
|
if (typeof $btnStreamStats === 'undefined') {
|
||||||
$btnStreamStats = StreamUiHandler.cloneStreamHudButton($orgButton, t('stream-stats'), BxIcon.STREAM_STATS);
|
$btnStreamStats = StreamUiHandler.cloneStreamHudButton($orgButton, t('stream-stats'), BxIcon.STREAM_STATS);
|
||||||
$btnStreamStats?.addEventListener('click', e => {
|
$btnStreamStats?.addEventListener('click', async (e) => {
|
||||||
hideGripHandle();
|
hideGripHandle();
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
|
||||||
// Toggle Stream Stats
|
// Toggle Stream Stats
|
||||||
streamStats.toggle();
|
await streamStats.toggle();
|
||||||
|
|
||||||
const btnStreamStatsOn = (!streamStats.isHidden() && !streamStats.isGlancing());
|
const btnStreamStatsOn = (!streamStats.isHidden() && !streamStats.isGlancing());
|
||||||
$btnStreamStats!.classList.toggle('bx-stream-menu-button-on', btnStreamStatsOn);
|
$btnStreamStats!.classList.toggle('bx-stream-menu-button-on', btnStreamStatsOn);
|
||||||
|
@ -1,9 +1,11 @@
|
|||||||
import { GamepadKey } from "@/enums/mkb";
|
import { GamepadKey } from "@/enums/mkb";
|
||||||
import { EmulatedMkbHandler } from "@/modules/mkb/mkb-handler";
|
import { PrefKey } from "@/enums/pref-keys";
|
||||||
|
import { VIRTUAL_GAMEPAD_ID } from "@/modules/mkb/mkb-handler";
|
||||||
import { BxEvent } from "@/utils/bx-event";
|
import { BxEvent } from "@/utils/bx-event";
|
||||||
import { STATES } from "@/utils/global";
|
import { STATES } from "@/utils/global";
|
||||||
import { CE, isElementVisible } from "@/utils/html";
|
import { CE, isElementVisible } from "@/utils/html";
|
||||||
import { setNearby } from "@/utils/navigation-utils";
|
import { setNearby } from "@/utils/navigation-utils";
|
||||||
|
import { getPref } from "@/utils/settings-storages/global-settings-storage";
|
||||||
|
|
||||||
export enum NavigationDirection {
|
export enum NavigationDirection {
|
||||||
UP = 1,
|
UP = 1,
|
||||||
@ -80,7 +82,7 @@ export abstract class NavigationDialog {
|
|||||||
}
|
}
|
||||||
|
|
||||||
handleGamepad(button: GamepadKey): boolean {
|
handleGamepad(button: GamepadKey): boolean {
|
||||||
return true;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -154,6 +156,57 @@ export class NavigationDialogManager {
|
|||||||
|
|
||||||
// Hide dialog when the Guide menu is shown
|
// Hide dialog when the Guide menu is shown
|
||||||
window.addEventListener(BxEvent.XCLOUD_GUIDE_MENU_SHOWN, e => this.hide());
|
window.addEventListener(BxEvent.XCLOUD_GUIDE_MENU_SHOWN, e => this.hide());
|
||||||
|
|
||||||
|
// Calculate minimum width of controller-friendly <select> elements
|
||||||
|
if (getPref(PrefKey.UI_CONTROLLER_FRIENDLY)) {
|
||||||
|
const observer = new MutationObserver(mutationList => {
|
||||||
|
if (mutationList.length === 0 || mutationList[0].addedNodes.length === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get dialog
|
||||||
|
const $dialog = mutationList[0].addedNodes[0];
|
||||||
|
if (!$dialog || !($dialog instanceof HTMLElement)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find un-calculated <select> elements
|
||||||
|
this.calculateSelectBoxes($dialog);
|
||||||
|
});
|
||||||
|
observer.observe(this.$container, {childList: true});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
calculateSelectBoxes($root: HTMLElement) {
|
||||||
|
const $selects = $root.querySelectorAll('.bx-select:not([data-calculated]) select');
|
||||||
|
$selects.forEach($select => {
|
||||||
|
const $parent = $select.parentElement! as HTMLElement;
|
||||||
|
|
||||||
|
// Don't apply to select.bx-full-width elements
|
||||||
|
if ($parent.classList.contains('bx-full-width')) {
|
||||||
|
$parent.dataset.calculated = 'true';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const rect = $select.getBoundingClientRect();
|
||||||
|
|
||||||
|
let $label;
|
||||||
|
let width = Math.ceil(rect.width);
|
||||||
|
if (!width) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (($select as HTMLSelectElement).multiple) {
|
||||||
|
$label = $parent.querySelector('.bx-select-value') as HTMLElement;
|
||||||
|
width += 20; // Add checkbox's width
|
||||||
|
} else {
|
||||||
|
$label = $parent.querySelector('div') as HTMLElement;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set min-width
|
||||||
|
$label.style.minWidth = width + 'px';
|
||||||
|
$parent.dataset.calculated = 'true';
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
handleEvent(event: Event) {
|
handleEvent(event: Event) {
|
||||||
@ -210,7 +263,7 @@ export class NavigationDialogManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Ignore virtual controller
|
// Ignore virtual controller
|
||||||
if (gamepad.id === EmulatedMkbHandler.VIRTUAL_GAMEPAD_ID) {
|
if (gamepad.id === VIRTUAL_GAMEPAD_ID) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
135
src/modules/ui/dialog/remote-play-dialog.ts
Normal file
135
src/modules/ui/dialog/remote-play-dialog.ts
Normal file
@ -0,0 +1,135 @@
|
|||||||
|
import { ButtonStyle, CE, createButton } from "@/utils/html";
|
||||||
|
import { NavigationDialog, type NavigationElement } from "./navigation-dialog";
|
||||||
|
import { PrefKey } from "@/enums/pref-keys";
|
||||||
|
import { BxIcon } from "@/utils/bx-icon";
|
||||||
|
import { getPref, setPref } from "@/utils/settings-storages/global-settings-storage";
|
||||||
|
import { t } from "@/utils/translation";
|
||||||
|
import { RemotePlayConsoleState, RemotePlayManager } from "@/modules/remote-play-manager";
|
||||||
|
import { BxSelectElement } from "@/web-components/bx-select";
|
||||||
|
import { BxEvent } from "@/utils/bx-event";
|
||||||
|
|
||||||
|
|
||||||
|
export class RemotePlayNavigationDialog extends NavigationDialog {
|
||||||
|
private static instance: RemotePlayNavigationDialog;
|
||||||
|
public static getInstance(): RemotePlayNavigationDialog {
|
||||||
|
if (!RemotePlayNavigationDialog.instance) {
|
||||||
|
RemotePlayNavigationDialog.instance = new RemotePlayNavigationDialog();
|
||||||
|
}
|
||||||
|
return RemotePlayNavigationDialog.instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
private readonly STATE_LABELS: Record<RemotePlayConsoleState, string> = {
|
||||||
|
[RemotePlayConsoleState.ON]: t('powered-on'),
|
||||||
|
[RemotePlayConsoleState.OFF]: t('powered-off'),
|
||||||
|
[RemotePlayConsoleState.STANDBY]: t('standby'),
|
||||||
|
[RemotePlayConsoleState.UNKNOWN]: t('unknown'),
|
||||||
|
};
|
||||||
|
|
||||||
|
$container!: HTMLElement;
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
this.setupDialog();
|
||||||
|
}
|
||||||
|
|
||||||
|
private setupDialog() {
|
||||||
|
const $fragment = CE('div', {'class': 'bx-remote-play-container'});
|
||||||
|
|
||||||
|
const $settingNote = CE('p', {});
|
||||||
|
|
||||||
|
const currentResolution = getPref(PrefKey.REMOTE_PLAY_RESOLUTION);
|
||||||
|
let $resolutions : HTMLSelectElement | NavigationElement = CE<HTMLSelectElement>('select', {},
|
||||||
|
CE('option', {value: '1080p'}, '1080p'),
|
||||||
|
CE('option', {value: '720p'}, '720p'),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (getPref(PrefKey.UI_CONTROLLER_FRIENDLY)) {
|
||||||
|
$resolutions = BxSelectElement.wrap($resolutions as HTMLSelectElement);
|
||||||
|
}
|
||||||
|
|
||||||
|
$resolutions.addEventListener('input', (e: Event) => {
|
||||||
|
const value = (e.target as HTMLSelectElement).value;
|
||||||
|
|
||||||
|
$settingNote.textContent = value === '1080p' ? '✅ ' + t('can-stream-xbox-360-games') : '❌ ' + t('cant-stream-xbox-360-games');
|
||||||
|
setPref(PrefKey.REMOTE_PLAY_RESOLUTION, value);
|
||||||
|
});
|
||||||
|
|
||||||
|
($resolutions as any).value = currentResolution;
|
||||||
|
BxEvent.dispatch($resolutions, 'input', {
|
||||||
|
manualTrigger: true,
|
||||||
|
});
|
||||||
|
|
||||||
|
const $qualitySettings = CE('div', {
|
||||||
|
class: 'bx-remote-play-settings',
|
||||||
|
}, CE('div', {},
|
||||||
|
CE('label', {}, t('target-resolution'), $settingNote),
|
||||||
|
$resolutions,
|
||||||
|
));
|
||||||
|
|
||||||
|
$fragment.appendChild($qualitySettings);
|
||||||
|
|
||||||
|
// Render consoles list
|
||||||
|
const manager = RemotePlayManager.getInstance();
|
||||||
|
const consoles = manager.getConsoles();
|
||||||
|
|
||||||
|
for (let con of consoles) {
|
||||||
|
const $child = CE('div', {class: 'bx-remote-play-device-wrapper'},
|
||||||
|
CE('div', {class: 'bx-remote-play-device-info'},
|
||||||
|
CE('div', {},
|
||||||
|
CE('span', {class: 'bx-remote-play-device-name'}, con.deviceName),
|
||||||
|
CE('span', {class: 'bx-remote-play-console-type'}, con.consoleType.replace('Xbox', ''))
|
||||||
|
),
|
||||||
|
CE('div', {class: 'bx-remote-play-power-state'}, this.STATE_LABELS[con.powerState]),
|
||||||
|
),
|
||||||
|
|
||||||
|
// Connect button
|
||||||
|
createButton({
|
||||||
|
classes: ['bx-remote-play-connect-button'],
|
||||||
|
label: t('console-connect'),
|
||||||
|
style: ButtonStyle.PRIMARY | ButtonStyle.FOCUSABLE,
|
||||||
|
onClick: e => manager.play(con.serverId),
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
|
||||||
|
$fragment.appendChild($child);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add buttons
|
||||||
|
$fragment.appendChild(
|
||||||
|
CE('div', {
|
||||||
|
class: 'bx-remote-play-buttons',
|
||||||
|
_nearby: {
|
||||||
|
orientation: 'horizontal',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
createButton({
|
||||||
|
icon: BxIcon.QUESTION,
|
||||||
|
style: ButtonStyle.GHOST | ButtonStyle.FOCUSABLE,
|
||||||
|
url: 'https://better-xcloud.github.io/remote-play',
|
||||||
|
label: t('help'),
|
||||||
|
}),
|
||||||
|
|
||||||
|
createButton({
|
||||||
|
style: ButtonStyle.GHOST | ButtonStyle.FOCUSABLE,
|
||||||
|
label: t('close'),
|
||||||
|
onClick: e => this.hide(),
|
||||||
|
}),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
this.$container = $fragment;
|
||||||
|
}
|
||||||
|
|
||||||
|
getDialog(): NavigationDialog {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
getContent(): HTMLElement {
|
||||||
|
return this.$container;
|
||||||
|
}
|
||||||
|
|
||||||
|
focusIfNeeded(): void {
|
||||||
|
const $btnConnect = this.$container.querySelector('.bx-remote-play-device-wrapper button') as HTMLElement;
|
||||||
|
$btnConnect && $btnConnect.focus();
|
||||||
|
}
|
||||||
|
}
|
@ -1,3 +1,5 @@
|
|||||||
|
import { isFullVersion } from "@macros/build" with {type: "macro"};
|
||||||
|
|
||||||
import { onChangeVideoPlayerType, updateVideoPlayer } from "@/modules/stream/stream-settings-utils";
|
import { onChangeVideoPlayerType, updateVideoPlayer } from "@/modules/stream/stream-settings-utils";
|
||||||
import { ButtonStyle, CE, createButton, createSvgIcon, removeChildElements, type BxButton } from "@/utils/html";
|
import { ButtonStyle, CE, createButton, createSvgIcon, removeChildElements, type BxButton } from "@/utils/html";
|
||||||
import { NavigationDialog, NavigationDirection } from "./navigation-dialog";
|
import { NavigationDialog, NavigationDirection } from "./navigation-dialog";
|
||||||
@ -10,7 +12,7 @@ import { TouchController } from "@/modules/touch-controller";
|
|||||||
import { VibrationManager } from "@/modules/vibration-manager";
|
import { VibrationManager } from "@/modules/vibration-manager";
|
||||||
import { BxEvent } from "@/utils/bx-event";
|
import { BxEvent } from "@/utils/bx-event";
|
||||||
import { BxIcon } from "@/utils/bx-icon";
|
import { BxIcon } from "@/utils/bx-icon";
|
||||||
import { STATES, AppInterface, deepClone, SCRIPT_VERSION, STORAGE } from "@/utils/global";
|
import { STATES, AppInterface, deepClone, SCRIPT_VERSION, STORAGE, SCRIPT_VARIANT } from "@/utils/global";
|
||||||
import { t, Translations } from "@/utils/translation";
|
import { t, Translations } from "@/utils/translation";
|
||||||
import { BxSelectElement } from "@/web-components/bx-select";
|
import { BxSelectElement } from "@/web-components/bx-select";
|
||||||
import { setNearby } from "@/utils/navigation-utils";
|
import { setNearby } from "@/utils/navigation-utils";
|
||||||
@ -38,6 +40,7 @@ type SettingTabContentItem = Partial<{
|
|||||||
onChange: (e: any, value: number) => void;
|
onChange: (e: any, value: number) => void;
|
||||||
onCreated: (setting: SettingTabContentItem, $control: any) => void;
|
onCreated: (setting: SettingTabContentItem, $control: any) => void;
|
||||||
params: any;
|
params: any;
|
||||||
|
requiredVariants?: BuildVariant | Array<BuildVariant>;
|
||||||
}>
|
}>
|
||||||
|
|
||||||
type SettingTabContent = {
|
type SettingTabContent = {
|
||||||
@ -48,12 +51,14 @@ type SettingTabContent = {
|
|||||||
helpUrl?: string;
|
helpUrl?: string;
|
||||||
content?: any;
|
content?: any;
|
||||||
items?: Array<SettingTabContentItem | PrefKey | (($parent: HTMLElement) => void) | false>;
|
items?: Array<SettingTabContentItem | PrefKey | (($parent: HTMLElement) => void) | false>;
|
||||||
|
requiredVariants?: BuildVariant | Array<BuildVariant>;
|
||||||
};
|
};
|
||||||
|
|
||||||
type SettingTab = {
|
type SettingTab = {
|
||||||
icon: SVGElement;
|
icon: SVGElement;
|
||||||
group: 'global';
|
group: 'global';
|
||||||
items: Array<SettingTabContent | false>;
|
items: Array<SettingTabContent | false>;
|
||||||
|
requiredVariants?: BuildVariant | Array<BuildVariant>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export class SettingsNavigationDialog extends NavigationDialog {
|
export class SettingsNavigationDialog extends NavigationDialog {
|
||||||
@ -205,12 +210,14 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
PrefKey.STREAM_COMBINE_SOURCES,
|
PrefKey.STREAM_COMBINE_SOURCES,
|
||||||
],
|
],
|
||||||
}, {
|
}, {
|
||||||
|
requiredVariants: 'full',
|
||||||
group: 'co-op',
|
group: 'co-op',
|
||||||
label: t('local-co-op'),
|
label: t('local-co-op'),
|
||||||
items: [
|
items: [
|
||||||
PrefKey.LOCAL_CO_OP_ENABLED,
|
PrefKey.LOCAL_CO_OP_ENABLED,
|
||||||
],
|
],
|
||||||
}, {
|
}, {
|
||||||
|
requiredVariants: 'full',
|
||||||
group: 'mkb',
|
group: 'mkb',
|
||||||
label: t('mouse-and-keyboard'),
|
label: t('mouse-and-keyboard'),
|
||||||
items: [
|
items: [
|
||||||
@ -219,6 +226,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
PrefKey.MKB_HIDE_IDLE_CURSOR,
|
PrefKey.MKB_HIDE_IDLE_CURSOR,
|
||||||
],
|
],
|
||||||
}, {
|
}, {
|
||||||
|
requiredVariants: 'full',
|
||||||
group: 'touch-control',
|
group: 'touch-control',
|
||||||
label: t('touch-controller'),
|
label: t('touch-controller'),
|
||||||
note: !STATES.userAgent.capabilities.touch ? '⚠️ ' + t('device-unsupported-touch') : null,
|
note: !STATES.userAgent.capabilities.touch ? '⚠️ ' + t('device-unsupported-touch') : null,
|
||||||
@ -247,6 +255,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
PrefKey.UI_HIDE_SECTIONS,
|
PrefKey.UI_HIDE_SECTIONS,
|
||||||
],
|
],
|
||||||
}, {
|
}, {
|
||||||
|
requiredVariants: 'full',
|
||||||
group: 'game-bar',
|
group: 'game-bar',
|
||||||
label: t('game-bar'),
|
label: t('game-bar'),
|
||||||
items: [
|
items: [
|
||||||
@ -357,6 +366,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
}];
|
}];
|
||||||
|
|
||||||
private readonly TAB_DISPLAY_ITEMS: Array<SettingTabContent | false> = [{
|
private readonly TAB_DISPLAY_ITEMS: Array<SettingTabContent | false> = [{
|
||||||
|
requiredVariants: 'full',
|
||||||
group: 'audio',
|
group: 'audio',
|
||||||
label: t('audio'),
|
label: t('audio'),
|
||||||
helpUrl: 'https://better-xcloud.github.io/ingame-features/#audio',
|
helpUrl: 'https://better-xcloud.github.io/ingame-features/#audio',
|
||||||
@ -441,7 +451,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
}],
|
}],
|
||||||
},
|
},
|
||||||
|
|
||||||
STATES.userAgent.capabilities.touch && {
|
isFullVersion() && STATES.userAgent.capabilities.touch && {
|
||||||
group: 'touch-control',
|
group: 'touch-control',
|
||||||
label: t('touch-controller'),
|
label: t('touch-controller'),
|
||||||
items: [{
|
items: [{
|
||||||
@ -499,18 +509,19 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
group: 'mkb',
|
group: 'mkb',
|
||||||
label: t('virtual-controller'),
|
label: t('virtual-controller'),
|
||||||
helpUrl: 'https://better-xcloud.github.io/mouse-and-keyboard/',
|
helpUrl: 'https://better-xcloud.github.io/mouse-and-keyboard/',
|
||||||
content: MkbRemapper.INSTANCE.render(),
|
content: isFullVersion() && MkbRemapper.INSTANCE.render(),
|
||||||
}];
|
}];
|
||||||
|
|
||||||
private readonly TAB_NATIVE_MKB_ITEMS: Array<SettingTabContent | false> = [{
|
private readonly TAB_NATIVE_MKB_ITEMS: Array<SettingTabContent | false> = [{
|
||||||
|
requiredVariants: 'full',
|
||||||
group: 'native-mkb',
|
group: 'native-mkb',
|
||||||
label: t('native-mkb'),
|
label: t('native-mkb'),
|
||||||
items: [{
|
items: [isFullVersion() && {
|
||||||
pref: PrefKey.NATIVE_MKB_SCROLL_VERTICAL_SENSITIVITY,
|
pref: PrefKey.NATIVE_MKB_SCROLL_VERTICAL_SENSITIVITY,
|
||||||
onChange: (e: any, value: number) => {
|
onChange: (e: any, value: number) => {
|
||||||
NativeMkbHandler.getInstance().setVerticalScrollMultiplier(value / 100);
|
NativeMkbHandler.getInstance().setVerticalScrollMultiplier(value / 100);
|
||||||
},
|
},
|
||||||
}, {
|
}, isFullVersion() && {
|
||||||
pref: PrefKey.NATIVE_MKB_SCROLL_HORIZONTAL_SENSITIVITY,
|
pref: PrefKey.NATIVE_MKB_SCROLL_HORIZONTAL_SENSITIVITY,
|
||||||
onChange: (e: any, value: number) => {
|
onChange: (e: any, value: number) => {
|
||||||
NativeMkbHandler.getInstance().setHorizontalScrollMultiplier(value / 100);
|
NativeMkbHandler.getInstance().setHorizontalScrollMultiplier(value / 100);
|
||||||
@ -519,9 +530,10 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
}];
|
}];
|
||||||
|
|
||||||
private readonly TAB_SHORTCUTS_ITEMS: Array<SettingTabContent | false> = [{
|
private readonly TAB_SHORTCUTS_ITEMS: Array<SettingTabContent | false> = [{
|
||||||
|
requiredVariants: 'full',
|
||||||
group: 'controller-shortcuts',
|
group: 'controller-shortcuts',
|
||||||
label: t('controller-shortcuts'),
|
label: t('controller-shortcuts'),
|
||||||
content: ControllerShortcut.renderSettings(),
|
content: isFullVersion() && ControllerShortcut.renderSettings(),
|
||||||
}];
|
}];
|
||||||
|
|
||||||
private readonly TAB_STATS_ITEMS: Array<SettingTabContent | false> = [{
|
private readonly TAB_STATS_ITEMS: Array<SettingTabContent | false> = [{
|
||||||
@ -575,24 +587,28 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
icon: BxIcon.CONTROLLER,
|
icon: BxIcon.CONTROLLER,
|
||||||
group: 'controller',
|
group: 'controller',
|
||||||
items: this.TAB_CONTROLLER_ITEMS,
|
items: this.TAB_CONTROLLER_ITEMS,
|
||||||
|
requiredVariants: 'full',
|
||||||
},
|
},
|
||||||
|
|
||||||
getPref(PrefKey.MKB_ENABLED) && {
|
isFullVersion() && getPref(PrefKey.MKB_ENABLED) && {
|
||||||
icon: BxIcon.VIRTUAL_CONTROLLER,
|
icon: BxIcon.VIRTUAL_CONTROLLER,
|
||||||
group: 'mkb',
|
group: 'mkb',
|
||||||
items: this.TAB_VIRTUAL_CONTROLLER_ITEMS,
|
items: this.TAB_VIRTUAL_CONTROLLER_ITEMS,
|
||||||
|
requiredVariants: 'full',
|
||||||
},
|
},
|
||||||
|
|
||||||
AppInterface && getPref(PrefKey.NATIVE_MKB_ENABLED) === 'on' && {
|
isFullVersion() && AppInterface && getPref(PrefKey.NATIVE_MKB_ENABLED) === 'on' && {
|
||||||
icon: BxIcon.NATIVE_MKB,
|
icon: BxIcon.NATIVE_MKB,
|
||||||
group: 'native-mkb',
|
group: 'native-mkb',
|
||||||
items: this.TAB_NATIVE_MKB_ITEMS,
|
items: this.TAB_NATIVE_MKB_ITEMS,
|
||||||
|
requiredVariants: 'full',
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
icon: BxIcon.COMMAND,
|
icon: BxIcon.COMMAND,
|
||||||
group: 'shortcuts',
|
group: 'shortcuts',
|
||||||
items: this.TAB_SHORTCUTS_ITEMS,
|
items: this.TAB_SHORTCUTS_ITEMS,
|
||||||
|
requiredVariants: 'full',
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
@ -715,6 +731,15 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private isSupportedVariant(requiredVariants: BuildVariant | Array<BuildVariant> | undefined) {
|
||||||
|
if (typeof requiredVariants === 'undefined') {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
requiredVariants = typeof requiredVariants === 'string' ? [requiredVariants] : requiredVariants;
|
||||||
|
return requiredVariants.includes(SCRIPT_VARIANT);
|
||||||
|
}
|
||||||
|
|
||||||
private async renderSuggestions(e: Event) {
|
private async renderSuggestions(e: Event) {
|
||||||
const $btnSuggest = (e.target as HTMLElement).closest('div')!;
|
const $btnSuggest = (e.target as HTMLElement).closest('div')!;
|
||||||
$btnSuggest.toggleAttribute('bx-open');
|
$btnSuggest.toggleAttribute('bx-open');
|
||||||
@ -943,6 +968,11 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
for (const $child of Array.from(this.$settings.children)) {
|
for (const $child of Array.from(this.$settings.children)) {
|
||||||
if ($child.getAttribute('data-tab-group') === settingTab.group) {
|
if ($child.getAttribute('data-tab-group') === settingTab.group) {
|
||||||
$child.classList.remove('bx-gone');
|
$child.classList.remove('bx-gone');
|
||||||
|
|
||||||
|
// Calculate size of controller-friendly select boxes
|
||||||
|
if (getPref(PrefKey.UI_CONTROLLER_FRIENDLY)) {
|
||||||
|
this.dialogManager.calculateSelectBoxes($child as HTMLElement);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
$child.classList.add('bx-gone');
|
$child.classList.add('bx-gone');
|
||||||
}
|
}
|
||||||
@ -961,7 +991,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
|
|
||||||
private onGlobalSettingChanged(e: Event) {
|
private onGlobalSettingChanged(e: Event) {
|
||||||
// Clear PatcherCache;
|
// Clear PatcherCache;
|
||||||
PatcherCache.clear();
|
isFullVersion() && PatcherCache.clear();
|
||||||
|
|
||||||
this.$btnReload.classList.add('bx-danger');
|
this.$btnReload.classList.add('bx-danger');
|
||||||
|
|
||||||
@ -1096,6 +1126,10 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
prefDefinition = getPrefDefinition(pref);
|
prefDefinition = getPrefDefinition(pref);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (prefDefinition && !this.isSupportedVariant(prefDefinition.requiredVariants)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
let label = prefDefinition?.label || setting.label;
|
let label = prefDefinition?.label || setting.label;
|
||||||
let note = prefDefinition?.note || setting.note;
|
let note = prefDefinition?.note || setting.note;
|
||||||
const experimental = prefDefinition?.experimental || setting.experimental;
|
const experimental = prefDefinition?.experimental || setting.experimental;
|
||||||
@ -1118,6 +1152,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
}
|
}
|
||||||
|
|
||||||
let $label;
|
let $label;
|
||||||
|
|
||||||
const $row = CE('label', {
|
const $row = CE('label', {
|
||||||
class: 'bx-settings-row',
|
class: 'bx-settings-row',
|
||||||
for: `bx_setting_${pref}`,
|
for: `bx_setting_${pref}`,
|
||||||
@ -1128,10 +1163,9 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
},
|
},
|
||||||
$label = CE('span', {class: 'bx-settings-label'},
|
$label = CE('span', {class: 'bx-settings-label'},
|
||||||
label,
|
label,
|
||||||
note && CE('div', {class: 'bx-settings-dialog-note'}, note),
|
note ? CE('div', {class: 'bx-settings-dialog-note'}, note) : prefDefinition?.unsupported && CE('div', {class: 'bx-settings-dialog-note'}, t('browser-unsupported-feature')),
|
||||||
setting.unsupported && CE('div', {class: 'bx-settings-dialog-note'}, t('browser-unsupported-feature')),
|
|
||||||
),
|
),
|
||||||
!setting.unsupported && $control,
|
!prefDefinition?.unsupported && $control,
|
||||||
);
|
);
|
||||||
|
|
||||||
// Make link inside <label> focusable
|
// Make link inside <label> focusable
|
||||||
@ -1144,7 +1178,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
}
|
}
|
||||||
|
|
||||||
$tabContent.appendChild($row);
|
$tabContent.appendChild($row);
|
||||||
setting.onCreated && setting.onCreated(setting, $control);
|
!prefDefinition?.unsupported && setting.onCreated && setting.onCreated(setting, $control);
|
||||||
}
|
}
|
||||||
|
|
||||||
private setupDialog() {
|
private setupDialog() {
|
||||||
@ -1232,6 +1266,11 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Don't render unsupported build variant
|
||||||
|
if (!this.isSupportedVariant(settingTab.requiredVariants)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// Don't render other tabs in unsupported regions
|
// Don't render other tabs in unsupported regions
|
||||||
if (settingTab.group !== 'global' && !this.renderFullSettings) {
|
if (settingTab.group !== 'global' && !this.renderFullSettings) {
|
||||||
continue;
|
continue;
|
||||||
@ -1250,6 +1289,10 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!this.isSupportedVariant(settingTabContent.requiredVariants)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// Don't render other settings in unsupported regions
|
// Don't render other settings in unsupported regions
|
||||||
if (!this.renderFullSettings && settingTab.group === 'global' && settingTabContent.group !== 'general' && settingTabContent.group !== 'footer') {
|
if (!this.renderFullSettings && settingTab.group === 'global' && settingTabContent.group !== 'general' && settingTabContent.group !== 'footer') {
|
||||||
continue;
|
continue;
|
||||||
@ -1260,6 +1303,11 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
// If label is "Better xCloud" => create a link to Releases page
|
// If label is "Better xCloud" => create a link to Releases page
|
||||||
if (label === t('better-xcloud')) {
|
if (label === t('better-xcloud')) {
|
||||||
label += ' ' + SCRIPT_VERSION;
|
label += ' ' + SCRIPT_VERSION;
|
||||||
|
|
||||||
|
if (SCRIPT_VARIANT === 'lite') {
|
||||||
|
label += ' (Lite)';
|
||||||
|
}
|
||||||
|
|
||||||
label = createButton({
|
label = createButton({
|
||||||
label: label,
|
label: label,
|
||||||
url: 'https://github.com/redphx/better-xcloud/releases',
|
url: 'https://github.com/redphx/better-xcloud/releases',
|
||||||
|
@ -1,27 +1,11 @@
|
|||||||
import { BxEvent } from "@/utils/bx-event";
|
import { BxEvent } from "@/utils/bx-event";
|
||||||
import { BxIcon } from "@/utils/bx-icon";
|
import { BxIcon } from "@/utils/bx-icon";
|
||||||
import { CE, createSvgIcon, getReactProps, isElementVisible } from "@/utils/html";
|
import { CE, createSvgIcon, getReactProps, isElementVisible, secondsToHms } from "@/utils/html";
|
||||||
import { XcloudApi } from "@/utils/xcloud-api";
|
import { XcloudApi } from "@/utils/xcloud-api";
|
||||||
|
|
||||||
export class GameTile {
|
export class GameTile {
|
||||||
static #timeout: number | null;
|
static #timeout: number | null;
|
||||||
|
|
||||||
static #secondsToHms(seconds: number) {
|
|
||||||
let h = Math.floor(seconds / 3600);
|
|
||||||
seconds %= 3600;
|
|
||||||
let m = Math.floor(seconds / 60);
|
|
||||||
let s = seconds % 60;
|
|
||||||
|
|
||||||
const output = [];
|
|
||||||
h > 0 && output.push(`${h}h`);
|
|
||||||
m > 0 && output.push(`${m}m`);
|
|
||||||
if (s > 0 || output.length === 0) {
|
|
||||||
output.push(`${s}s`);
|
|
||||||
}
|
|
||||||
|
|
||||||
return output.join(' ');
|
|
||||||
}
|
|
||||||
|
|
||||||
static async #showWaitTime($elm: HTMLElement, productId: string) {
|
static async #showWaitTime($elm: HTMLElement, productId: string) {
|
||||||
if (($elm as any).hasWaitTime) {
|
if (($elm as any).hasWaitTime) {
|
||||||
return;
|
return;
|
||||||
@ -42,7 +26,7 @@ export class GameTile {
|
|||||||
if (typeof totalWaitTime === 'number' && isElementVisible($elm)) {
|
if (typeof totalWaitTime === 'number' && isElementVisible($elm)) {
|
||||||
const $div = CE('div', {'class': 'bx-game-tile-wait-time'},
|
const $div = CE('div', {'class': 'bx-game-tile-wait-time'},
|
||||||
createSvgIcon(BxIcon.PLAYTIME),
|
createSvgIcon(BxIcon.PLAYTIME),
|
||||||
CE('span', {}, GameTile.#secondsToHms(totalWaitTime)),
|
CE('span', {}, secondsToHms(totalWaitTime)),
|
||||||
);
|
);
|
||||||
$elm.insertAdjacentElement('afterbegin', $div);
|
$elm.insertAdjacentElement('afterbegin', $div);
|
||||||
}
|
}
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
import { isFullVersion } from "@macros/build" with {type: "macro"};
|
||||||
|
|
||||||
import { BxEvent } from "@/utils/bx-event";
|
import { BxEvent } from "@/utils/bx-event";
|
||||||
import { AppInterface, STATES } from "@/utils/global";
|
import { AppInterface, STATES } from "@/utils/global";
|
||||||
import { createButton, ButtonStyle, CE } from "@/utils/html";
|
import { createButton, ButtonStyle, CE } from "@/utils/html";
|
||||||
@ -22,7 +24,7 @@ export class GuideMenu {
|
|||||||
}, {once: true});
|
}, {once: true});
|
||||||
|
|
||||||
// Close all xCloud's dialogs
|
// Close all xCloud's dialogs
|
||||||
window.BX_EXPOSED.dialogRoutes.closeAll();
|
GuideMenu.#closeGuideMenu();
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
|
|
||||||
@ -53,7 +55,7 @@ export class GuideMenu {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Close all xCloud's dialogs
|
// Close all xCloud's dialogs
|
||||||
window.BX_EXPOSED.dialogRoutes.closeAll();
|
GuideMenu.#closeGuideMenu();
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
|
|
||||||
@ -66,7 +68,7 @@ export class GuideMenu {
|
|||||||
confirm(t('back-to-home-confirm')) && (window.location.href = window.location.href.substring(0, 31));
|
confirm(t('back-to-home-confirm')) && (window.location.href = window.location.href.substring(0, 31));
|
||||||
|
|
||||||
// Close all xCloud's dialogs
|
// Close all xCloud's dialogs
|
||||||
window.BX_EXPOSED.dialogRoutes.closeAll();
|
GuideMenu.#closeGuideMenu();
|
||||||
},
|
},
|
||||||
attributes: {
|
attributes: {
|
||||||
'data-state': 'playing',
|
'data-state': 'playing',
|
||||||
@ -76,6 +78,17 @@ export class GuideMenu {
|
|||||||
|
|
||||||
static #$renderedButtons: HTMLElement;
|
static #$renderedButtons: HTMLElement;
|
||||||
|
|
||||||
|
static #closeGuideMenu() {
|
||||||
|
if (window.BX_EXPOSED.dialogRoutes) {
|
||||||
|
window.BX_EXPOSED.dialogRoutes.closeAll();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Use alternative method for Lite version
|
||||||
|
const $btnClose = document.querySelector('#gamepass-dialog-root button[class^=Header-module__closeButton]') as HTMLElement;
|
||||||
|
$btnClose && $btnClose.click();
|
||||||
|
}
|
||||||
|
|
||||||
static #renderButtons() {
|
static #renderButtons() {
|
||||||
if (GuideMenu.#$renderedButtons) {
|
if (GuideMenu.#$renderedButtons) {
|
||||||
return GuideMenu.#$renderedButtons;
|
return GuideMenu.#$renderedButtons;
|
||||||
@ -115,9 +128,11 @@ export class GuideMenu {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static #injectHome($root: HTMLElement, isPlaying = false) {
|
static #injectHome($root: HTMLElement, isPlaying = false) {
|
||||||
const $achievementsProgress = $root.querySelector('button[class*=AchievementsButton-module__progressBarContainer]');
|
if (isFullVersion()) {
|
||||||
if ($achievementsProgress) {
|
const $achievementsProgress = $root.querySelector('button[class*=AchievementsButton-module__progressBarContainer]');
|
||||||
TrueAchievements.injectAchievementsProgress($achievementsProgress as HTMLElement);
|
if ($achievementsProgress) {
|
||||||
|
TrueAchievements.injectAchievementsProgress($achievementsProgress as HTMLElement);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find the element to add buttons to
|
// Find the element to add buttons to
|
||||||
@ -162,7 +177,7 @@ export class GuideMenu {
|
|||||||
static observe($addedElm: HTMLElement) {
|
static observe($addedElm: HTMLElement) {
|
||||||
const className = $addedElm.className;
|
const className = $addedElm.className;
|
||||||
|
|
||||||
if (className.includes('AchievementsButton-module__progressBarContainer')) {
|
if (isFullVersion() && className.includes('AchievementsButton-module__progressBarContainer')) {
|
||||||
TrueAchievements.injectAchievementsProgress($addedElm);
|
TrueAchievements.injectAchievementsProgress($addedElm);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -174,10 +189,12 @@ export class GuideMenu {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Achievement Details page
|
// Achievement Details page
|
||||||
const $achievDetailPage = $addedElm.querySelector('div[class*=AchievementDetailPage]');
|
if (isFullVersion()) {
|
||||||
if ($achievDetailPage) {
|
const $achievDetailPage = $addedElm.querySelector('div[class*=AchievementDetailPage]');
|
||||||
TrueAchievements.injectAchievementDetailPage($achievDetailPage as HTMLElement);
|
if ($achievDetailPage) {
|
||||||
return;
|
TrueAchievements.injectAchievementDetailPage($achievDetailPage as HTMLElement);
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find navigation bar
|
// Find navigation bar
|
||||||
|
@ -2,7 +2,7 @@ import { SCRIPT_VERSION } from "@utils/global";
|
|||||||
import { createButton, ButtonStyle, CE, isElementVisible } from "@utils/html";
|
import { createButton, ButtonStyle, CE, isElementVisible } from "@utils/html";
|
||||||
import { BxIcon } from "@utils/bx-icon";
|
import { BxIcon } from "@utils/bx-icon";
|
||||||
import { getPreferredServerRegion } from "@utils/region";
|
import { getPreferredServerRegion } from "@utils/region";
|
||||||
import { RemotePlay } from "@modules/remote-play";
|
import { RemotePlayManager } from "@/modules/remote-play-manager";
|
||||||
import { t } from "@utils/translation";
|
import { t } from "@utils/translation";
|
||||||
import { SettingsNavigationDialog } from "./dialog/settings-dialog";
|
import { SettingsNavigationDialog } from "./dialog/settings-dialog";
|
||||||
import { PrefKey } from "@/enums/pref-keys";
|
import { PrefKey } from "@/enums/pref-keys";
|
||||||
@ -15,7 +15,7 @@ export class HeaderSection {
|
|||||||
title: t('remote-play'),
|
title: t('remote-play'),
|
||||||
style: ButtonStyle.GHOST | ButtonStyle.FOCUSABLE | ButtonStyle.CIRCULAR,
|
style: ButtonStyle.GHOST | ButtonStyle.FOCUSABLE | ButtonStyle.CIRCULAR,
|
||||||
onClick: e => {
|
onClick: e => {
|
||||||
RemotePlay.togglePopup();
|
RemotePlayManager.getInstance().togglePopup();
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -63,7 +63,7 @@ export class HeaderSection {
|
|||||||
static checkHeader() {
|
static checkHeader() {
|
||||||
let $target = document.querySelector('#PageContent div[class*=EdgewaterHeader-module__rightSectionSpacing]');
|
let $target = document.querySelector('#PageContent div[class*=EdgewaterHeader-module__rightSectionSpacing]');
|
||||||
if (!$target) {
|
if (!$target) {
|
||||||
$target = document.querySelector("div[class^=UnsupportedMarketPage-module__buttons]");
|
$target = document.querySelector('div[class^=UnsupportedMarketPage-module__buttons]');
|
||||||
}
|
}
|
||||||
|
|
||||||
$target && HeaderSection.#injectSettingsButton($target as HTMLElement);
|
$target && HeaderSection.#injectSettingsButton($target as HTMLElement);
|
||||||
|
@ -1,12 +1,11 @@
|
|||||||
import { BX_FLAGS } from "@/utils/bx-flags";
|
import { BX_FLAGS } from "@/utils/bx-flags";
|
||||||
import { BxIcon } from "@/utils/bx-icon";
|
import { BxIcon } from "@/utils/bx-icon";
|
||||||
import { AppInterface } from "@/utils/global";
|
import { AppInterface } from "@/utils/global";
|
||||||
import { ButtonStyle, createButton } from "@/utils/html";
|
import { ButtonStyle, CE, createButton } from "@/utils/html";
|
||||||
import { t } from "@/utils/translation";
|
import { t } from "@/utils/translation";
|
||||||
|
|
||||||
export class ProductDetailsPage {
|
export class ProductDetailsPage {
|
||||||
private static $btnShortcut = AppInterface && createButton({
|
private static $btnShortcut = AppInterface && createButton({
|
||||||
classes: ['bx-button-shortcut'],
|
|
||||||
icon: BxIcon.CREATE_SHORTCUT,
|
icon: BxIcon.CREATE_SHORTCUT,
|
||||||
label: t('create-shortcut'),
|
label: t('create-shortcut'),
|
||||||
style: ButtonStyle.FOCUSABLE,
|
style: ButtonStyle.FOCUSABLE,
|
||||||
@ -17,7 +16,6 @@ export class ProductDetailsPage {
|
|||||||
});
|
});
|
||||||
|
|
||||||
private static $btnWallpaper = AppInterface && createButton({
|
private static $btnWallpaper = AppInterface && createButton({
|
||||||
classes: ['bx-button-shortcut'],
|
|
||||||
icon: BxIcon.DOWNLOAD,
|
icon: BxIcon.DOWNLOAD,
|
||||||
label: t('wallpaper'),
|
label: t('wallpaper'),
|
||||||
style: ButtonStyle.FOCUSABLE,
|
style: ButtonStyle.FOCUSABLE,
|
||||||
@ -48,17 +46,12 @@ export class ProductDetailsPage {
|
|||||||
// Find action buttons container
|
// Find action buttons container
|
||||||
const $container = document.querySelector('div[class*=ActionButtons-module__container]');
|
const $container = document.querySelector('div[class*=ActionButtons-module__container]');
|
||||||
if ($container && $container.parentElement) {
|
if ($container && $container.parentElement) {
|
||||||
const fragment = document.createDocumentFragment();
|
$container.parentElement.appendChild(CE('div', {
|
||||||
|
class: 'bx-product-details-buttons',
|
||||||
// Shortcut button
|
},
|
||||||
if (BX_FLAGS.DeviceInfo.deviceType === 'android') {
|
BX_FLAGS.DeviceInfo.deviceType === 'android' && ProductDetailsPage.$btnShortcut,
|
||||||
fragment.appendChild(ProductDetailsPage.$btnShortcut);
|
ProductDetailsPage.$btnWallpaper,
|
||||||
}
|
));
|
||||||
|
|
||||||
// Wallpaper button
|
|
||||||
fragment.appendChild(ProductDetailsPage.$btnWallpaper);
|
|
||||||
|
|
||||||
$container.parentElement.appendChild(fragment);
|
|
||||||
}
|
}
|
||||||
}, 500);
|
}, 500);
|
||||||
}
|
}
|
||||||
|
4
src/types/index.d.ts
vendored
4
src/types/index.d.ts
vendored
@ -1,7 +1,9 @@
|
|||||||
|
type BuildVariant = 'full' | 'lite';
|
||||||
|
|
||||||
// Get type of an array's element
|
// Get type of an array's element
|
||||||
type ArrayElement<ArrayType extends readonly unknown[]> = ArrayType extends readonly (infer ElementType)[] ? ElementType : never;
|
type ArrayElement<ArrayType extends readonly unknown[]> = ArrayType extends readonly (infer ElementType)[] ? ElementType : never;
|
||||||
|
|
||||||
type PartialRecord<K extends keyof any, T> = Partial<Record<K, T>>
|
type PartialRecord<K extends keyof any, T> = Partial<Record<K, T>>;
|
||||||
|
|
||||||
interface Window {
|
interface Window {
|
||||||
AppInterface: any;
|
AppInterface: any;
|
||||||
|
1
src/types/setting-definition.d.ts
vendored
1
src/types/setting-definition.d.ts
vendored
@ -24,6 +24,7 @@ export type SettingDefinition = {
|
|||||||
suggest: PartialRecord<SuggestedSettingCategory, any>,
|
suggest: PartialRecord<SuggestedSettingCategory, any>,
|
||||||
ready: (setting: SettingDefinition) => void;
|
ready: (setting: SettingDefinition) => void;
|
||||||
type: SettingElementType,
|
type: SettingElementType,
|
||||||
|
requiredVariants: BuildVariant | Array<BuildVariant>;
|
||||||
// migrate?: (this: Preferences, savedPrefs: any, value: any) => void;
|
// migrate?: (this: Preferences, savedPrefs: any, value: any) => void;
|
||||||
}> & (
|
}> & (
|
||||||
{} | {
|
{} | {
|
||||||
|
2
src/types/stream-stats.d.ts
vendored
2
src/types/stream-stats.d.ts
vendored
@ -3,6 +3,8 @@ type RTCBasicStat = {
|
|||||||
bytesReceived: number,
|
bytesReceived: number,
|
||||||
clockRate: number,
|
clockRate: number,
|
||||||
codecId: string,
|
codecId: string,
|
||||||
|
frameWidth: number,
|
||||||
|
frameHeight: number,
|
||||||
framesDecoded: number,
|
framesDecoded: number,
|
||||||
id: string,
|
id: string,
|
||||||
kind: string,
|
kind: string,
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
import { isFullVersion } from "@macros/build" with {type: "macro"};
|
||||||
|
|
||||||
import { ControllerShortcut } from "@/modules/controller-shortcut";
|
import { ControllerShortcut } from "@/modules/controller-shortcut";
|
||||||
import { BxEvent } from "@utils/bx-event";
|
import { BxEvent } from "@utils/bx-event";
|
||||||
import { deepClone, STATES } from "@utils/global";
|
import { deepClone, STATES } from "@utils/global";
|
||||||
@ -20,7 +22,7 @@ export type SupportedInputTypeValue = (typeof SupportedInputType)[keyof typeof S
|
|||||||
export const BxExposed = {
|
export const BxExposed = {
|
||||||
getTitleInfo: () => STATES.currentStream.titleInfo,
|
getTitleInfo: () => STATES.currentStream.titleInfo,
|
||||||
|
|
||||||
modifyTitleInfo: (titleInfo: XcloudTitleInfo): XcloudTitleInfo => {
|
modifyTitleInfo: isFullVersion() && function(titleInfo: XcloudTitleInfo): XcloudTitleInfo {
|
||||||
// Clone the object since the original is read-only
|
// Clone the object since the original is read-only
|
||||||
titleInfo = deepClone(titleInfo);
|
titleInfo = deepClone(titleInfo);
|
||||||
|
|
||||||
@ -110,8 +112,8 @@ export const BxExposed = {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
handleControllerShortcut: ControllerShortcut.handle,
|
handleControllerShortcut: isFullVersion() && ControllerShortcut.handle,
|
||||||
resetControllerShortcut: ControllerShortcut.reset,
|
resetControllerShortcut: isFullVersion() && ControllerShortcut.reset,
|
||||||
|
|
||||||
overrideSettings: {
|
overrideSettings: {
|
||||||
'Tv_settings': {
|
'Tv_settings': {
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { EmulatedMkbHandler } from "@modules/mkb/mkb-handler";
|
import { VIRTUAL_GAMEPAD_ID } from "@modules/mkb/mkb-handler";
|
||||||
import { t } from "@utils/translation";
|
import { t } from "@utils/translation";
|
||||||
import { Toast } from "@utils/toast";
|
import { Toast } from "@utils/toast";
|
||||||
import { BxLogger } from "@utils/bx-logger";
|
import { BxLogger } from "@utils/bx-logger";
|
||||||
@ -8,7 +8,7 @@ import { getPref } from "./settings-storages/global-settings-storage";
|
|||||||
// Show a toast when connecting/disconecting controller
|
// Show a toast when connecting/disconecting controller
|
||||||
export function showGamepadToast(gamepad: Gamepad) {
|
export function showGamepadToast(gamepad: Gamepad) {
|
||||||
// Don't show Toast for virtual controller
|
// Don't show Toast for virtual controller
|
||||||
if (gamepad.id === EmulatedMkbHandler.VIRTUAL_GAMEPAD_ID) {
|
if (gamepad.id === VIRTUAL_GAMEPAD_ID) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2,6 +2,7 @@ import type { BaseSettingsStore } from "./settings-storages/base-settings-storag
|
|||||||
import { UserAgent } from "./user-agent";
|
import { UserAgent } from "./user-agent";
|
||||||
|
|
||||||
export const SCRIPT_VERSION = Bun.env.SCRIPT_VERSION!;
|
export const SCRIPT_VERSION = Bun.env.SCRIPT_VERSION!;
|
||||||
|
export const SCRIPT_VARIANT = Bun.env.BUILD_VARIANT! as BuildVariant;
|
||||||
|
|
||||||
export const AppInterface = window.AppInterface;
|
export const AppInterface = window.AppInterface;
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import { BxEvent } from "@utils/bx-event";
|
import { BxEvent } from "@utils/bx-event";
|
||||||
import { LoadingScreen } from "@modules/loading-screen";
|
import { LoadingScreen } from "@modules/loading-screen";
|
||||||
import { RemotePlay } from "@modules/remote-play";
|
import { RemotePlayManager } from "@/modules/remote-play-manager";
|
||||||
import { HeaderSection } from "@/modules/ui/header";
|
import { HeaderSection } from "@/modules/ui/header";
|
||||||
import { NavigationDialogManager } from "@/modules/ui/dialog/navigation-dialog";
|
import { NavigationDialogManager } from "@/modules/ui/dialog/navigation-dialog";
|
||||||
|
|
||||||
@ -24,7 +24,7 @@ export function onHistoryChanged(e: PopStateEvent) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
window.setTimeout(RemotePlay.detect, 10);
|
window.setTimeout(RemotePlayManager.detect, 10);
|
||||||
|
|
||||||
// Hide Global settings
|
// Hide Global settings
|
||||||
const $settings = document.querySelector('.bx-settings-container');
|
const $settings = document.querySelector('.bx-settings-container');
|
||||||
@ -35,9 +35,6 @@ export function onHistoryChanged(e: PopStateEvent) {
|
|||||||
// Hide Navigation dialog
|
// Hide Navigation dialog
|
||||||
NavigationDialogManager.getInstance().hide();
|
NavigationDialogManager.getInstance().hide();
|
||||||
|
|
||||||
// Hide Remote Play popup
|
|
||||||
RemotePlay.detachPopup();
|
|
||||||
|
|
||||||
LoadingScreen.reset();
|
LoadingScreen.reset();
|
||||||
window.setTimeout(HeaderSection.watchHeader, 2000);
|
window.setTimeout(HeaderSection.watchHeader, 2000);
|
||||||
|
|
||||||
|
@ -163,7 +163,7 @@ export function escapeHtml(html: string): string {
|
|||||||
|
|
||||||
export function isElementVisible($elm: HTMLElement): boolean {
|
export function isElementVisible($elm: HTMLElement): boolean {
|
||||||
const rect = $elm.getBoundingClientRect();
|
const rect = $elm.getBoundingClientRect();
|
||||||
return !!rect.width && !!rect.height;
|
return (rect.x >= 0 || rect.y >= 0) && !!rect.width && !!rect.height;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const CTN = document.createTextNode.bind(document);
|
export const CTN = document.createTextNode.bind(document);
|
||||||
@ -181,9 +181,47 @@ export function clearFocus() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
export function clearDataSet($elm: HTMLElement) {
|
export function clearDataSet($elm: HTMLElement) {
|
||||||
Object.keys($elm.dataset).forEach(key => {
|
Object.keys($elm.dataset).forEach(key => {
|
||||||
delete $elm.dataset[key];
|
delete $elm.dataset[key];
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// https://stackoverflow.com/a/20732091
|
||||||
|
const FILE_SIZE_UNITS = ['B', 'KB', 'MB', 'GB', 'TB'];
|
||||||
|
export function humanFileSize(size: number) {
|
||||||
|
const i = size == 0 ? 0 : Math.floor(Math.log(size) / Math.log(1024));
|
||||||
|
return (size / Math.pow(1024, i)).toFixed(2) + ' ' + FILE_SIZE_UNITS[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
export function secondsToHm(seconds: number) {
|
||||||
|
let h = Math.floor(seconds / 3600);
|
||||||
|
let m = Math.floor(seconds % 3600 / 60) + 1;
|
||||||
|
|
||||||
|
if (m === 60) {
|
||||||
|
h += 1;
|
||||||
|
m = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
const output = [];
|
||||||
|
h > 0 && output.push(`${h}h`);
|
||||||
|
m > 0 && output.push(`${m}m`);
|
||||||
|
|
||||||
|
return output.join(' ');
|
||||||
|
}
|
||||||
|
|
||||||
|
export function secondsToHms(seconds: number) {
|
||||||
|
let h = Math.floor(seconds / 3600);
|
||||||
|
seconds %= 3600;
|
||||||
|
let m = Math.floor(seconds / 60);
|
||||||
|
let s = seconds % 60;
|
||||||
|
|
||||||
|
const output = [];
|
||||||
|
h > 0 && output.push(`${h}h`);
|
||||||
|
m > 0 && output.push(`${m}m`);
|
||||||
|
if (s > 0 || output.length === 0) {
|
||||||
|
output.push(`${s}s`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return output.join(' ');
|
||||||
|
}
|
||||||
|
@ -254,6 +254,7 @@ export function patchPointerLockApi() {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// const nativeRequestPointerLock = HTMLElement.prototype.requestPointerLock;
|
// const nativeRequestPointerLock = HTMLElement.prototype.requestPointerLock;
|
||||||
|
// @ts-ignore
|
||||||
HTMLElement.prototype.requestPointerLock = function() {
|
HTMLElement.prototype.requestPointerLock = function() {
|
||||||
pointerLockElement = document.documentElement;
|
pointerLockElement = document.documentElement;
|
||||||
window.dispatchEvent(new Event(BxEvent.POINTER_LOCK_REQUESTED));
|
window.dispatchEvent(new Event(BxEvent.POINTER_LOCK_REQUESTED));
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
import { isFullVersion } from "@macros/build" with {type: "macro"};
|
||||||
|
|
||||||
import { BxEvent } from "@utils/bx-event";
|
import { BxEvent } from "@utils/bx-event";
|
||||||
import { BX_FLAGS, NATIVE_FETCH } from "@utils/bx-flags";
|
import { BX_FLAGS, NATIVE_FETCH } from "@utils/bx-flags";
|
||||||
import { TouchController } from "@modules/touch-controller";
|
import { TouchController } from "@modules/touch-controller";
|
||||||
@ -222,7 +224,7 @@ export function interceptHttpRequests() {
|
|||||||
for (let i = 1; i < obj.length; i++) {
|
for (let i = 1; i < obj.length; i++) {
|
||||||
gamepassAllGames.push(obj[i].id);
|
gamepassAllGames.push(obj[i].id);
|
||||||
}
|
}
|
||||||
} else if (url.includes(GamePassCloudGallery.TOUCH)) {
|
} else if (isFullVersion() && url.includes(GamePassCloudGallery.TOUCH)) {
|
||||||
try {
|
try {
|
||||||
let customList = TouchController.getCustomList();
|
let customList = TouchController.getCustomList();
|
||||||
|
|
||||||
@ -262,7 +264,7 @@ export function interceptHttpRequests() {
|
|||||||
requestType = 'xcloud';
|
requestType = 'xcloud';
|
||||||
}
|
}
|
||||||
|
|
||||||
if (requestType === 'xhome') {
|
if (isFullVersion() && requestType === 'xhome') {
|
||||||
return XhomeInterceptor.handle(request as Request);
|
return XhomeInterceptor.handle(request as Request);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -31,7 +31,7 @@ export class Screenshot {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static updateCanvasFilters(filters: string) {
|
static updateCanvasFilters(filters: string) {
|
||||||
Screenshot.#canvasContext.filter = filters;
|
Screenshot.#canvasContext && (Screenshot.#canvasContext.filter = filters);
|
||||||
}
|
}
|
||||||
|
|
||||||
static #onAnimationEnd(e: Event) {
|
static #onAnimationEnd(e: Event) {
|
||||||
|
@ -3,6 +3,7 @@ import type { NumberStepperParams, SettingDefinitions } from "@/types/setting-de
|
|||||||
import { BxEvent } from "../bx-event";
|
import { BxEvent } from "../bx-event";
|
||||||
import { SettingElementType } from "../setting-element";
|
import { SettingElementType } from "../setting-element";
|
||||||
import { t } from "../translation";
|
import { t } from "../translation";
|
||||||
|
import { SCRIPT_VARIANT } from "../global";
|
||||||
|
|
||||||
export class BaseSettingsStore {
|
export class BaseSettingsStore {
|
||||||
private storage: Storage;
|
private storage: Storage;
|
||||||
@ -18,6 +19,11 @@ export class BaseSettingsStore {
|
|||||||
for (settingId in definitions) {
|
for (settingId in definitions) {
|
||||||
const setting = definitions[settingId];
|
const setting = definitions[settingId];
|
||||||
|
|
||||||
|
// Convert requiredVariants to array
|
||||||
|
if (typeof setting.requiredVariants === 'string') {
|
||||||
|
setting.requiredVariants = [setting.requiredVariants];
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
if (setting.migrate && settingId in savedPrefs) {
|
if (setting.migrate && settingId in savedPrefs) {
|
||||||
setting.migrate.call(this, savedPrefs, savedPrefs[settingId]);
|
setting.migrate.call(this, savedPrefs, savedPrefs[settingId]);
|
||||||
@ -58,9 +64,16 @@ export class BaseSettingsStore {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const definition = this.definitions[key];
|
||||||
|
|
||||||
|
// Return default value if build variant is different
|
||||||
|
if (definition.requiredVariants && !definition.requiredVariants.includes(SCRIPT_VARIANT)) {
|
||||||
|
return definition.default;
|
||||||
|
}
|
||||||
|
|
||||||
// Return default value if the feature is not supported
|
// Return default value if the feature is not supported
|
||||||
if (checkUnsupported && this.definitions[key].unsupported) {
|
if (checkUnsupported && definition.unsupported) {
|
||||||
return this.definitions[key].default;
|
return definition.default;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!(key in this.settings)) {
|
if (!(key in this.settings)) {
|
||||||
|
@ -3,7 +3,6 @@ import { PrefKey, StorageKey } from "@/enums/pref-keys";
|
|||||||
import { StreamPlayerType, StreamVideoProcessing } from "@/enums/stream-player";
|
import { StreamPlayerType, StreamVideoProcessing } from "@/enums/stream-player";
|
||||||
import { UiSection } from "@/enums/ui-sections";
|
import { UiSection } from "@/enums/ui-sections";
|
||||||
import { UserAgentProfile } from "@/enums/user-agent";
|
import { UserAgentProfile } from "@/enums/user-agent";
|
||||||
import { StreamStat } from "@/modules/stream/stream-stats";
|
|
||||||
import { type SettingDefinition, type SettingDefinitions } from "@/types/setting-definition";
|
import { type SettingDefinition, type SettingDefinitions } from "@/types/setting-definition";
|
||||||
import { BX_FLAGS } from "../bx-flags";
|
import { BX_FLAGS } from "../bx-flags";
|
||||||
import { STATES, AppInterface, STORAGE } from "../global";
|
import { STATES, AppInterface, STORAGE } from "../global";
|
||||||
@ -12,6 +11,7 @@ import { t, SUPPORTED_LANGUAGES } from "../translation";
|
|||||||
import { UserAgent } from "../user-agent";
|
import { UserAgent } from "../user-agent";
|
||||||
import { BaseSettingsStore as BaseSettingsStorage } from "./base-settings-storage";
|
import { BaseSettingsStore as BaseSettingsStorage } from "./base-settings-storage";
|
||||||
import { SettingElementType } from "../setting-element";
|
import { SettingElementType } from "../setting-element";
|
||||||
|
import { StreamStat } from "../stream-stats-collector";
|
||||||
|
|
||||||
|
|
||||||
export const enum StreamResolution {
|
export const enum StreamResolution {
|
||||||
@ -96,7 +96,7 @@ function getSupportedCodecProfiles() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export class GlobalSettingsStorage extends BaseSettingsStorage {
|
export class GlobalSettingsStorage extends BaseSettingsStorage {
|
||||||
private static readonly DEFINITIONS: SettingDefinitions = {
|
private static readonly DEFINITIONS = {
|
||||||
[PrefKey.LAST_UPDATE_CHECK]: {
|
[PrefKey.LAST_UPDATE_CHECK]: {
|
||||||
default: 0,
|
default: 0,
|
||||||
},
|
},
|
||||||
@ -135,8 +135,8 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
'da-DK': 'dansk',
|
'da-DK': 'dansk',
|
||||||
'de-DE': 'Deutsch',
|
'de-DE': 'Deutsch',
|
||||||
'el-GR': 'Ελληνικά',
|
'el-GR': 'Ελληνικά',
|
||||||
'en-GB': 'English (United Kingdom)',
|
'en-GB': 'English (UK)',
|
||||||
'en-US': 'English (United States)',
|
'en-US': 'English (US)',
|
||||||
'es-ES': 'español (España)',
|
'es-ES': 'español (España)',
|
||||||
'es-MX': 'español (Latinoamérica)',
|
'es-MX': 'español (Latinoamérica)',
|
||||||
'fi-FI': 'suomi',
|
'fi-FI': 'suomi',
|
||||||
@ -197,6 +197,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.SCREENSHOT_APPLY_FILTERS]: {
|
[PrefKey.SCREENSHOT_APPLY_FILTERS]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('screenshot-apply-filters'),
|
label: t('screenshot-apply-filters'),
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
@ -211,6 +212,8 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.STREAM_COMBINE_SOURCES]: {
|
[PrefKey.STREAM_COMBINE_SOURCES]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
|
|
||||||
label: t('combine-audio-video-streams'),
|
label: t('combine-audio-video-streams'),
|
||||||
default: false,
|
default: false,
|
||||||
experimental: true,
|
experimental: true,
|
||||||
@ -218,6 +221,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.STREAM_TOUCH_CONTROLLER]: {
|
[PrefKey.STREAM_TOUCH_CONTROLLER]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('tc-availability'),
|
label: t('tc-availability'),
|
||||||
default: StreamTouchController.ALL,
|
default: StreamTouchController.ALL,
|
||||||
options: {
|
options: {
|
||||||
@ -233,11 +237,13 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
[PrefKey.STREAM_TOUCH_CONTROLLER_AUTO_OFF]: {
|
[PrefKey.STREAM_TOUCH_CONTROLLER_AUTO_OFF]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('tc-auto-off'),
|
label: t('tc-auto-off'),
|
||||||
default: false,
|
default: false,
|
||||||
unsupported: !STATES.userAgent.capabilities.touch,
|
unsupported: !STATES.userAgent.capabilities.touch,
|
||||||
},
|
},
|
||||||
[PrefKey.STREAM_TOUCH_CONTROLLER_DEFAULT_OPACITY]: {
|
[PrefKey.STREAM_TOUCH_CONTROLLER_DEFAULT_OPACITY]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
label: t('tc-default-opacity'),
|
label: t('tc-default-opacity'),
|
||||||
default: 100,
|
default: 100,
|
||||||
@ -252,6 +258,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
unsupported: !STATES.userAgent.capabilities.touch,
|
unsupported: !STATES.userAgent.capabilities.touch,
|
||||||
},
|
},
|
||||||
[PrefKey.STREAM_TOUCH_CONTROLLER_STYLE_STANDARD]: {
|
[PrefKey.STREAM_TOUCH_CONTROLLER_STYLE_STANDARD]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('tc-standard-layout-style'),
|
label: t('tc-standard-layout-style'),
|
||||||
default: 'default',
|
default: 'default',
|
||||||
options: {
|
options: {
|
||||||
@ -262,6 +269,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
unsupported: !STATES.userAgent.capabilities.touch,
|
unsupported: !STATES.userAgent.capabilities.touch,
|
||||||
},
|
},
|
||||||
[PrefKey.STREAM_TOUCH_CONTROLLER_STYLE_CUSTOM]: {
|
[PrefKey.STREAM_TOUCH_CONTROLLER_STYLE_CUSTOM]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('tc-custom-layout-style'),
|
label: t('tc-custom-layout-style'),
|
||||||
default: 'default',
|
default: 'default',
|
||||||
options: {
|
options: {
|
||||||
@ -276,15 +284,18 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
[PrefKey.MKB_HIDE_IDLE_CURSOR]: {
|
[PrefKey.MKB_HIDE_IDLE_CURSOR]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('hide-idle-cursor'),
|
label: t('hide-idle-cursor'),
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
[PrefKey.STREAM_DISABLE_FEEDBACK_DIALOG]: {
|
[PrefKey.STREAM_DISABLE_FEEDBACK_DIALOG]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('disable-post-stream-feedback-dialog'),
|
label: t('disable-post-stream-feedback-dialog'),
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.BITRATE_VIDEO_MAX]: {
|
[PrefKey.BITRATE_VIDEO_MAX]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
label: t('bitrate-video-maximum'),
|
label: t('bitrate-video-maximum'),
|
||||||
note: '⚠️ ' + t('unexpected-behavior'),
|
note: '⚠️ ' + t('unexpected-behavior'),
|
||||||
@ -306,10 +317,11 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
suggest: {
|
suggest: {
|
||||||
highest: 0,
|
highest: 0,
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.GAME_BAR_POSITION]: {
|
[PrefKey.GAME_BAR_POSITION]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('position'),
|
label: t('position'),
|
||||||
default: 'bottom-left',
|
default: 'bottom-left',
|
||||||
options: {
|
options: {
|
||||||
@ -320,6 +332,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.LOCAL_CO_OP_ENABLED]: {
|
[PrefKey.LOCAL_CO_OP_ENABLED]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('enable-local-co-op-support'),
|
label: t('enable-local-co-op-support'),
|
||||||
default: false,
|
default: false,
|
||||||
note: CE<HTMLAnchorElement>('a', {
|
note: CE<HTMLAnchorElement>('a', {
|
||||||
@ -341,15 +354,18 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.CONTROLLER_ENABLE_SHORTCUTS]: {
|
[PrefKey.CONTROLLER_ENABLE_SHORTCUTS]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.CONTROLLER_ENABLE_VIBRATION]: {
|
[PrefKey.CONTROLLER_ENABLE_VIBRATION]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('controller-vibration'),
|
label: t('controller-vibration'),
|
||||||
default: true,
|
default: true,
|
||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.CONTROLLER_DEVICE_VIBRATION]: {
|
[PrefKey.CONTROLLER_DEVICE_VIBRATION]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('device-vibration'),
|
label: t('device-vibration'),
|
||||||
default: ControllerDeviceVibration.OFF,
|
default: ControllerDeviceVibration.OFF,
|
||||||
options: {
|
options: {
|
||||||
@ -360,6 +376,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.CONTROLLER_VIBRATION_INTENSITY]: {
|
[PrefKey.CONTROLLER_VIBRATION_INTENSITY]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('vibration-intensity'),
|
label: t('vibration-intensity'),
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 100,
|
default: 100,
|
||||||
@ -373,6 +390,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.MKB_ENABLED]: {
|
[PrefKey.MKB_ENABLED]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('enable-mkb'),
|
label: t('enable-mkb'),
|
||||||
default: false,
|
default: false,
|
||||||
unsupported: ((): string | boolean => {
|
unsupported: ((): string | boolean => {
|
||||||
@ -398,6 +416,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.NATIVE_MKB_ENABLED]: {
|
[PrefKey.NATIVE_MKB_ENABLED]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('native-mkb'),
|
label: t('native-mkb'),
|
||||||
default: 'default',
|
default: 'default',
|
||||||
options: {
|
options: {
|
||||||
@ -419,6 +438,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.NATIVE_MKB_SCROLL_HORIZONTAL_SENSITIVITY]: {
|
[PrefKey.NATIVE_MKB_SCROLL_HORIZONTAL_SENSITIVITY]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('horizontal-scroll-sensitivity'),
|
label: t('horizontal-scroll-sensitivity'),
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 0,
|
default: 0,
|
||||||
@ -438,6 +458,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.NATIVE_MKB_SCROLL_VERTICAL_SENSITIVITY]: {
|
[PrefKey.NATIVE_MKB_SCROLL_VERTICAL_SENSITIVITY]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('vertical-scroll-sensitivity'),
|
label: t('vertical-scroll-sensitivity'),
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 0,
|
default: 0,
|
||||||
@ -457,10 +478,12 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.MKB_DEFAULT_PRESET_ID]: {
|
[PrefKey.MKB_DEFAULT_PRESET_ID]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
default: 0,
|
default: 0,
|
||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.MKB_ABSOLUTE_MOUSE]: {
|
[PrefKey.MKB_ABSOLUTE_MOUSE]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -470,6 +493,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.UI_LOADING_SCREEN_GAME_ART]: {
|
[PrefKey.UI_LOADING_SCREEN_GAME_ART]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('show-game-art'),
|
label: t('show-game-art'),
|
||||||
default: true,
|
default: true,
|
||||||
},
|
},
|
||||||
@ -493,6 +517,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.UI_LAYOUT]: {
|
[PrefKey.UI_LAYOUT]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('layout'),
|
label: t('layout'),
|
||||||
default: 'default',
|
default: 'default',
|
||||||
options: {
|
options: {
|
||||||
@ -508,11 +533,13 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.UI_HOME_CONTEXT_MENU_DISABLED]: {
|
[PrefKey.UI_HOME_CONTEXT_MENU_DISABLED]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('disable-home-context-menu'),
|
label: t('disable-home-context-menu'),
|
||||||
default: STATES.browser.capabilities.touch,
|
default: STATES.browser.capabilities.touch,
|
||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.UI_HIDE_SECTIONS]: {
|
[PrefKey.UI_HIDE_SECTIONS]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('hide-sections'),
|
label: t('hide-sections'),
|
||||||
default: [],
|
default: [],
|
||||||
multipleOptions: {
|
multipleOptions: {
|
||||||
@ -529,6 +556,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.UI_GAME_CARD_SHOW_WAIT_TIME]: {
|
[PrefKey.UI_GAME_CARD_SHOW_WAIT_TIME]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('show-wait-time-in-game-card'),
|
label: t('show-wait-time-in-game-card'),
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
@ -663,6 +691,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
[PrefKey.AUDIO_ENABLE_VOLUME_CONTROL]: {
|
[PrefKey.AUDIO_ENABLE_VOLUME_CONTROL]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('enable-volume-control'),
|
label: t('enable-volume-control'),
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
@ -684,16 +713,28 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
label: t('stats'),
|
label: t('stats'),
|
||||||
default: [StreamStat.PING, StreamStat.FPS, StreamStat.BITRATE, StreamStat.DECODE_TIME, StreamStat.PACKETS_LOST, StreamStat.FRAMES_LOST],
|
default: [StreamStat.PING, StreamStat.FPS, StreamStat.BITRATE, StreamStat.DECODE_TIME, StreamStat.PACKETS_LOST, StreamStat.FRAMES_LOST],
|
||||||
multipleOptions: {
|
multipleOptions: {
|
||||||
|
[StreamStat.CLOCK]: `${StreamStat.CLOCK.toUpperCase()}: ${t('clock')}`,
|
||||||
|
[StreamStat.PLAYTIME]: `${StreamStat.PLAYTIME.toUpperCase()}: ${t('playtime')}`,
|
||||||
|
[StreamStat.BATTERY]: `${StreamStat.BATTERY.toUpperCase()}: ${t('battery')}`,
|
||||||
[StreamStat.PING]: `${StreamStat.PING.toUpperCase()}: ${t('stat-ping')}`,
|
[StreamStat.PING]: `${StreamStat.PING.toUpperCase()}: ${t('stat-ping')}`,
|
||||||
[StreamStat.FPS]: `${StreamStat.FPS.toUpperCase()}: ${t('stat-fps')}`,
|
[StreamStat.FPS]: `${StreamStat.FPS.toUpperCase()}: ${t('stat-fps')}`,
|
||||||
[StreamStat.BITRATE]: `${StreamStat.BITRATE.toUpperCase()}: ${t('stat-bitrate')}`,
|
[StreamStat.BITRATE]: `${StreamStat.BITRATE.toUpperCase()}: ${t('stat-bitrate')}`,
|
||||||
[StreamStat.DECODE_TIME]: `${StreamStat.DECODE_TIME.toUpperCase()}: ${t('stat-decode-time')}`,
|
[StreamStat.DECODE_TIME]: `${StreamStat.DECODE_TIME.toUpperCase()}: ${t('stat-decode-time')}`,
|
||||||
[StreamStat.PACKETS_LOST]: `${StreamStat.PACKETS_LOST.toUpperCase()}: ${t('stat-packets-lost')}`,
|
[StreamStat.PACKETS_LOST]: `${StreamStat.PACKETS_LOST.toUpperCase()}: ${t('stat-packets-lost')}`,
|
||||||
[StreamStat.FRAMES_LOST]: `${StreamStat.FRAMES_LOST.toUpperCase()}: ${t('stat-frames-lost')}`,
|
[StreamStat.FRAMES_LOST]: `${StreamStat.FRAMES_LOST.toUpperCase()}: ${t('stat-frames-lost')}`,
|
||||||
|
[StreamStat.DOWNLOAD]: `${StreamStat.DOWNLOAD.toUpperCase()}: ${t('download')}`,
|
||||||
|
[StreamStat.UPLOAD]: `${StreamStat.UPLOAD.toUpperCase()}: ${t('upload')}`,
|
||||||
},
|
},
|
||||||
params: {
|
params: {
|
||||||
size: 6,
|
size: 6,
|
||||||
},
|
},
|
||||||
|
ready: setting => {
|
||||||
|
// Remove Battery option in unsupported browser
|
||||||
|
const multipleOptions = (setting as any).multipleOptions;
|
||||||
|
if (!STATES.browser.capabilities.batteryApi) {
|
||||||
|
delete multipleOptions[StreamStat.BATTERY];
|
||||||
|
}
|
||||||
|
},
|
||||||
},
|
},
|
||||||
[PrefKey.STATS_SHOW_WHEN_PLAYING]: {
|
[PrefKey.STATS_SHOW_WHEN_PLAYING]: {
|
||||||
label: t('show-stats-on-startup'),
|
label: t('show-stats-on-startup'),
|
||||||
@ -743,11 +784,13 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.REMOTE_PLAY_ENABLED]: {
|
[PrefKey.REMOTE_PLAY_ENABLED]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: t('enable-remote-play-feature'),
|
label: t('enable-remote-play-feature'),
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.REMOTE_PLAY_RESOLUTION]: {
|
[PrefKey.REMOTE_PLAY_RESOLUTION]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
default: StreamResolution.DIM_1080P,
|
default: StreamResolution.DIM_1080P,
|
||||||
options: {
|
options: {
|
||||||
[StreamResolution.DIM_1080P]: '1080p',
|
[StreamResolution.DIM_1080P]: '1080p',
|
||||||
@ -756,11 +799,12 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.GAME_FORTNITE_FORCE_CONSOLE]: {
|
[PrefKey.GAME_FORTNITE_FORCE_CONSOLE]: {
|
||||||
|
requiredVariants: 'full',
|
||||||
label: '🎮 ' + t('fortnite-force-console-version'),
|
label: '🎮 ' + t('fortnite-force-console-version'),
|
||||||
default: false,
|
default: false,
|
||||||
note: t('fortnite-allow-stw-mode'),
|
note: t('fortnite-allow-stw-mode'),
|
||||||
},
|
},
|
||||||
};
|
} satisfies SettingDefinitions;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super(StorageKey.GLOBAL, GlobalSettingsStorage.DEFINITIONS);
|
super(StorageKey.GLOBAL, GlobalSettingsStorage.DEFINITIONS);
|
||||||
|
303
src/utils/stream-stats-collector.ts
Normal file
303
src/utils/stream-stats-collector.ts
Normal file
@ -0,0 +1,303 @@
|
|||||||
|
import { BxEvent } from "./bx-event";
|
||||||
|
import { STATES } from "./global";
|
||||||
|
import { humanFileSize, secondsToHm } from "./html";
|
||||||
|
|
||||||
|
export enum StreamStat {
|
||||||
|
PING = 'ping',
|
||||||
|
FPS = 'fps',
|
||||||
|
BITRATE = 'btr',
|
||||||
|
DECODE_TIME = 'dt',
|
||||||
|
PACKETS_LOST = 'pl',
|
||||||
|
FRAMES_LOST = 'fl',
|
||||||
|
DOWNLOAD = 'dl',
|
||||||
|
UPLOAD = 'ul',
|
||||||
|
PLAYTIME = 'play',
|
||||||
|
BATTERY = 'batt',
|
||||||
|
CLOCK = 'time',
|
||||||
|
};
|
||||||
|
|
||||||
|
export type StreamStatGrade = '' | 'bad' | 'ok' | 'good';
|
||||||
|
|
||||||
|
type CurrentStats = {
|
||||||
|
[StreamStat.PING]: {
|
||||||
|
current: number;
|
||||||
|
calculateGrade: () => StreamStatGrade;
|
||||||
|
toString: () => string;
|
||||||
|
};
|
||||||
|
|
||||||
|
[StreamStat.FPS]: {
|
||||||
|
current: number;
|
||||||
|
toString: () => string;
|
||||||
|
};
|
||||||
|
|
||||||
|
[StreamStat.BITRATE]: {
|
||||||
|
current: number;
|
||||||
|
toString: () => string;
|
||||||
|
};
|
||||||
|
|
||||||
|
[StreamStat.FRAMES_LOST]: {
|
||||||
|
received: number;
|
||||||
|
dropped: number;
|
||||||
|
toString: () => string;
|
||||||
|
};
|
||||||
|
|
||||||
|
[StreamStat.PACKETS_LOST]: {
|
||||||
|
received: number;
|
||||||
|
dropped: number;
|
||||||
|
toString: () => string;
|
||||||
|
};
|
||||||
|
|
||||||
|
[StreamStat.DECODE_TIME]: {
|
||||||
|
current: number;
|
||||||
|
total: number;
|
||||||
|
calculateGrade: () => StreamStatGrade;
|
||||||
|
toString: () => string;
|
||||||
|
};
|
||||||
|
|
||||||
|
[StreamStat.DOWNLOAD]: {
|
||||||
|
total: number;
|
||||||
|
toString: () => string;
|
||||||
|
};
|
||||||
|
|
||||||
|
[StreamStat.UPLOAD]: {
|
||||||
|
total: number;
|
||||||
|
toString: () => string;
|
||||||
|
};
|
||||||
|
|
||||||
|
[StreamStat.PLAYTIME]: {
|
||||||
|
seconds: number;
|
||||||
|
startTime: number;
|
||||||
|
toString: () => string;
|
||||||
|
};
|
||||||
|
|
||||||
|
[StreamStat.BATTERY]: {
|
||||||
|
current: number;
|
||||||
|
start: number;
|
||||||
|
isCharging: boolean;
|
||||||
|
toString: () => string;
|
||||||
|
},
|
||||||
|
|
||||||
|
[StreamStat.CLOCK]: {
|
||||||
|
toString: () => string;
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
export class StreamStatsCollector {
|
||||||
|
private static instance: StreamStatsCollector;
|
||||||
|
public static getInstance(): StreamStatsCollector {
|
||||||
|
if (!StreamStatsCollector.instance) {
|
||||||
|
StreamStatsCollector.instance = new StreamStatsCollector();
|
||||||
|
}
|
||||||
|
|
||||||
|
return StreamStatsCollector.instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Collect in background - 60 seconds
|
||||||
|
static readonly INTERVAL_BACKGROUND = 60 * 1000;
|
||||||
|
|
||||||
|
private currentStats: CurrentStats = {
|
||||||
|
[StreamStat.PING]: {
|
||||||
|
current: -1,
|
||||||
|
calculateGrade() {
|
||||||
|
return (this.current >= 100) ? 'bad' : (this.current > 75) ? 'ok' : (this.current > 40) ? 'good' : '';
|
||||||
|
},
|
||||||
|
toString() {
|
||||||
|
return this.current === -1 ? '???' : this.current.toString();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
[StreamStat.FPS]: {
|
||||||
|
current: 0,
|
||||||
|
toString() {
|
||||||
|
return this.current.toString();
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
[StreamStat.BITRATE]: {
|
||||||
|
current: 0,
|
||||||
|
toString() {
|
||||||
|
return `${this.current.toFixed(2)} Mbps`;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
[StreamStat.FRAMES_LOST]: {
|
||||||
|
received: 0,
|
||||||
|
dropped: 0,
|
||||||
|
toString() {
|
||||||
|
const framesDroppedPercentage = (this.dropped * 100 / ((this.dropped + this.received) || 1)).toFixed(2);
|
||||||
|
return framesDroppedPercentage === '0.00' ? this.dropped.toString() : `${this.dropped} (${framesDroppedPercentage}%)`;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
[StreamStat.PACKETS_LOST]: {
|
||||||
|
received: 0,
|
||||||
|
dropped: 0,
|
||||||
|
toString() {
|
||||||
|
const packetsLostPercentage = (this.dropped * 100 / ((this.dropped + this.received) || 1)).toFixed(2);
|
||||||
|
return packetsLostPercentage === '0.00' ? this.dropped.toString() : `${this.dropped} (${packetsLostPercentage}%)`;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
[StreamStat.DECODE_TIME]: {
|
||||||
|
current: 0,
|
||||||
|
total: 0,
|
||||||
|
calculateGrade() {
|
||||||
|
return (this.current > 12) ? 'bad' : (this.current > 9) ? 'ok' : (this.current > 6) ? 'good' : '';
|
||||||
|
},
|
||||||
|
toString() {
|
||||||
|
return isNaN(this.current) ? '??ms' : `${this.current.toFixed(2)}ms`;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
[StreamStat.DOWNLOAD]: {
|
||||||
|
total: 0,
|
||||||
|
toString() {
|
||||||
|
return humanFileSize(this.total);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
[StreamStat.UPLOAD]: {
|
||||||
|
total: 0,
|
||||||
|
toString() {
|
||||||
|
return humanFileSize(this.total);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
[StreamStat.PLAYTIME]: {
|
||||||
|
seconds: 0,
|
||||||
|
startTime: 0,
|
||||||
|
toString() {
|
||||||
|
return secondsToHm(this.seconds);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
[StreamStat.BATTERY]: {
|
||||||
|
current: 100,
|
||||||
|
start: 100,
|
||||||
|
isCharging: false,
|
||||||
|
toString() {
|
||||||
|
let text = `${this.current}%`;
|
||||||
|
|
||||||
|
if (this.current !== this.start) {
|
||||||
|
const diffLevel = Math.round(this.current - this.start);
|
||||||
|
const sign = diffLevel > 0 ? '+' : '';
|
||||||
|
text += ` (${sign}${diffLevel}%)`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return text;
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
[StreamStat.CLOCK]: {
|
||||||
|
toString() {
|
||||||
|
return new Date().toLocaleTimeString([], {
|
||||||
|
hour: '2-digit',
|
||||||
|
minute:'2-digit',
|
||||||
|
hour12: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
private lastVideoStat?: RTCBasicStat | null;
|
||||||
|
|
||||||
|
async collect() {
|
||||||
|
const stats = await STATES.currentStream.peerConnection?.getStats();
|
||||||
|
|
||||||
|
stats?.forEach(stat => {
|
||||||
|
if (stat.type === 'inbound-rtp' && stat.kind === 'video') {
|
||||||
|
// FPS
|
||||||
|
const fps = this.currentStats[StreamStat.FPS];
|
||||||
|
fps.current = stat.framesPerSecond || 0;
|
||||||
|
|
||||||
|
// Packets Lost
|
||||||
|
// packetsLost can be negative, but we don't care about that
|
||||||
|
const pl = this.currentStats[StreamStat.PACKETS_LOST];
|
||||||
|
pl.dropped = Math.max(0, stat.packetsLost);
|
||||||
|
pl.received = stat.packetsReceived;
|
||||||
|
|
||||||
|
// Frames lost
|
||||||
|
const fl = this.currentStats[StreamStat.FRAMES_LOST];
|
||||||
|
fl.dropped = stat.framesDropped;
|
||||||
|
fl.received = stat.framesReceived;
|
||||||
|
|
||||||
|
if (!this.lastVideoStat) {
|
||||||
|
this.lastVideoStat = stat;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const lastStat = this.lastVideoStat;
|
||||||
|
|
||||||
|
// Bitrate
|
||||||
|
const btr = this.currentStats[StreamStat.BITRATE];
|
||||||
|
const timeDiff = stat.timestamp - lastStat.timestamp;
|
||||||
|
btr.current = 8 * (stat.bytesReceived - lastStat.bytesReceived) / timeDiff / 1000;
|
||||||
|
|
||||||
|
// Decode time
|
||||||
|
const dt = this.currentStats[StreamStat.DECODE_TIME];
|
||||||
|
dt.total = stat.totalDecodeTime - lastStat.totalDecodeTime;
|
||||||
|
const framesDecodedDiff = stat.framesDecoded - lastStat.framesDecoded;
|
||||||
|
dt.current = dt.total / framesDecodedDiff * 1000;
|
||||||
|
|
||||||
|
this.lastVideoStat = stat;
|
||||||
|
} else if (stat.type === 'candidate-pair' && stat.packetsReceived > 0 && stat.state === 'succeeded') {
|
||||||
|
// Round Trip Time
|
||||||
|
const ping = this.currentStats[StreamStat.PING];
|
||||||
|
ping.current = stat.currentRoundTripTime ? stat.currentRoundTripTime * 1000 : -1;
|
||||||
|
|
||||||
|
// Download
|
||||||
|
const dl = this.currentStats[StreamStat.DOWNLOAD];
|
||||||
|
dl.total = stat.bytesReceived;
|
||||||
|
|
||||||
|
// Upload
|
||||||
|
const ul = this.currentStats[StreamStat.UPLOAD];
|
||||||
|
ul.total = stat.bytesSent;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Battery
|
||||||
|
let batteryLevel = 100;
|
||||||
|
let isCharging = false;
|
||||||
|
if (STATES.browser.capabilities.batteryApi) {
|
||||||
|
try {
|
||||||
|
const bm = await (navigator as NavigatorBattery).getBattery();
|
||||||
|
isCharging = bm.charging;
|
||||||
|
batteryLevel = Math.round(bm.level * 100);
|
||||||
|
} catch(e) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
const battery = this.currentStats[StreamStat.BATTERY];
|
||||||
|
battery.current = batteryLevel;
|
||||||
|
battery.isCharging = isCharging;
|
||||||
|
|
||||||
|
// Playtime
|
||||||
|
const playTime = this.currentStats[StreamStat.PLAYTIME];
|
||||||
|
const now = +new Date;
|
||||||
|
playTime.seconds = Math.ceil((now - playTime.startTime) / 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
getStat<T extends StreamStat>(kind: T): CurrentStats[T] {
|
||||||
|
return this.currentStats[kind];
|
||||||
|
}
|
||||||
|
|
||||||
|
reset() {
|
||||||
|
const playTime = this.currentStats[StreamStat.PLAYTIME];
|
||||||
|
playTime.seconds = 0;
|
||||||
|
playTime.startTime = +new Date;
|
||||||
|
|
||||||
|
// Get battery level
|
||||||
|
try {
|
||||||
|
STATES.browser.capabilities.batteryApi && (navigator as NavigatorBattery).getBattery().then(bm => {
|
||||||
|
this.currentStats[StreamStat.BATTERY].start = Math.round(bm.level * 100);
|
||||||
|
});
|
||||||
|
} catch(e) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
static setupEvents() {
|
||||||
|
window.addEventListener(BxEvent.STREAM_PLAYING, e => {
|
||||||
|
const statsCollector = StreamStatsCollector.getInstance();
|
||||||
|
statsCollector.reset();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
@ -2,7 +2,7 @@ import { NATIVE_FETCH } from "./bx-flags";
|
|||||||
import { BxLogger } from "./bx-logger";
|
import { BxLogger } from "./bx-logger";
|
||||||
|
|
||||||
export const SUPPORTED_LANGUAGES = {
|
export const SUPPORTED_LANGUAGES = {
|
||||||
'en-US': 'English (United States)',
|
'en-US': 'English (US)',
|
||||||
|
|
||||||
'ca-CA': 'Català',
|
'ca-CA': 'Català',
|
||||||
'da-DK': 'dansk',
|
'da-DK': 'dansk',
|
||||||
@ -40,13 +40,7 @@ const Texts = {
|
|||||||
"auto": "Auto",
|
"auto": "Auto",
|
||||||
"back-to-home": "Back to home",
|
"back-to-home": "Back to home",
|
||||||
"back-to-home-confirm": "Do you want to go back to the home page (without disconnecting)?",
|
"back-to-home-confirm": "Do you want to go back to the home page (without disconnecting)?",
|
||||||
"badge-audio": "Audio",
|
"battery": "Battery",
|
||||||
"badge-battery": "Battery",
|
|
||||||
"badge-in": "In",
|
|
||||||
"badge-out": "Out",
|
|
||||||
"badge-playtime": "Playtime",
|
|
||||||
"badge-server": "Server",
|
|
||||||
"badge-video": "Video",
|
|
||||||
"battery-saving": "Battery saving",
|
"battery-saving": "Battery saving",
|
||||||
"better-xcloud": "Better xCloud",
|
"better-xcloud": "Better xCloud",
|
||||||
"bitrate-audio-maximum": "Maximum audio bitrate",
|
"bitrate-audio-maximum": "Maximum audio bitrate",
|
||||||
@ -63,6 +57,7 @@ const Texts = {
|
|||||||
"clarity-boost": "Clarity boost",
|
"clarity-boost": "Clarity boost",
|
||||||
"clarity-boost-warning": "These settings don't work when the Clarity Boost mode is ON",
|
"clarity-boost-warning": "These settings don't work when the Clarity Boost mode is ON",
|
||||||
"clear": "Clear",
|
"clear": "Clear",
|
||||||
|
"clock": "Clock",
|
||||||
"close": "Close",
|
"close": "Close",
|
||||||
"close-app": "Close app",
|
"close-app": "Close app",
|
||||||
"combine-audio-video-streams": "Combine audio & video streams",
|
"combine-audio-video-streams": "Combine audio & video streams",
|
||||||
@ -97,6 +92,7 @@ const Texts = {
|
|||||||
"disable-xcloud-analytics": "Disable xCloud analytics",
|
"disable-xcloud-analytics": "Disable xCloud analytics",
|
||||||
"disabled": "Disabled",
|
"disabled": "Disabled",
|
||||||
"disconnected": "Disconnected",
|
"disconnected": "Disconnected",
|
||||||
|
"download": "Download",
|
||||||
"edit": "Edit",
|
"edit": "Edit",
|
||||||
"enable-controller-shortcuts": "Enable controller shortcuts",
|
"enable-controller-shortcuts": "Enable controller shortcuts",
|
||||||
"enable-local-co-op-support": "Enable local co-op support",
|
"enable-local-co-op-support": "Enable local co-op support",
|
||||||
@ -161,22 +157,22 @@ const Texts = {
|
|||||||
(e: any) => `Version ${e.version} available`,
|
(e: any) => `Version ${e.version} available`,
|
||||||
,
|
,
|
||||||
,
|
,
|
||||||
|
(e: any) => `Version ${e.version} verfügbar`,
|
||||||
,
|
,
|
||||||
,
|
(e: any) => `Versión ${e.version} disponible`,
|
||||||
,
|
(e: any) => `Version ${e.version} disponible`,
|
||||||
,
|
(e: any) => `Disponibile la versione ${e.version}`,
|
||||||
,
|
|
||||||
(e: any) => `Ver ${e.version} が利用可能です`,
|
(e: any) => `Ver ${e.version} が利用可能です`,
|
||||||
(e: any) => `${e.version} 버전 사용가능`,
|
(e: any) => `${e.version} 버전 사용가능`,
|
||||||
|
(e: any) => `Dostępna jest nowa wersja ${e.version}`,
|
||||||
|
(e: any) => `Versão ${e.version} disponível`,
|
||||||
,
|
,
|
||||||
,
|
(e: any) => `เวอร์ชัน ${e.version} พร้อมใช้งานแล้ว`,
|
||||||
,
|
(e: any) => `${e.version} sayılı yeni sürüm mevcut`,
|
||||||
,
|
(e: any) => `Доступна версія ${e.version}`,
|
||||||
,
|
|
||||||
,
|
|
||||||
(e: any) => `Đã có phiên bản ${e.version}`,
|
(e: any) => `Đã có phiên bản ${e.version}`,
|
||||||
,
|
(e: any) => `版本 ${e.version} 可供更新`,
|
||||||
,
|
(e: any) => `已可更新為 ${e.version} 版`,
|
||||||
],
|
],
|
||||||
"no-consoles-found": "No consoles found",
|
"no-consoles-found": "No consoles found",
|
||||||
"normal": "Normal",
|
"normal": "Normal",
|
||||||
@ -186,6 +182,7 @@ const Texts = {
|
|||||||
"opacity": "Opacity",
|
"opacity": "Opacity",
|
||||||
"other": "Other",
|
"other": "Other",
|
||||||
"playing": "Playing",
|
"playing": "Playing",
|
||||||
|
"playtime": "Playtime",
|
||||||
"poland": "Poland",
|
"poland": "Poland",
|
||||||
"position": "Position",
|
"position": "Position",
|
||||||
"powered-off": "Powered off",
|
"powered-off": "Powered off",
|
||||||
@ -230,9 +227,9 @@ const Texts = {
|
|||||||
(e: any) => `${e.device} の推奨設定`,
|
(e: any) => `${e.device} の推奨設定`,
|
||||||
(e: any) => `다음 기기에서 권장되는 설정: ${e.device}`,
|
(e: any) => `다음 기기에서 권장되는 설정: ${e.device}`,
|
||||||
(e: any) => `Zalecane ustawienia dla ${e.device}`,
|
(e: any) => `Zalecane ustawienia dla ${e.device}`,
|
||||||
,
|
(e: any) => `Configurações recomendadas para ${e.device}`,
|
||||||
(e: any) => `Рекомендуемые настройки для ${e.device}`,
|
(e: any) => `Рекомендуемые настройки для ${e.device}`,
|
||||||
,
|
(e: any) => `การตั้งค่าที่แนะนำสำหรับ ${e.device}`,
|
||||||
(e: any) => `${e.device} için önerilen ayarlar`,
|
(e: any) => `${e.device} için önerilen ayarlar`,
|
||||||
(e: any) => `Рекомендовані налаштування для ${e.device}`,
|
(e: any) => `Рекомендовані налаштування для ${e.device}`,
|
||||||
(e: any) => `Cấu hình được đề xuất cho ${e.device}`,
|
(e: any) => `Cấu hình được đề xuất cho ${e.device}`,
|
||||||
@ -350,6 +347,7 @@ const Texts = {
|
|||||||
"unlimited": "Unlimited",
|
"unlimited": "Unlimited",
|
||||||
"unmuted": "Unmuted",
|
"unmuted": "Unmuted",
|
||||||
"unsharp-masking": "Unsharp masking",
|
"unsharp-masking": "Unsharp masking",
|
||||||
|
"upload": "Upload",
|
||||||
"use-mouse-absolute-position": "Use mouse's absolute position",
|
"use-mouse-absolute-position": "Use mouse's absolute position",
|
||||||
"use-this-at-your-own-risk": "Use this at your own risk",
|
"use-this-at-your-own-risk": "Use this at your own risk",
|
||||||
"user-agent-profile": "User-Agent profile",
|
"user-agent-profile": "User-Agent profile",
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { BxIcon } from "./bx-icon";
|
import { BxIcon } from "./bx-icon";
|
||||||
import { AppInterface, STATES } from "./global";
|
import { AppInterface, SCRIPT_VARIANT, STATES } from "./global";
|
||||||
import { ButtonStyle, CE, clearDataSet, createButton, getReactProps } from "./html";
|
import { ButtonStyle, CE, clearDataSet, createButton, getReactProps } from "./html";
|
||||||
import { t } from "./translation";
|
import { t } from "./translation";
|
||||||
|
|
||||||
@ -27,7 +27,7 @@ export class TrueAchievements {
|
|||||||
TrueAchievements.open(true, dataset.xboxTitleId, dataset.id);
|
TrueAchievements.open(true, dataset.xboxTitleId, dataset.id);
|
||||||
|
|
||||||
// Close all xCloud's dialogs
|
// Close all xCloud's dialogs
|
||||||
window.BX_EXPOSED.dialogRoutes.closeAll();
|
window.BX_EXPOSED.dialogRoutes?.closeAll();
|
||||||
}
|
}
|
||||||
|
|
||||||
private static $hiddenLink = CE<HTMLAnchorElement>('a', {
|
private static $hiddenLink = CE<HTMLAnchorElement>('a', {
|
||||||
@ -53,6 +53,11 @@ export class TrueAchievements {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static injectAchievementsProgress($elm: HTMLElement) {
|
static injectAchievementsProgress($elm: HTMLElement) {
|
||||||
|
// Only do this in Full version
|
||||||
|
if (SCRIPT_VARIANT !== 'full') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const $parent = $elm.parentElement!;
|
const $parent = $elm.parentElement!;
|
||||||
|
|
||||||
// Wrap xCloud's element with our own
|
// Wrap xCloud's element with our own
|
||||||
@ -89,6 +94,11 @@ export class TrueAchievements {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static injectAchievementDetailPage($parent: HTMLElement) {
|
static injectAchievementDetailPage($parent: HTMLElement) {
|
||||||
|
// Only do this in Full version
|
||||||
|
if (SCRIPT_VARIANT !== 'full') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const props = getReactProps($parent);
|
const props = getReactProps($parent);
|
||||||
if (!props) {
|
if (!props) {
|
||||||
return;
|
return;
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
|
import { isFullVersion } from "@macros/build" with {type: "macro"};
|
||||||
|
|
||||||
import { LoadingScreen } from "@modules/loading-screen";
|
import { LoadingScreen } from "@modules/loading-screen";
|
||||||
import { RemotePlay } from "@modules/remote-play";
|
import { RemotePlayManager } from "@/modules/remote-play-manager";
|
||||||
import { StreamBadges } from "@modules/stream/stream-badges";
|
import { StreamBadges } from "@modules/stream/stream-badges";
|
||||||
import { TouchController } from "@modules/touch-controller";
|
import { TouchController } from "@modules/touch-controller";
|
||||||
import { BxEvent } from "./bx-event";
|
import { BxEvent } from "./bx-event";
|
||||||
@ -30,7 +32,7 @@ class XcloudInterceptor {
|
|||||||
const obj = await response.clone().json();
|
const obj = await response.clone().json();
|
||||||
|
|
||||||
// Store xCloud token
|
// Store xCloud token
|
||||||
RemotePlay.XCLOUD_TOKEN = obj.gsToken;
|
RemotePlayManager.getInstance().xcloudToken = obj.gsToken;
|
||||||
|
|
||||||
// Get server list
|
// Get server list
|
||||||
const serverEmojis = {
|
const serverEmojis = {
|
||||||
@ -147,7 +149,7 @@ class XcloudInterceptor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Touch controller for all games
|
// Touch controller for all games
|
||||||
if (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === StreamTouchController.ALL) {
|
if (isFullVersion() && getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === StreamTouchController.ALL) {
|
||||||
const titleInfo = STATES.currentStream.titleInfo;
|
const titleInfo = STATES.currentStream.titleInfo;
|
||||||
if (titleInfo?.details.hasTouchSupport) {
|
if (titleInfo?.details.hasTouchSupport) {
|
||||||
TouchController.disable();
|
TouchController.disable();
|
||||||
@ -187,7 +189,7 @@ class XcloudInterceptor {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Enable touch controller
|
// Enable touch controller
|
||||||
if (TouchController.isEnabled()) {
|
if (isFullVersion() && TouchController.isEnabled()) {
|
||||||
overrides.inputConfiguration.enableTouchInput = true;
|
overrides.inputConfiguration.enableTouchInput = true;
|
||||||
overrides.inputConfiguration.maxTouchPoints = 10;
|
overrides.inputConfiguration.maxTouchPoints = 10;
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,3 @@
|
|||||||
import { RemotePlay } from "@/modules/remote-play";
|
|
||||||
import { TouchController } from "@/modules/touch-controller";
|
import { TouchController } from "@/modules/touch-controller";
|
||||||
import { BxEvent } from "./bx-event";
|
import { BxEvent } from "./bx-event";
|
||||||
import { SupportedInputType } from "./bx-exposed";
|
import { SupportedInputType } from "./bx-exposed";
|
||||||
@ -8,10 +7,51 @@ import { patchIceCandidates } from "./network";
|
|||||||
import { PrefKey } from "@/enums/pref-keys";
|
import { PrefKey } from "@/enums/pref-keys";
|
||||||
import { getPref, StreamResolution, StreamTouchController } from "./settings-storages/global-settings-storage";
|
import { getPref, StreamResolution, StreamTouchController } from "./settings-storages/global-settings-storage";
|
||||||
import type { RemotePlayConsoleAddresses } from "@/types/network";
|
import type { RemotePlayConsoleAddresses } from "@/types/network";
|
||||||
|
import { RemotePlayManager } from "@/modules/remote-play-manager";
|
||||||
|
|
||||||
export class XhomeInterceptor {
|
export class XhomeInterceptor {
|
||||||
static #consoleAddrs: RemotePlayConsoleAddresses = {};
|
static #consoleAddrs: RemotePlayConsoleAddresses = {};
|
||||||
|
|
||||||
|
private static readonly BASE_DEVICE_INFO = {
|
||||||
|
appInfo: {
|
||||||
|
env: {
|
||||||
|
clientAppId: window.location.host,
|
||||||
|
clientAppType: 'browser',
|
||||||
|
clientAppVersion: '24.17.36',
|
||||||
|
clientSdkVersion: '10.1.14',
|
||||||
|
httpEnvironment: 'prod',
|
||||||
|
sdkInstallId: '',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
dev: {
|
||||||
|
displayInfo: {
|
||||||
|
dimensions: {
|
||||||
|
widthInPixels: 1920,
|
||||||
|
heightInPixels: 1080,
|
||||||
|
},
|
||||||
|
pixelDensity: {
|
||||||
|
dpiX: 1,
|
||||||
|
dpiY: 1,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
hw: {
|
||||||
|
make: 'Microsoft',
|
||||||
|
model: 'unknown',
|
||||||
|
sdktype: 'web',
|
||||||
|
},
|
||||||
|
os: {
|
||||||
|
name: 'windows',
|
||||||
|
ver: '22631.2715',
|
||||||
|
platform: 'desktop',
|
||||||
|
},
|
||||||
|
browser: {
|
||||||
|
browserName: 'chrome',
|
||||||
|
browserVersion: '125.0',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
static async #handleLogin(request: Request) {
|
static async #handleLogin(request: Request) {
|
||||||
try {
|
try {
|
||||||
const clone = (request as Request).clone();
|
const clone = (request as Request).clone();
|
||||||
@ -42,7 +82,7 @@ export class XhomeInterceptor {
|
|||||||
|
|
||||||
const processPorts = (port: number): number[] => {
|
const processPorts = (port: number): number[] => {
|
||||||
const ports = new Set<number>();
|
const ports = new Set<number>();
|
||||||
ports.add(port);
|
port && ports.add(port);
|
||||||
ports.add(9002);
|
ports.add(9002);
|
||||||
|
|
||||||
return Array.from(ports);
|
return Array.from(ports);
|
||||||
@ -111,7 +151,7 @@ export class XhomeInterceptor {
|
|||||||
for (const pair of (clone.headers as any).entries()) {
|
for (const pair of (clone.headers as any).entries()) {
|
||||||
headers[pair[0]] = pair[1];
|
headers[pair[0]] = pair[1];
|
||||||
}
|
}
|
||||||
headers.authorization = `Bearer ${RemotePlay.XCLOUD_TOKEN}`;
|
headers.authorization = `Bearer ${RemotePlayManager.getInstance().xcloudToken}`;
|
||||||
|
|
||||||
const index = request.url.indexOf('.xboxlive.com');
|
const index = request.url.indexOf('.xboxlive.com');
|
||||||
request = new Request('https://wus.core.gssv-play-prod' + request.url.substring(index), {
|
request = new Request('https://wus.core.gssv-play-prod' + request.url.substring(index), {
|
||||||
@ -146,10 +186,10 @@ export class XhomeInterceptor {
|
|||||||
headers[pair[0]] = pair[1];
|
headers[pair[0]] = pair[1];
|
||||||
}
|
}
|
||||||
// Add xHome token to headers
|
// Add xHome token to headers
|
||||||
headers.authorization = `Bearer ${RemotePlay.XHOME_TOKEN}`;
|
headers.authorization = `Bearer ${RemotePlayManager.getInstance().xhomeToken}`;
|
||||||
|
|
||||||
// Patch resolution
|
// Patch resolution
|
||||||
const deviceInfo = RemotePlay.BASE_DEVICE_INFO;
|
const deviceInfo = XhomeInterceptor.BASE_DEVICE_INFO;
|
||||||
if (getPref(PrefKey.REMOTE_PLAY_RESOLUTION) === StreamResolution.DIM_720P) {
|
if (getPref(PrefKey.REMOTE_PLAY_RESOLUTION) === StreamResolution.DIM_720P) {
|
||||||
deviceInfo.dev.os.name = 'android';
|
deviceInfo.dev.os.name = 'android';
|
||||||
}
|
}
|
||||||
|
@ -164,6 +164,10 @@ export class BxSelectElement {
|
|||||||
Object.defineProperty($div, 'value', {
|
Object.defineProperty($div, 'value', {
|
||||||
get() {
|
get() {
|
||||||
return $select.value;
|
return $select.value;
|
||||||
|
},
|
||||||
|
|
||||||
|
set(value) {
|
||||||
|
($div as any).setValue(value);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user