Compare commits

...

179 Commits

Author SHA1 Message Date
5d177bd76c Fix "guideAchievementsDefaultLocked" patch 2024-11-18 22:01:24 +07:00
f18c5c14ed Bump version to 5.9.5 2024-11-06 19:22:07 +07:00
d0ceed00f8 Add "Server locations" link 2024-11-06 19:21:38 +07:00
fce8af4b3b Fix custom buttons disappearing in Guide Menu (#551) 2024-11-06 07:36:32 +07:00
57686f9d8e Bump version to 5.9.4 2024-11-01 20:08:15 +07:00
f0e7272a82 Use "brand" instead of "manufacturer" 2024-11-01 19:57:08 +07:00
b0ecc7171b Update device suggestion URL 2024-11-01 19:52:25 +07:00
17c08792e1 Redirect to /en-US/play if visiting from an unsupported region 2024-11-01 17:02:30 +07:00
e8376b52fe Add patch to modify __PRELOADED_STATE__ 2024-11-01 16:55:58 +07:00
f6581abe34 Show "Remote Play" on web's title instead of "Fortnite" 2024-11-01 08:53:24 +07:00
b090d325ae Migrate PatcherCache to singleton class 2024-11-01 07:22:21 +07:00
ec3daa09fd Use hash of client.js file for calculating patch's signature 2024-11-01 07:16:11 +07:00
b2a2e4d27e Add in-game language support for Bulgarian, Romanian and Thai 2024-10-30 08:45:59 +07:00
4f3430c43c Bump version to 5.9.3 2024-10-29 20:35:15 +07:00
15c6d3c74b Update dists 2024-10-29 20:24:04 +07:00
b170b95145 Fix bugs in NumberStepper 2024-10-29 20:20:34 +07:00
4217b89194 Reduce the amount of event listeners in NumberStepper 2024-10-29 20:11:54 +07:00
38211168e9 Reduce width of controller-friendly select box if it has <optgroup> 2024-10-29 16:56:34 +07:00
392dc2cf86 Categorize servers by continents 2024-10-29 16:51:29 +07:00
67de264aa9 Revert "Use gl.texSubImage2D()"
This reverts commit 3e2c1bb2a4.
2024-10-27 10:20:11 +07:00
3e2c1bb2a4 Use gl.texSubImage2D() 2024-10-27 09:36:34 +07:00
5653914d19 Move WebGL2's drawFrame() function to animate() function 2024-10-26 21:53:03 +07:00
4a8f66f2a1 Upgrade bun 2024-10-25 08:59:05 +07:00
70f43ba8f2 Add emoji flag for SwedenCentral server 2024-10-25 07:33:35 +07:00
4d49639622 Bump version to 5.9.2 2024-10-24 20:58:53 +07:00
22f1ebdd08 Fix Remote Play not working when using different network (#538) 2024-10-24 20:38:01 +07:00
bae51eff3d Bump version to 5.9.1 2024-10-23 21:09:26 +07:00
adc9897210 Update translations 2024-10-23 21:08:20 +07:00
53442557e1 Fix Virtual Controller Remapper's bug (contd) 2024-10-23 20:51:04 +07:00
5b67b4c37d Fix Virtual Controller Remapper's bug (contd) 2024-10-23 20:14:10 +07:00
5a06933143 Update dists 2024-10-23 20:03:26 +07:00
6440c91cdf Fix Virtual Controller Remapper's bug 2024-10-23 20:00:27 +07:00
b06dc6e219 Update polling rate's default text 2024-10-22 20:47:45 +07:00
540a50fb3a Bump version to 5.9.0 2024-10-22 20:14:16 +07:00
e5178830cb Update translations 2024-10-22 20:13:59 +07:00
75549bc477 Update dists 2024-10-22 20:08:18 +07:00
8a3d48d4a3 Optimize Game slug generator by using cached RegEx 2024-10-22 20:07:26 +07:00
33c3b2810a Update NumberStepper 2024-10-22 16:52:23 +07:00
95881dd241 Upgrade bun 2024-10-22 16:49:47 +07:00
c89ebb78a4 Update dists 2024-10-22 10:43:14 +07:00
222ad1c34e Remove "disableSendMetadata" patch 2024-10-22 10:42:43 +07:00
6cfff0274d Replace forEach() with for() 2024-10-22 10:42:09 +07:00
01502363ab Add "disableSendMetadata" patch 2024-10-22 09:14:11 +07:00
9ab63c4a53 Update polling rate in controller-shortcut.js 2024-10-21 22:16:18 +07:00
89a968d688 Update dists 2024-10-21 22:01:53 +07:00
5e98c756d4 Add Polling rate setting 2024-10-21 22:01:32 +07:00
831fd98d02 Remove CONTROLLER_ENABLE_SHORTCUTS 2024-10-21 20:53:27 +07:00
de76364a46 Optimize + refactor code 2024-10-21 20:50:12 +07:00
075b15aa48 Update better-xcloud.user.js 2024-10-21 17:16:45 +07:00
9388d7fbf4 Update better-xcloud.user.js 2024-10-21 11:04:44 +07:00
2d8361ba73 Update better-xcloud.user.js 2024-10-21 08:15:49 +07:00
79c7af10d4 Update better-xcloud.user.js 2024-10-20 21:33:05 +07:00
6bd658e8a6 Bump version to 5.8.6 2024-10-19 18:54:12 +07:00
7e6b89b357 Typo 2024-10-19 18:53:49 +07:00
4271583a5a Add MINIFY_SYNTAX flag in build.ts 2024-10-19 18:46:13 +07:00
1b2cf70248 Minor fix 2024-10-19 18:40:59 +07:00
87447df7fd Fix Server badge not updating between sessions 2024-10-19 18:40:47 +07:00
8664c1a60f Fix taking screenshot not working when limiting FPS 2024-10-19 18:04:05 +07:00
602c31dc7f Update dist 2024-10-19 16:56:02 +07:00
bbaea5f629 Fix Game Bar keep clearing focus even when not playing 2024-10-19 16:54:21 +07:00
03efa528c8 Android: add Shortcut & Wallpaper menu to Game Card's context menu 2024-10-19 16:53:55 +07:00
63aaca7d61 Fix crashing in "disableTouchContextMenu" patch 2024-10-18 22:19:06 +07:00
15ae88e9e6 Disable long touch activating context menu 2024-10-18 21:50:22 +07:00
7578671cc3 Remove "ui_home_context_menu_disabled" setting as it's no longer needed 2024-10-18 21:41:21 +07:00
82cfb11a6d Update dist 2024-10-18 17:32:08 +07:00
15700e736d Fix "Smart TV" User-Agent profile (#527) 2024-10-18 16:57:32 +07:00
b27cfc8215 Refactor utils/html 2024-10-18 16:54:29 +07:00
1e644504ec Upgrade bun 2024-10-18 16:52:41 +07:00
7206d11825 Test: hide <video> when using WebGL2 renderer 2024-10-17 20:19:27 +07:00
fa19a5a68e Bump version to 5.8.5 2024-10-15 19:48:18 +07:00
3f834f74b6 Update "skipFeedbackDialog" patch 2024-10-15 16:49:38 +07:00
749d5d720e Update dist 2024-10-14 21:08:35 +07:00
b969d52a3c Show max FPS value in Stats bar 2024-10-14 21:06:52 +07:00
e5bd7e64a7 Refactor xCloud & xHome interceptors 2024-10-14 20:08:47 +07:00
82ee00b4ae Update dist 2024-10-14 17:17:32 +07:00
8e88af5f8c Set indent of built scripts to 1 space 2024-10-14 17:14:43 +07:00
927eae3f2f Refactor getInstance() methods 2024-10-14 16:56:05 +07:00
9f440e9cf4 Don't call animate() when hiding renderer 2024-10-14 16:47:03 +07:00
1acb30e3af Refactor Game Bar 2024-10-14 16:45:57 +07:00
34159fad22 Update better-xcloud.lite.user.js 2024-10-13 20:04:42 +07:00
741538ebcf Bump version to 5.8.4 2024-10-13 20:00:36 +07:00
6d2e04aff1 Refactor Game Bar actions 2024-10-13 19:15:29 +07:00
f2bc98229f Update version 2024-10-13 17:46:48 +07:00
49fb8e2818 Refactor "data-enabled" to "data-activated" 2024-10-13 17:32:38 +07:00
d012d96675 Add Game Bar action to toggle renderer's visibility 2024-10-13 17:05:27 +07:00
c129feaf2d Refactor WebGL2Player 2024-10-13 16:26:33 +07:00
4f7b23912d Refactor BxLogger 2024-10-13 16:06:01 +07:00
e4d73f9e36 Replace "#" with "private" 2024-10-13 10:51:50 +07:00
2eea9ce8f5 Bump version to 5.8.3 2024-10-12 18:41:41 +07:00
27abab8473 Change "FPS" unit to "fps" 2024-10-12 18:41:28 +07:00
0c34173815 Add "Limit video player's FPS" feature 2024-10-12 16:15:51 +07:00
0164423e45 Test WebGL2 shader 2024-10-12 11:14:55 +07:00
71dcaf4b07 Optimize Clarity boost shader 2024-10-11 17:11:32 +07:00
8f49c48e74 Bump version to 5.8.2 2024-10-11 07:11:37 +07:00
6fa1f73702 Optimize built scripts 2024-10-10 21:43:42 +07:00
728abced45 Add jitter stat 2024-10-10 21:35:36 +07:00
411e43ceb0 Disable inputPollingDurationStats 2024-10-10 20:55:57 +07:00
baa22dbefc Optimize Clarity Boost shader 2024-10-10 17:28:19 +07:00
97fb7a114f Set Sharpness's suggested value to 2 2024-10-09 09:02:52 +07:00
39b2f814b6 Fix stream badge always show "IPv6" even when connecting to IPv4 server #517 2024-10-09 06:30:09 +07:00
3d34bb3edf Bump version to 5.8.1 2024-10-08 20:00:39 +07:00
ab1c93eb3a Upgrade bun 2024-10-08 19:59:53 +07:00
739adfce41 Update translations 2024-10-08 19:55:03 +07:00
2e77f19006 Update scripts 2024-10-08 07:19:20 +07:00
8a40d361d9 Add unsupportedNote property 2024-10-08 07:19:09 +07:00
98fa273b48 Don't render MKB settings on unsupported devices 2024-10-08 07:01:58 +07:00
1e6527413c Update scripts 2024-10-07 21:40:09 +07:00
b9134bc141 Add "MSFS2020: force native MKB support" setting 2024-10-07 21:39:42 +07:00
336a965653 Update translations 2024-10-07 21:21:37 +07:00
3a91210ba7 Bump version to 5.8.0 2024-10-06 20:35:56 +07:00
14f2d8a741 Upgrade bun 2024-10-06 20:35:08 +07:00
c24d1620b6 Update scripts 2024-10-06 20:34:25 +07:00
63f30111cb Update translations 2024-10-06 20:28:19 +07:00
d30a628fb1 Update scripts 2024-10-06 20:25:50 +07:00
5b80170c8b Fix Stream menu's grip handle 2024-10-06 20:20:11 +07:00
203346c0a1 Fix Quick glancing activated when using Touch control dialog 2024-10-06 20:16:08 +07:00
9719454ea1 Fix not hiding Stream menu's grip handle sometimes 2024-10-06 20:10:02 +07:00
59a178bb16 Fix Stats button in Stream menu not updating state 2024-10-06 20:01:53 +07:00
fd1494ebfa Remove Battery option in unsupported browser 2024-10-06 17:02:18 +07:00
8e6dec4b70 Update label's style in Stats bar 2024-10-06 16:15:52 +07:00
6e905621f6 Fix not able to click on checkbox in controller-friendly select box 2024-10-06 15:52:52 +07:00
76b205a65a New stats: clock, play time, battery, download, upload 2024-10-06 15:50:39 +07:00
af41dc7c5e Add build.sh 2024-10-05 10:41:18 +07:00
d0f43db1fd Bump version to 5.7.8 2024-10-02 21:24:23 +07:00
eed0aa9d9e Fix not disabling unsupported features in Settings dialog 2024-10-02 07:17:17 +07:00
9007663a3a Lite: remove NativeMkbHandler code in built script 2024-10-01 17:47:01 +07:00
8f6bc5cb1b Detach VIRTUAL_GAMEPAD_ID from EmulatedMkbHandler 2024-10-01 17:22:33 +07:00
12d8d766dc Lite: remove XhomeInterceptor and TouchController in built script 2024-10-01 17:09:07 +07:00
aeffccaf67 Update better-xcloud.lite.user.js 2024-10-01 16:51:44 +07:00
b2736d574d Disable PatcherCache in Lite version 2024-10-01 16:49:40 +07:00
98cf893956 Fix Settings dialog opening during gameplay 2024-09-30 17:18:40 +07:00
086afafedf Update dist 2024-09-30 17:12:22 +07:00
bd58355ef5 Create better-xcloud.lite.user.js 2024-09-30 17:11:05 +07:00
109cd63a7b Bump version to 5.7.7 2024-09-26 19:50:28 +07:00
8ea6b7f81a Update better-xcloud.user.js 2024-09-26 19:49:58 +07:00
e7c10d43f5 Fix buttons layout in product details page 2024-09-26 19:46:30 +07:00
2f7a57e084 Update translations 2024-09-26 19:22:51 +07:00
c99e38b097 Update better-xcloud.user.js 2024-09-25 20:20:45 +07:00
f6ec6d7c9b Fix not calculating controller-friendly <select>'s size when switching tab 2024-09-25 20:20:06 +07:00
e69fa19ef3 Update better-xcloud.user.js 2024-09-25 19:44:33 +07:00
cc422b31a4 build: collapse if/else blocks without curly braces 2024-09-25 19:43:19 +07:00
9609d0ae7b Fix duplicated CSS strings 2024-09-25 19:43:07 +07:00
506fd71433 Update better-xcloud.user.js 2024-09-25 08:48:57 +07:00
f40b8cb0b2 build: add more minify steps 2024-09-25 08:47:01 +07:00
49a6c036a3 Bump version to 5.7.6 2024-09-24 21:13:56 +07:00
f5a5a79a82 Check offscreen element in isElementVisible() 2024-09-24 20:58:32 +07:00
7ec449160a Update better-xcloud.user.js 2024-09-24 19:53:20 +07:00
fecc5411da Remote Play dialog: update styles 2024-09-24 19:53:02 +07:00
f704452171 Remote Play dialog: replace radio buttons with select box 2024-09-24 19:47:55 +07:00
135193813c Shorten language names 2024-09-24 19:34:20 +07:00
bb57f72e64 Calculate minimum width of controller-friendly <select> elements 2024-09-24 19:31:56 +07:00
69d7cbfffb Bump version to 5.7.5 2024-09-20 17:46:32 +07:00
92e6828cb2 Update better-xcloud.user.js 2024-09-20 17:25:12 +07:00
12ad81e9c7 Update translations 2024-09-20 17:16:32 +07:00
102e0bd318 Use "let" keyword in Patcher to reduce the size of generated script 2024-09-20 16:53:48 +07:00
9308963bc2 Remote Play: Prevent adding "Fortnite" to the "Jump back in" list 2024-09-20 16:42:27 +07:00
c90e013dc1 Upgrade bun 2024-09-20 16:42:03 +07:00
037927b9be Fix not able to control Remote Play dialog using controller (#509) 2024-09-20 07:05:39 +07:00
dabab9acb1 Bump version to 5.7.4 2024-09-19 19:59:12 +07:00
a4a52c6bc3 Update better-xcloud.user.js 2024-09-19 19:58:49 +07:00
eebd7434ea Remove Close icon in Remote Play dialog 2024-09-19 19:58:45 +07:00
ec1805f832 Refactor Remote Play 2024-09-19 18:01:27 +07:00
34f959d5ae Update better-xcloud.user.js 2024-09-18 20:15:02 +07:00
784a31ce43 Migrate Remote Play popup to Navigation dialog 2024-09-18 20:14:49 +07:00
df266d32fc Update better-xcloud.user.js 2024-09-12 22:03:35 +07:00
a6ccd6666e Check next Remote Play server when the console list is empty 2024-09-12 22:03:21 +07:00
fe609034d6 Remote Play: don't accept candidates with port 0 2024-09-11 08:24:50 +07:00
97ec29faa0 Upgrade bun 2024-09-11 08:09:27 +07:00
a34ae75131 Bump version to 5.7.3 2024-09-07 18:36:05 +07:00
139543aaa5 Update better-xcloud.user.js 2024-09-07 18:29:45 +07:00
8099115959 Set Achievements list's default filter to "Locked" 2024-09-07 18:15:04 +07:00
21efa5ffdc Minor fix in Game Bar 2024-09-07 17:27:23 +07:00
07ebf3926b Update script in app when clicking on the "Version x available" button 2024-09-07 16:43:56 +07:00
714178a82d Bump version to 5.7.2 2024-09-06 20:55:12 +07:00
5c2c13e0e6 Update better-xcloud.user.js 2024-09-06 20:52:35 +07:00
3f423325b9 Add Game Bar action to mute/unmute speaker (#491) 2024-09-06 20:44:28 +07:00
870a205ead Update better-xcloud.user.js 2024-09-06 18:17:39 +07:00
421af05882 Update TA button's logic & layout in the Guide Menu 2024-09-06 18:07:13 +07:00
756d105f74 Clear focus on Game Bar after activating it 2024-09-06 17:03:55 +07:00
101 changed files with 18097 additions and 10980 deletions

16
build.sh Executable file
View 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

122
build.ts
View File

@ -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,10 @@ enum BuildTarget {
WEBOS = 'webos', WEBOS = 'webos',
} }
type BuildVariant = 'full' | 'lite';
const MINIFY_SYNTAX = true;
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 +41,67 @@ 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, (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, '{}');
// Remove blank lines
str = str.replaceAll(/\n([\s]*)\n/g, "\n");
// Minify WebGL shaders & JS strings
// Replace "\n " with "\n"
str = str.replaceAll(/\\n+\s*/g, '\\n');
// Remove comment line
str = str.replaceAll(/\\n\/\/.*?(?=\\n)/g, '');
// Replace ${"time".toUpperCase()} with "TIME"
str = str.replaceAll(/\$\{"([^"]+)"\.toUpperCase\(\)\}/g, (match, p1) => {
return p1.toUpperCase();
});
// Replace " (e) =>" to " e =>"
// str = str.replaceAll(/ \(([^\s,.$()]+)\) =>/g, ' $1 =>');
// Set indent to 1 space
if (MINIFY_SYNTAX) {
// Collapse if/else blocks without curly braces
str = str.replaceAll(/((if \(.*?\)|else)\n\s+)/g, '$2 ');
str = str.replaceAll(/\n(\s+)/g, (match, p1) => {
const len = p1.length / 2;
return '\n' + ' '.repeat(len);
});
}
// Fix unicode regex in Patcher.optimizeGameSlugGenerator
str = str.replaceAll('^\\™', '^\\\\u2122');
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 +111,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 +119,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';
@ -69,10 +135,11 @@ const build = async (target: BuildTarget, version: string, config: any={}) => {
outdir: outDir, outdir: outDir,
naming: outputScriptName, naming: outputScriptName,
minify: { minify: {
syntax: true, syntax: MINIFY_SYNTAX,
}, },
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,13 +154,19 @@ 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);
// Create meta file (don't build if it's beta version) // Create meta file (don't build if it's beta version)
if (!version.includes('beta')) { if (!version.includes('beta') && variant === 'full') {
await Bun.write(outDir + '/' + outputMetaName, txtMetaHeader.replace('[[VERSION]]', version)); await Bun.write(outDir + '/' + outputMetaName, txtMetaHeader.replace('[[VERSION]]', version));
} }
@ -118,28 +191,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 +241,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);
*/

BIN
bun.lockb

Binary file not shown.

6068
dist/better-xcloud.lite.user.js vendored Normal file

File diff suppressed because one or more lines are too long

View File

@ -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.1 // @version 5.9.5
// ==/UserScript== // ==/UserScript==

File diff suppressed because one or more lines are too long

View File

@ -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.12",
"@types/node": "^22.5.2", "@types/node": "^22.7.9",
"@types/stylus": "^0.48.42", "@types/stylus": "^0.48.43",
"eslint": "^9.9.1", "eslint": "^9.13.0",
"eslint-plugin-compat": "^6.0.0", "eslint-plugin-compat": "^6.0.1",
"stylus": "^0.63.0" "stylus": "^0.64.0"
}, },
"peerDependencies": { "peerDependencies": {
"typescript": "^5.5.4" "typescript": "^5.6.3"
} }
} }

View File

@ -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;
}
}

View File

@ -76,21 +76,21 @@
} }
/* Touch controller buttons */ /* Touch controller buttons */
div[data-enabled] { div[data-activated] {
button { button {
display: none; display: none;
} }
} }
/* Show enabled button */ /* Show default button */
div[data-enabled='true'] { div[data-activated='false'] {
button:first-of-type { button:first-of-type {
display: block; display: block;
} }
} }
/* Show enable button */ /* Show activated button */
div[data-enabled='false'] { div[data-activated='true'] {
button:last-of-type { button:last-of-type {
display: block; display: block;
} }

View File

@ -1,10 +1,40 @@
.bx-guide-home-achievements-progress {
display: flex;
gap: 10px;
flex-direction: row;
.bx-button {
margin-bottom: 0 !important;
}
html[data-xds-platform=tv] & {
flex-direction: column;
}
html:not([data-xds-platform=tv]) & {
flex-direction: row;
> button:first-of-type {
flex: 1;
}
> button:last-of-type {
width: 40px;
span {
display: none;
}
}
}
}
.bx-guide-home-buttons { .bx-guide-home-buttons {
> div { > div {
display: flex; display: flex;
flex-direction: row; flex-direction: row;
gap: 12px; gap: 12px;
body[data-media-type=tv] & { html[data-xds-platform=tv] & {
flex-direction: column; flex-direction: column;
button { button {
@ -12,7 +42,7 @@
} }
} }
body:not([data-media-type=tv]) & { html:not([data-xds-platform=tv]) & {
button { button {
span { span {
display: none; display: none;

21
src/assets/css/misc.styl Normal file
View 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;
}
}
}

View File

@ -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 {

View File

@ -5,7 +5,7 @@
display: inline-block; display: inline-block;
min-width: 40px; min-width: 40px;
font-family: var(--bx-monospaced-font); font-family: var(--bx-monospaced-font);
font-size: 12px; font-size: 13px;
margin: 0 4px; margin: 0 4px;
} }

View File

@ -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;
}

View File

@ -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;

View File

@ -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;

View File

@ -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,34 @@ 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*="[jit]"] > .bx-stat-jit,
&[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$="[jit]"] > .bx-stat-jit,
&[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 +151,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 +158,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;

View File

@ -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';

View File

@ -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,

View 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";

View File

@ -0,0 +1,8 @@
<svg xmlns='http://www.w3.org/2000/svg' fill='#fff' stroke='none ' fill-rule='evenodd' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 32 32'>
<clipPath id='A'>
<path d='M0 0h32v32H0z'/>
</clipPath>
<g clip-path='url(#A)'>
<path d='M6.123 3.549a1.07 1.07 0 0 0-.798-.359c-.585 0-1.067.482-1.067 1.067 0 .27.102.53.286.727l2.565 2.823C2.267 10.779.184 15.36.092 15.568c-.123.276-.123.591 0 .867.047.105 1.176 2.609 3.687 5.12 3.345 3.344 7.57 5.112 12.221 5.112a16.97 16.97 0 0 0 6.943-1.444l2.933 3.228c.202.228.493.359.798.359.585 0 1.067-.482 1.067-1.067a1.07 1.07 0 0 0-.286-.727L6.123 3.549zm6.31 10.112l5.556 6.114c-.612.322-1.294.49-1.986.49a4.29 4.29 0 0 1-4.267-4.266c0-.831.242-1.643.697-2.338zM16 24.533c-4.104 0-7.689-1.492-10.657-4.433A17.73 17.73 0 0 1 2.267 16c.625-1.172 2.621-4.452 6.313-6.584l2.4 2.633c-.878 1.125-1.356 2.512-1.356 3.939 0 3.511 2.89 6.4 6.4 6.4 1.221 0 2.416-.349 3.444-1.005l1.964 2.16a14.92 14.92 0 0 1-5.432.99zm.8-12.724a1.07 1.07 0 0 1-.867-1.048c0-.585.482-1.067 1.067-1.067a1.12 1.12 0 0 1 .2.019c2.784.54 4.896 2.863 5.169 5.686a1.07 1.07 0 0 1-.962 1.161c-.034.002-.067.002-.1 0a1.07 1.07 0 0 1-1.067-.968 4.29 4.29 0 0 0-3.44-3.783zm15.104 4.626c-.056.125-1.407 3.116-4.448 5.84a1.07 1.07 0 0 1-.724.283c-.585 0-1.067-.482-1.067-1.067a1.07 1.07 0 0 1 .368-.806A17.7 17.7 0 0 0 29.74 16a17.73 17.73 0 0 0-3.083-4.103C23.689 8.959 20.104 7.467 16 7.467a15.82 15.82 0 0 0-2.581.209 1.06 1.06 0 0 1-.186.016 1.07 1.07 0 0 1-1.067-1.066 1.07 1.07 0 0 1 .901-1.054A17.89 17.89 0 0 1 16 5.333c4.651 0 8.876 1.768 12.221 5.114 2.511 2.51 3.64 5.016 3.687 5.121.123.276.123.591 0 .867h-.004z' fill-rule='nonzero'/>
</g>
</svg>

After

Width:  |  Height:  |  Size: 1.7 KiB

8
src/assets/svg/eye.svg Normal file
View File

@ -0,0 +1,8 @@
<svg xmlns='http://www.w3.org/2000/svg' fill='#fff' stroke='none ' fill-rule='evenodd' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 32 32'>
<clipPath id='A'>
<path d='M0 0h32v32H0z'/>
</clipPath>
<g clip-path='url(#A)'>
<path d='M31.908 15.568c-.047-.105-1.176-2.611-3.687-5.121C24.876 7.101 20.651 5.333 16 5.333S7.124 7.101 3.779 10.447c-2.511 2.51-3.646 5.02-3.687 5.121-.123.276-.123.591 0 .867.047.105 1.176 2.609 3.687 5.12 3.345 3.344 7.57 5.112 12.221 5.112s8.876-1.768 12.221-5.112c2.511-2.511 3.64-5.015 3.687-5.12.123-.276.123-.591 0-.867zM16 24.533c-4.104 0-7.689-1.492-10.657-4.433-1.218-1.211-2.254-2.592-3.076-4.1.822-1.508 1.858-2.889 3.076-4.1C8.311 8.959 11.896 7.467 16 7.467s7.689 1.492 10.657 4.433c1.221 1.211 2.259 2.592 3.083 4.1-.961 1.795-5.149 8.533-13.74 8.533zM16 9.6c-3.511 0-6.4 2.889-6.4 6.4s2.889 6.4 6.4 6.4 6.4-2.889 6.4-6.4A6.44 6.44 0 0 0 16 9.6zm0 10.667A4.29 4.29 0 0 1 11.733 16 4.29 4.29 0 0 1 16 11.733 4.29 4.29 0 0 1 20.267 16 4.29 4.29 0 0 1 16 20.267z' fill-rule='nonzero'/>
</g>
</svg>

After

Width:  |  Height:  |  Size: 1.1 KiB

View File

@ -0,0 +1,3 @@
<svg xmlns='http://www.w3.org/2000/svg' fill='#fff' stroke='none' fill-rule='evenodd' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 32 32'>
<path d='M5.462 3.4c-.205-.23-.499-.363-.808-.363-.592 0-1.079.488-1.079 1.08a1.08 1.08 0 0 0 .289.736l4.247 4.672H2.504a2.17 2.17 0 0 0-2.16 2.16v8.637a2.17 2.17 0 0 0 2.16 2.16h6.107l9.426 7.33a1.08 1.08 0 0 0 .662.227c.592 0 1.08-.487 1.08-1.079v-6.601l5.679 6.247a1.08 1.08 0 0 0 .808.363c.592 0 1.08-.487 1.08-1.079a1.08 1.08 0 0 0-.29-.736L5.462 3.4zm-2.958 8.285h5.398v8.637H2.504v-8.637zM17.62 26.752l-7.558-5.878V11.67l7.558 8.313v6.769zm5.668-8.607c1.072-1.218 1.072-3.063 0-4.281a1.08 1.08 0 0 1-.293-.74c0-.592.487-1.079 1.079-1.079a1.08 1.08 0 0 1 .834.393 5.42 5.42 0 0 1 0 7.137 1.08 1.08 0 0 1-.81.365c-.593 0-1.08-.488-1.08-1.08 0-.263.096-.517.27-.715zM12.469 7.888c-.147-.19-.228-.423-.228-.663a1.08 1.08 0 0 1 .417-.853l5.379-4.184a1.08 1.08 0 0 1 .662-.227c.593 0 1.08.488 1.08 1.08v10.105c0 .593-.487 1.08-1.08 1.08s-1.079-.487-1.079-1.08V5.255l-3.636 2.834c-.469.362-1.153.273-1.515-.196v-.005zm19.187 8.115a10.79 10.79 0 0 1-2.749 7.199 1.08 1.08 0 0 1-.793.347c-.593 0-1.08-.487-1.08-1.079 0-.26.094-.511.264-.708 2.918-3.262 2.918-8.253 0-11.516-.184-.2-.287-.461-.287-.733 0-.592.487-1.08 1.08-1.08a1.08 1.08 0 0 1 .816.373 10.78 10.78 0 0 1 2.749 7.197z' fill-rule='nonzero'/>
</svg>

After

Width:  |  Height:  |  Size: 1.4 KiB

View File

@ -37,11 +37,11 @@ export enum PrefKey {
LOCAL_CO_OP_ENABLED = 'local_co_op_enabled', LOCAL_CO_OP_ENABLED = 'local_co_op_enabled',
// LOCAL_CO_OP_SEPARATE_TOUCH_CONTROLLER = 'local_co_op_separate_touch_controller', // LOCAL_CO_OP_SEPARATE_TOUCH_CONTROLLER = 'local_co_op_separate_touch_controller',
CONTROLLER_ENABLE_SHORTCUTS = 'controller_enable_shortcuts',
CONTROLLER_ENABLE_VIBRATION = 'controller_enable_vibration', CONTROLLER_ENABLE_VIBRATION = 'controller_enable_vibration',
CONTROLLER_DEVICE_VIBRATION = 'controller_device_vibration', CONTROLLER_DEVICE_VIBRATION = 'controller_device_vibration',
CONTROLLER_VIBRATION_INTENSITY = 'controller_vibration_intensity', CONTROLLER_VIBRATION_INTENSITY = 'controller_vibration_intensity',
CONTROLLER_SHOW_CONNECTION_STATUS = 'controller_show_connection_status', CONTROLLER_SHOW_CONNECTION_STATUS = 'controller_show_connection_status',
CONTROLLER_POLLING_RATE = 'controller_polling_rate',
NATIVE_MKB_ENABLED = 'native_mkb_enabled', NATIVE_MKB_ENABLED = 'native_mkb_enabled',
NATIVE_MKB_SCROLL_HORIZONTAL_SENSITIVITY = 'native_mkb_scroll_x_sensitivity', NATIVE_MKB_SCROLL_HORIZONTAL_SENSITIVITY = 'native_mkb_scroll_x_sensitivity',
@ -69,12 +69,12 @@ export enum PrefKey {
UI_SCROLLBAR_HIDE = 'ui_scrollbar_hide', UI_SCROLLBAR_HIDE = 'ui_scrollbar_hide',
UI_HIDE_SECTIONS = 'ui_hide_sections', UI_HIDE_SECTIONS = 'ui_hide_sections',
UI_HOME_CONTEXT_MENU_DISABLED = 'ui_home_context_menu_disabled',
UI_GAME_CARD_SHOW_WAIT_TIME = 'ui_game_card_show_wait_time', UI_GAME_CARD_SHOW_WAIT_TIME = 'ui_game_card_show_wait_time',
VIDEO_PLAYER_TYPE = 'video_player_type', VIDEO_PLAYER_TYPE = 'video_player_type',
VIDEO_PROCESSING = 'video_processing', VIDEO_PROCESSING = 'video_processing',
VIDEO_POWER_PREFERENCE = 'video_power_preference', VIDEO_POWER_PREFERENCE = 'video_power_preference',
VIDEO_MAX_FPS = 'video_max_fps',
VIDEO_SHARPNESS = 'video_sharpness', VIDEO_SHARPNESS = 'video_sharpness',
VIDEO_RATIO = 'video_ratio', VIDEO_RATIO = 'video_ratio',
VIDEO_BRIGHTNESS = 'video_brightness', VIDEO_BRIGHTNESS = 'video_brightness',
@ -98,4 +98,5 @@ export enum PrefKey {
REMOTE_PLAY_RESOLUTION = 'xhome_resolution', REMOTE_PLAY_RESOLUTION = 'xhome_resolution',
GAME_FORTNITE_FORCE_CONSOLE = 'game_fortnite_force_console', GAME_FORTNITE_FORCE_CONSOLE = 'game_fortnite_force_console',
GAME_MSFS2020_FORCE_NATIVE_MKB = 'game_msfs2020_force_native_mkb',
} }

View File

@ -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";
@ -5,26 +7,24 @@ import { BxExposed } from "@utils/bx-exposed";
import { t } from "@utils/translation"; import { t } from "@utils/translation";
import { interceptHttpRequests } from "@utils/network"; import { interceptHttpRequests } from "@utils/network";
import { CE } from "@utils/html"; import { CE } from "@utils/html";
import { showGamepadToast } from "@utils/gamepad"; import { showGamepadToast, updatePollingRate } from "@utils/gamepad";
import { EmulatedMkbHandler } from "@modules/mkb/mkb-handler"; import { EmulatedMkbHandler } from "@modules/mkb/mkb-handler";
import { StreamBadges } from "@modules/stream/stream-badges"; import { StreamBadges } from "@modules/stream/stream-badges";
import { StreamStats } from "@modules/stream/stream-stats"; import { StreamStats } from "@modules/stream/stream-stats";
import { addCss, preloadFonts } from "@utils/css"; import { addCss, preloadFonts } from "@utils/css";
import { Toast } from "@utils/toast";
import { LoadingScreen } from "@modules/loading-screen"; import { LoadingScreen } from "@modules/loading-screen";
import { MouseCursorHider } from "@modules/mkb/mouse-cursor-hider"; 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 { disableAdobeAudienceManager, patchAudioContext, patchCanvasContext, patchMeControl, patchPointerLockApi, patchRtcCodecs, patchRtcPeerConnection, patchVideoApi } from "@utils/monkey-patches"; import { disableAdobeAudienceManager, patchAudioContext, patchCanvasContext, patchMeControl, patchPointerLockApi, patchRtcCodecs, patchRtcPeerConnection, patchVideoApi } from "@utils/monkey-patches";
import { AppInterface, STATES } from "@utils/global"; import { AppInterface, STATES } from "@utils/global";
import { BxLogger } from "@utils/bx-logger"; import { BxLogger } from "@utils/bx-logger";
import { GameBar } from "./modules/game-bar/game-bar"; import { GameBar } from "./modules/game-bar/game-bar";
import { Screenshot } from "./utils/screenshot"; import { ScreenshotManager } from "./utils/screenshot-manager";
import { NativeMkbHandler } from "./modules/mkb/native-mkb-handler"; import { NativeMkbHandler } from "./modules/mkb/native-mkb-handler";
import { GuideMenu } from "./modules/ui/guide-menu"; import { GuideMenu } from "./modules/ui/guide-menu";
import { updateVideoPlayer } from "./modules/stream/stream-settings-utils"; import { updateVideoPlayer } from "./modules/stream/stream-settings-utils";
@ -35,12 +35,12 @@ 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";
import { RootDialogObserver } from "./utils/root-dialog-observer";
// Handle login page // Handle login page
if (window.location.pathname.includes('/auth/msa')) { if (window.location.pathname.includes('/auth/msa')) {
@ -63,12 +63,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 +117,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 +160,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);
@ -165,7 +168,7 @@ document.addEventListener('readystatechange', e => {
// Hide "Play with Friends" skeleton section // Hide "Play with Friends" skeleton section
if (getPref(PrefKey.UI_HIDE_SECTIONS).includes(UiSection.FRIENDS)) { if (getPref(PrefKey.UI_HIDE_SECTIONS).includes(UiSection.FRIENDS)) {
const $parent = document.querySelector('div[class*=PlayWithFriendsSkeleton]')?.closest('div[class*=HomePage-module]') as HTMLElement; const $parent = document.querySelector('div[class*=PlayWithFriendsSkeleton]')?.closest<HTMLElement>('div[class*=HomePage-module]');
$parent && ($parent.style.display = 'none'); $parent && ($parent.style.display = 'none');
} }
@ -189,8 +192,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<HTMLElement>('div[class^=UnsupportedMarketPage-module__container]');
}); 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 +230,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;
ScreenshotManager.getInstance().updateCanvasSize($video.videoWidth, $video.videoHeight);
}
updateVideoPlayer(); updateVideoPlayer();
}); });
@ -285,9 +293,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();
@ -298,11 +308,14 @@ function unload() {
window.BX_EXPOSED.stopTakRendering = false; window.BX_EXPOSED.stopTakRendering = false;
NavigationDialogManager.getInstance().hide(); NavigationDialogManager.getInstance().hide();
StreamStats.getInstance().onStoppedPlaying(); StreamStats.getInstance().destroy();
StreamBadges.getInstance().destroy();
MouseCursorHider.stop(); if (isFullVersion()) {
TouchController.reset(); MouseCursorHider.stop();
GameBar.getInstance().disable(); TouchController.reset();
(getPref(PrefKey.GAME_BAR_POSITION) !== 'off') && GameBar.getInstance().disable();
}
} }
window.addEventListener(BxEvent.STREAM_STOPPED, unload); window.addEventListener(BxEvent.STREAM_STOPPED, unload);
@ -310,62 +323,15 @@ 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(); ScreenshotManager.getInstance().takeScreenshot();
}); });
function observeRootDialog($root: HTMLElement) {
let beingShown = false;
const observer = new MutationObserver(mutationList => {
for (const mutation of mutationList) {
if (mutation.type !== 'childList') {
continue;
}
BX_FLAGS.Debug && BxLogger.warning('RootDialog', 'added', mutation.addedNodes);
if (mutation.addedNodes.length === 1) {
const $addedElm = mutation.addedNodes[0];
if ($addedElm instanceof HTMLElement && $addedElm.className) {
// Make sure it's Guide dialog
if ($root.querySelector('div[class*=GuideDialog]')) {
GuideMenu.observe($addedElm);
}
}
}
const shown = !!($root.firstElementChild && $root.firstElementChild.childElementCount > 0);
if (shown !== beingShown) {
beingShown = shown;
BxEvent.dispatch(window, shown ? BxEvent.XCLOUD_DIALOG_SHOWN : BxEvent.XCLOUD_DIALOG_DISMISSED);
}
}
});
observer.observe($root, {subtree: true, childList: true});
}
function waitForRootDialog() {
const observer = new MutationObserver(mutationList => {
for (const mutation of mutationList) {
if (mutation.type !== 'childList') {
continue;
}
const $target = mutation.target as HTMLElement;
if ($target.id && $target.id === 'gamepass-dialog-root') {
observer.disconnect();
observeRootDialog($target);
break;
}
};
});
observer.observe(document.documentElement, {subtree: true, childList: true});
}
function main() { function main() {
waitForRootDialog(); if (getPref(PrefKey.GAME_MSFS2020_FORCE_NATIVE_MKB)) {
BX_FLAGS.ForceNativeMkbTitles.push('9PMQDM08SNK9');
}
// Monkey patches // Monkey patches
patchRtcPeerConnection(); patchRtcPeerConnection();
@ -373,7 +339,7 @@ function main() {
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 +348,54 @@ function main() {
disableAdobeAudienceManager(); disableAdobeAudienceManager();
} }
STATES.userAgent.capabilities.touch && TouchController.updateCustomList(); RootDialogObserver.waitForRootDialog();
overridePreloadState();
VibrationManager.initialSetup();
// Check for Update
BX_FLAGS.CheckForUpdate && checkForUpdate();
// Setup UI // Setup UI
addCss(); addCss();
Toast.setup();
(getPref(PrefKey.GAME_BAR_POSITION) !== 'off') && GameBar.getInstance();
Screenshot.setup();
GuideMenu.addEventListeners(); GuideMenu.getInstance().addEventListeners();
StreamStatsCollector.setupEvents();
StreamBadges.setupEvents(); StreamBadges.setupEvents();
StreamStats.setupEvents(); StreamStats.setupEvents();
EmulatedMkbHandler.setupEvents();
Patcher.init(); if (isFullVersion()) {
updatePollingRate();
STATES.userAgent.capabilities.touch && TouchController.updateCustomList();
disablePwa(); 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();

View File

@ -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();

View File

@ -1,9 +1,8 @@
import { Screenshot } from "@utils/screenshot"; import { ScreenshotManager } from "@/utils/screenshot-manager";
import { GamepadKey } from "@enums/mkb"; 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',
@ -38,66 +38,67 @@ const enum ShortcutAction {
} }
export class ControllerShortcut { export class ControllerShortcut {
static readonly #STORAGE_KEY = 'better_xcloud_controller_shortcuts'; private static readonly STORAGE_KEY = 'better_xcloud_controller_shortcuts';
static #buttonsCache: {[key: string]: boolean[]} = {}; private static buttonsCache: {[key: string]: boolean[]} = {};
static #buttonsStatus: {[key: string]: boolean[]} = {}; private static buttonsStatus: {[key: string]: boolean[]} = {};
static #$selectProfile: HTMLSelectElement; private static $selectProfile: HTMLSelectElement;
static #$selectActions: Partial<{[key in GamepadKey]: HTMLSelectElement}> = {}; private static $selectActions: Partial<{[key in GamepadKey]: HTMLSelectElement}> = {};
static #$container: HTMLElement; private static $container: HTMLElement;
static #ACTIONS: {[key: string]: (ShortcutAction | null)[]} | null = null; private static ACTIONS: {[key: string]: (ShortcutAction | null)[]} | null = null;
static reset(index: number) { static reset(index: number) {
ControllerShortcut.#buttonsCache[index] = []; ControllerShortcut.buttonsCache[index] = [];
ControllerShortcut.#buttonsStatus[index] = []; ControllerShortcut.buttonsStatus[index] = [];
} }
static handle(gamepad: Gamepad): boolean { static handle(gamepad: Gamepad): boolean {
if (!ControllerShortcut.#ACTIONS) { if (!ControllerShortcut.ACTIONS) {
ControllerShortcut.#ACTIONS = ControllerShortcut.#getActionsFromStorage(); ControllerShortcut.ACTIONS = ControllerShortcut.getActionsFromStorage();
} }
const gamepadIndex = gamepad.index; const gamepadIndex = gamepad.index;
const actions = ControllerShortcut.#ACTIONS![gamepad.id]; const actions = ControllerShortcut.ACTIONS![gamepad.id];
if (!actions) { if (!actions) {
return false; return false;
} }
// Move the buttons status from the previous frame to the cache // Move the buttons status from the previous frame to the cache
ControllerShortcut.#buttonsCache[gamepadIndex] = ControllerShortcut.#buttonsStatus[gamepadIndex].slice(0); ControllerShortcut.buttonsCache[gamepadIndex] = ControllerShortcut.buttonsStatus[gamepadIndex].slice(0);
// Clear the buttons status // Clear the buttons status
ControllerShortcut.#buttonsStatus[gamepadIndex] = []; ControllerShortcut.buttonsStatus[gamepadIndex] = [];
const pressed: boolean[] = []; const pressed: boolean[] = [];
let otherButtonPressed = false; let otherButtonPressed = false;
gamepad.buttons.forEach((button, index) => { const entries = gamepad.buttons.entries();
for (const [index, button] of entries) {
// Only add the newly pressed button to the array (holding doesn't count) // Only add the newly pressed button to the array (holding doesn't count)
if (button.pressed && index !== GamepadKey.HOME) { if (button.pressed && index !== GamepadKey.HOME) {
otherButtonPressed = true; otherButtonPressed = true;
pressed[index] = true; pressed[index] = true;
// If this is newly pressed button -> run action // If this is newly pressed button -> run action
if (actions[index] && !ControllerShortcut.#buttonsCache[gamepadIndex][index]) { if (actions[index] && !ControllerShortcut.buttonsCache[gamepadIndex][index]) {
setTimeout(() => ControllerShortcut.#runAction(actions[index]!), 0); setTimeout(() => ControllerShortcut.runAction(actions[index]!), 0);
} }
} }
}); };
ControllerShortcut.#buttonsStatus[gamepadIndex] = pressed; ControllerShortcut.buttonsStatus[gamepadIndex] = pressed;
return otherButtonPressed; return otherButtonPressed;
} }
static #runAction(action: ShortcutAction) { private static runAction(action: ShortcutAction) {
switch (action) { switch (action) {
case ShortcutAction.BETTER_XCLOUD_SETTINGS_SHOW: case ShortcutAction.BETTER_XCLOUD_SETTINGS_SHOW:
SettingsNavigationDialog.getInstance().show(); SettingsNavigationDialog.getInstance().show();
break; break;
case ShortcutAction.STREAM_SCREENSHOT_CAPTURE: case ShortcutAction.STREAM_SCREENSHOT_CAPTURE:
Screenshot.takeScreenshot(); ScreenshotManager.getInstance().takeScreenshot();
break; break;
case ShortcutAction.STREAM_STATS_TOGGLE: case ShortcutAction.STREAM_STATS_TOGGLE:
@ -134,8 +135,8 @@ export class ControllerShortcut {
} }
} }
static #updateAction(profile: string, button: GamepadKey, action: ShortcutAction | null) { private static updateAction(profile: string, button: GamepadKey, action: ShortcutAction | null) {
const actions = ControllerShortcut.#ACTIONS!; const actions = ControllerShortcut.ACTIONS!;
if (!(profile in actions)) { if (!(profile in actions)) {
actions[profile] = []; actions[profile] = [];
} }
@ -147,9 +148,9 @@ export class ControllerShortcut {
actions[profile][button] = action; actions[profile][button] = action;
// Remove empty profiles // Remove empty profiles
for (const key in ControllerShortcut.#ACTIONS) { for (const key in ControllerShortcut.ACTIONS) {
let empty = true; let empty = true;
for (const value of ControllerShortcut.#ACTIONS[key]) { for (const value of ControllerShortcut.ACTIONS[key]) {
if (!!value) { if (!!value) {
empty = false; empty = false;
break; break;
@ -157,19 +158,17 @@ export class ControllerShortcut {
} }
if (empty) { if (empty) {
delete ControllerShortcut.#ACTIONS[key]; delete ControllerShortcut.ACTIONS[key];
} }
} }
// Save to storage // Save to storage
window.localStorage.setItem(ControllerShortcut.#STORAGE_KEY, JSON.stringify(ControllerShortcut.#ACTIONS)); window.localStorage.setItem(ControllerShortcut.STORAGE_KEY, JSON.stringify(ControllerShortcut.ACTIONS));
console.log(ControllerShortcut.#ACTIONS);
} }
static #updateProfileList(e?: GamepadEvent) { private static updateProfileList(e?: GamepadEvent) {
const $select = ControllerShortcut.#$selectProfile; const $select = ControllerShortcut.$selectProfile;
const $container = ControllerShortcut.#$container; const $container = ControllerShortcut.$container;
const $fragment = document.createDocumentFragment(); const $fragment = document.createDocumentFragment();
@ -185,7 +184,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;
} }
@ -205,16 +204,16 @@ export class ControllerShortcut {
} }
static #switchProfile(profile: string) { private static switchProfile(profile: string) {
let actions = ControllerShortcut.#ACTIONS![profile]; let actions = ControllerShortcut.ACTIONS![profile];
if (!actions) { if (!actions) {
actions = []; actions = [];
} }
// Reset selects' values // Reset selects' values
let button: any; let button: any;
for (button in ControllerShortcut.#$selectActions) { for (button in ControllerShortcut.$selectActions) {
const $select = ControllerShortcut.#$selectActions[button as GamepadKey]!; const $select = ControllerShortcut.$selectActions[button as GamepadKey]!;
$select.value = actions[button] || ''; $select.value = actions[button] || '';
BxEvent.dispatch($select, 'input', { BxEvent.dispatch($select, 'input', {
@ -224,15 +223,15 @@ export class ControllerShortcut {
} }
} }
static #getActionsFromStorage() { private static getActionsFromStorage() {
return JSON.parse(window.localStorage.getItem(ControllerShortcut.#STORAGE_KEY) || '{}'); return JSON.parse(window.localStorage.getItem(ControllerShortcut.STORAGE_KEY) || '{}');
} }
static renderSettings() { static renderSettings() {
const PREF_CONTROLLER_FRIENDLY_UI = getPref(PrefKey.UI_CONTROLLER_FRIENDLY); const PREF_CONTROLLER_FRIENDLY_UI = getPref(PrefKey.UI_CONTROLLER_FRIENDLY);
// Read actions from localStorage // Read actions from localStorage
ControllerShortcut.#ACTIONS = ControllerShortcut.#getActionsFromStorage(); ControllerShortcut.ACTIONS = ControllerShortcut.getActionsFromStorage();
const buttons: Map<GamepadKey, PrompFont> = new Map(); const buttons: Map<GamepadKey, PrompFont> = new Map();
buttons.set(GamepadKey.Y, PrompFont.Y); buttons.set(GamepadKey.Y, PrompFont.Y);
@ -314,6 +313,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',
@ -339,7 +339,7 @@ export class ControllerShortcut {
); );
$selectProfile.addEventListener('input', e => { $selectProfile.addEventListener('input', e => {
ControllerShortcut.#switchProfile($selectProfile.value); ControllerShortcut.switchProfile($selectProfile.value);
}); });
const onActionChanged = (e: Event) => { const onActionChanged = (e: Event) => {
@ -360,7 +360,7 @@ export class ControllerShortcut {
($fakeSelect.firstElementChild as HTMLOptionElement).text = fakeText; ($fakeSelect.firstElementChild as HTMLOptionElement).text = fakeText;
} }
!(e as any).ignoreOnChange && ControllerShortcut.#updateAction(profile, button as GamepadKey, action); !(e as any).ignoreOnChange && ControllerShortcut.updateAction(profile, button as GamepadKey, action);
}; };
@ -386,10 +386,12 @@ export class ControllerShortcut {
$select.dataset.button = button.toString(); $select.dataset.button = button.toString();
$select.addEventListener('input', onActionChanged); $select.addEventListener('input', onActionChanged);
ControllerShortcut.#$selectActions[button] = $select; ControllerShortcut.$selectActions[button] = $select;
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,
@ -409,14 +411,14 @@ export class ControllerShortcut {
$container.appendChild($remap); $container.appendChild($remap);
ControllerShortcut.#$selectProfile = $selectProfile; ControllerShortcut.$selectProfile = $selectProfile;
ControllerShortcut.#$container = $container; ControllerShortcut.$container = $container;
// Detect when gamepad connected/disconnect // Detect when gamepad connected/disconnect
window.addEventListener('gamepadconnected', ControllerShortcut.#updateProfileList); window.addEventListener('gamepadconnected', ControllerShortcut.updateProfileList);
window.addEventListener('gamepaddisconnected', ControllerShortcut.#updateProfileList); window.addEventListener('gamepaddisconnected', ControllerShortcut.updateProfileList);
ControllerShortcut.#updateProfileList(); ControllerShortcut.updateProfileList();
return $container; return $container;
} }

View File

@ -30,7 +30,7 @@ export class Dialog {
} = options; } = options;
// Create dialog overlay // Create dialog overlay
const $overlay = document.querySelector('.bx-dialog-overlay') as HTMLElement; const $overlay = document.querySelector<HTMLElement>('.bx-dialog-overlay');
if (!$overlay) { if (!$overlay) {
this.$overlay = CE('div', {'class': 'bx-dialog-overlay bx-gone'}); this.$overlay = CE('div', {'class': 'bx-dialog-overlay bx-gone'});

View File

@ -1,6 +1,16 @@
import { BxEvent } from "@/utils/bx-event";
export abstract class BaseGameBarAction { export abstract class BaseGameBarAction {
abstract $content: HTMLElement;
constructor() {} constructor() {}
reset() {} reset() {}
abstract render(): HTMLElement; onClick(e: Event) {
BxEvent.dispatch(window, BxEvent.GAME_BAR_ACTION_ACTIVATED);
};
render(): HTMLElement {
return this.$content;
};
} }

View File

@ -1,7 +1,6 @@
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 { createButton, ButtonStyle, CE } from "@utils/html"; import { createButton, ButtonStyle, CE } from "@utils/html";
import { t } from "@utils/translation";
import { BaseGameBarAction } from "./action-base"; import { BaseGameBarAction } from "./action-base";
import { MicrophoneShortcut, MicrophoneState } from "../shortcuts/shortcut-microphone"; import { MicrophoneShortcut, MicrophoneState } from "../shortcuts/shortcut-microphone";
@ -9,58 +8,42 @@ import { MicrophoneShortcut, MicrophoneState } from "../shortcuts/shortcut-micro
export class MicrophoneAction extends BaseGameBarAction { export class MicrophoneAction extends BaseGameBarAction {
$content: HTMLElement; $content: HTMLElement;
visible: boolean = false;
constructor() { constructor() {
super(); super();
const onClick = (e: Event) => {
BxEvent.dispatch(window, BxEvent.GAME_BAR_ACTION_ACTIVATED);
const enabled = MicrophoneShortcut.toggle(false);
this.$content.setAttribute('data-enabled', enabled.toString());
};
const $btnDefault = createButton({ const $btnDefault = createButton({
style: ButtonStyle.GHOST, style: ButtonStyle.GHOST,
icon: BxIcon.MICROPHONE, icon: BxIcon.MICROPHONE,
title: t('show-touch-controller'), onClick: this.onClick.bind(this),
onClick: onClick,
classes: ['bx-activated'], classes: ['bx-activated'],
}); });
const $btnMuted = createButton({ const $btnMuted = createButton({
style: ButtonStyle.GHOST, style: ButtonStyle.GHOST,
icon: BxIcon.MICROPHONE_MUTED, icon: BxIcon.MICROPHONE_MUTED,
title: t('hide-touch-controller'), onClick: this.onClick.bind(this),
onClick: onClick,
}); });
this.$content = CE('div', {}, this.$content = CE('div', {}, $btnMuted, $btnDefault);
$btnDefault,
$btnMuted,
);
this.reset();
window.addEventListener(BxEvent.MICROPHONE_STATE_CHANGED, e => { window.addEventListener(BxEvent.MICROPHONE_STATE_CHANGED, e => {
const microphoneState = (e as any).microphoneState; const microphoneState = (e as any).microphoneState;
const enabled = microphoneState === MicrophoneState.ENABLED; const enabled = microphoneState === MicrophoneState.ENABLED;
this.$content.dataset.activated = enabled.toString();
this.$content.setAttribute('data-enabled', enabled.toString());
// Show the button in Game Bar if the mic is enabled // Show the button in Game Bar if the mic is enabled
this.$content.classList.remove('bx-gone'); this.$content.classList.remove('bx-gone');
}); });
} }
render(): HTMLElement { onClick(e: Event) {
return this.$content; super.onClick(e);
const enabled = MicrophoneShortcut.toggle(false);
this.$content.dataset.activated = enabled.toString();
} }
reset(): void { reset(): void {
this.visible = false;
this.$content.classList.add('bx-gone'); this.$content.classList.add('bx-gone');
this.$content.setAttribute('data-enabled', 'false'); this.$content.dataset.activated = 'false';
} }
} }

View File

@ -0,0 +1,38 @@
import { BxIcon } from "@utils/bx-icon";
import { createButton, ButtonStyle, CE } from "@utils/html";
import { BaseGameBarAction } from "./action-base";
import { RendererShortcut } from "../shortcuts/shortcut-renderer";
export class RendererAction extends BaseGameBarAction {
$content: HTMLElement;
constructor() {
super();
const $btnDefault = createButton({
style: ButtonStyle.GHOST,
icon: BxIcon.EYE,
onClick: this.onClick.bind(this),
});
const $btnActivated = createButton({
style: ButtonStyle.GHOST,
icon: BxIcon.EYE_SLASH,
onClick: this.onClick.bind(this),
classes: ['bx-activated'],
});
this.$content = CE('div', {}, $btnDefault, $btnActivated);
}
onClick(e: Event) {
super.onClick(e);
const isVisible = RendererShortcut.toggleVisibility();
this.$content.dataset.activated = (!isVisible).toString();
}
reset(): void {
this.$content.dataset.activated = 'false';
}
}

View File

@ -1,9 +1,8 @@
import { BxEvent } from "@utils/bx-event";
import { BxIcon } from "@utils/bx-icon"; import { BxIcon } from "@utils/bx-icon";
import { createButton, ButtonStyle } from "@utils/html"; import { createButton, ButtonStyle } from "@utils/html";
import { BaseGameBarAction } from "./action-base"; import { BaseGameBarAction } from "./action-base";
import { t } from "@utils/translation"; import { t } from "@utils/translation";
import { Screenshot } from "@/utils/screenshot"; import { ScreenshotManager } from "@/utils/screenshot-manager";
export class ScreenshotAction extends BaseGameBarAction { export class ScreenshotAction extends BaseGameBarAction {
$content: HTMLElement; $content: HTMLElement;
@ -11,20 +10,16 @@ export class ScreenshotAction extends BaseGameBarAction {
constructor() { constructor() {
super(); super();
const onClick = (e: Event) => {
BxEvent.dispatch(window, BxEvent.GAME_BAR_ACTION_ACTIVATED);
Screenshot.takeScreenshot();
};
this.$content = createButton({ this.$content = createButton({
style: ButtonStyle.GHOST, style: ButtonStyle.GHOST,
icon: BxIcon.SCREENSHOT, icon: BxIcon.SCREENSHOT,
title: t('take-screenshot'), title: t('take-screenshot'),
onClick: onClick, onClick: this.onClick.bind(this),
}); });
} }
render(): HTMLElement { onClick(e: Event): void {
return this.$content; super.onClick(e);
ScreenshotManager.getInstance().takeScreenshot();
} }
} }

View File

@ -0,0 +1,45 @@
import { BxEvent } from "@utils/bx-event";
import { BxIcon } from "@utils/bx-icon";
import { createButton, ButtonStyle, CE } from "@utils/html";
import { BaseGameBarAction } from "./action-base";
import { SoundShortcut, SpeakerState } from "../shortcuts/shortcut-sound";
export class SpeakerAction extends BaseGameBarAction {
$content: HTMLElement;
constructor() {
super();
const $btnEnable = createButton({
style: ButtonStyle.GHOST,
icon: BxIcon.AUDIO,
onClick: this.onClick.bind(this),
});
const $btnMuted = createButton({
style: ButtonStyle.GHOST,
icon: BxIcon.SPEAKER_MUTED,
onClick: this.onClick.bind(this),
classes: ['bx-activated'],
});
this.$content = CE('div', {}, $btnEnable, $btnMuted);
window.addEventListener(BxEvent.SPEAKER_STATE_CHANGED, e => {
const speakerState = (e as any).speakerState;
const enabled = speakerState === SpeakerState.ENABLED;
this.$content.dataset.activated = (!enabled).toString();
});
}
onClick(e: Event) {
super.onClick(e);
SoundShortcut.muteUnmute();
}
reset(): void {
this.$content.dataset.activated = 'false';
}
}

View File

@ -1,4 +1,3 @@
import { BxEvent } from "@utils/bx-event";
import { BxIcon } from "@utils/bx-icon"; import { BxIcon } from "@utils/bx-icon";
import { createButton, ButtonStyle, CE } from "@utils/html"; import { createButton, ButtonStyle, CE } from "@utils/html";
import { TouchController } from "@modules/touch-controller"; import { TouchController } from "@modules/touch-controller";
@ -11,44 +10,31 @@ export class TouchControlAction extends BaseGameBarAction {
constructor() { constructor() {
super(); super();
const onClick = (e: Event) => {
BxEvent.dispatch(window, BxEvent.GAME_BAR_ACTION_ACTIVATED);
const $parent = (e as any).target.closest('div[data-enabled]');
let enabled = $parent.getAttribute('data-enabled', 'true') === 'true';
$parent.setAttribute('data-enabled', (!enabled).toString());
TouchController.toggleVisibility(enabled);
};
const $btnEnable = createButton({ const $btnEnable = createButton({
style: ButtonStyle.GHOST, style: ButtonStyle.GHOST,
icon: BxIcon.TOUCH_CONTROL_ENABLE, icon: BxIcon.TOUCH_CONTROL_ENABLE,
title: t('show-touch-controller'), title: t('show-touch-controller'),
onClick: onClick, onClick: this.onClick.bind(this),
classes: ['bx-activated'],
}); });
const $btnDisable = createButton({ const $btnDisable = createButton({
style: ButtonStyle.GHOST, style: ButtonStyle.GHOST,
icon: BxIcon.TOUCH_CONTROL_DISABLE, icon: BxIcon.TOUCH_CONTROL_DISABLE,
title: t('hide-touch-controller'), title: t('hide-touch-controller'),
onClick: onClick, onClick: this.onClick.bind(this),
classes: ['bx-activated'],
}); });
this.$content = CE('div', {}, this.$content = CE('div', {}, $btnEnable, $btnDisable);
$btnEnable,
$btnDisable,
);
this.reset();
} }
render(): HTMLElement { onClick(e: Event) {
return this.$content; super.onClick(e);
const isVisible = TouchController.toggleVisibility();
this.$content.dataset.activated = (!isVisible).toString();
} }
reset(): void { reset(): void {
this.$content.setAttribute('data-enabled', 'true'); this.$content.dataset.activated = 'false';
} }
} }

View File

@ -1,7 +1,5 @@
import { BxEvent } from "@/utils/bx-event";
import { BxIcon } from "@/utils/bx-icon"; import { BxIcon } from "@/utils/bx-icon";
import { createButton, ButtonStyle } from "@/utils/html"; import { createButton, ButtonStyle } from "@/utils/html";
import { t } from "@/utils/translation";
import { BaseGameBarAction } from "./action-base"; import { BaseGameBarAction } from "./action-base";
import { TrueAchievements } from "@/utils/true-achievements"; import { TrueAchievements } from "@/utils/true-achievements";
@ -11,20 +9,15 @@ export class TrueAchievementsAction extends BaseGameBarAction {
constructor() { constructor() {
super(); super();
const onClick = (e: Event) => {
BxEvent.dispatch(window, BxEvent.GAME_BAR_ACTION_ACTIVATED);
TrueAchievements.open(false);
};
this.$content = createButton({ this.$content = createButton({
style: ButtonStyle.GHOST, style: ButtonStyle.GHOST,
icon: BxIcon.TRUE_ACHIEVEMENTS, icon: BxIcon.TRUE_ACHIEVEMENTS,
title: t('true-achievements'), onClick: this.onClick.bind(this),
onClick: onClick,
}); });
} }
render(): HTMLElement { onClick(e: Event) {
return this.$content; super.onClick(e);
TrueAchievements.getInstance().open(false);
} }
} }

View File

@ -7,44 +7,46 @@ import type { BaseGameBarAction } from "./action-base";
import { STATES } from "@utils/global"; import { STATES } from "@utils/global";
import { MicrophoneAction } from "./action-microphone"; import { MicrophoneAction } from "./action-microphone";
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, type GameBarPosition } from "@/utils/settings-storages/global-settings-storage";
import { TrueAchievementsAction } from "./action-true-achievements"; import { TrueAchievementsAction } from "./action-true-achievements";
import { SpeakerAction } from "./action-speaker";
import { RendererAction } from "./action-renderer";
import { BxLogger } from "@/utils/bx-logger";
export class GameBar { export class GameBar {
private static instance: GameBar; private static instance: GameBar;
public static getInstance(): GameBar { public static getInstance = () => GameBar.instance ?? (GameBar.instance = new GameBar());
if (!GameBar.instance) { private readonly LOG_TAG = 'GameBar';
GameBar.instance = new GameBar();
}
return GameBar.instance;
}
private static readonly VISIBLE_DURATION = 2000; private static readonly VISIBLE_DURATION = 2000;
private $gameBar: HTMLElement; private $gameBar: HTMLElement;
private $container: HTMLElement; private $container: HTMLElement;
private timeout: number | null = null; private timeoutId: number | null = null;
private actions: BaseGameBarAction[] = []; private actions: BaseGameBarAction[] = [];
private constructor() { private constructor() {
BxLogger.info(this.LOG_TAG, 'constructor()');
let $container; let $container;
const position = getPref(PrefKey.GAME_BAR_POSITION); const position = getPref(PrefKey.GAME_BAR_POSITION) as GameBarPosition;
const $gameBar = CE('div', {id: 'bx-game-bar', class: 'bx-gone', 'data-position': position}, const $gameBar = CE('div', {id: 'bx-game-bar', class: 'bx-gone', 'data-position': position},
$container = CE('div', {class: 'bx-game-bar-container bx-offscreen'}), $container = CE('div', {class: 'bx-game-bar-container bx-offscreen'}),
createSvgIcon(position === 'bottom-left' ? BxIcon.CARET_RIGHT : BxIcon.CARET_LEFT), createSvgIcon(position === 'bottom-left' ? BxIcon.CARET_RIGHT : BxIcon.CARET_LEFT),
); );
this.actions = [ this.actions = [
new ScreenshotAction(), new ScreenshotAction(),
...(STATES.userAgent.capabilities.touch && (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) !== StreamTouchController.OFF) ? [new TouchControlAction()] : []), ...(STATES.userAgent.capabilities.touch && (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) !== StreamTouchController.OFF) ? [new TouchControlAction()] : []),
new TrueAchievementsAction(), new SpeakerAction(),
new RendererAction(),
new MicrophoneAction(), new MicrophoneAction(),
new TrueAchievementsAction(),
]; ];
// Reverse the action list if Game Bar's position is on the right side // Reverse the action list if Game Bar's position is on the right side
@ -74,11 +76,7 @@ export class GameBar {
// Add animation when hiding game bar // Add animation when hiding game bar
$container.addEventListener('transitionend', e => { $container.addEventListener('transitionend', e => {
const classList = $container.classList; $container.classList.replace('bx-hide', 'bx-offscreen');
if (classList.contains('bx-hide')) {
classList.remove('bx-offscreen', 'bx-hide');
classList.add('bx-offscreen');
}
}); });
document.documentElement.appendChild($gameBar); document.documentElement.appendChild($gameBar);
@ -87,45 +85,38 @@ export class GameBar {
// Enable/disable Game Bar when playing/pausing // Enable/disable Game Bar when playing/pausing
getPref(PrefKey.GAME_BAR_POSITION) !== 'off' && window.addEventListener(BxEvent.XCLOUD_POLLING_MODE_CHANGED, ((e: Event) => { getPref(PrefKey.GAME_BAR_POSITION) !== 'off' && window.addEventListener(BxEvent.XCLOUD_POLLING_MODE_CHANGED, ((e: Event) => {
if (!STATES.isPlaying) {
this.disable();
return;
}
// Toggle Game bar // Toggle Game bar
const mode = (e as any).mode; if (STATES.isPlaying) {
mode !== 'none' ? this.disable() : this.enable(); const mode = (e as any).mode;
mode !== 'none' ? this.disable() : this.enable();
}
}).bind(this)); }).bind(this));
} }
private beginHideTimeout() { private beginHideTimeout() {
this.clearHideTimeout(); this.clearHideTimeout();
this.timeout = window.setTimeout(() => { this.timeoutId = window.setTimeout(() => {
this.timeout = null; this.timeoutId = null;
this.hideBar(); this.hideBar();
}, GameBar.VISIBLE_DURATION); }, GameBar.VISIBLE_DURATION);
} }
private clearHideTimeout() { private clearHideTimeout() {
this.timeout && clearTimeout(this.timeout); this.timeoutId && clearTimeout(this.timeoutId);
this.timeout = null; this.timeoutId = null;
} }
enable() { enable() {
this.$gameBar && this.$gameBar.classList.remove('bx-gone'); this.$gameBar.classList.remove('bx-gone');
} }
disable() { disable() {
this.hideBar(); this.hideBar();
this.$gameBar && this.$gameBar.classList.add('bx-gone'); this.$gameBar.classList.add('bx-gone');
} }
showBar() { showBar() {
if (!this.$container) {
return;
}
this.$container.classList.remove('bx-offscreen', 'bx-hide' , 'bx-gone'); this.$container.classList.remove('bx-offscreen', 'bx-hide' , 'bx-gone');
this.$container.classList.add('bx-show'); this.$container.classList.add('bx-show');
@ -133,12 +124,8 @@ export class GameBar {
} }
hideBar() { hideBar() {
if (!this.$container) { this.clearHideTimeout();
return; this.$container.classList.replace('bx-show', 'bx-hide');
}
this.$container.classList.remove('bx-show');
this.$container.classList.add('bx-hide');
} }
// Reset all states // Reset all states

View File

@ -7,13 +7,13 @@ import { getPref } from "@/utils/settings-storages/global-settings-storage";
import { compressCss } from "@macros/build" with {type: "macro"}; import { compressCss } from "@macros/build" with {type: "macro"};
export class LoadingScreen { export class LoadingScreen {
static #$bgStyle: HTMLElement; private static $bgStyle: HTMLElement;
static #$waitTimeBox: HTMLElement; private static $waitTimeBox: HTMLElement;
static #waitTimeInterval?: number | null = null; private static waitTimeInterval?: number | null = null;
static #orgWebTitle: string; private static orgWebTitle: string;
static #secondsToString(seconds: number) { private static secondsToString(seconds: number) {
const m = Math.floor(seconds / 60); const m = Math.floor(seconds / 60);
const s = Math.floor(seconds % 60); const s = Math.floor(seconds % 60);
@ -28,23 +28,23 @@ export class LoadingScreen {
return; return;
} }
if (!LoadingScreen.#$bgStyle) { if (!LoadingScreen.$bgStyle) {
const $bgStyle = CE('style'); const $bgStyle = CE('style');
document.documentElement.appendChild($bgStyle); document.documentElement.appendChild($bgStyle);
LoadingScreen.#$bgStyle = $bgStyle; LoadingScreen.$bgStyle = $bgStyle;
} }
LoadingScreen.#setBackground(titleInfo.product.heroImageUrl || titleInfo.product.titledHeroImageUrl || titleInfo.product.tileImageUrl); LoadingScreen.setBackground(titleInfo.product.heroImageUrl || titleInfo.product.titledHeroImageUrl || titleInfo.product.tileImageUrl);
if (getPref(PrefKey.UI_LOADING_SCREEN_ROCKET) === 'hide') { if (getPref(PrefKey.UI_LOADING_SCREEN_ROCKET) === 'hide') {
LoadingScreen.#hideRocket(); LoadingScreen.hideRocket();
} }
} }
static #hideRocket() { private 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,17 +53,16 @@ export class LoadingScreen {
display: none; display: none;
} }
`); `);
$bgStyle.textContent! += css;
} }
static #setBackground(imageUrl: string) { private static setBackground(imageUrl: string) {
// Setup style tag // Setup style tag
let $bgStyle = LoadingScreen.#$bgStyle; let $bgStyle = LoadingScreen.$bgStyle;
// 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 => {
@ -91,14 +89,14 @@ export class LoadingScreen {
static setupWaitTime(waitTime: number) { static setupWaitTime(waitTime: number) {
// Hide rocket when queing // Hide rocket when queing
if (getPref(PrefKey.UI_LOADING_SCREEN_ROCKET) === 'hide-queue') { if (getPref(PrefKey.UI_LOADING_SCREEN_ROCKET) === 'hide-queue') {
LoadingScreen.#hideRocket(); LoadingScreen.hideRocket();
} }
let secondsLeft = waitTime; let secondsLeft = waitTime;
let $countDown; let $countDown;
let $estimated; let $estimated;
LoadingScreen.#orgWebTitle = document.title; LoadingScreen.orgWebTitle = document.title;
const endDate = new Date(); const endDate = new Date();
const timeZoneOffsetSeconds = endDate.getTimezoneOffset() * 60; const timeZoneOffsetSeconds = endDate.getTimezoneOffset() * 60;
@ -106,9 +104,9 @@ export class LoadingScreen {
let endDateStr = endDate.toISOString().slice(0, 19); let endDateStr = endDate.toISOString().slice(0, 19);
endDateStr = endDateStr.substring(0, 10) + ' ' + endDateStr.substring(11, 19); endDateStr = endDateStr.substring(0, 10) + ' ' + endDateStr.substring(11, 19);
endDateStr += ` (${LoadingScreen.#secondsToString(waitTime)})`; endDateStr += ` (${LoadingScreen.secondsToString(waitTime)})`;
let $waitTimeBox = LoadingScreen.#$waitTimeBox; let $waitTimeBox = LoadingScreen.$waitTimeBox;
if (!$waitTimeBox) { if (!$waitTimeBox) {
$waitTimeBox = CE('div', {'class': 'bx-wait-time-box'}, $waitTimeBox = CE('div', {'class': 'bx-wait-time-box'},
CE('label', {}, t('server')), CE('label', {}, t('server')),
@ -120,7 +118,7 @@ export class LoadingScreen {
); );
document.documentElement.appendChild($waitTimeBox); document.documentElement.appendChild($waitTimeBox);
LoadingScreen.#$waitTimeBox = $waitTimeBox; LoadingScreen.$waitTimeBox = $waitTimeBox;
} else { } else {
$waitTimeBox.classList.remove('bx-gone'); $waitTimeBox.classList.remove('bx-gone');
$estimated = $waitTimeBox.querySelector('.bx-wait-time-estimated')!; $estimated = $waitTimeBox.querySelector('.bx-wait-time-estimated')!;
@ -128,36 +126,36 @@ export class LoadingScreen {
} }
$estimated.textContent = endDateStr; $estimated.textContent = endDateStr;
$countDown.textContent = LoadingScreen.#secondsToString(secondsLeft); $countDown.textContent = LoadingScreen.secondsToString(secondsLeft);
document.title = `[${$countDown.textContent}] ${LoadingScreen.#orgWebTitle}`; document.title = `[${$countDown.textContent}] ${LoadingScreen.orgWebTitle}`;
LoadingScreen.#waitTimeInterval = window.setInterval(() => { LoadingScreen.waitTimeInterval = window.setInterval(() => {
secondsLeft--; secondsLeft--;
$countDown.textContent = LoadingScreen.#secondsToString(secondsLeft); $countDown.textContent = LoadingScreen.secondsToString(secondsLeft);
document.title = `[${$countDown.textContent}] ${LoadingScreen.#orgWebTitle}`; document.title = `[${$countDown.textContent}] ${LoadingScreen.orgWebTitle}`;
if (secondsLeft <= 0) { if (secondsLeft <= 0) {
LoadingScreen.#waitTimeInterval && clearInterval(LoadingScreen.#waitTimeInterval); LoadingScreen.waitTimeInterval && clearInterval(LoadingScreen.waitTimeInterval);
LoadingScreen.#waitTimeInterval = null; LoadingScreen.waitTimeInterval = null;
} }
}, 1000); }, 1000);
} }
static hide() { static hide() {
LoadingScreen.#orgWebTitle && (document.title = LoadingScreen.#orgWebTitle); LoadingScreen.orgWebTitle && (document.title = LoadingScreen.orgWebTitle);
LoadingScreen.#$waitTimeBox && LoadingScreen.#$waitTimeBox.classList.add('bx-gone'); LoadingScreen.$waitTimeBox && LoadingScreen.$waitTimeBox.classList.add('bx-gone');
if (getPref(PrefKey.UI_LOADING_SCREEN_GAME_ART) && LoadingScreen.#$bgStyle) { if (getPref(PrefKey.UI_LOADING_SCREEN_GAME_ART) && LoadingScreen.$bgStyle) {
const $rocketBg = document.querySelector('#game-stream rect[width="800"]'); const $rocketBg = document.querySelector('#game-stream rect[width="800"]');
$rocketBg && $rocketBg.addEventListener('transitionend', e => { $rocketBg && $rocketBg.addEventListener('transitionend', e => {
LoadingScreen.#$bgStyle.textContent += compressCss(` LoadingScreen.$bgStyle.textContent += compressCss(`
#game-stream { #game-stream {
background: #000 !important; background: #000 !important;
} }
`); `);
}); });
LoadingScreen.#$bgStyle.textContent += compressCss(` LoadingScreen.$bgStyle.textContent += compressCss(`
#game-stream rect[width="800"] { #game-stream rect[width="800"] {
opacity: 1 !important; opacity: 1 !important;
} }
@ -168,10 +166,10 @@ export class LoadingScreen {
} }
static reset() { static reset() {
LoadingScreen.#$bgStyle && (LoadingScreen.#$bgStyle.textContent = ''); LoadingScreen.$bgStyle && (LoadingScreen.$bgStyle.textContent = '');
LoadingScreen.#$waitTimeBox && LoadingScreen.#$waitTimeBox.classList.add('bx-gone'); LoadingScreen.$waitTimeBox && LoadingScreen.$waitTimeBox.classList.add('bx-gone');
LoadingScreen.#waitTimeInterval && clearInterval(LoadingScreen.#waitTimeInterval); LoadingScreen.waitTimeInterval && clearInterval(LoadingScreen.waitTimeInterval);
LoadingScreen.#waitTimeInterval = null; LoadingScreen.waitTimeInterval = null;
} }
} }

View File

@ -1,10 +1,11 @@
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";
import { BxEvent } from "@utils/bx-event"; import { BxEvent } from "@utils/bx-event";
import { Toast } from "@utils/toast"; import { Toast } from "@utils/toast";
import { t } from "@utils/translation"; import { t } from "@utils/translation";
import { LocalDb } from "@utils/local-db";
import { KeyHelper } from "./key-helper"; import { KeyHelper } from "./key-helper";
import type { MkbStoredPreset } from "@/types/mkb"; import type { MkbStoredPreset } from "@/types/mkb";
import { AppInterface, STATES } from "@utils/global"; import { AppInterface, STATES } from "@utils/global";
@ -17,8 +18,7 @@ import { SettingsNavigationDialog } from "../ui/dialog/settings-dialog";
import { NavigationDialogManager } from "../ui/dialog/navigation-dialog"; import { NavigationDialogManager } from "../ui/dialog/navigation-dialog";
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 { MkbPresetsDb } from "@/utils/local-db/mkb-presets-db";
const LOG_TAG = 'MkbHandler';
const PointerToMouseButton = { const PointerToMouseButton = {
1: 0, 1: 0,
@ -26,6 +26,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
@ -121,14 +122,9 @@ This class uses some code from Yuzu emulator to handle mouse's movements
Source: https://github.com/yuzu-emu/yuzu-mainline/blob/master/src/input_common/drivers/mouse.cpp Source: https://github.com/yuzu-emu/yuzu-mainline/blob/master/src/input_common/drivers/mouse.cpp
*/ */
export class EmulatedMkbHandler extends MkbHandler { export class EmulatedMkbHandler extends MkbHandler {
static #instance: EmulatedMkbHandler; private static instance: EmulatedMkbHandler;
public static getInstance(): EmulatedMkbHandler { public static getInstance = () => EmulatedMkbHandler.instance ?? (EmulatedMkbHandler.instance = new EmulatedMkbHandler());
if (!EmulatedMkbHandler.#instance) { private static readonly LOG_TAG = 'EmulatedMkbHandler';
EmulatedMkbHandler.#instance = new EmulatedMkbHandler();
}
return EmulatedMkbHandler.#instance;
}
#CURRENT_PRESET_DATA = MkbPreset.convert(MkbPreset.DEFAULT_PRESET); #CURRENT_PRESET_DATA = MkbPreset.convert(MkbPreset.DEFAULT_PRESET);
@ -136,10 +132,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,
@ -172,8 +166,9 @@ export class EmulatedMkbHandler extends MkbHandler {
#RIGHT_STICK_X: GamepadKey[] = []; #RIGHT_STICK_X: GamepadKey[] = [];
#RIGHT_STICK_Y: GamepadKey[] = []; #RIGHT_STICK_Y: GamepadKey[] = [];
constructor() { private constructor() {
super(); super();
BxLogger.info(EmulatedMkbHandler.LOG_TAG, 'constructor()');
this.#STICK_MAP = { this.#STICK_MAP = {
[GamepadKey.LS_LEFT]: [this.#LEFT_STICK_X, 0, -1], [GamepadKey.LS_LEFT]: [this.#LEFT_STICK_X, 0, -1],
@ -436,7 +431,7 @@ export class EmulatedMkbHandler extends MkbHandler {
#getCurrentPreset = (): Promise<MkbStoredPreset> => { #getCurrentPreset = (): Promise<MkbStoredPreset> => {
return new Promise(resolve => { return new Promise(resolve => {
const presetId = getPref(PrefKey.MKB_DEFAULT_PRESET_ID); const presetId = getPref(PrefKey.MKB_DEFAULT_PRESET_ID);
LocalDb.INSTANCE.getPreset(presetId).then((preset: MkbStoredPreset) => { MkbPresetsDb.getInstance().getPreset(presetId).then((preset: MkbStoredPreset) => {
resolve(preset); resolve(preset);
}); });
}); });
@ -678,14 +673,14 @@ 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') {
AppInterface && NativeMkbHandler.getInstance().init(); AppInterface && NativeMkbHandler.getInstance().init();
} }
} else if (getPref(PrefKey.MKB_ENABLED) && (AppInterface || !UserAgent.isMobile())) { } else if (getPref(PrefKey.MKB_ENABLED) && (AppInterface || !UserAgent.isMobile())) {
BxLogger.info(LOG_TAG, 'Emulate MKB'); BxLogger.info(EmulatedMkbHandler.LOG_TAG, 'Emulate MKB');
EmulatedMkbHandler.getInstance().init(); EmulatedMkbHandler.getInstance().init();
} }
}); });

View File

@ -130,7 +130,6 @@ export class MkbPreset {
mouse[MkbPresetKey.MOUSE_MAP_TO] = MkbPreset.MOUSE_SETTINGS[MkbPresetKey.MOUSE_MAP_TO].default; mouse[MkbPresetKey.MOUSE_MAP_TO] = MkbPreset.MOUSE_SETTINGS[MkbPresetKey.MOUSE_MAP_TO].default;
} }
console.log(obj);
return obj; return obj;
} }
} }

View File

@ -1,10 +1,9 @@
import { CE, createButton, ButtonStyle } from "@utils/html"; import { CE, createButton, ButtonStyle, removeChildElements } from "@utils/html";
import { t } from "@utils/translation"; import { t } from "@utils/translation";
import { Dialog } from "@modules/dialog"; import { Dialog } from "@modules/dialog";
import { KeyHelper } from "./key-helper"; import { KeyHelper } from "./key-helper";
import { MkbPreset } from "./mkb-preset"; import { MkbPreset } from "./mkb-preset";
import { EmulatedMkbHandler } from "./mkb-handler"; import { EmulatedMkbHandler } from "./mkb-handler";
import { LocalDb } from "@utils/local-db";
import { BxIcon } from "@utils/bx-icon"; import { BxIcon } from "@utils/bx-icon";
import type { MkbPresetData, MkbStoredPresets } from "@/types/mkb"; import type { MkbPresetData, MkbStoredPresets } from "@/types/mkb";
import { MkbPresetKey, GamepadKey, GamepadKeyName } from "@enums/mkb"; import { MkbPresetKey, GamepadKey, GamepadKeyName } from "@enums/mkb";
@ -12,18 +11,10 @@ import { deepClone } from "@utils/global";
import { SettingElement } from "@/utils/setting-element"; import { SettingElement } from "@/utils/setting-element";
import { PrefKey } from "@/enums/pref-keys"; import { PrefKey } from "@/enums/pref-keys";
import { getPref, setPref } from "@/utils/settings-storages/global-settings-storage"; import { getPref, setPref } from "@/utils/settings-storages/global-settings-storage";
import { MkbPresetsDb } from "@/utils/local-db/mkb-presets-db";
import { BxLogger } from "@/utils/bx-logger";
type MkbRemapperElements = {
wrapper: HTMLElement | null;
presetsSelect: HTMLSelectElement | null;
activateButton: HTMLButtonElement | null;
currentBindingKey: HTMLElement | null;
allKeyElements: HTMLElement[];
allMouseElements: {[key in MkbPresetKey]?: HTMLElement};
};
type MkbRemapperStates = { type MkbRemapperStates = {
currentPresetId: number; currentPresetId: number;
presets: MkbStoredPresets; presets: MkbStoredPresets;
@ -33,7 +24,7 @@ type MkbRemapperStates = {
}; };
export class MkbRemapper { export class MkbRemapper {
readonly #BUTTON_ORDERS = [ private readonly BUTTON_ORDERS = [
GamepadKey.UP, GamepadKey.UP,
GamepadKey.DOWN, GamepadKey.DOWN,
GamepadKey.LEFT, GamepadKey.LEFT,
@ -66,169 +57,170 @@ export class MkbRemapper {
GamepadKey.RS_RIGHT, GamepadKey.RS_RIGHT,
]; ];
static #instance: MkbRemapper; private static instance: MkbRemapper;
static get INSTANCE() { public static getInstance = () => MkbRemapper.instance ?? (MkbRemapper.instance = new MkbRemapper());
if (!MkbRemapper.#instance) { private readonly LOG_TAG = 'MkbRemapper';
MkbRemapper.#instance = new MkbRemapper();
}
return MkbRemapper.#instance; private states: MkbRemapperStates = {
};
#STATE: MkbRemapperStates = {
currentPresetId: 0, currentPresetId: 0,
presets: {}, presets: {},
editingPresetData: null, editingPresetData: null,
isEditing: false, isEditing: false,
}; };
#$: MkbRemapperElements = { private $wrapper!: HTMLElement;
wrapper: null, private $presetsSelect!: HTMLSelectElement;
presetsSelect: null, private $activateButton!: HTMLButtonElement;
activateButton: null,
currentBindingKey: null, private $currentBindingKey!: HTMLElement;
allKeyElements: [], private allKeyElements: HTMLElement[] = [];
allMouseElements: {}, private allMouseElements: {[key in MkbPresetKey]?: HTMLElement} = {};
};
bindingDialog: Dialog; bindingDialog: Dialog;
constructor() { private constructor() {
this.#STATE.currentPresetId = getPref(PrefKey.MKB_DEFAULT_PRESET_ID); BxLogger.info(this.LOG_TAG, 'constructor()');
this.states.currentPresetId = getPref(PrefKey.MKB_DEFAULT_PRESET_ID);
this.bindingDialog = new Dialog({ this.bindingDialog = new Dialog({
className: 'bx-binding-dialog', className: 'bx-binding-dialog',
content: CE('div', {}, content: CE('div', {},
CE('p', {}, t('press-to-bind')), CE('p', {}, t('press-to-bind')),
CE('i', {}, t('press-esc-to-cancel')), CE('i', {}, t('press-esc-to-cancel')),
), ),
hideCloseButton: true, hideCloseButton: true,
}); });
} }
#clearEventListeners = () => { private clearEventListeners = () => {
window.removeEventListener('keydown', this.#onKeyDown); window.removeEventListener('keydown', this.onKeyDown);
window.removeEventListener('mousedown', this.#onMouseDown); window.removeEventListener('mousedown', this.onMouseDown);
window.removeEventListener('wheel', this.#onWheel); window.removeEventListener('wheel', this.onWheel);
}; };
#bindKey = ($elm: HTMLElement, key: any) => { private bindKey = ($elm: HTMLElement, key: any) => {
const buttonIndex = parseInt($elm.getAttribute('data-button-index')!); const buttonIndex = parseInt($elm.dataset.buttonIndex!);
const keySlot = parseInt($elm.getAttribute('data-key-slot')!); const keySlot = parseInt($elm.dataset.keySlot!);
// Ignore if bind the save key to the same element // Ignore if bind the save key to the same element
if ($elm.getAttribute('data-key-code') === key.code) { if ($elm.dataset.keyCode! === key.code) {
return; return;
} }
// Unbind duplicated keys // Unbind duplicated keys
for (const $otherElm of this.#$.allKeyElements) { for (const $otherElm of this.allKeyElements) {
if ($otherElm.getAttribute('data-key-code') === key.code) { if ($otherElm.dataset.keyCode === key.code) {
this.#unbindKey($otherElm); this.unbindKey($otherElm);
} }
} }
this.#STATE.editingPresetData!.mapping[buttonIndex][keySlot] = key.code; this.states.editingPresetData!.mapping[buttonIndex][keySlot] = key.code;
$elm.textContent = key.name; $elm.textContent = key.name;
$elm.setAttribute('data-key-code', key.code); $elm.dataset.keyCode = key.code;
} }
#unbindKey = ($elm: HTMLElement) => { private unbindKey = ($elm: HTMLElement) => {
const buttonIndex = parseInt($elm.getAttribute('data-button-index')!); const buttonIndex = parseInt($elm.dataset.buttonIndex!);
const keySlot = parseInt($elm.getAttribute('data-key-slot')!); const keySlot = parseInt($elm.dataset.keySlot!);
// Remove key from preset // Remove key from preset
this.#STATE.editingPresetData!.mapping[buttonIndex][keySlot] = null; this.states.editingPresetData!.mapping[buttonIndex][keySlot] = null;
$elm.textContent = ''; $elm.textContent = '';
$elm.removeAttribute('data-key-code'); delete $elm.dataset.keyCode;
} }
#onWheel = (e: WheelEvent) => { private onWheel = (e: WheelEvent) => {
e.preventDefault(); e.preventDefault();
this.#clearEventListeners(); this.clearEventListeners();
this.#bindKey(this.#$.currentBindingKey!, KeyHelper.getKeyFromEvent(e)); this.bindKey(this.$currentBindingKey!, KeyHelper.getKeyFromEvent(e));
window.setTimeout(() => this.bindingDialog.hide(), 200); window.setTimeout(() => this.bindingDialog.hide(), 200);
}; };
#onMouseDown = (e: MouseEvent) => { private onMouseDown = (e: MouseEvent) => {
e.preventDefault(); e.preventDefault();
this.#clearEventListeners(); this.clearEventListeners();
this.#bindKey(this.#$.currentBindingKey!, KeyHelper.getKeyFromEvent(e)); this.bindKey(this.$currentBindingKey!, KeyHelper.getKeyFromEvent(e));
window.setTimeout(() => this.bindingDialog.hide(), 200); window.setTimeout(() => this.bindingDialog.hide(), 200);
}; };
#onKeyDown = (e: KeyboardEvent) => { private onKeyDown = (e: KeyboardEvent) => {
e.preventDefault(); e.preventDefault();
e.stopPropagation(); e.stopPropagation();
this.#clearEventListeners(); this.clearEventListeners();
if (e.code !== 'Escape') { if (e.code !== 'Escape') {
this.#bindKey(this.#$.currentBindingKey!, KeyHelper.getKeyFromEvent(e)); this.bindKey(this.$currentBindingKey!, KeyHelper.getKeyFromEvent(e));
} }
window.setTimeout(() => this.bindingDialog.hide(), 200); window.setTimeout(() => this.bindingDialog.hide(), 200);
}; };
#onBindingKey = (e: MouseEvent) => { private onBindingKey = (e: MouseEvent) => {
if (!this.#STATE.isEditing || e.button !== 0) { if (!this.states.isEditing || e.button !== 0) {
return; return;
} }
console.log(e); console.log(e);
this.#$.currentBindingKey = e.target as HTMLElement; this.$currentBindingKey = e.target as HTMLElement;
window.addEventListener('keydown', this.#onKeyDown); window.addEventListener('keydown', this.onKeyDown);
window.addEventListener('mousedown', this.#onMouseDown); window.addEventListener('mousedown', this.onMouseDown);
window.addEventListener('wheel', this.#onWheel); window.addEventListener('wheel', this.onWheel);
this.bindingDialog.show({title: this.#$.currentBindingKey.getAttribute('data-prompt')!}); this.bindingDialog.show({title: this.$currentBindingKey.dataset.prompt!});
}; };
#onContextMenu = (e: Event) => { private onContextMenu = (e: Event) => {
e.preventDefault(); e.preventDefault();
if (!this.#STATE.isEditing) { if (!this.states.isEditing) {
return; return;
} }
this.#unbindKey(e.target as HTMLElement); this.unbindKey(e.target as HTMLElement);
}; };
#getPreset = (presetId: number) => { private getPreset = (presetId: number) => {
return this.#STATE.presets[presetId]; return this.states.presets[presetId];
} }
#getCurrentPreset = () => { private getCurrentPreset = () => {
return this.#getPreset(this.#STATE.currentPresetId); let preset = this.getPreset(this.states.currentPresetId);
if (!preset) {
// Get the first preset in the list
const firstPresetId = parseInt(Object.keys(this.states.presets)[0]);
preset = this.states.presets[firstPresetId];
this.states.currentPresetId = firstPresetId;
setPref(PrefKey.MKB_DEFAULT_PRESET_ID, firstPresetId);
}
return preset;
} }
#switchPreset = (presetId: number) => { private switchPreset = (presetId: number) => {
this.#STATE.currentPresetId = presetId; this.states.currentPresetId = presetId;
const presetData = this.#getCurrentPreset().data; const presetData = this.getCurrentPreset().data;
for (const $elm of this.#$.allKeyElements) { for (const $elm of this.allKeyElements) {
const buttonIndex = parseInt($elm.getAttribute('data-button-index')!); const buttonIndex = parseInt($elm.dataset.buttonIndex!);
const keySlot = parseInt($elm.getAttribute('data-key-slot')!); const keySlot = parseInt($elm.dataset.keySlot!);
const buttonKeys = presetData.mapping[buttonIndex]; const buttonKeys = presetData.mapping[buttonIndex];
if (buttonKeys && buttonKeys[keySlot]) { if (buttonKeys && buttonKeys[keySlot]) {
$elm.textContent = KeyHelper.codeToKeyName(buttonKeys[keySlot]!); $elm.textContent = KeyHelper.codeToKeyName(buttonKeys[keySlot]!);
$elm.setAttribute('data-key-code', buttonKeys[keySlot]!); $elm.dataset.keyCode = buttonKeys[keySlot]!;
} else { } else {
$elm.textContent = ''; $elm.textContent = '';
$elm.removeAttribute('data-key-code'); delete $elm.dataset.keyCode;
} }
} }
let key: MkbPresetKey; let key: MkbPresetKey;
for (key in this.#$.allMouseElements) { for (key in this.allMouseElements) {
const $elm = this.#$.allMouseElements[key]!; const $elm = this.allMouseElements[key]!;
let value = presetData.mouse[key]; let value = presetData.mouse[key];
if (typeof value === 'undefined') { if (typeof value === 'undefined') {
value = MkbPreset.MOUSE_SETTINGS[key].default; value = MkbPreset.MOUSE_SETTINGS[key].default;
@ -238,74 +230,72 @@ export class MkbRemapper {
} }
// Update state of Activate button // Update state of Activate button
const activated = getPref(PrefKey.MKB_DEFAULT_PRESET_ID) === this.#STATE.currentPresetId; const activated = getPref(PrefKey.MKB_DEFAULT_PRESET_ID) === this.states.currentPresetId;
this.#$.activateButton!.disabled = activated; this.$activateButton.disabled = activated;
this.#$.activateButton!.querySelector('span')!.textContent = activated ? t('activated') : t('activate'); this.$activateButton.querySelector('span')!.textContent = activated ? t('activated') : t('activate');
} }
#refresh() { private async refresh() {
// Clear presets select // Clear presets select
while (this.#$.presetsSelect!.firstChild) { removeChildElements(this.$presetsSelect);
this.#$.presetsSelect!.removeChild(this.#$.presetsSelect!.firstChild);
const presets = await MkbPresetsDb.getInstance().getPresets();
this.states.presets = presets;
const fragment = document.createDocumentFragment();
let defaultPresetId;
if (this.states.currentPresetId === 0) {
this.states.currentPresetId = parseInt(Object.keys(presets)[0]);
defaultPresetId = this.states.currentPresetId;
setPref(PrefKey.MKB_DEFAULT_PRESET_ID, defaultPresetId);
EmulatedMkbHandler.getInstance().refreshPresetData();
} else {
defaultPresetId = getPref(PrefKey.MKB_DEFAULT_PRESET_ID);
} }
LocalDb.INSTANCE.getPresets().then(presets => { for (let id in presets) {
this.#STATE.presets = presets; const preset = presets[id];
const $fragment = document.createDocumentFragment(); let name = preset.name;
if (id === defaultPresetId) {
let defaultPresetId; name = `🎮 ` + name;
if (this.#STATE.currentPresetId === 0) {
this.#STATE.currentPresetId = parseInt(Object.keys(presets)[0]);
defaultPresetId = this.#STATE.currentPresetId;
setPref(PrefKey.MKB_DEFAULT_PRESET_ID, defaultPresetId);
EmulatedMkbHandler.getInstance().refreshPresetData();
} else {
defaultPresetId = getPref(PrefKey.MKB_DEFAULT_PRESET_ID);
} }
for (let id in presets) { const $options = CE<HTMLOptionElement>('option', {value: id}, name);
const preset = presets[id]; $options.selected = parseInt(id) === this.states.currentPresetId;
let name = preset.name;
if (id === defaultPresetId) {
name = `🎮 ` + name;
}
const $options = CE<HTMLOptionElement>('option', {value: id}, name); fragment.appendChild($options);
$options.selected = parseInt(id) === this.#STATE.currentPresetId; };
$fragment.appendChild($options); this.$presetsSelect.appendChild(fragment);
};
this.#$.presetsSelect!.appendChild($fragment); // Update state of Activate button
const activated = defaultPresetId === this.states.currentPresetId;
this.$activateButton.disabled = activated;
this.$activateButton.querySelector('span')!.textContent = activated ? t('activated') : t('activate');
// Update state of Activate button !this.states.isEditing && this.switchPreset(this.states.currentPresetId);
const activated = defaultPresetId === this.#STATE.currentPresetId;
this.#$.activateButton!.disabled = activated;
this.#$.activateButton!.querySelector('span')!.textContent = activated ? t('activated') : t('activate');
!this.#STATE.isEditing && this.#switchPreset(this.#STATE.currentPresetId);
});
} }
#toggleEditing = (force?: boolean) => { private toggleEditing = (force?: boolean) => {
this.#STATE.isEditing = typeof force !== 'undefined' ? force : !this.#STATE.isEditing; this.states.isEditing = typeof force !== 'undefined' ? force : !this.states.isEditing;
this.#$.wrapper!.classList.toggle('bx-editing', this.#STATE.isEditing); this.$wrapper.classList.toggle('bx-editing', this.states.isEditing);
if (this.#STATE.isEditing) { if (this.states.isEditing) {
this.#STATE.editingPresetData = deepClone(this.#getCurrentPreset().data); this.states.editingPresetData = deepClone(this.getCurrentPreset().data);
} else { } else {
this.#STATE.editingPresetData = null; this.states.editingPresetData = null;
} }
const childElements = this.#$.wrapper!.querySelectorAll('select, button, input'); const childElements = this.$wrapper.querySelectorAll('select, button, input');
for (const $elm of Array.from(childElements)) { for (const $elm of Array.from(childElements)) {
if ($elm.parentElement!.parentElement!.classList.contains('bx-mkb-action-buttons')) { if ($elm.parentElement!.parentElement!.classList.contains('bx-mkb-action-buttons')) {
continue; continue;
} }
let disable = !this.#STATE.isEditing; let disable = !this.states.isEditing;
if ($elm.parentElement!.classList.contains('bx-mkb-preset-tools')) { if ($elm.parentElement!.classList.contains('bx-mkb-preset-tools')) {
disable = !disable; disable = !disable;
@ -316,14 +306,14 @@ export class MkbRemapper {
} }
render() { render() {
this.#$.wrapper = CE('div', {'class': 'bx-mkb-settings'}); this.$wrapper = CE('div', {class: 'bx-mkb-settings'});
this.#$.presetsSelect = CE<HTMLSelectElement>('select', {tabindex: -1}); this.$presetsSelect = CE<HTMLSelectElement>('select', {tabindex: -1});
this.#$.presetsSelect!.addEventListener('change', e => { this.$presetsSelect.addEventListener('change', e => {
this.#switchPreset(parseInt((e.target as HTMLSelectElement).value)); this.switchPreset(parseInt((e.target as HTMLSelectElement).value));
}); });
const promptNewName = (value?: string) => { const promptNewName = (value: string) => {
let newName: string | null = ''; let newName: string | null = '';
while (!newName) { while (!newName) {
newName = prompt(t('prompt-preset-name'), value); newName = prompt(t('prompt-preset-name'), value);
@ -336,15 +326,15 @@ export class MkbRemapper {
return newName ? newName : false; return newName ? newName : false;
}; };
const $header = CE('div', {'class': 'bx-mkb-preset-tools'}, const $header = CE('div', {class: 'bx-mkb-preset-tools'},
this.#$.presetsSelect, this.$presetsSelect,
// Rename button // Rename button
createButton({ createButton({
title: t('rename'), title: t('rename'),
icon: BxIcon.CURSOR_TEXT, icon: BxIcon.CURSOR_TEXT,
tabIndex: -1, tabIndex: -1,
onClick: e => { onClick: async () => {
const preset = this.#getCurrentPreset(); const preset = this.getCurrentPreset();
let newName = promptNewName(preset.name); let newName = promptNewName(preset.name);
if (!newName || newName === preset.name) { if (!newName || newName === preset.name) {
@ -353,28 +343,30 @@ export class MkbRemapper {
// Update preset with new name // Update preset with new name
preset.name = newName; preset.name = newName;
LocalDb.INSTANCE.updatePreset(preset).then(id => this.#refresh());
await MkbPresetsDb.getInstance().updatePreset(preset);
await this.refresh();
}, },
}), }),
// New button // New button
createButton({ createButton({
icon: BxIcon.NEW, icon: BxIcon.NEW,
title: t('new'), title: t('new'),
tabIndex: -1, tabIndex: -1,
onClick: e => { onClick: e => {
let newName = promptNewName(''); let newName = promptNewName('');
if (!newName) { if (!newName) {
return; return;
} }
// Create new preset selected name // Create new preset selected name
LocalDb.INSTANCE.newPreset(newName, MkbPreset.DEFAULT_PRESET).then(id => { MkbPresetsDb.getInstance().newPreset(newName, MkbPreset.DEFAULT_PRESET).then(id => {
this.#STATE.currentPresetId = id; this.states.currentPresetId = id;
this.#refresh(); this.refresh();
}); });
}, },
}), }),
// Copy button // Copy button
createButton({ createButton({
@ -382,7 +374,7 @@ export class MkbRemapper {
title: t('copy'), title: t('copy'),
tabIndex: -1, tabIndex: -1,
onClick: e => { onClick: e => {
const preset = this.#getCurrentPreset(); const preset = this.getCurrentPreset();
let newName = promptNewName(`${preset.name} (2)`); let newName = promptNewName(`${preset.name} (2)`);
if (!newName) { if (!newName) {
@ -390,9 +382,9 @@ export class MkbRemapper {
} }
// Create new preset selected name // Create new preset selected name
LocalDb.INSTANCE.newPreset(newName, preset.data).then(id => { MkbPresetsDb.getInstance().newPreset(newName, preset.data).then(id => {
this.#STATE.currentPresetId = id; this.states.currentPresetId = id;
this.#refresh(); this.refresh();
}); });
}, },
}), }),
@ -408,23 +400,23 @@ export class MkbRemapper {
return; return;
} }
LocalDb.INSTANCE.deletePreset(this.#STATE.currentPresetId).then(id => { MkbPresetsDb.getInstance().deletePreset(this.states.currentPresetId).then(id => {
this.#STATE.currentPresetId = 0; this.states.currentPresetId = 0;
this.#refresh(); this.refresh();
}); });
}, },
}), }),
); );
this.#$.wrapper!.appendChild($header); this.$wrapper.appendChild($header);
const $rows = CE('div', {'class': 'bx-mkb-settings-rows'}, const $rows = CE('div', {class: 'bx-mkb-settings-rows'},
CE('i', {'class': 'bx-mkb-note'}, t('right-click-to-unbind')), CE('i', {class: 'bx-mkb-note'}, t('right-click-to-unbind')),
); );
// Render keys // Render keys
const keysPerButton = 2; const keysPerButton = 2;
for (const buttonIndex of this.#BUTTON_ORDERS) { for (const buttonIndex of this.BUTTON_ORDERS) {
const [buttonName, buttonPrompt] = GamepadKeyName[buttonIndex]; const [buttonName, buttonPrompt] = GamepadKeyName[buttonIndex];
let $elm; let $elm;
@ -437,22 +429,22 @@ export class MkbRemapper {
'data-key-slot': i, 'data-key-slot': i,
}, ' '); }, ' ');
$elm.addEventListener('mouseup', this.#onBindingKey); $elm.addEventListener('mouseup', this.onBindingKey);
$elm.addEventListener('contextmenu', this.#onContextMenu); $elm.addEventListener('contextmenu', this.onContextMenu);
$fragment.appendChild($elm); $fragment.appendChild($elm);
this.#$.allKeyElements.push($elm); this.allKeyElements.push($elm);
} }
const $keyRow = CE('div', {'class': 'bx-mkb-key-row'}, const $keyRow = CE('div', {class: 'bx-mkb-key-row'},
CE('label', {'title': buttonName}, buttonPrompt), CE('label', {title: buttonName}, buttonPrompt),
$fragment, $fragment,
); );
$rows.appendChild($keyRow); $rows.appendChild($keyRow);
} }
$rows.appendChild(CE('i', {'class': 'bx-mkb-note'}, t('mkb-adjust-ingame-settings')),); $rows.appendChild(CE('i', {class: 'bx-mkb-note'}, t('mkb-adjust-ingame-settings')),);
// Render mouse settings // Render mouse settings
const $mouseSettings = document.createDocumentFragment(); const $mouseSettings = document.createDocumentFragment();
@ -463,7 +455,7 @@ export class MkbRemapper {
let $elm; let $elm;
const onChange = (e: Event, value: any) => { const onChange = (e: Event, value: any) => {
(this.#STATE.editingPresetData!.mouse as any)[key] = value; (this.states.editingPresetData!.mouse as any)[key] = value;
}; };
const $row = CE('label', { const $row = CE('label', {
class: 'bx-settings-row', class: 'bx-settings-row',
@ -474,32 +466,32 @@ export class MkbRemapper {
); );
$mouseSettings.appendChild($row); $mouseSettings.appendChild($row);
this.#$.allMouseElements[key as MkbPresetKey] = $elm; this.allMouseElements[key as MkbPresetKey] = $elm;
} }
$rows.appendChild($mouseSettings); $rows.appendChild($mouseSettings);
this.#$.wrapper!.appendChild($rows); this.$wrapper.appendChild($rows);
// Render action buttons // Render action buttons
const $actionButtons = CE('div', {'class': 'bx-mkb-action-buttons'}, const $actionButtons = CE('div', {class: 'bx-mkb-action-buttons'},
CE('div', {}, CE('div', {},
// Edit button // Edit button
createButton({ createButton({
label: t('edit'), label: t('edit'),
tabIndex: -1, tabIndex: -1,
onClick: e => this.#toggleEditing(true), onClick: e => this.toggleEditing(true),
}), }),
// Activate button // Activate button
this.#$.activateButton = createButton({ this.$activateButton = createButton({
label: t('activate'), label: t('activate'),
style: ButtonStyle.PRIMARY, style: ButtonStyle.PRIMARY,
tabIndex: -1, tabIndex: -1,
onClick: e => { onClick: e => {
setPref(PrefKey.MKB_DEFAULT_PRESET_ID, this.#STATE.currentPresetId); setPref(PrefKey.MKB_DEFAULT_PRESET_ID, this.states.currentPresetId);
EmulatedMkbHandler.getInstance().refreshPresetData(); EmulatedMkbHandler.getInstance().refreshPresetData();
this.#refresh(); this.refresh();
}, },
}), }),
), ),
@ -512,8 +504,8 @@ export class MkbRemapper {
tabIndex: -1, tabIndex: -1,
onClick: e => { onClick: e => {
// Restore preset // Restore preset
this.#switchPreset(this.#STATE.currentPresetId); this.switchPreset(this.states.currentPresetId);
this.#toggleEditing(false); this.toggleEditing(false);
}, },
}), }),
@ -523,27 +515,27 @@ export class MkbRemapper {
style: ButtonStyle.PRIMARY, style: ButtonStyle.PRIMARY,
tabIndex: -1, tabIndex: -1,
onClick: e => { onClick: e => {
const updatedPreset = deepClone(this.#getCurrentPreset()); const updatedPreset = deepClone(this.getCurrentPreset());
updatedPreset.data = this.#STATE.editingPresetData as MkbPresetData; updatedPreset.data = this.states.editingPresetData as MkbPresetData;
LocalDb.INSTANCE.updatePreset(updatedPreset).then(id => { MkbPresetsDb.getInstance().updatePreset(updatedPreset).then(id => {
// If this is the default preset => refresh preset data // If this is the default preset => refresh preset data
if (id === getPref(PrefKey.MKB_DEFAULT_PRESET_ID)) { if (id === getPref(PrefKey.MKB_DEFAULT_PRESET_ID)) {
EmulatedMkbHandler.getInstance().refreshPresetData(); EmulatedMkbHandler.getInstance().refreshPresetData();
} }
this.#toggleEditing(false); this.toggleEditing(false);
this.#refresh(); this.refresh();
}); });
}, },
}), }),
), ),
); );
this.#$.wrapper!.appendChild($actionButtons); this.$wrapper.appendChild($actionButtons);
this.#toggleEditing(false); this.toggleEditing(false);
this.#refresh(); this.refresh();
return this.#$.wrapper; return this.$wrapper;
} }
} }

View File

@ -7,6 +7,7 @@ import { BxEvent } from "@/utils/bx-event";
import { ButtonStyle, CE, createButton } from "@/utils/html"; import { ButtonStyle, CE, createButton } from "@/utils/html";
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 { BxLogger } from "@/utils/bx-logger";
type NativeMouseData = { type NativeMouseData = {
X: number, X: number,
@ -23,6 +24,9 @@ type XcloudInputSink = {
export class NativeMkbHandler extends MkbHandler { export class NativeMkbHandler extends MkbHandler {
private static instance: NativeMkbHandler; private static instance: NativeMkbHandler;
public static getInstance = () => NativeMkbHandler.instance ?? (NativeMkbHandler.instance = new NativeMkbHandler());
private readonly LOG_TAG = 'NativeMkbHandler';
#pointerClient: PointerClient | undefined; #pointerClient: PointerClient | undefined;
#enabled: boolean = false; #enabled: boolean = false;
@ -37,12 +41,9 @@ export class NativeMkbHandler extends MkbHandler {
#$message?: HTMLElement; #$message?: HTMLElement;
public static getInstance(): NativeMkbHandler { private constructor() {
if (!NativeMkbHandler.instance) { super();
NativeMkbHandler.instance = new NativeMkbHandler(); BxLogger.info(this.LOG_TAG, 'constructor()');
}
return NativeMkbHandler.instance;
} }
#onKeyboardEvent(e: KeyboardEvent) { #onKeyboardEvent(e: KeyboardEvent) {

View File

@ -2,8 +2,6 @@ import { BxLogger } from "@/utils/bx-logger";
import { Toast } from "@/utils/toast"; import { Toast } from "@/utils/toast";
import type { MkbHandler } from "./base-mkb-handler"; import type { MkbHandler } from "./base-mkb-handler";
const LOG_TAG = 'PointerClient';
enum PointerAction { enum PointerAction {
MOVE = 1, MOVE = 1,
BUTTON_PRESS = 2, BUTTON_PRESS = 2,
@ -15,45 +13,44 @@ enum PointerAction {
export class PointerClient { export class PointerClient {
private static instance: PointerClient; private static instance: PointerClient;
public static getInstance(): PointerClient { public static getInstance = () => PointerClient.instance ?? (PointerClient.instance = new PointerClient());
if (!PointerClient.instance) { private readonly LOG_TAG = 'PointerClient';
PointerClient.instance = new PointerClient();
}
return PointerClient.instance; private socket: WebSocket | undefined | null;
private mkbHandler: MkbHandler | undefined;
private constructor() {
BxLogger.info(this.LOG_TAG, 'constructor()');
} }
#socket: WebSocket | undefined | null;
#mkbHandler: MkbHandler | undefined;
start(port: number, mkbHandler: MkbHandler) { start(port: number, mkbHandler: MkbHandler) {
if (!port) { if (!port) {
throw new Error('PointerServer port is 0'); throw new Error('PointerServer port is 0');
} }
this.#mkbHandler = mkbHandler; this.mkbHandler = mkbHandler;
// Create WebSocket connection. // Create WebSocket connection.
this.#socket = new WebSocket(`ws://localhost:${port}`); this.socket = new WebSocket(`ws://localhost:${port}`);
this.#socket.binaryType = 'arraybuffer'; this.socket.binaryType = 'arraybuffer';
// Connection opened // Connection opened
this.#socket.addEventListener('open', (event) => { this.socket.addEventListener('open', (event) => {
BxLogger.info(LOG_TAG, 'connected') BxLogger.info(this.LOG_TAG, 'connected')
}); });
// Error // Error
this.#socket.addEventListener('error', (event) => { this.socket.addEventListener('error', (event) => {
BxLogger.error(LOG_TAG, event); BxLogger.error(this.LOG_TAG, event);
Toast.show('Cannot setup mouse: ' + event); Toast.show('Cannot setup mouse: ' + event);
}); });
this.#socket.addEventListener('close', (event) => { this.socket.addEventListener('close', (event) => {
this.#socket = null; this.socket = null;
}); });
// Listen for messages // Listen for messages
this.#socket.addEventListener('message', (event) => { this.socket.addEventListener('message', (event) => {
const dataView = new DataView(event.data); const dataView = new DataView(event.data);
let messageType = dataView.getInt8(0); let messageType = dataView.getInt8(0);
@ -84,7 +81,7 @@ export class PointerClient {
offset += Int16Array.BYTES_PER_ELEMENT; offset += Int16Array.BYTES_PER_ELEMENT;
const y = dataView.getInt16(offset); const y = dataView.getInt16(offset);
this.#mkbHandler?.handleMouseMove({ this.mkbHandler?.handleMouseMove({
movementX: x, movementX: x,
movementY: y, movementY: y,
}); });
@ -94,7 +91,7 @@ export class PointerClient {
onPress(messageType: PointerAction, dataView: DataView, offset: number) { onPress(messageType: PointerAction, dataView: DataView, offset: number) {
const button = dataView.getUint8(offset); const button = dataView.getUint8(offset);
this.#mkbHandler?.handleMouseClick({ this.mkbHandler?.handleMouseClick({
pointerButton: button, pointerButton: button,
pressed: messageType === PointerAction.BUTTON_PRESS, pressed: messageType === PointerAction.BUTTON_PRESS,
}); });
@ -108,7 +105,7 @@ export class PointerClient {
offset += Int16Array.BYTES_PER_ELEMENT; offset += Int16Array.BYTES_PER_ELEMENT;
const hScroll = dataView.getInt16(offset); const hScroll = dataView.getInt16(offset);
this.#mkbHandler?.handleMouseWheel({ this.mkbHandler?.handleMouseWheel({
vertical: vScroll, vertical: vScroll,
horizontal: hScroll, horizontal: hScroll,
}); });
@ -118,13 +115,13 @@ export class PointerClient {
onPointerCaptureChanged(dataView: DataView, offset: number) { onPointerCaptureChanged(dataView: DataView, offset: number) {
const hasCapture = dataView.getInt8(offset) === 1; const hasCapture = dataView.getInt8(offset) === 1;
!hasCapture && this.#mkbHandler?.stop(); !hasCapture && this.mkbHandler?.stop();
} }
stop() { stop() {
try { try {
this.#socket?.close(); this.socket?.close();
} catch (e) {} } catch (e) {}
this.#socket = null; this.socket = null;
} }
} }

View File

@ -17,6 +17,7 @@ import { UiSection } from "@/enums/ui-sections.js";
import { PrefKey } from "@/enums/pref-keys.js"; import { PrefKey } from "@/enums/pref-keys.js";
import { getPref, StreamTouchController } from "@/utils/settings-storages/global-settings-storage"; import { getPref, StreamTouchController } from "@/utils/settings-storages/global-settings-storage";
import { GamePassCloudGallery } from "@/enums/game-pass-gallery.js"; import { GamePassCloudGallery } from "@/enums/game-pass-gallery.js";
import { t } from "@/utils/translation.js";
type PatchArray = (keyof typeof PATCHES)[]; type PatchArray = (keyof typeof PATCHES)[];
@ -54,7 +55,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 +70,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 +79,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 +100,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 +112,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 +132,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 +144,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 +152,43 @@ 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 = 'titleTemplate:void 0,title:';
const index = str.indexOf(text);
if (index < 0) {
return false;
}
str = PatcherUtils.insertAt(str, index + text.length, `!!window.BX_REMOTE_PLAY_CONFIG ? "${t('remote-play')} - Better xCloud" :`);
return str;
},
// 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;
} }
@ -183,16 +202,22 @@ if (!!window.BX_REMOTE_PLAY_CONFIG) {
return false; return false;
} }
const nextIndex = str.indexOf('setTimeout(this.pollGamepads', index); const setTimeoutIndex = str.indexOf('setTimeout(this.pollGamepads', index);
if (nextIndex === -1) { if (setTimeoutIndex < 0) {
return false; return false;
} }
let codeBlock = str.substring(index, nextIndex); let codeBlock = str.substring(index, setTimeoutIndex);
// Patch polling rate
const tmp = str.substring(setTimeoutIndex, setTimeoutIndex + 150);
const tmpPatched = tmp.replaceAll('Math.max(0,4-', 'Math.max(0,window.BX_CONTROLLER_POLLING_RATE-');
str = PatcherUtils.replaceWith(str, setTimeoutIndex, tmp, tmpPatched);
// Block gamepad stats collecting // Block gamepad stats collecting
if (getPref(PrefKey.BLOCK_TRACKING)) { if (getPref(PrefKey.BLOCK_TRACKING)) {
codeBlock = codeBlock.replaceAll('this.inputPollingIntervalStats.addValue', ''); codeBlock = codeBlock.replace('this.inputPollingIntervalStats.addValue', '');
codeBlock = codeBlock.replace('this.inputPollingDurationStats.addValue', '');
} }
// Map the Share button on Xbox Series controller with the capturing screenshot feature // Map the Share button on Xbox Series controller with the capturing screenshot feature
@ -200,17 +225,18 @@ if (!!window.BX_REMOTE_PLAY_CONFIG) {
if (match) { if (match) {
const gamepadVar = match[1]; const gamepadVar = match[1];
const newCode = renderString(codeControllerShortcuts, { const newCode = renderString(codeControllerShortcuts, {
gamepadVar, gamepadVar,
}); });
codeBlock = codeBlock.replace('this.gamepadTimestamps.set', newCode + 'this.gamepadTimestamps.set'); codeBlock = codeBlock.replace('this.gamepadTimestamps.set', newCode + 'this.gamepadTimestamps.set');
} }
return str.substring(0, index) + codeBlock + str.substring(nextIndex); str = str.substring(0, index) + codeBlock + str.substring(setTimeoutIndex);
return str;
}, },
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 +252,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 +263,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 +304,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 +317,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 +342,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 +389,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 +401,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 +413,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 +453,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 +463,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 +485,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 +509,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 +531,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 +559,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 +571,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 +582,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 +594,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 +607,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 +624,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,17 +638,17 @@ true` + text;
}, },
skipFeedbackDialog(str: string) { skipFeedbackDialog(str: string) {
const text = '&&this.shouldTransitionToFeedback('; let text = 'shouldTransitionToFeedback(e){';
if (!str.includes(text)) { if (!str.includes(text)) {
return false; return false;
} }
str = str.replace(text, '&& false ' + text); str = str.replace(text, text + 'return !1;');
return str; return str;
}, },
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 +658,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 +668,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 +680,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 +691,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 +701,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;
} }
@ -809,10 +835,10 @@ true` + text;
[UiSection.MOST_POPULAR]: GamePassCloudGallery.MOST_POPULAR, [UiSection.MOST_POPULAR]: GamePassCloudGallery.MOST_POPULAR,
}; };
PREF_HIDE_SECTIONS.forEach(section => { for (const section of PREF_HIDE_SECTIONS) {
const galleryId = sections[section]; const galleryId = sections[section];
galleryId && siglIds.push(galleryId); galleryId && siglIds.push(galleryId);
}); };
const checkSyntax = siglIds.map(item => `siglId === "${item}"`).join(' || '); const checkSyntax = siglIds.map(item => `siglId === "${item}"`).join(' || ');
@ -830,7 +856,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 +920,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;
} }
@ -912,6 +938,63 @@ if (this.baseStorageKey in window.BX_EXPOSED.overrideSettings) {
str = PatcherUtils.insertAt(str, index, 'window.BxEvent.dispatch(window, window.BxEvent.XCLOUD_ROUTER_HISTORY_READY, {history: this.history});'); str = PatcherUtils.insertAt(str, index, 'window.BxEvent.dispatch(window, window.BxEvent.XCLOUD_ROUTER_HISTORY_READY, {history: this.history});');
return str; return str;
}, },
// Set Achievements list's filter default to "Locked"
guideAchievementsDefaultLocked(str: string) {
let index = str.indexOf('FilterButton-module__container');
index >= 0 && (index = PatcherUtils.lastIndexOf(str, '"All"', index, 150));
if (index < 0) {
return false;
}
str = PatcherUtils.replaceWith(str, index, '"All"', '"Locked"');
index = str.indexOf('"Guide_Achievements_Unlocked_Empty","Guide_Achievements_Locked_Empty"');
index >= 0 && (index = PatcherUtils.indexOf(str, '"All"', index, 250));
if (index < 0) {
return false;
}
str = PatcherUtils.replaceWith(str, index, '"All"', '"Locked"');
return str;
},
// Disable long touch activating context menu
disableTouchContextMenu(str: string) {
let index = str.indexOf('"ContextualCardActions-module__container');
index >= 0 && (index = str.indexOf('addEventListener("touchstart"', index));
index >= 0 && (index = PatcherUtils.lastIndexOf(str, 'return ', index, 50));
if (index < 0) {
return false;
}
str = PatcherUtils.replaceWith(str, index, 'return', 'return () => {};');
return str;
},
// Optimize Game slug generator by using cached RegEx
optimizeGameSlugGenerator(str: string) {
let text = '/[;,/?:@&=+_`~$%#^*()!^\\u2122\\xae\\xa9]/g';
if (!str.includes(text)) {
return false;
}
str = str.replace(text, 'window.BX_EXPOSED.GameSlugRegexes[0]');
str = str.replace('/ {2,}/g', 'window.BX_EXPOSED.GameSlugRegexes[1]');
str = str.replace('/ /g', 'window.BX_EXPOSED.GameSlugRegexes[2]');
return str;
},
modifyPreloadedState(str: string) {
let text = '=window.__PRELOADED_STATE__;';
if (!str.includes(text)) {
return false;
}
str = str.replace(text, '=window.BX_EXPOSED.modifyPreloadedState(window.__PRELOADED_STATE__);');
return str;
},
}; };
let PATCH_ORDERS: PatchArray = [ let PATCH_ORDERS: PatchArray = [
@ -922,6 +1005,10 @@ let PATCH_ORDERS: PatchArray = [
'exposeInputSink', 'exposeInputSink',
] : []), ] : []),
'modifyPreloadedState',
'optimizeGameSlugGenerator',
'detectBrowserRouterReady', 'detectBrowserRouterReady',
'patchRequestInfoCrash', 'patchRequestInfoCrash',
@ -933,6 +1020,8 @@ let PATCH_ORDERS: PatchArray = [
'exposeStreamSession', 'exposeStreamSession',
'exposeDialogRoutes', 'exposeDialogRoutes',
'guideAchievementsDefaultLocked',
'enableTvRoutes', 'enableTvRoutes',
AppInterface && 'detectProductDetailsPage', AppInterface && 'detectProductDetailsPage',
@ -948,6 +1037,10 @@ let PATCH_ORDERS: PatchArray = [
getPref(PrefKey.UI_HIDE_SECTIONS).includes(UiSection.TOUCH) && 'ignorePlayWithTouchSection', getPref(PrefKey.UI_HIDE_SECTIONS).includes(UiSection.TOUCH) && 'ignorePlayWithTouchSection',
(getPref(PrefKey.UI_HIDE_SECTIONS).includes(UiSection.NATIVE_MKB) || getPref(PrefKey.UI_HIDE_SECTIONS).includes(UiSection.MOST_POPULAR)) && 'ignoreSiglSections', (getPref(PrefKey.UI_HIDE_SECTIONS).includes(UiSection.NATIVE_MKB) || getPref(PrefKey.UI_HIDE_SECTIONS).includes(UiSection.MOST_POPULAR)) && 'ignoreSiglSections',
...(STATES.userAgent.capabilities.touch ? [
'disableTouchContextMenu',
] : []),
...(getPref(PrefKey.BLOCK_TRACKING) ? [ ...(getPref(PrefKey.BLOCK_TRACKING) ? [
'disableAiTrack', 'disableAiTrack',
'disableTelemetry', 'disableTelemetry',
@ -962,6 +1055,8 @@ let PATCH_ORDERS: PatchArray = [
'remotePlayKeepAlive', 'remotePlayKeepAlive',
'remotePlayDirectConnectUrl', 'remotePlayDirectConnectUrl',
'remotePlayDisableAchievementToast', 'remotePlayDisableAchievementToast',
'remotePlayRecentlyUsedTitleIds',
'remotePlayWebTitle',
STATES.userAgent.capabilities.touch && 'patchUpdateInputConfigurationAsync', STATES.userAgent.capabilities.touch && 'patchUpdateInputConfigurationAsync',
] : []), ] : []),
@ -1034,7 +1129,7 @@ export class Patcher {
return nativeBind.apply(this, arguments); return nativeBind.apply(this, arguments);
} }
PatcherCache.init(); PatcherCache.getInstance().init();
if (typeof arguments[1] === 'function') { if (typeof arguments[1] === 'function') {
BxLogger.info(LOG_TAG, 'Restored Function.prototype.bind()'); BxLogger.info(LOG_TAG, 'Restored Function.prototype.bind()');
@ -1059,11 +1154,12 @@ export class Patcher {
let appliedPatches: PatchArray; let appliedPatches: PatchArray;
const patchesMap: Record<string, PatchArray> = {}; const patchesMap: Record<string, PatchArray> = {};
const patcherCache = PatcherCache.getInstance();
for (let id in item[1]) { for (let id in item[1]) {
appliedPatches = []; appliedPatches = [];
const cachedPatches = PatcherCache.getPatches(id); const cachedPatches = patcherCache.getPatches(id);
if (cachedPatches) { if (cachedPatches) {
patchesToCheck = cachedPatches.slice(0); patchesToCheck = cachedPatches.slice(0);
patchesToCheck.push(...PATCH_ORDERS); patchesToCheck.push(...PATCH_ORDERS);
@ -1130,7 +1226,7 @@ export class Patcher {
} }
if (Object.keys(patchesMap).length) { if (Object.keys(patchesMap).length) {
PatcherCache.saveToCache(patchesMap); patcherCache.saveToCache(patchesMap);
} }
} }
@ -1140,51 +1236,65 @@ export class Patcher {
} }
export class PatcherCache { export class PatcherCache {
static #KEY_CACHE = 'better_xcloud_patches_cache'; private static instance: PatcherCache;
static #KEY_SIGNATURE = 'better_xcloud_patches_cache_signature'; public static getInstance = () => PatcherCache.instance ?? (PatcherCache.instance = new PatcherCache());
static #CACHE: any; private readonly KEY_CACHE = 'better_xcloud_patches_cache';
private readonly KEY_SIGNATURE = 'better_xcloud_patches_cache_signature';
static #isInitialized = false; private CACHE: any;
private isInitialized = false;
/** /**
* Get patch's signature * Get patch's signature
*/ */
static #getSignature(): number { private getSignature(): number {
const scriptVersion = SCRIPT_VERSION; const scriptVersion = SCRIPT_VERSION;
const webVersion = (document.querySelector('meta[name=gamepass-app-version]') as HTMLMetaElement)?.content;
const patches = JSON.stringify(ALL_PATCHES); const patches = JSON.stringify(ALL_PATCHES);
// Get client.js's hash
let webVersion = '';
const $link = document.querySelector<HTMLLinkElement>('link[data-chunk="client"][href*="/client."]');
if ($link) {
const match = /\/client\.([^\.]+)\.js/.exec($link.href);
match && (webVersion = match[1]);
} else {
// Get version from <meta>
// Sometimes this value is missing
webVersion = (document.querySelector<HTMLMetaElement>('meta[name=gamepass-app-version]'))?.content ?? '';
}
// Calculate signature // Calculate signature
const sig = hashCode(scriptVersion + webVersion + patches) const sig = hashCode(scriptVersion + webVersion + patches)
return sig; return sig;
} }
static clear() { clear() {
// Clear cache // Clear cache
window.localStorage.removeItem(PatcherCache.#KEY_CACHE); window.localStorage.removeItem(this.KEY_CACHE);
PatcherCache.#CACHE = {}; this.CACHE = {};
} }
static checkSignature() { private checkSignature() {
const storedSig = window.localStorage.getItem(PatcherCache.#KEY_SIGNATURE) || 0; const storedSig = window.localStorage.getItem(this.KEY_SIGNATURE) || 0;
const currentSig = PatcherCache.#getSignature(); const currentSig = this.getSignature();
if (currentSig !== parseInt(storedSig as string)) { if (currentSig !== parseInt(storedSig as string)) {
// Save new signature // Save new signature
BxLogger.warning(LOG_TAG, 'Signature changed'); BxLogger.warning(LOG_TAG, 'Signature changed');
window.localStorage.setItem(PatcherCache.#KEY_SIGNATURE, currentSig.toString()); window.localStorage.setItem(this.KEY_SIGNATURE, currentSig.toString());
PatcherCache.clear(); this.clear();
} else { } else {
BxLogger.info(LOG_TAG, 'Signature unchanged'); BxLogger.info(LOG_TAG, 'Signature unchanged');
} }
} }
static #cleanupPatches(patches: PatchArray): PatchArray { private cleanupPatches(patches: PatchArray): PatchArray {
return patches.filter(item => { return patches.filter(item => {
for (const id in PatcherCache.#CACHE) { for (const id in this.CACHE) {
const cached = PatcherCache.#CACHE[id]; const cached = this.CACHE[id];
if (cached.includes(item)) { if (cached.includes(item)) {
return false; return false;
@ -1195,17 +1305,17 @@ export class PatcherCache {
}); });
} }
static getPatches(id: string): PatchArray { getPatches(id: string): PatchArray {
return PatcherCache.#CACHE[id]; return this.CACHE[id];
} }
static saveToCache(subCache: Record<string, PatchArray>) { saveToCache(subCache: Record<string, PatchArray>) {
for (const id in subCache) { for (const id in subCache) {
const patchNames = subCache[id]; const patchNames = subCache[id];
let data = PatcherCache.#CACHE[id]; let data = this.CACHE[id];
if (!data) { if (!data) {
PatcherCache.#CACHE[id] = patchNames; this.CACHE[id] = patchNames;
} else { } else {
for (const patchName of patchNames) { for (const patchName of patchNames) {
if (!data.includes(patchName)) { if (!data.includes(patchName)) {
@ -1216,20 +1326,20 @@ export class PatcherCache {
} }
// Save to storage // Save to storage
window.localStorage.setItem(PatcherCache.#KEY_CACHE, JSON.stringify(PatcherCache.#CACHE)); window.localStorage.setItem(this.KEY_CACHE, JSON.stringify(this.CACHE));
} }
static init() { init() {
if (PatcherCache.#isInitialized) { if (this.isInitialized) {
return; return;
} }
PatcherCache.#isInitialized = true; this.isInitialized = true;
PatcherCache.checkSignature(); this.checkSignature();
// Read cache from storage // Read cache from storage
PatcherCache.#CACHE = JSON.parse(window.localStorage.getItem(PatcherCache.#KEY_CACHE) || '{}'); this.CACHE = JSON.parse(window.localStorage.getItem(this.KEY_CACHE) || '{}');
BxLogger.info(LOG_TAG, PatcherCache.#CACHE); BxLogger.info(LOG_TAG, this.CACHE);
if (window.location.pathname.includes('/play/')) { if (window.location.pathname.includes('/play/')) {
PATCH_ORDERS.push(...PLAYING_PATCH_ORDERS); PATCH_ORDERS.push(...PLAYING_PATCH_ORDERS);
@ -1238,8 +1348,8 @@ export class PatcherCache {
} }
// Remove cached patches from PATCH_ORDERS & PLAYING_PATCH_ORDERS // Remove cached patches from PATCH_ORDERS & PLAYING_PATCH_ORDERS
PATCH_ORDERS = PatcherCache.#cleanupPatches(PATCH_ORDERS); PATCH_ORDERS = this.cleanupPatches(PATCH_ORDERS);
PLAYING_PATCH_ORDERS = PatcherCache.#cleanupPatches(PLAYING_PATCH_ORDERS); PLAYING_PATCH_ORDERS = this.cleanupPatches(PLAYING_PATCH_ORDERS);
BxLogger.info(LOG_TAG, PATCH_ORDERS.slice(0)); BxLogger.info(LOG_TAG, PATCH_ORDERS.slice(0));
BxLogger.info(LOG_TAG, PLAYING_PATCH_ORDERS.slice(0)); BxLogger.info(LOG_TAG, PLAYING_PATCH_ORDERS.slice(0));

View File

@ -85,7 +85,7 @@ if (btnHome) {
this.inputSink.onGamepadInput(performance.now() - intervalMs, fakeGamepadMappings); this.inputSink.onGamepadInput(performance.now() - intervalMs, fakeGamepadMappings);
} else { } else {
intervalMs = 4; intervalMs = window.BX_CONTROLLER_POLLING_RATE;
} }
} }

View File

@ -1,121 +1,100 @@
const int FILTER_UNSHARP_MASKING = 1; #version 300 es
const int FILTER_CAS = 2;
precision highp float; precision mediump float;
uniform sampler2D data; uniform sampler2D data;
uniform vec2 iResolution; uniform vec2 iResolution;
const int FILTER_UNSHARP_MASKING = 1;
// const int FILTER_CAS = 2;
// constrast = 0.8
const float CAS_CONTRAST_PEAK = 0.8 * -3.0 + 8.0;
// Luminosity factor
const vec3 LUMINOSITY_FACTOR = vec3(0.2126, 0.7152, 0.0722);
uniform int filterId; uniform int filterId;
uniform float sharpenFactor; uniform float sharpenFactor;
uniform float brightness; uniform float brightness;
uniform float contrast; uniform float contrast;
uniform float saturation; uniform float saturation;
vec3 textureAt(sampler2D tex, vec2 coord) { out vec4 fragColor;
return texture2D(tex, coord / iResolution.xy).rgb;
} vec3 clarityBoost(sampler2D tex, vec2 coord, vec3 e) {
vec2 texelSize = 1.0 / iResolution.xy;
vec3 clarityBoost(sampler2D tex, vec2 coord)
{
// Load a collection of samples in a 3x3 neighorhood, where e is the current pixel. // Load a collection of samples in a 3x3 neighorhood, where e is the current pixel.
// a b c // a b c
// d e f // d e f
// g h i // g h i
vec3 a = textureAt(tex, coord + vec2(-1, 1)); vec3 a = texture(tex, coord + texelSize * vec2(-1, 1)).rgb;
vec3 b = textureAt(tex, coord + vec2(0, 1)); vec3 b = texture(tex, coord + texelSize * vec2(0, 1)).rgb;
vec3 c = textureAt(tex, coord + vec2(1, 1)); vec3 c = texture(tex, coord + texelSize * vec2(1, 1)).rgb;
vec3 d = textureAt(tex, coord + vec2(-1, 0)); vec3 d = texture(tex, coord + texelSize * vec2(-1, 0)).rgb;
vec3 e = textureAt(tex, coord); vec3 f = texture(tex, coord + texelSize * vec2(1, 0)).rgb;
vec3 f = textureAt(tex, coord + vec2(1, 0));
vec3 g = textureAt(tex, coord + vec2(-1, -1)); vec3 g = texture(tex, coord + texelSize * vec2(-1, -1)).rgb;
vec3 h = textureAt(tex, coord + vec2(0, -1)); vec3 h = texture(tex, coord + texelSize * vec2(0, -1)).rgb;
vec3 i = textureAt(tex, coord + vec2(1, -1)); vec3 i = texture(tex, coord + texelSize * vec2(1, -1)).rgb;
if (filterId == FILTER_CAS) { // USM
// Soft min and max. if (filterId == FILTER_UNSHARP_MASKING) {
// a b c b vec3 gaussianBlur = (a + c + g + i) * 1.0 + (b + d + f + h) * 2.0 + e * 4.0;
// d e f * 0.5 + d e f * 0.5 gaussianBlur /= 16.0;
// g h i h
// These are 2.0x bigger (factored out the extra multiply).
vec3 minRgb = min(min(min(d, e), min(f, b)), h);
vec3 minRgb2 = min(min(a, c), min(g, i));
minRgb += min(minRgb, minRgb2);
vec3 maxRgb = max(max(max(d, e), max(f, b)), h);
vec3 maxRgb2 = max(max(a, c), max(g, i));
maxRgb += max(maxRgb, maxRgb2);
// Smooth minimum distance to signal limit divided by smooth max.
vec3 reciprocalMaxRgb = 1.0 / maxRgb;
vec3 amplifyRgb = clamp(min(minRgb, 2.0 - maxRgb) * reciprocalMaxRgb, 0.0, 1.0);
// Shaping amount of sharpening.
amplifyRgb = inversesqrt(amplifyRgb);
float contrast = 0.8;
float peak = -3.0 * contrast + 8.0;
vec3 weightRgb = -(1.0 / (amplifyRgb * peak));
vec3 reciprocalWeightRgb = 1.0 / (4.0 * weightRgb + 1.0);
// 0 w 0
// Filter shape: w 1 w
// 0 w 0
vec3 window = (b + d) + (f + h);
vec3 outColor = clamp((window * weightRgb + e) * reciprocalWeightRgb, 0.0, 1.0);
outColor = mix(e, outColor, sharpenFactor / 2.0);
return outColor;
} else if (filterId == FILTER_UNSHARP_MASKING) {
vec3 gaussianBlur = (a * 1.0 + b * 2.0 + c * 1.0 +
d * 2.0 + e * 4.0 + f * 2.0 +
g * 1.0 + h * 2.0 + i * 1.0) / 16.0;
// Return edge detection // Return edge detection
return e + (e - gaussianBlur) * sharpenFactor / 3.0; return e + (e - gaussianBlur) * sharpenFactor / 3.0;
} }
return e; // CAS
} // Soft min and max.
// a b c b
// d e f * 0.5 + d e f * 0.5
// g h i h
// These are 2.0x bigger (factored out the extra multiply).
vec3 minRgb = min(min(min(d, e), min(f, b)), h);
minRgb += min(min(a, c), min(g, i));
vec3 adjustBrightness(vec3 color) { vec3 maxRgb = max(max(max(d, e), max(f, b)), h);
return (1.0 + brightness) * color; maxRgb += max(max(a, c), max(g, i));
}
vec3 adjustContrast(vec3 color) { // Smooth minimum distance to signal limit divided by smooth max.
return 0.5 + (1.0 + contrast) * (color - 0.5); vec3 reciprocalMaxRgb = 1.0 / maxRgb;
} vec3 amplifyRgb = clamp(min(minRgb, 2.0 - maxRgb) * reciprocalMaxRgb, 0.0, 1.0);
vec3 adjustSaturation(vec3 color) { // Shaping amount of sharpening.
const vec3 luminosityFactor = vec3(0.2126, 0.7152, 0.0722); amplifyRgb = inversesqrt(amplifyRgb);
vec3 grayscale = vec3(dot(color, luminosityFactor));
return mix(grayscale, color, 1.0 + saturation); vec3 weightRgb = -(1.0 / (amplifyRgb * CAS_CONTRAST_PEAK));
vec3 reciprocalWeightRgb = 1.0 / (4.0 * weightRgb + 1.0);
// 0 w 0
// Filter shape: w 1 w
// 0 w 0
vec3 window = b + d + f + h;
vec3 outColor = clamp((window * weightRgb + e) * reciprocalWeightRgb, 0.0, 1.0);
return mix(e, outColor, sharpenFactor / 2.0);
} }
void main() { void main() {
vec3 color; vec2 uv = gl_FragCoord.xy / iResolution.xy;
// Get current pixel
vec3 color = texture(data, uv).rgb;
if (sharpenFactor > 0.0) { // Clarity boost
color = clarityBoost(data, gl_FragCoord.xy); color = sharpenFactor > 0.0 ? clarityBoost(data, uv, color) : color;
} else {
color = textureAt(data, gl_FragCoord.xy);
}
if (saturation != 0.0) { // Saturation
color = adjustSaturation(color); color = saturation != 1.0 ? mix(vec3(dot(color, LUMINOSITY_FACTOR)), color, saturation) : color;
}
if (contrast != 0.0) { // Contrast
color = adjustContrast(color); color = contrast * (color - 0.5) + 0.5;
}
if (brightness != 0.0) { // Brightness
color = adjustBrightness(color); color = brightness * color;
}
gl_FragColor = vec4(color, 1.0); fragColor = vec4(color, 1.0);
} }

View File

@ -1,5 +1,7 @@
attribute vec2 position; #version 300 es
in vec4 position;
void main() { void main() {
gl_Position = vec4(position, 0, 1); gl_Position = position;
} }

View File

@ -5,19 +5,19 @@ import { PrefKey } from "@/enums/pref-keys";
import { getPref } from "@/utils/settings-storages/global-settings-storage"; import { getPref } from "@/utils/settings-storages/global-settings-storage";
const LOG_TAG = 'WebGL2Player';
export class WebGL2Player { export class WebGL2Player {
#$video: HTMLVideoElement; private readonly LOG_TAG = 'WebGL2Player';
#$canvas: HTMLCanvasElement;
#gl: WebGL2RenderingContext | null = null; private $video: HTMLVideoElement;
#resources: Array<any> = []; private $canvas: HTMLCanvasElement;
#program: WebGLProgram | null = null;
#stopped: boolean = false; private gl: WebGL2RenderingContext | null = null;
private resources: Array<any> = [];
private program: WebGLProgram | null = null;
#options = { private stopped: boolean = false;
private options = {
filterId: 1, filterId: 1,
sharpenFactor: 0, sharpenFactor: 0,
brightness: 0.0, brightness: 0.0,
@ -25,112 +25,133 @@ export class WebGL2Player {
saturation: 0.0, saturation: 0.0,
}; };
#animFrameId: number | null = null; private targetFps = 60;
private frameInterval = 0;
private lastFrameTime = 0;
private animFrameId: number | null = null;
constructor($video: HTMLVideoElement) { constructor($video: HTMLVideoElement) {
BxLogger.info(LOG_TAG, 'Initialize'); BxLogger.info(this.LOG_TAG, 'Initialize');
this.#$video = $video; this.$video = $video;
const $canvas = document.createElement('canvas'); const $canvas = document.createElement('canvas');
$canvas.width = $video.videoWidth; $canvas.width = $video.videoWidth;
$canvas.height = $video.videoHeight; $canvas.height = $video.videoHeight;
this.#$canvas = $canvas; this.$canvas = $canvas;
this.#setupShaders(); this.setupShaders();
this.#setupRendering(); this.setupRendering();
$video.insertAdjacentElement('afterend', $canvas); $video.insertAdjacentElement('afterend', $canvas);
} }
setFilter(filterId: number, update = true) { setFilter(filterId: number, update = true) {
this.#options.filterId = filterId; this.options.filterId = filterId;
update && this.updateCanvas(); update && this.updateCanvas();
} }
setSharpness(sharpness: number, update = true) { setSharpness(sharpness: number, update = true) {
this.#options.sharpenFactor = sharpness; this.options.sharpenFactor = sharpness;
update && this.updateCanvas(); update && this.updateCanvas();
} }
setBrightness(brightness: number, update = true) { setBrightness(brightness: number, update = true) {
this.#options.brightness = (brightness - 100) / 100; this.options.brightness = 1 + (brightness - 100) / 100;
update && this.updateCanvas(); update && this.updateCanvas();
} }
setContrast(contrast: number, update = true) { setContrast(contrast: number, update = true) {
this.#options.contrast = (contrast - 100) / 100; this.options.contrast = 1 + (contrast - 100) / 100;
update && this.updateCanvas(); update && this.updateCanvas();
} }
setSaturation(saturation: number, update = true) { setSaturation(saturation: number, update = true) {
this.#options.saturation = (saturation - 100) / 100; this.options.saturation = 1 + (saturation - 100) / 100;
update && this.updateCanvas(); update && this.updateCanvas();
} }
setTargetFps(target: number) {
this.targetFps = target;
this.lastFrameTime = 0;
this.frameInterval = target ? Math.floor(1000 / target) : 0;
}
getCanvas() { getCanvas() {
return this.#$canvas; return this.$canvas;
} }
updateCanvas() { updateCanvas() {
const gl = this.#gl!; const gl = this.gl!;
const program = this.#program!; const program = this.program!;
gl.uniform2f(gl.getUniformLocation(program, 'iResolution'), this.#$canvas.width, this.#$canvas.height); gl.uniform2f(gl.getUniformLocation(program, 'iResolution'), this.$canvas.width, this.$canvas.height);
gl.uniform1i(gl.getUniformLocation(program, 'filterId'), this.#options.filterId); gl.uniform1i(gl.getUniformLocation(program, 'filterId'), this.options.filterId);
gl.uniform1f(gl.getUniformLocation(program, 'sharpenFactor'), this.#options.sharpenFactor); gl.uniform1f(gl.getUniformLocation(program, 'sharpenFactor'), this.options.sharpenFactor);
gl.uniform1f(gl.getUniformLocation(program, 'brightness'), this.#options.brightness); gl.uniform1f(gl.getUniformLocation(program, 'brightness'), this.options.brightness);
gl.uniform1f(gl.getUniformLocation(program, 'contrast'), this.#options.contrast); gl.uniform1f(gl.getUniformLocation(program, 'contrast'), this.options.contrast);
gl.uniform1f(gl.getUniformLocation(program, 'saturation'), this.#options.saturation); gl.uniform1f(gl.getUniformLocation(program, 'saturation'), this.options.saturation);
} }
drawFrame() { forceDrawFrame() {
const gl = this.#gl!; const gl = this.gl!;
const $video = this.#$video; gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGB, gl.RGB, gl.UNSIGNED_BYTE, this.$video);
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGB, gl.RGB, gl.UNSIGNED_BYTE, $video);
gl.drawArrays(gl.TRIANGLES, 0, 6); gl.drawArrays(gl.TRIANGLES, 0, 6);
} }
#setupRendering() { private setupRendering() {
let animate: any; let frameCallback: any;
if ('requestVideoFrameCallback' in HTMLVideoElement.prototype) { if ('requestVideoFrameCallback' in HTMLVideoElement.prototype) {
const $video = this.#$video; const $video = this.$video;
animate = () => { frameCallback = $video.requestVideoFrameCallback.bind($video);
if (this.#stopped) {
return;
}
this.drawFrame();
this.#animFrameId = $video.requestVideoFrameCallback(animate);
}
this.#animFrameId = $video.requestVideoFrameCallback(animate);
} else { } else {
animate = () => { frameCallback = requestAnimationFrame;
if (this.#stopped) { }
return;
}
this.drawFrame(); let animate = () => {
this.#animFrameId = requestAnimationFrame(animate); if (this.stopped) {
return;
} }
this.#animFrameId = requestAnimationFrame(animate); let draw = true;
// Don't draw when FPS is 0
if (this.targetFps === 0) {
draw = false;
} else if (this.targetFps < 60) {
// Limit FPS
const currentTime = performance.now();
const timeSinceLastFrame = currentTime - this.lastFrameTime;
if (timeSinceLastFrame < this.frameInterval) {
draw = false;
} else {
this.lastFrameTime = currentTime;
}
}
if (draw) {
const gl = this.gl!;
gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGB, gl.RGB, gl.UNSIGNED_BYTE, this.$video);
gl.drawArrays(gl.TRIANGLES, 0, 6);
}
this.animFrameId = frameCallback(animate);
} }
this.animFrameId = frameCallback(animate);
} }
#setupShaders() { private setupShaders() {
BxLogger.info(LOG_TAG, 'Setting up', getPref(PrefKey.VIDEO_POWER_PREFERENCE)); BxLogger.info(this.LOG_TAG, 'Setting up', getPref(PrefKey.VIDEO_POWER_PREFERENCE));
const gl = this.#$canvas.getContext('webgl', { const gl = this.$canvas.getContext('webgl2', {
isBx: true, isBx: true,
antialias: true, antialias: true,
alpha: false, alpha: false,
powerPreference: getPref(PrefKey.VIDEO_POWER_PREFERENCE), powerPreference: getPref(PrefKey.VIDEO_POWER_PREFERENCE),
}) as WebGL2RenderingContext; }) as WebGL2RenderingContext;
this.#gl = gl; this.gl = gl;
gl.viewport(0, 0, gl.drawingBufferWidth, gl.drawingBufferWidth); gl.viewport(0, 0, gl.drawingBufferWidth, gl.drawingBufferWidth);
@ -145,7 +166,7 @@ export class WebGL2Player {
// Create and link program // Create and link program
const program = gl.createProgram()!; const program = gl.createProgram()!;
this.#program = program; this.program = program;
gl.attachShader(program, vShader); gl.attachShader(program, vShader);
gl.attachShader(program, fShader); gl.attachShader(program, fShader);
@ -162,24 +183,17 @@ export class WebGL2Player {
// Vertices: A screen-filling quad made from two triangles // Vertices: A screen-filling quad made from two triangles
const buffer = gl.createBuffer(); const buffer = gl.createBuffer();
this.#resources.push(buffer); this.resources.push(buffer);
gl.bindBuffer(gl.ARRAY_BUFFER, buffer); gl.bindBuffer(gl.ARRAY_BUFFER, buffer);
gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([ gl.bufferData(gl.ARRAY_BUFFER, new Float32Array([-1, -1, 1, -1, -1, 1, -1, 1, 1, -1, 1, 1]), gl.STATIC_DRAW);
-1, -1,
1, -1,
-1, 1,
-1, 1,
1, -1,
1, 1,
]), gl.STATIC_DRAW);
gl.enableVertexAttribArray(0); gl.enableVertexAttribArray(0);
gl.vertexAttribPointer(0, 2, gl.FLOAT, false, 0, 0); gl.vertexAttribPointer(0, 2, gl.FLOAT, false, 0, 0);
// Texture to contain the video data // Texture to contain the video data
const texture = gl.createTexture(); const texture = gl.createTexture();
this.#resources.push(texture); this.resources.push(texture);
gl.bindTexture(gl.TEXTURE_2D, texture); gl.bindTexture(gl.TEXTURE_2D, texture);
gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, true); gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, true);
@ -197,40 +211,40 @@ export class WebGL2Player {
resume() { resume() {
this.stop(); this.stop();
this.#stopped = false; this.stopped = false;
BxLogger.info(LOG_TAG, 'Resume'); BxLogger.info(this.LOG_TAG, 'Resume');
this.#$canvas.classList.remove('bx-gone'); this.$canvas.classList.remove('bx-gone');
this.#setupRendering(); this.setupRendering();
} }
stop() { stop() {
BxLogger.info(LOG_TAG, 'Stop'); BxLogger.info(this.LOG_TAG, 'Stop');
this.#$canvas.classList.add('bx-gone'); this.$canvas.classList.add('bx-gone');
this.#stopped = true; this.stopped = true;
if (this.#animFrameId) { if (this.animFrameId) {
if ('requestVideoFrameCallback' in HTMLVideoElement.prototype) { if ('requestVideoFrameCallback' in HTMLVideoElement.prototype) {
this.#$video.cancelVideoFrameCallback(this.#animFrameId); this.$video.cancelVideoFrameCallback(this.animFrameId);
} else { } else {
cancelAnimationFrame(this.#animFrameId); cancelAnimationFrame(this.animFrameId);
} }
this.#animFrameId = null; this.animFrameId = null;
} }
} }
destroy() { destroy() {
BxLogger.info(LOG_TAG, 'Destroy'); BxLogger.info(this.LOG_TAG, 'Destroy');
this.stop(); this.stop();
const gl = this.#gl; const gl = this.gl;
if (gl) { if (gl) {
gl.getExtension('WEBGL_lose_context')?.loseContext(); gl.getExtension('WEBGL_lose_context')?.loseContext();
gl.useProgram(null);
for (const resource of this.#resources) { for (const resource of this.resources) {
if (resource instanceof WebGLProgram) { if (resource instanceof WebGLProgram) {
gl.useProgram(null);
gl.deleteProgram(resource); gl.deleteProgram(resource);
} else if (resource instanceof WebGLShader) { } else if (resource instanceof WebGLShader) {
gl.deleteShader(resource); gl.deleteShader(resource);
@ -241,14 +255,14 @@ export class WebGL2Player {
} }
} }
this.#gl = null; this.gl = null;
} }
if (this.#$canvas.isConnected) { if (this.$canvas.isConnected) {
this.#$canvas.parentElement?.removeChild(this.#$canvas); this.$canvas.parentElement?.removeChild(this.$canvas);
} }
this.#$canvas.width = 1; this.$canvas.width = 1;
this.#$canvas.height = 1; this.$canvas.height = 1;
} }
} }

View File

@ -0,0 +1,229 @@
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";
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.instance ?? (RemotePlayManager.instance = new RemotePlayManager());
private readonly LOG_TAG = 'RemotePlayManager';
private isInitialized = false;
private XCLOUD_TOKEN!: string;
private XHOME_TOKEN!: string;
private consoles!: Array<RemotePlayConsole>;
private regions: Array<RemotePlayRegion> = [];
private constructor() {
BxLogger.info(this.LOG_TAG, 'constructor()');
}
initialize() {
if (this.isInitialized) {
return;
}
this.isInitialized = true;
this.getXhomeToken(() => {
this.getConsolesList(() => {
BxLogger.info(this.LOG_TAG, 'Consoles', this.consoles);
STATES.supportedRegion && HeaderSection.getInstance().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;
}
}

View File

@ -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;
}
}

View File

@ -0,0 +1,18 @@
import { PrefKey } from "@/enums/pref-keys";
import { getPref } from "@/utils/settings-storages/global-settings-storage";
import { limitVideoPlayerFps } from "../stream/stream-settings-utils";
export class RendererShortcut {
static toggleVisibility(): boolean {
const $mediaContainer = document.querySelector('#game-stream div[data-testid="media-container"]');
if (!$mediaContainer) {
return true;
}
$mediaContainer.classList.toggle('bx-gone');
const isShowing = !$mediaContainer.classList.contains('bx-gone');
// Switch FPS
limitVideoPlayerFps(isShowing ? getPref(PrefKey.VIDEO_MAX_FPS) : 0);
return isShowing;
}
}

View File

@ -4,6 +4,12 @@ import { Toast } from "@utils/toast";
import { ceilToNearest, floorToNearest } from "@/utils/utils"; import { ceilToNearest, floorToNearest } from "@/utils/utils";
import { PrefKey } from "@/enums/pref-keys"; import { PrefKey } from "@/enums/pref-keys";
import { getPref, setPref } from "@/utils/settings-storages/global-settings-storage"; import { getPref, setPref } from "@/utils/settings-storages/global-settings-storage";
import { BxEvent } from "@/utils/bx-event";
export enum SpeakerState {
ENABLED,
MUTED,
}
export class SoundShortcut { export class SoundShortcut {
static adjustGainNodeVolume(amount: number): number { static adjustGainNodeVolume(amount: number): number {
@ -64,21 +70,23 @@ export class SoundShortcut {
SoundShortcut.setGainNodeVolume(targetValue); SoundShortcut.setGainNodeVolume(targetValue);
Toast.show(`${t('stream')} ${t('volume')}`, status, {instant: true}); Toast.show(`${t('stream')} ${t('volume')}`, status, {instant: true});
BxEvent.dispatch(window, BxEvent.SPEAKER_STATE_CHANGED, {
speakerState: targetValue === 0 ? SpeakerState.MUTED : SpeakerState.ENABLED,
})
return; return;
} }
let $media: HTMLMediaElement; const $media = document.querySelector<HTMLAudioElement>('div[data-testid=media-container] audio') ?? document.querySelector<HTMLAudioElement>('div[data-testid=media-container] video');
$media = document.querySelector('div[data-testid=media-container] audio') as HTMLAudioElement;
if (!$media) {
$media = document.querySelector('div[data-testid=media-container] video') as HTMLAudioElement;
}
if ($media) { if ($media) {
$media.muted = !$media.muted; $media.muted = !$media.muted;
const status = $media.muted ? t('muted') : t('unmuted'); const status = $media.muted ? t('muted') : t('unmuted');
Toast.show(`${t('stream')} ${t('volume')}`, status, {instant: true}); Toast.show(`${t('stream')} ${t('volume')}`, status, {instant: true});
BxEvent.dispatch(window, BxEvent.SPEAKER_STATE_CHANGED, {
speakerState: $media.muted ? SpeakerState.MUTED : SpeakerState.ENABLED,
})
} }
} }
} }

View File

@ -1,10 +1,13 @@
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 { ScreenshotManager } from "@/utils/screenshot-manager";
import { StreamPlayerType, StreamVideoProcessing } from "@enums/stream-player"; import { StreamPlayerType, StreamVideoProcessing } from "@enums/stream-player";
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 { BX_FLAGS } from "@/utils/bx-flags";
export type StreamPlayerOptions = Partial<{ export type StreamPlayerOptions = Partial<{
processing: string, processing: string,
@ -15,35 +18,35 @@ export type StreamPlayerOptions = Partial<{
}>; }>;
export class StreamPlayer { export class StreamPlayer {
#$video: HTMLVideoElement; private $video: HTMLVideoElement;
#playerType: StreamPlayerType = StreamPlayerType.VIDEO; private playerType: StreamPlayerType = StreamPlayerType.VIDEO;
#options: StreamPlayerOptions = {}; private options: StreamPlayerOptions = {};
#webGL2Player: WebGL2Player | null = null; private webGL2Player: WebGL2Player | null = null;
#$videoCss: HTMLStyleElement | null = null; private $videoCss: HTMLStyleElement | null = null;
#$usmMatrix: SVGFEConvolveMatrixElement | null = null; private $usmMatrix: SVGFEConvolveMatrixElement | null = null;
constructor($video: HTMLVideoElement, type: StreamPlayerType, options: StreamPlayerOptions) { constructor($video: HTMLVideoElement, type: StreamPlayerType, options: StreamPlayerOptions) {
this.#setupVideoElements(); this.setupVideoElements();
this.#$video = $video; this.$video = $video;
this.#options = options || {}; this.options = options || {};
this.setPlayerType(type); this.setPlayerType(type);
} }
#setupVideoElements() { private setupVideoElements() {
this.#$videoCss = document.getElementById('bx-video-css') as HTMLStyleElement; this.$videoCss = document.getElementById('bx-video-css') as HTMLStyleElement;
if (this.#$videoCss) { if (this.$videoCss) {
this.#$usmMatrix = this.#$videoCss.querySelector('#bx-filter-usm-matrix') as any; this.$usmMatrix = this.$videoCss.querySelector('#bx-filter-usm-matrix') as any;
return; return;
} }
const $fragment = document.createDocumentFragment(); const $fragment = document.createDocumentFragment();
this.#$videoCss = CE<HTMLStyleElement>('style', {id: 'bx-video-css'}); this.$videoCss = CE<HTMLStyleElement>('style', {id: 'bx-video-css'});
$fragment.appendChild(this.#$videoCss); $fragment.appendChild(this.$videoCss);
// Setup SVG filters // Setup SVG filters
const $svg = CE('svg', { const $svg = CE('svg', {
@ -54,7 +57,7 @@ export class StreamPlayer {
CE('filter', { CE('filter', {
id: 'bx-filter-usm', id: 'bx-filter-usm',
xmlns: 'http://www.w3.org/2000/svg', xmlns: 'http://www.w3.org/2000/svg',
}, this.#$usmMatrix = CE('feConvolveMatrix', { }, this.$usmMatrix = CE('feConvolveMatrix', {
id: 'bx-filter-usm-matrix', id: 'bx-filter-usm-matrix',
order: '3', order: '3',
xmlns: 'http://www.w3.org/2000/svg', xmlns: 'http://www.w3.org/2000/svg',
@ -65,29 +68,29 @@ export class StreamPlayer {
document.documentElement.appendChild($fragment); document.documentElement.appendChild($fragment);
} }
#getVideoPlayerFilterStyle() { private getVideoPlayerFilterStyle() {
const filters = []; const filters = [];
const sharpness = this.#options.sharpness || 0; const sharpness = this.options.sharpness || 0;
if (this.#options.processing === StreamVideoProcessing.USM && sharpness != 0) { if (this.options.processing === StreamVideoProcessing.USM && sharpness != 0) {
const level = (7 - ((sharpness / 2) - 1) * 0.5).toFixed(1); // 5, 5.5, 6, 6.5, 7 const level = (7 - ((sharpness / 2) - 1) * 0.5).toFixed(1); // 5, 5.5, 6, 6.5, 7
const matrix = `0 -1 0 -1 ${level} -1 0 -1 0`; const matrix = `0 -1 0 -1 ${level} -1 0 -1 0`;
this.#$usmMatrix?.setAttributeNS(null, 'kernelMatrix', matrix); this.$usmMatrix?.setAttributeNS(null, 'kernelMatrix', matrix);
filters.push(`url(#bx-filter-usm)`); filters.push(`url(#bx-filter-usm)`);
} }
const saturation = this.#options.saturation || 100; const saturation = this.options.saturation || 100;
if (saturation != 100) { if (saturation != 100) {
filters.push(`saturate(${saturation}%)`); filters.push(`saturate(${saturation}%)`);
} }
const contrast = this.#options.contrast || 100; const contrast = this.options.contrast || 100;
if (contrast != 100) { if (contrast != 100) {
filters.push(`contrast(${contrast}%)`); filters.push(`contrast(${contrast}%)`);
} }
const brightness = this.#options.brightness || 100; const brightness = this.options.brightness || 100;
if (brightness != 100) { if (brightness != 100) {
filters.push(`brightness(${brightness}%)`); filters.push(`brightness(${brightness}%)`);
} }
@ -95,14 +98,14 @@ export class StreamPlayer {
return filters.join(' '); return filters.join(' ');
} }
#resizePlayer() { private resizePlayer() {
const PREF_RATIO = getPref(PrefKey.VIDEO_RATIO); const PREF_RATIO = getPref(PrefKey.VIDEO_RATIO);
const $video = this.#$video; const $video = this.$video;
const isNativeTouchGame = STATES.currentStream.titleInfo?.details.hasNativeTouchSupport; const isNativeTouchGame = STATES.currentStream.titleInfo?.details.hasNativeTouchSupport;
let $webGL2Canvas; let $webGL2Canvas;
if (this.#playerType == StreamPlayerType.WEBGL2) { if (this.playerType == StreamPlayerType.WEBGL2) {
$webGL2Canvas = this.#webGL2Player?.getCanvas()!; $webGL2Canvas = this.webGL2Player?.getCanvas()!;
} }
let targetWidth; let targetWidth;
@ -164,67 +167,69 @@ export class StreamPlayer {
} }
// Update video dimensions // Update video dimensions
if (isNativeTouchGame && this.#playerType == StreamPlayerType.WEBGL2) { if (isNativeTouchGame && this.playerType == StreamPlayerType.WEBGL2) {
window.BX_EXPOSED.streamSession.updateDimensions(); window.BX_EXPOSED.streamSession.updateDimensions();
} }
} }
setPlayerType(type: StreamPlayerType, refreshPlayer: boolean = false) { setPlayerType(type: StreamPlayerType, refreshPlayer: boolean = false) {
if (this.#playerType !== type) { if (this.playerType !== type) {
const videoClass = BX_FLAGS.DeviceInfo.deviceType === 'android-tv' ? 'bx-pixel' : 'bx-gone';
// Switch from Video -> WebGL2 // Switch from Video -> WebGL2
if (type === StreamPlayerType.WEBGL2) { if (type === StreamPlayerType.WEBGL2) {
// Initialize WebGL2 player // Initialize WebGL2 player
if (!this.#webGL2Player) { if (!this.webGL2Player) {
this.#webGL2Player = new WebGL2Player(this.#$video); this.webGL2Player = new WebGL2Player(this.$video);
} else { } else {
this.#webGL2Player.resume(); this.webGL2Player.resume();
} }
this.#$videoCss!.textContent = ''; this.$videoCss!.textContent = '';
this.#$video.classList.add('bx-pixel'); this.$video.classList.add(videoClass);
} else { } else {
// Cleanup WebGL2 Player // Cleanup WebGL2 Player
this.#webGL2Player?.stop(); this.webGL2Player?.stop();
this.#$video.classList.remove('bx-pixel'); this.$video.classList.remove(videoClass);
} }
} }
this.#playerType = type; this.playerType = type;
refreshPlayer && this.refreshPlayer(); refreshPlayer && this.refreshPlayer();
} }
setOptions(options: StreamPlayerOptions, refreshPlayer: boolean = false) { setOptions(options: StreamPlayerOptions, refreshPlayer: boolean = false) {
this.#options = options; this.options = options;
refreshPlayer && this.refreshPlayer(); refreshPlayer && this.refreshPlayer();
} }
updateOptions(options: StreamPlayerOptions, refreshPlayer: boolean = false) { updateOptions(options: StreamPlayerOptions, refreshPlayer: boolean = false) {
this.#options = Object.assign(this.#options, options); this.options = Object.assign(this.options, options);
refreshPlayer && this.refreshPlayer(); refreshPlayer && this.refreshPlayer();
} }
getPlayerElement(playerType?: StreamPlayerType) { getPlayerElement(playerType?: StreamPlayerType) {
if (typeof playerType === 'undefined') { if (typeof playerType === 'undefined') {
playerType = this.#playerType; playerType = this.playerType;
} }
if (playerType === StreamPlayerType.WEBGL2) { if (playerType === StreamPlayerType.WEBGL2) {
return this.#webGL2Player?.getCanvas(); return this.webGL2Player?.getCanvas();
} }
return this.#$video; return this.$video;
} }
getWebGL2Player() { getWebGL2Player() {
return this.#webGL2Player; return this.webGL2Player;
} }
refreshPlayer() { refreshPlayer() {
if (this.#playerType === StreamPlayerType.WEBGL2) { if (this.playerType === StreamPlayerType.WEBGL2) {
const options = this.#options; const options = this.options;
const webGL2Player = this.#webGL2Player!; const webGL2Player = this.webGL2Player!;
if (options.processing === StreamVideoProcessing.USM) { if (options.processing === StreamVideoProcessing.USM) {
webGL2Player.setFilter(1); webGL2Player.setFilter(1);
@ -232,22 +237,22 @@ export class StreamPlayer {
webGL2Player.setFilter(2); webGL2Player.setFilter(2);
} }
Screenshot.updateCanvasFilters('none'); isFullVersion() && ScreenshotManager.getInstance().updateCanvasFilters('none');
webGL2Player.setSharpness(options.sharpness || 0); webGL2Player.setSharpness(options.sharpness || 0);
webGL2Player.setSaturation(options.saturation || 100); webGL2Player.setSaturation(options.saturation || 100);
webGL2Player.setContrast(options.contrast || 100); webGL2Player.setContrast(options.contrast || 100);
webGL2Player.setBrightness(options.brightness || 100); webGL2Player.setBrightness(options.brightness || 100);
} else { } else {
let filters = this.#getVideoPlayerFilterStyle(); let filters = this.getVideoPlayerFilterStyle();
let videoCss = ''; let videoCss = '';
if (filters) { if (filters) {
videoCss += `filter: ${filters} !important;`; videoCss += `filter: ${filters} !important;`;
} }
// 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); ScreenshotManager.getInstance().updateCanvasFilters(filters);
} }
let css = ''; let css = '';
@ -255,26 +260,26 @@ export class StreamPlayer {
css = `#game-stream video { ${videoCss} }`; css = `#game-stream video { ${videoCss} }`;
} }
this.#$videoCss!.textContent = css; this.$videoCss!.textContent = css;
} }
this.#resizePlayer(); this.resizePlayer();
} }
reloadPlayer() { reloadPlayer() {
this.#cleanUpWebGL2Player(); this.cleanUpWebGL2Player();
this.#playerType = StreamPlayerType.VIDEO; this.playerType = StreamPlayerType.VIDEO;
this.setPlayerType(StreamPlayerType.WEBGL2, false); this.setPlayerType(StreamPlayerType.WEBGL2, false);
} }
#cleanUpWebGL2Player() { private cleanUpWebGL2Player() {
// Clean up WebGL2 Player // Clean up WebGL2 Player
this.#webGL2Player?.destroy(); this.webGL2Player?.destroy();
this.#webGL2Player = null; this.webGL2Player = null;
} }
destroy() { destroy() {
this.#cleanUpWebGL2Player(); this.cleanUpWebGL2Player();
} }
} }

View File

@ -1,126 +1,154 @@
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?: {
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;
public static getInstance(): StreamBadges { public static getInstance = () => StreamBadges.instance ?? (StreamBadges.instance = new StreamBadges());
if (!StreamBadges.instance) { private readonly LOG_TAG = 'StreamBadges';
StreamBadges.instance = new StreamBadges();
}
return StreamBadges.instance; private serverInfo: StreamServerInfo = {};
private badges: Record<StreamBadge, StreamBadgeInfo> = {
[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',
},
};
private $container: HTMLElement | undefined;
private intervalId?: number | null;
private readonly REFRESH_INTERVAL = 3 * 1000;
private constructor() {
BxLogger.info(this.LOG_TAG, 'constructor()');
} }
#ipv6 = false;
#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;
startTimestamp = 0;
#$container: HTMLElement | undefined;
#cachedDoms: Partial<{[key in StreamBadge]: HTMLElement}> = {};
#interval?: number | null;
readonly #REFRESH_INTERVAL = 3000;
setRegion(region: string) { setRegion(region: string) {
this.#region = region; this.serverInfo.server = {
region: region,
};
} }
#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 +158,49 @@ 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) { destroy() {
let h = Math.floor(seconds / 3600); this.serverInfo = {};
let m = Math.floor(seconds % 3600 / 60) + 1; delete this.$container;
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 +208,51 @@ 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.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 => {
for (const item of BADGES) {
if (!item) { if (!item) {
return; continue;
} }
const $badge = this.#renderBadge(...(item as [StreamBadge, string, string])); let $badge: HTMLElement;
$container.appendChild($badge); if (!(item instanceof HTMLElement)) {
}); $badge = this.renderBadge(...(item as [StreamBadge, string]));
} else {
$badge = item;
}
this.#$container = $container; $container.appendChild($badge);
await this.#start(); };
this.$container = $container;
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 +269,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 +284,89 @@ 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
let text = '';
const isIpv6 = allCandidates[candidateId].includes(':');
const server = this.serverInfo.server;
if (server && server.region) {
text += server.region;
}
text += '@' + (isIpv6 ? '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());
}); });
*/
} }
} }

View File

@ -7,9 +7,10 @@ import { getPref, setPref } from "@/utils/settings-storages/global-settings-stor
export function onChangeVideoPlayerType() { export function onChangeVideoPlayerType() {
const playerType = getPref(PrefKey.VIDEO_PLAYER_TYPE); const playerType = getPref(PrefKey.VIDEO_PLAYER_TYPE);
const $videoProcessing = document.getElementById('bx_setting_video_processing') as HTMLSelectElement; const $videoProcessing = document.getElementById(`bx_setting_${PrefKey.VIDEO_PROCESSING}`) as HTMLSelectElement;
const $videoSharpness = document.getElementById('bx_setting_video_sharpness') as HTMLElement; const $videoSharpness = document.getElementById(`bx_setting_${PrefKey.VIDEO_SHARPNESS}`) as HTMLElement;
const $videoPowerPreference = document.getElementById('bx_setting_video_power_preference') as HTMLElement; const $videoPowerPreference = document.getElementById(`bx_setting_${PrefKey.VIDEO_POWER_PREFERENCE}`) as HTMLElement;
const $videoMaxFps = document.getElementById(`bx_setting_${PrefKey.VIDEO_MAX_FPS}`) as HTMLElement;
if (!$videoProcessing) { if (!$videoProcessing) {
return; return;
@ -17,7 +18,7 @@ export function onChangeVideoPlayerType() {
let isDisabled = false; let isDisabled = false;
const $optCas = $videoProcessing.querySelector(`option[value=${StreamVideoProcessing.CAS}]`) as HTMLOptionElement; const $optCas = $videoProcessing.querySelector<HTMLOptionElement>(`option[value=${StreamVideoProcessing.CAS}]`);
if (playerType === StreamPlayerType.WEBGL2) { if (playerType === StreamPlayerType.WEBGL2) {
$optCas && ($optCas.disabled = false); $optCas && ($optCas.disabled = false);
@ -38,17 +39,26 @@ export function onChangeVideoPlayerType() {
// Hide Power Preference setting if renderer isn't WebGL2 // Hide Power Preference setting if renderer isn't WebGL2
$videoPowerPreference.closest('.bx-settings-row')!.classList.toggle('bx-gone', playerType !== StreamPlayerType.WEBGL2); $videoPowerPreference.closest('.bx-settings-row')!.classList.toggle('bx-gone', playerType !== StreamPlayerType.WEBGL2);
$videoMaxFps.closest('.bx-settings-row')!.classList.toggle('bx-gone', playerType !== StreamPlayerType.WEBGL2);
updateVideoPlayer(); updateVideoPlayer();
} }
export function limitVideoPlayerFps(targetFps: number) {
const streamPlayer = STATES.currentStream.streamPlayer;
streamPlayer?.getWebGL2Player()?.setTargetFps(targetFps);
}
export function updateVideoPlayer() { export function updateVideoPlayer() {
const streamPlayer = STATES.currentStream.streamPlayer; const streamPlayer = STATES.currentStream.streamPlayer;
if (!streamPlayer) { if (!streamPlayer) {
return; return;
} }
limitVideoPlayerFps(getPref(PrefKey.VIDEO_MAX_FPS));
const options = { const options = {
processing: getPref(PrefKey.VIDEO_PROCESSING), processing: getPref(PrefKey.VIDEO_PROCESSING),
sharpness: getPref(PrefKey.VIDEO_SHARPNESS), sharpness: getPref(PrefKey.VIDEO_SHARPNESS),
@ -60,6 +70,7 @@ export function updateVideoPlayer() {
streamPlayer.setPlayerType(getPref(PrefKey.VIDEO_PLAYER_TYPE)); streamPlayer.setPlayerType(getPref(PrefKey.VIDEO_PLAYER_TYPE));
streamPlayer.updateOptions(options); streamPlayer.updateOptions(options);
streamPlayer.refreshPlayer(); streamPlayer.refreshPlayer();
} }
window.addEventListener('resize', updateVideoPlayer); window.addEventListener('resize', updateVideoPlayer);

View File

@ -4,92 +4,123 @@ 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"
import { BxLogger } from "@/utils/bx-logger"
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;
public static getInstance(): StreamStats { public static getInstance = () => StreamStats.instance ?? (StreamStats.instance = new StreamStats());
if (!StreamStats.instance) { private readonly LOG_TAG = 'StreamStats';
StreamStats.instance = new StreamStats();
}
return StreamStats.instance; private intervalId?: number | null;
private readonly REFRESH_INTERVAL = 1 * 1000;
private stats = {
[StreamStat.CLOCK]: {
name: t('clock'),
$element: CE('span'),
},
[StreamStat.PLAYTIME]: {
name: t('playtime'),
$element: CE('span'),
},
[StreamStat.BATTERY]: {
name: t('battery'),
$element: CE('span'),
},
[StreamStat.PING]: {
name: t('stat-ping'),
$element: CE('span'),
},
[StreamStat.JITTER]: {
name: t('jitter'),
$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('downloaded'),
$element: CE('span'),
},
[StreamStat.UPLOAD]: {
name: t('uploaded'),
$element: CE('span'),
},
};
private $container!: HTMLElement;
quickGlanceObserver?: MutationObserver | null;
private constructor() {
BxLogger.info(this.LOG_TAG, 'constructor()');
this.render();
} }
#timeoutId?: number | null; async start(glancing=false) {
readonly #updateInterval = 1000;
#$container: HTMLElement | undefined;
#$fps: HTMLElement | undefined;
#$ping: HTMLElement | undefined;
#$dt: HTMLElement | undefined;
#$pl: HTMLElement | undefined;
#$fl: HTMLElement | undefined;
#$br: HTMLElement | undefined;
#lastVideoStat?: RTCBasicStat | null;
#quickGlanceObserver?: MutationObserver | null;
constructor() {
this.#render();
}
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();
} }
} }
onStoppedPlaying() { destroy() {
this.stop(); this.stop();
this.quickGlanceStop(); this.quickGlanceStop();
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 +129,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 +153,52 @@ 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.destroy();
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 && 'grades' in value) {
const framesReceived = stat.framesReceived; grade = statsCollector.calculateGrade(value.current, value.grades);
const framesDroppedPercentage = (framesDropped * 100 / ((framesDropped + framesReceived) || 1)).toFixed(2);
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 +207,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 +234,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) {

View File

@ -39,7 +39,7 @@ export class StreamUiHandler {
return; return;
} }
const $streamHud = (e.target as HTMLElement).closest('#StreamHud') as HTMLElement; const $streamHud = (e.target as HTMLElement).closest<HTMLElement>('#StreamHud');
if (!$streamHud) { if (!$streamHud) {
return; return;
} }
@ -58,13 +58,13 @@ export class StreamUiHandler {
$container.addEventListener('transitionend', onTransitionEnd); $container.addEventListener('transitionend', onTransitionEnd);
} }
const $button = $container.querySelector('button') as HTMLElement; const $button = $container.querySelector<HTMLButtonElement>('button');
if (!$button) { if (!$button) {
return null; return null;
} }
$button.setAttribute('title', label); $button.setAttribute('title', label);
const $orgSvg = $button.querySelector('svg') as SVGElement; const $orgSvg = $button.querySelector<SVGElement>('svg');
if (!$orgSvg) { if (!$orgSvg) {
return null; return null;
} }
@ -102,7 +102,7 @@ export class StreamUiHandler {
} }
private static async handleStreamMenu() { private static async handleStreamMenu() {
const $btnCloseHud = document.querySelector('button[class*=StreamMenu-module__backButton]') as HTMLElement; const $btnCloseHud = document.querySelector<HTMLElement>('button[class*=StreamMenu-module__backButton]');
if (!$btnCloseHud) { if (!$btnCloseHud) {
return; return;
} }
@ -135,27 +135,21 @@ 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<HTMLElement>('div[class^=HUDButton]');
if (!$orgButton) { if (!$orgButton) {
return; return;
} }
const hideGripHandle = () => { const hideGripHandle = () => {
if (!$gripHandle) { // Grip handle
return; const $gripHandle = document.querySelector<HTMLElement>('#StreamHud button[class^=GripHandle]');
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);
@ -227,9 +221,10 @@ export class StreamUiHandler {
} }
const observer = new MutationObserver(mutationList => { const observer = new MutationObserver(mutationList => {
mutationList.forEach(item => { let item: MutationRecord;
for (item of mutationList) {
if (item.type !== 'childList') { if (item.type !== 'childList') {
return; continue;
} }
item.addedNodes.forEach(async $node => { item.addedNodes.forEach(async $node => {
@ -269,7 +264,7 @@ export class StreamUiHandler {
// Handle System Menu bar // Handle System Menu bar
StreamUiHandler.handleSystemMenu($elm); StreamUiHandler.handleSystemMenu($elm);
}); });
}); };
}); });
observer.observe($screen, {subtree: true, childList: true}); observer.observe($screen, {subtree: true, childList: true});

View File

@ -85,16 +85,24 @@ export class TouchController {
document.querySelector('#BabylonCanvasContainer-main')?.parentElement?.classList.remove('bx-offscreen'); document.querySelector('#BabylonCanvasContainer-main')?.parentElement?.classList.remove('bx-offscreen');
} }
/*
static #hide() { static #hide() {
document.querySelector('#BabylonCanvasContainer-main')?.parentElement?.classList.add('bx-offscreen'); document.querySelector('#BabylonCanvasContainer-main')?.parentElement?.classList.add('bx-offscreen');
} }
*/
static toggleVisibility(status: boolean) { static toggleVisibility(): boolean {
if (!TouchController.#dataChannel) { if (!TouchController.#dataChannel) {
return; return false;
} }
status ? TouchController.#hide() : TouchController.#show(); const $container = document.querySelector('#BabylonCanvasContainer-main')?.parentElement;
if (!$container) {
return false;
}
$container.classList.toggle('bx-offscreen');
return !$container.classList.contains('bx-offscreen');
} }
static reset() { static reset() {

View File

@ -1,9 +1,12 @@
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 { BxLogger } from "@/utils/bx-logger";
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,18 +83,14 @@ export abstract class NavigationDialog {
} }
handleGamepad(button: GamepadKey): boolean { handleGamepad(button: GamepadKey): boolean {
return true; return false;
} }
} }
export class NavigationDialogManager { export class NavigationDialogManager {
private static instance: NavigationDialogManager; private static instance: NavigationDialogManager;
public static getInstance(): NavigationDialogManager { public static getInstance = () => NavigationDialogManager.instance ?? (NavigationDialogManager.instance = new NavigationDialogManager());
if (!NavigationDialogManager.instance) { private readonly LOG_TAG = 'NavigationDialogManager';
NavigationDialogManager.instance = new NavigationDialogManager();
}
return NavigationDialogManager.instance;
}
private static readonly GAMEPAD_POLLING_INTERVAL = 50; private static readonly GAMEPAD_POLLING_INTERVAL = 50;
private static readonly GAMEPAD_KEYS = [ private static readonly GAMEPAD_KEYS = [
@ -139,7 +138,9 @@ export class NavigationDialogManager {
private $container: HTMLElement; private $container: HTMLElement;
private dialog: NavigationDialog | null = null; private dialog: NavigationDialog | null = null;
constructor() { private constructor() {
BxLogger.info(this.LOG_TAG, 'constructor()');
this.$overlay = CE('div', {class: 'bx-navigation-dialog-overlay bx-gone'}); this.$overlay = CE('div', {class: 'bx-navigation-dialog-overlay bx-gone'});
this.$overlay.addEventListener('click', e => { this.$overlay.addEventListener('click', e => {
e.preventDefault(); e.preventDefault();
@ -154,6 +155,63 @@ 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 = Array.from($root.querySelectorAll('.bx-select:not([data-calculated]) select'));
for (const $select of selects) {
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: HTMLElement;
let width = Math.ceil(rect.width);
if (!width) {
return;
}
if (($select as HTMLSelectElement).multiple) {
$label = $parent.querySelector<HTMLElement>('.bx-select-value')!;
width += 20; // Add checkbox's width
} else {
$label = $parent.querySelector<HTMLElement>('div')!;
}
// Reduce width if it has <optgroup>
if ($select.querySelector('optgroup')) {
width -= 15;
}
// Set min-width
$label.style.minWidth = width + 'px';
$parent.dataset.calculated = 'true';
};
} }
handleEvent(event: Event) { handleEvent(event: Event) {
@ -181,7 +239,7 @@ export class NavigationDialogManager {
} else if (keyCode === 'Enter' || keyCode === 'NumpadEnter' || keyCode === 'Space') { } else if (keyCode === 'Enter' || keyCode === 'NumpadEnter' || keyCode === 'Space') {
if (!($target instanceof HTMLInputElement && $target.type === 'text')) { if (!($target instanceof HTMLInputElement && $target.type === 'text')) {
handled = true; handled = true;
$target.dispatchEvent(new MouseEvent('click')); $target.dispatchEvent(new MouseEvent('click', {bubbles: true}));
} }
} else if (keyCode === 'Escape') { } else if (keyCode === 'Escape') {
handled = true; handled = true;
@ -210,7 +268,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;
} }
@ -308,7 +366,7 @@ export class NavigationDialogManager {
} }
if (releasedButton === GamepadKey.A) { if (releasedButton === GamepadKey.A) {
document.activeElement && document.activeElement.dispatchEvent(new MouseEvent('click')); document.activeElement && document.activeElement.dispatchEvent(new MouseEvent('click', {bubbles: true}));
return; return;
} else if (releasedButton === GamepadKey.B) { } else if (releasedButton === GamepadKey.B) {
this.hide(); this.hide();

View File

@ -0,0 +1,133 @@
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";
import { BxLogger } from "@/utils/bx-logger";
export class RemotePlayNavigationDialog extends NavigationDialog {
private static instance: RemotePlayNavigationDialog;
public static getInstance = () => RemotePlayNavigationDialog.instance ?? (RemotePlayNavigationDialog.instance = new RemotePlayNavigationDialog());
private readonly LOG_TAG = 'RemotePlayNavigationDialog';
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;
private constructor() {
super();
BxLogger.info(this.LOG_TAG, 'constructor()');
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<HTMLElement>('.bx-remote-play-device-wrapper button');
$btnConnect && $btnConnect.focus();
}
}

View File

@ -1,5 +1,7 @@
import { onChangeVideoPlayerType, updateVideoPlayer } from "@/modules/stream/stream-settings-utils"; import { isFullVersion } from "@macros/build" with {type: "macro"};
import { ButtonStyle, CE, createButton, createSvgIcon, removeChildElements } from "@/utils/html";
import { limitVideoPlayerFps, onChangeVideoPlayerType, updateVideoPlayer } from "@/modules/stream/stream-settings-utils";
import { ButtonStyle, CE, createButton, createSvgIcon, removeChildElements, type BxButton } from "@/utils/html";
import { NavigationDialog, NavigationDirection } from "./navigation-dialog"; import { NavigationDialog, NavigationDirection } from "./navigation-dialog";
import { ControllerShortcut } from "@/modules/controller-shortcut"; import { ControllerShortcut } from "@/modules/controller-shortcut";
import { MkbRemapper } from "@/modules/mkb/mkb-remapper"; import { MkbRemapper } from "@/modules/mkb/mkb-remapper";
@ -10,14 +12,14 @@ 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";
import { PatcherCache } from "@/modules/patcher"; import { PatcherCache } from "@/modules/patcher";
import { UserAgentProfile } from "@/enums/user-agent"; import { UserAgentProfile } from "@/enums/user-agent";
import { UserAgent } from "@/utils/user-agent"; import { UserAgent } from "@/utils/user-agent";
import { BX_FLAGS, NATIVE_FETCH } from "@/utils/bx-flags"; import { BX_FLAGS, NATIVE_FETCH, type BxFlags } from "@/utils/bx-flags";
import { copyToClipboard } from "@/utils/utils"; import { copyToClipboard } from "@/utils/utils";
import { GamepadKey } from "@/enums/mkb"; import { GamepadKey } from "@/enums/mkb";
import { PrefKey, StorageKey } from "@/enums/pref-keys"; import { PrefKey, StorageKey } from "@/enums/pref-keys";
@ -25,49 +27,55 @@ import { ControllerDeviceVibration, getPref, getPrefDefinition, setPref, StreamT
import { SettingElement, type BxHtmlSettingElement } from "@/utils/setting-element"; import { SettingElement, type BxHtmlSettingElement } from "@/utils/setting-element";
import type { RecommendedSettings, SettingDefinition, SuggestedSettingCategory as SuggestedSettingProfile } from "@/types/setting-definition"; import type { RecommendedSettings, SettingDefinition, SuggestedSettingCategory as SuggestedSettingProfile } from "@/types/setting-definition";
import { FullscreenText } from "../fullscreen-text"; import { FullscreenText } from "../fullscreen-text";
import { BxLogger } from "@/utils/bx-logger";
import { updatePollingRate } from "@/utils/gamepad";
type SettingTabContentItem = Partial<{ type SettingTabContentItem = Partial<{
pref: PrefKey; pref: PrefKey;
label: string; label: string;
note: string; note: string | (() => HTMLElement);
experimental: string; experimental: string;
content: HTMLElement | (() => HTMLElement); content: HTMLElement | (() => HTMLElement);
options: {[key: string]: string}; options: {[key: string]: string};
unsupported: boolean; unsupported: boolean;
unsupportedNote: string;
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 = {
group: 'general' | 'server' | 'stream' | 'game-bar' | 'co-op' | 'mkb' | 'touch-control' | 'loading-screen' | 'ui' | 'other' | 'advanced' | 'footer' | 'audio' | 'video' | 'controller' | 'native-mkb' | 'stats' | 'controller-shortcuts'; group: 'general' | 'server' | 'stream' | 'game-bar' | 'co-op' | 'mkb' | 'touch-control' | 'loading-screen' | 'ui' | 'other' | 'advanced' | 'footer' | 'audio' | 'video' | 'controller' | 'native-mkb' | 'stats' | 'controller-shortcuts';
label?: string; label?: string;
note?: string | Text | null;
unsupported?: boolean; unsupported?: boolean;
unsupportedNote?: string | Text | null;
helpUrl?: string; helpUrl?: string;
content?: any; content?: any;
lazyContent?: boolean | (() => HTMLElement);
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: SettingTabGroup,
items: Array<SettingTabContent | false>; items: Array<SettingTabContent | false> | (() => Array<SettingTabContent | false>);
requiredVariants?: BuildVariant | Array<BuildVariant>;
lazyContent?: boolean;
}; };
type SettingTabGroup = 'global' | 'stream' | 'controller' | 'mkb' | 'native-mkb' | 'shortcuts' | 'stats';
export class SettingsNavigationDialog extends NavigationDialog { export class SettingsNavigationDialog extends NavigationDialog {
private static instance: SettingsNavigationDialog; private static instance: SettingsNavigationDialog;
public static getInstance(): SettingsNavigationDialog { public static getInstance = () => SettingsNavigationDialog.instance ?? (SettingsNavigationDialog.instance = new SettingsNavigationDialog());
if (!SettingsNavigationDialog.instance) { private readonly LOG_TAG = 'SettingsNavigationDialog';
SettingsNavigationDialog.instance = new SettingsNavigationDialog();
}
return SettingsNavigationDialog.instance;
}
$container!: HTMLElement; $container!: HTMLElement;
private $tabs!: HTMLElement; private $tabs!: HTMLElement;
private $settings!: HTMLElement; private $tabContents!: HTMLElement;
private $btnReload!: HTMLElement; private $btnReload!: HTMLElement;
private $btnGlobalReload!: HTMLButtonElement; private $btnGlobalReload!: HTMLButtonElement;
@ -97,12 +105,19 @@ export class SettingsNavigationDialog extends NavigationDialog {
// "New version available" button // "New version available" button
if (!SCRIPT_VERSION.includes('beta') && PREF_LATEST_VERSION && PREF_LATEST_VERSION != SCRIPT_VERSION) { if (!SCRIPT_VERSION.includes('beta') && PREF_LATEST_VERSION && PREF_LATEST_VERSION != SCRIPT_VERSION) {
// Show new version indicator // Show new version button
topButtons.push(createButton({ const opts = {
label: `🌟 Version ${PREF_LATEST_VERSION} available`, label: '🌟 ' + t('new-version-available', {version: PREF_LATEST_VERSION}),
style: ButtonStyle.PRIMARY | ButtonStyle.FOCUSABLE | ButtonStyle.FULL_WIDTH, style: ButtonStyle.PRIMARY | ButtonStyle.FOCUSABLE | ButtonStyle.FULL_WIDTH,
url: 'https://github.com/redphx/better-xcloud/releases/latest', } as BxButton;
}));
if (AppInterface && AppInterface.updateLatestScript) {
opts.onClick = e => AppInterface.updateLatestScript();
} else {
opts.url = 'https://github.com/redphx/better-xcloud/releases/latest';
}
topButtons.push(createButton(opts));
} }
// Buttons for Android app // Buttons for Android app
@ -198,24 +213,33 @@ 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'),
unsupportedNote: !STATES.userAgent.capabilities.mkb ? CE('a', {
href: 'https://github.com/redphx/better-xcloud/issues/206#issuecomment-1920475657',
target: '_blank',
}, '⚠️ ' + t('browser-unsupported-feature')) : null,
unsupported: !STATES.userAgent.capabilities.mkb,
items: [ items: [
PrefKey.NATIVE_MKB_ENABLED, PrefKey.NATIVE_MKB_ENABLED,
PrefKey.GAME_MSFS2020_FORCE_NATIVE_MKB,
PrefKey.MKB_ENABLED, PrefKey.MKB_ENABLED,
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,
unsupported: !STATES.userAgent.capabilities.touch, unsupported: !STATES.userAgent.capabilities.touch,
unsupportedNote: !STATES.userAgent.capabilities.touch ? '⚠️ ' + t('device-unsupported-touch') : null,
items: [ items: [
PrefKey.STREAM_TOUCH_CONTROLLER, PrefKey.STREAM_TOUCH_CONTROLLER,
PrefKey.STREAM_TOUCH_CONTROLLER_AUTO_OFF, PrefKey.STREAM_TOUCH_CONTROLLER_AUTO_OFF,
@ -229,7 +253,6 @@ export class SettingsNavigationDialog extends NavigationDialog {
items: [ items: [
PrefKey.UI_LAYOUT, PrefKey.UI_LAYOUT,
PrefKey.UI_GAME_CARD_SHOW_WAIT_TIME, PrefKey.UI_GAME_CARD_SHOW_WAIT_TIME,
PrefKey.UI_HOME_CONTEXT_MENU_DISABLED,
PrefKey.CONTROLLER_SHOW_CONNECTION_STATUS, PrefKey.CONTROLLER_SHOW_CONNECTION_STATUS,
PrefKey.STREAM_SIMPLIFY_MENU, PrefKey.STREAM_SIMPLIFY_MENU,
PrefKey.SKIP_SPLASH_VIDEO, PrefKey.SKIP_SPLASH_VIDEO,
@ -240,6 +263,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: [
@ -309,8 +333,8 @@ export class SettingsNavigationDialog extends NavigationDialog {
// xCloud version // xCloud version
($parent) => { ($parent) => {
try { try {
const appVersion = (document.querySelector('meta[name=gamepass-app-version]') as HTMLMetaElement).content; const appVersion = document.querySelector<HTMLMetaElement>('meta[name=gamepass-app-version]')!.content;
const appDate = new Date((document.querySelector('meta[name=gamepass-app-date]') as HTMLMetaElement).content).toISOString().substring(0, 10); const appDate = new Date(document.querySelector<HTMLMetaElement>('meta[name=gamepass-app-date]')!.content).toISOString().substring(0, 10);
$parent.appendChild(CE('div', { $parent.appendChild(CE('div', {
class: 'bx-settings-app-version', class: 'bx-settings-app-version',
}, `xCloud website version ${appVersion} (${appDate})`)); }, `xCloud website version ${appVersion} (${appDate})`));
@ -350,6 +374,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',
@ -362,7 +387,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
disabled: !getPref(PrefKey.AUDIO_ENABLE_VOLUME_CONTROL), disabled: !getPref(PrefKey.AUDIO_ENABLE_VOLUME_CONTROL),
}, },
onCreated: (setting: SettingTabContentItem, $elm: HTMLElement) => { onCreated: (setting: SettingTabContentItem, $elm: HTMLElement) => {
const $range = $elm.querySelector('input[type=range') as HTMLInputElement; const $range = $elm.querySelector<HTMLInputElement>('input[type=range')!;
window.addEventListener(BxEvent.SETTINGS_CHANGED, e => { window.addEventListener(BxEvent.SETTINGS_CHANGED, e => {
const { storageKey, settingKey, settingValue } = e as any; const { storageKey, settingKey, settingValue } = e as any;
if (storageKey !== StorageKey.GLOBAL || settingKey !== PrefKey.AUDIO_VOLUME) { if (storageKey !== StorageKey.GLOBAL || settingKey !== PrefKey.AUDIO_VOLUME) {
@ -383,6 +408,11 @@ export class SettingsNavigationDialog extends NavigationDialog {
items: [{ items: [{
pref: PrefKey.VIDEO_PLAYER_TYPE, pref: PrefKey.VIDEO_PLAYER_TYPE,
onChange: onChangeVideoPlayerType, onChange: onChangeVideoPlayerType,
}, {
pref: PrefKey.VIDEO_MAX_FPS,
onChange: e => {
limitVideoPlayerFps(parseInt(e.target.value));
},
}, { }, {
pref: PrefKey.VIDEO_POWER_PREFERENCE, pref: PrefKey.VIDEO_POWER_PREFERENCE,
onChange: () => { onChange: () => {
@ -431,10 +461,13 @@ export class SettingsNavigationDialog extends NavigationDialog {
pref: PrefKey.CONTROLLER_VIBRATION_INTENSITY, pref: PrefKey.CONTROLLER_VIBRATION_INTENSITY,
unsupported: !VibrationManager.supportDeviceVibration(), unsupported: !VibrationManager.supportDeviceVibration(),
onChange: () => VibrationManager.updateGlobalVars(), onChange: () => VibrationManager.updateGlobalVars(),
}, isFullVersion() && {
pref: PrefKey.CONTROLLER_POLLING_RATE,
onChange: () => updatePollingRate(),
}], }],
}, },
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: [{
@ -488,17 +521,18 @@ export class SettingsNavigationDialog extends NavigationDialog {
}], }],
}]; }];
private readonly TAB_VIRTUAL_CONTROLLER_ITEMS: Array<SettingTabContent | false> = [{ private readonly TAB_VIRTUAL_CONTROLLER_ITEMS: (() => Array<SettingTabContent | false>) = () => [{
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: MkbRemapper.getInstance().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);
@ -508,13 +542,14 @@ export class SettingsNavigationDialog extends NavigationDialog {
onChange: (e: any, value: number) => { onChange: (e: any, value: number) => {
NativeMkbHandler.getInstance().setHorizontalScrollMultiplier(value / 100); NativeMkbHandler.getInstance().setHorizontalScrollMultiplier(value / 100);
}, },
}], }] : [],
}]; }];
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> = [{
@ -551,52 +586,59 @@ export class SettingsNavigationDialog extends NavigationDialog {
], ],
}]; }];
private readonly SETTINGS_UI: Array<SettingTab> = [ private readonly SETTINGS_UI: Record<SettingTabGroup, SettingTab> = {
{ global: {
icon: BxIcon.HOME,
group: 'global', group: 'global',
icon: BxIcon.HOME,
items: this.TAB_GLOBAL_ITEMS, items: this.TAB_GLOBAL_ITEMS,
}, },
{ stream: {
icon: BxIcon.DISPLAY,
group: 'stream', group: 'stream',
icon: BxIcon.DISPLAY,
items: this.TAB_DISPLAY_ITEMS, items: this.TAB_DISPLAY_ITEMS,
}, },
{ controller: {
icon: BxIcon.CONTROLLER,
group: 'controller', group: 'controller',
icon: BxIcon.CONTROLLER,
items: this.TAB_CONTROLLER_ITEMS, items: this.TAB_CONTROLLER_ITEMS,
requiredVariants: 'full',
}, },
getPref(PrefKey.MKB_ENABLED) && { mkb: isFullVersion() && getPref(PrefKey.MKB_ENABLED) && {
icon: BxIcon.VIRTUAL_CONTROLLER,
group: 'mkb', group: 'mkb',
icon: BxIcon.VIRTUAL_CONTROLLER,
items: this.TAB_VIRTUAL_CONTROLLER_ITEMS, items: this.TAB_VIRTUAL_CONTROLLER_ITEMS,
lazyContent: true,
requiredVariants: 'full',
}, },
AppInterface && getPref(PrefKey.NATIVE_MKB_ENABLED) === 'on' && { 'native-mkb': isFullVersion() && AppInterface && getPref(PrefKey.NATIVE_MKB_ENABLED) === 'on' && {
icon: BxIcon.NATIVE_MKB,
group: 'native-mkb', group: 'native-mkb',
icon: BxIcon.NATIVE_MKB,
items: this.TAB_NATIVE_MKB_ITEMS, items: this.TAB_NATIVE_MKB_ITEMS,
requiredVariants: 'full',
}, },
{ shortcuts: {
icon: BxIcon.COMMAND,
group: 'shortcuts', group: 'shortcuts',
icon: BxIcon.COMMAND,
items: this.TAB_SHORTCUTS_ITEMS, items: this.TAB_SHORTCUTS_ITEMS,
lazyContent: true,
requiredVariants: 'full',
}, },
{ stats: {
icon: BxIcon.STREAM_STATS,
group: 'stats', group: 'stats',
icon: BxIcon.STREAM_STATS,
items: this.TAB_STATS_ITEMS, items: this.TAB_STATS_ITEMS,
}, },
]; };
constructor() { private constructor() {
super(); super();
BxLogger.info(this.LOG_TAG, 'constructor()');
this.renderFullSettings = STATES.supportedRegion && STATES.isSignedIn; this.renderFullSettings = STATES.supportedRegion && STATES.isSignedIn;
this.setupDialog(); this.setupDialog();
@ -624,7 +666,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
} }
// Trigger event // Trigger event
const $selectUserAgent = document.querySelector(`#bx_setting_${PrefKey.USER_AGENT_PROFILE}`) as HTMLSelectElement; const $selectUserAgent = document.querySelector<HTMLSelectElement>(`#bx_setting_${PrefKey.USER_AGENT_PROFILE}`);
if ($selectUserAgent) { if ($selectUserAgent) {
$selectUserAgent.disabled = true; $selectUserAgent.disabled = true;
BxEvent.dispatch($selectUserAgent, 'input', {}); BxEvent.dispatch($selectUserAgent, 'input', {});
@ -641,10 +683,23 @@ export class SettingsNavigationDialog extends NavigationDialog {
window.location.reload(); window.location.reload();
} }
private async getRecommendedSettings(deviceCode: string): Promise<string | null> { private async getRecommendedSettings(androidInfo: BxFlags['DeviceInfo']['androidInfo']): Promise<string | null> {
function normalize(str: string) {
return str.toLowerCase()
.trim()
.replaceAll(/\s+/g, '-')
.replaceAll(/-+/g, '-');
}
// Get recommended settings from GitHub // Get recommended settings from GitHub
try { try {
const response = await NATIVE_FETCH(`https://raw.githubusercontent.com/redphx/better-xcloud/gh-pages/devices/${deviceCode.toLowerCase()}.json`); let {brand, board, model} = androidInfo!;
brand = normalize(brand);
board = normalize(board);
model = normalize(model);
const url = `https://raw.githubusercontent.com/redphx/better-xcloud/gh-pages/devices/${brand}/${board}-${model}.json`;
const response = await NATIVE_FETCH(url);
const json = (await response.json()) as RecommendedSettings; const json = (await response.json()) as RecommendedSettings;
const recommended: PartialRecord<PrefKey, any> = {}; const recommended: PartialRecord<PrefKey, any> = {};
@ -708,6 +763,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');
@ -719,8 +783,11 @@ export class SettingsNavigationDialog extends NavigationDialog {
} }
// Get labels // Get labels
for (const settingTab of this.SETTINGS_UI) { let settingTabGroup: keyof typeof this.SETTINGS_UI;
if (!settingTab || !settingTab.items) { for (settingTabGroup in this.SETTINGS_UI) {
const settingTab = this.SETTINGS_UI[settingTabGroup];
if (!settingTab || !settingTab.items || typeof settingTab.items === 'function') {
continue; continue;
} }
@ -750,11 +817,17 @@ export class SettingsNavigationDialog extends NavigationDialog {
if (BX_FLAGS.DeviceInfo.deviceType.includes('android')) { if (BX_FLAGS.DeviceInfo.deviceType.includes('android')) {
if (BX_FLAGS.DeviceInfo.androidInfo) { if (BX_FLAGS.DeviceInfo.androidInfo) {
const deviceCode = BX_FLAGS.DeviceInfo.androidInfo.board; recommendedDevice = await this.getRecommendedSettings(BX_FLAGS.DeviceInfo.androidInfo);
recommendedDevice = await this.getRecommendedSettings(deviceCode);
} }
} }
// recommendedDevice = await this.getRecommendedSettings('foster_e');
/*
recommendedDevice = await this.getRecommendedSettings({
manufacturer: 'Lenovo',
board: 'kona',
model: 'Lenovo TB-9707F',
});
*/
const hasRecommendedSettings = Object.keys(this.suggestedSettings.recommended).length > 0; const hasRecommendedSettings = Object.keys(this.suggestedSettings.recommended).length > 0;
@ -863,7 +936,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
let prefKey: PrefKey; let prefKey: PrefKey;
for (prefKey in settings) { for (prefKey in settings) {
const suggestedValue = settings[prefKey]; const suggestedValue = settings[prefKey];
const $checkBox = $content.querySelector(`#bx_suggest_${prefKey}`) as HTMLInputElement; const $checkBox = $content.querySelector<HTMLInputElement>(`#bx_suggest_${prefKey}`)!;
if (!$checkBox.checked || $checkBox.disabled) { if (!$checkBox.checked || $checkBox.disabled) {
continue; continue;
} }
@ -923,38 +996,64 @@ export class SettingsNavigationDialog extends NavigationDialog {
}, t('suggest-settings-link')), }, t('suggest-settings-link')),
); );
$btnSuggest?.insertAdjacentElement('afterend', $content); $btnSuggest.insertAdjacentElement('afterend', $content);
}
private onTabClicked(e: Event) {
const $svg = (e.target as SVGElement).closest('svg')!;
// Render tab content lazily
if (!!$svg.dataset.lazy) {
// Remove attribute
delete $svg.dataset.lazy;
// Render data
const settingTab = this.SETTINGS_UI[$svg.dataset.group as SettingTabGroup];
const items = (settingTab.items as Function)();
const $tabContent = this.renderTabContent.call(this, settingTab, items);
this.$tabContents.appendChild($tabContent);
}
// Switch tab
let $child: HTMLElement;
const children = Array.from(this.$tabContents.children) as HTMLElement[];
for ($child of children) {
if ($child.dataset.tabGroup === $svg.dataset.group) {
// Show tab content
$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 {
// Hide tab content
$child.classList.add('bx-gone');
}
}
// Highlight current tab button
for (const $child of Array.from(this.$tabs.children)) {
$child.classList.remove('bx-active');
}
$svg.classList.add('bx-active');
} }
private renderTab(settingTab: SettingTab) { private renderTab(settingTab: SettingTab) {
const $svg = createSvgIcon(settingTab.icon as any); const $svg = createSvgIcon(settingTab.icon as any);
$svg.dataset.group = settingTab.group; $svg.dataset.group = settingTab.group;
$svg.tabIndex = 0; $svg.tabIndex = 0;
settingTab.lazyContent && ($svg.dataset.lazy = settingTab.lazyContent.toString());
$svg.addEventListener('click', e => { $svg.addEventListener('click', this.onTabClicked.bind(this));
// Switch tab
for (const $child of Array.from(this.$settings.children)) {
if ($child.getAttribute('data-tab-group') === settingTab.group) {
$child.classList.remove('bx-gone');
} else {
$child.classList.add('bx-gone');
}
}
// Highlight current tab button
for (const $child of Array.from(this.$tabs.children)) {
$child.classList.remove('bx-active');
}
$svg.classList.add('bx-active');
});
return $svg; return $svg;
} }
private onGlobalSettingChanged(e: Event) { private onGlobalSettingChanged(e: Event) {
// Clear PatcherCache; // Clear PatcherCache;
PatcherCache.clear(); isFullVersion() && PatcherCache.getInstance().clear();
this.$btnReload.classList.add('bx-danger'); this.$btnReload.classList.add('bx-danger');
@ -964,13 +1063,37 @@ export class SettingsNavigationDialog extends NavigationDialog {
} }
private renderServerSetting(setting: SettingTabContentItem): HTMLElement { private renderServerSetting(setting: SettingTabContentItem): HTMLElement {
let selectedValue; let selectedValue =getPref(PrefKey.SERVER_REGION);
const continents: Record<ServerContinent, {
label: string,
children?: HTMLOptionElement[],
}> = {
'america-north': {
label: t('continent-north-america'),
},
'america-south': {
label: t('continent-south-america'),
},
'asia': {
label: t('continent-asia'),
},
'australia': {
label: t('continent-australia'),
},
'europe': {
label: t('continent-europe'),
},
'other': {
label: t('other'),
},
};
const $control = CE<HTMLSelectElement>('select', { const $control = CE<HTMLSelectElement>('select', {
id: `bx_setting_${setting.pref}`, id: `bx_setting_${setting.pref}`,
title: setting.label, title: setting.label,
tabindex: 0, tabindex: 0,
}); });
$control.name = $control.id; $control.name = $control.id;
$control.addEventListener('input', (e: Event) => { $control.addEventListener('input', (e: Event) => {
@ -978,8 +1101,6 @@ export class SettingsNavigationDialog extends NavigationDialog {
this.onGlobalSettingChanged(e); this.onGlobalSettingChanged(e);
}); });
selectedValue = getPref(PrefKey.SERVER_REGION);
setting.options = {}; setting.options = {};
for (const regionName in STATES.serverRegions) { for (const regionName in STATES.serverRegions) {
const region = STATES.serverRegions[regionName]; const region = STATES.serverRegions[regionName];
@ -996,15 +1117,29 @@ export class SettingsNavigationDialog extends NavigationDialog {
} }
setting.options[value] = label; setting.options[value] = label;
const $option = CE<HTMLOptionElement>('option', {value: value}, label);
const continent = continents[region.contintent];
if (!continent.children) {
continent.children = [];
}
continent.children.push($option);
} }
for (const value in setting.options) { const fragment = document.createDocumentFragment();
const label = setting.options[value]; let key: keyof typeof continents;
for (key in continents) {
const continent = continents[key];
if (!continent.children) {
continue;
}
const $option = CE('option', {value: value}, label); fragment.appendChild(CE('optgroup', {
$control.appendChild($option); label: continent.label,
}, ...continent.children));
} }
$control.appendChild(fragment);
$control.disabled = Object.keys(STATES.serverRegions).length === 0; $control.disabled = Object.keys(STATES.serverRegions).length === 0;
// Select preferred region // Select preferred region
@ -1089,10 +1224,24 @@ 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: string | undefined | (() => HTMLElement) | HTMLElement = prefDefinition?.note || setting.note;
let unsupportedNote: string | undefined | (() => HTMLElement) | HTMLElement = prefDefinition?.unsupportedNote || setting.unsupportedNote;
const experimental = prefDefinition?.experimental || setting.experimental; const experimental = prefDefinition?.experimental || setting.experimental;
// Render note lazily
if (typeof note === 'function') {
note = note();
}
if (typeof unsupportedNote === 'function') {
unsupportedNote = unsupportedNote();
}
if (settingTabContent.label && setting.pref) { if (settingTabContent.label && setting.pref) {
if (prefDefinition?.suggest) { if (prefDefinition?.suggest) {
typeof prefDefinition.suggest.lowest !== 'undefined' && (this.suggestedSettings.lowest[setting.pref] = prefDefinition.suggest.lowest); typeof prefDefinition.suggest.lowest !== 'undefined' && (this.suggestedSettings.lowest[setting.pref] = prefDefinition.suggest.lowest);
@ -1110,7 +1259,15 @@ export class SettingsNavigationDialog extends NavigationDialog {
} }
} }
let $note;
if (unsupportedNote) {
$note = CE('div', {class: 'bx-settings-dialog-note'}, unsupportedNote);
} else if (note) {
$note = CE('div', {class: 'bx-settings-dialog-note'}, note);
}
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}`,
@ -1121,10 +1278,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,
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
@ -1137,12 +1293,104 @@ 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 renderTabContent(settingTab: SettingTab, items: Array<SettingTabContent | false>): HTMLElement {
const $tabContent = CE('div', {
class: 'bx-gone',
'data-tab-group': settingTab.group,
});
for (const settingTabContent of items) {
if (!settingTabContent) {
continue;
}
if (!this.isSupportedVariant(settingTabContent.requiredVariants)) {
continue;
}
// Don't render other settings in unsupported regions
if (!this.renderFullSettings && settingTab.group === 'global' && settingTabContent.group !== 'general' && settingTabContent.group !== 'footer') {
continue;
}
let label = settingTabContent.label;
// If label is "Better xCloud" => create a link to Releases page
if (label === t('better-xcloud')) {
label += ' ' + SCRIPT_VERSION;
if (SCRIPT_VARIANT === 'lite') {
label += ' (Lite)';
}
label = createButton({
label: label,
url: 'https://github.com/redphx/better-xcloud/releases',
style: ButtonStyle.NORMAL_CASE | ButtonStyle.FROSTED | ButtonStyle.FOCUSABLE,
});
}
if (label) {
const $title = CE('h2', {
_nearby: {
orientation: 'horizontal',
}
},
CE('span', {}, label),
settingTabContent.helpUrl && createButton({
icon: BxIcon.QUESTION,
style: ButtonStyle.GHOST | ButtonStyle.FOCUSABLE,
url: settingTabContent.helpUrl,
title: t('help'),
}),
);
$tabContent.appendChild($title);
}
// Add note
if (settingTabContent.unsupportedNote) {
const $note = CE('b', {class: 'bx-note-unsupported'}, settingTabContent.unsupportedNote);
$tabContent.appendChild($note);
}
// Don't render settings if this is an unsupported feature
if (settingTabContent.unsupported) {
continue;
}
// Add content DOM
if (settingTabContent.content) {
$tabContent.appendChild(settingTabContent.content);
continue;
}
// Render list of settings
settingTabContent.items = settingTabContent.items || [];
for (const setting of settingTabContent.items) {
if (setting === false) {
continue;
}
if (typeof setting === 'function') {
setting.apply(this, [$tabContent]);
continue;
}
this.renderSettingRow(settingTab, $tabContent, settingTabContent, setting);
}
}
return $tabContent;
} }
private setupDialog() { private setupDialog() {
let $tabs: HTMLElement; let $tabs: HTMLElement;
let $settings: HTMLElement; let $tabContents: HTMLElement;
const $container = CE('div', { const $container = CE('div', {
class: 'bx-settings-dialog', class: 'bx-settings-dialog',
@ -1190,7 +1438,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
), ),
), ),
$settings = CE('div', { $tabContents = CE('div', {
class: 'bx-settings-tab-contents', class: 'bx-settings-tab-contents',
_nearby: { _nearby: {
orientation: 'vertical', orientation: 'vertical',
@ -1209,7 +1457,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
this.$container = $container; this.$container = $container;
this.$tabs = $tabs; this.$tabs = $tabs;
this.$settings = $settings; this.$tabContents = $tabContents;
// Close dialog when not clicking on any child elements in the dialog // Close dialog when not clicking on any child elements in the dialog
$container.addEventListener('click', e => { $container.addEventListener('click', e => {
@ -1220,11 +1468,19 @@ export class SettingsNavigationDialog extends NavigationDialog {
} }
}); });
for (const settingTab of this.SETTINGS_UI) { let settingTabGroup: keyof typeof this.SETTINGS_UI
for (settingTabGroup in this.SETTINGS_UI) {
const settingTab = this.SETTINGS_UI[settingTabGroup];
if (!settingTab) { if (!settingTab) {
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;
@ -1233,91 +1489,13 @@ export class SettingsNavigationDialog extends NavigationDialog {
const $svg = this.renderTab(settingTab); const $svg = this.renderTab(settingTab);
$tabs.appendChild($svg); $tabs.appendChild($svg);
const $tabContent = CE('div', { // Don't render lazy tab content
class: 'bx-gone', if (typeof settingTab.items === 'function') {
'data-tab-group': settingTab.group, continue;
});
for (const settingTabContent of settingTab.items) {
if (settingTabContent === false) {
continue;
}
// Don't render other settings in unsupported regions
if (!this.renderFullSettings && settingTab.group === 'global' && settingTabContent.group !== 'general' && settingTabContent.group !== 'footer') {
continue;
}
let label = settingTabContent.label;
// If label is "Better xCloud" => create a link to Releases page
if (label === t('better-xcloud')) {
label += ' ' + SCRIPT_VERSION;
label = createButton({
label: label,
url: 'https://github.com/redphx/better-xcloud/releases',
style: ButtonStyle.NORMAL_CASE | ButtonStyle.FROSTED | ButtonStyle.FOCUSABLE,
});
}
if (label) {
const $title = CE('h2', {
_nearby: {
orientation: 'horizontal',
}
},
CE('span', {}, label),
settingTabContent.helpUrl && createButton({
icon: BxIcon.QUESTION,
style: ButtonStyle.GHOST | ButtonStyle.FOCUSABLE,
url: settingTabContent.helpUrl,
title: t('help'),
}),
);
$tabContent.appendChild($title);
}
// Add note
if (settingTabContent.note) {
let $note;
if (typeof settingTabContent.note === 'string') {
$note = CE('b', {class: 'bx-note-unsupported'}, settingTabContent.note);
} else {
$note = settingTabContent.note;
}
$tabContent.appendChild($note);
}
// Don't render settings if this is an unsupported feature
if (settingTabContent.unsupported) {
continue;
}
// Add content DOM
if (settingTabContent.content) {
$tabContent.appendChild(settingTabContent.content);
continue;
}
// Render list of settings
settingTabContent.items = settingTabContent.items || [];
for (const setting of settingTabContent.items) {
if (setting === false) {
continue;
}
if (typeof setting === 'function') {
setting.apply(this, [$tabContent]);
continue;
}
this.renderSettingRow(settingTab, $tabContent, settingTabContent, setting);
}
} }
$settings.appendChild($tabContent); const $tabContent = this.renderTabContent.call(this, settingTab, settingTab.items);
$tabContents.appendChild($tabContent);
} }
// Select first tab // Select first tab
@ -1334,13 +1512,13 @@ export class SettingsNavigationDialog extends NavigationDialog {
} }
private focusActiveTab() { private focusActiveTab() {
const $currentTab = this.$tabs!.querySelector('.bx-active') as HTMLElement; const $currentTab = this.$tabs!.querySelector<HTMLElement>('.bx-active');
$currentTab && $currentTab.focus(); $currentTab && $currentTab.focus();
return true; return true;
} }
private focusVisibleSetting(type: 'first' | 'last' = 'first'): boolean { private focusVisibleSetting(type: 'first' | 'last' = 'first'): boolean {
const controls = Array.from(this.$settings.querySelectorAll('div[data-tab-group]:not(.bx-gone) > *')); const controls = Array.from(this.$tabContents.querySelectorAll('div[data-tab-group]:not(.bx-gone) > *'));
if (!controls.length) { if (!controls.length) {
return false; return false;
} }
@ -1386,7 +1564,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
} }
private jumpToSettingGroup(direction: 'next' | 'previous'): boolean { private jumpToSettingGroup(direction: 'next' | 'previous'): boolean {
const $tabContent = this.$settings.querySelector('div[data-tab-group]:not(.bx-gone)'); const $tabContent = this.$tabContents.querySelector('div[data-tab-group]:not(.bx-gone)');
if (!$tabContent) { if (!$tabContent) {
return false; return false;
} }
@ -1397,7 +1575,7 @@ export class SettingsNavigationDialog extends NavigationDialog {
$header = $tabContent.querySelector('h2'); $header = $tabContent.querySelector('h2');
} else { } else {
// Find the parent element // Find the parent element
const $parent = $focusing.closest('[data-tab-group] > *') as HTMLElement; const $parent = $focusing.closest<HTMLElement>('[data-tab-group] > *');
const siblingProperty = direction === 'next' ? 'nextSibling' : 'previousSibling'; const siblingProperty = direction === 'next' ? 'nextSibling' : 'previousSibling';
let $tmp = $parent; let $tmp = $parent;

View File

@ -1,17 +1,15 @@
import { BxLogger } from "@/utils/bx-logger";
import { CE } from "@/utils/html"; import { CE } from "@/utils/html";
export class FullscreenText { export class FullscreenText {
private static instance: FullscreenText; private static instance: FullscreenText;
public static getInstance(): FullscreenText { public static getInstance = () => FullscreenText.instance ?? (FullscreenText.instance = new FullscreenText());
if (!FullscreenText.instance) { private readonly LOG_TAG = 'FullscreenText';
FullscreenText.instance = new FullscreenText();
}
return FullscreenText.instance;
}
$text: HTMLElement; $text: HTMLElement;
constructor() { private constructor() {
BxLogger.info(this.LOG_TAG, 'constructor()');
this.$text = CE('div', { this.$text = CE('div', {
class: 'bx-fullscreen-text bx-gone', class: 'bx-fullscreen-text bx-gone',
}); });

View File

@ -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);
} }

View File

@ -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";
@ -11,91 +13,104 @@ export enum GuideMenuTab {
} }
export class GuideMenu { export class GuideMenu {
static #BUTTONS = { private static instance: GuideMenu;
scriptSettings: createButton({ public static getInstance = () => GuideMenu.instance ?? (GuideMenu.instance = new GuideMenu());
label: t('better-xcloud'),
style: ButtonStyle.FULL_WIDTH | ButtonStyle.FOCUSABLE | ButtonStyle.PRIMARY,
onClick: e => {
// Wait until the Guide dialog is closed
window.addEventListener(BxEvent.XCLOUD_DIALOG_DISMISSED, e => {
setTimeout(() => SettingsNavigationDialog.getInstance().show(), 50);
}, {once: true});
// Close all xCloud's dialogs private $renderedButtons?: HTMLElement;
window.BX_EXPOSED.dialogRoutes.closeAll();
},
}),
closeApp: AppInterface && createButton({ closeGuideMenu() {
icon: BxIcon.POWER, if (window.BX_EXPOSED.dialogRoutes) {
label: t('close-app'), window.BX_EXPOSED.dialogRoutes.closeAll();
title: t('close-app'), return;
style: ButtonStyle.FULL_WIDTH | ButtonStyle.FOCUSABLE | ButtonStyle.DANGER, }
onClick: e => {
AppInterface.closeApp();
},
attributes: { // Use alternative method for Lite version
'data-state': 'normal', const $btnClose = document.querySelector<HTMLElement>('#gamepass-dialog-root button[class^=Header-module__closeButton]');
}, $btnClose && $btnClose.click();
}),
reloadPage: createButton({
icon: BxIcon.REFRESH,
label: t('reload-page'),
title: t('reload-page'),
style: ButtonStyle.FULL_WIDTH | ButtonStyle.FOCUSABLE,
onClick: e => {
if (STATES.isPlaying) {
confirm(t('confirm-reload-stream')) && window.location.reload();
} else {
window.location.reload();
}
// Close all xCloud's dialogs
window.BX_EXPOSED.dialogRoutes.closeAll();
},
}),
backToHome: createButton({
icon: BxIcon.HOME,
label: t('back-to-home'),
title: t('back-to-home'),
style: ButtonStyle.FULL_WIDTH | ButtonStyle.FOCUSABLE,
onClick: e => {
confirm(t('back-to-home-confirm')) && (window.location.href = window.location.href.substring(0, 31));
// Close all xCloud's dialogs
window.BX_EXPOSED.dialogRoutes.closeAll();
},
attributes: {
'data-state': 'playing',
},
}),
} }
static #$renderedButtons: HTMLElement; private renderButtons() {
if (this.$renderedButtons) {
static #renderButtons() { return this.$renderedButtons;
if (GuideMenu.#$renderedButtons) {
return GuideMenu.#$renderedButtons;
} }
const buttons = {
scriptSettings: createButton({
label: t('better-xcloud'),
style: ButtonStyle.FULL_WIDTH | ButtonStyle.FOCUSABLE | ButtonStyle.PRIMARY,
onClick: (() => {
// Wait until the Guide dialog is closed
window.addEventListener(BxEvent.XCLOUD_DIALOG_DISMISSED, e => {
setTimeout(() => SettingsNavigationDialog.getInstance().show(), 50);
}, {once: true});
// Close all xCloud's dialogs
this.closeGuideMenu();
}).bind(this),
}),
closeApp: AppInterface && createButton({
icon: BxIcon.POWER,
label: t('close-app'),
title: t('close-app'),
style: ButtonStyle.FULL_WIDTH | ButtonStyle.FOCUSABLE | ButtonStyle.DANGER,
onClick: e => {
AppInterface.closeApp();
},
attributes: {
'data-state': 'normal',
},
}),
reloadPage: createButton({
icon: BxIcon.REFRESH,
label: t('reload-page'),
title: t('reload-page'),
style: ButtonStyle.FULL_WIDTH | ButtonStyle.FOCUSABLE,
onClick: (() => {
// Close all xCloud's dialogs
this.closeGuideMenu();
if (STATES.isPlaying) {
confirm(t('confirm-reload-stream')) && window.location.reload();
} else {
window.location.reload();
}
}).bind(this),
}),
backToHome: createButton({
icon: BxIcon.HOME,
label: t('back-to-home'),
title: t('back-to-home'),
style: ButtonStyle.FULL_WIDTH | ButtonStyle.FOCUSABLE,
onClick: (() => {
// Close all xCloud's dialogs
this.closeGuideMenu();
confirm(t('back-to-home-confirm')) && (window.location.href = window.location.href.substring(0, 31));
}).bind(this),
attributes: {
'data-state': 'playing',
},
}),
};
const buttonsLayout = [
buttons.scriptSettings,
[
buttons.backToHome,
buttons.reloadPage,
buttons.closeApp,
],
];
const $div = CE('div', { const $div = CE('div', {
class: 'bx-guide-home-buttons', class: 'bx-guide-home-buttons',
}); });
const buttons = [ for (const $button of buttonsLayout) {
GuideMenu.#BUTTONS.scriptSettings,
[
TrueAchievements.$button,
GuideMenu.#BUTTONS.backToHome,
GuideMenu.#BUTTONS.reloadPage,
GuideMenu.#BUTTONS.closeApp,
],
];
for (const $button of buttons) {
if (!$button) { if (!$button) {
continue; continue;
} }
@ -111,11 +126,18 @@ export class GuideMenu {
} }
} }
GuideMenu.#$renderedButtons = $div; this.$renderedButtons = $div;
return $div; return $div;
} }
static #injectHome($root: HTMLElement, isPlaying = false) { injectHome($root: HTMLElement, isPlaying = false) {
if (isFullVersion()) {
const $achievementsProgress = $root.querySelector('button[class*=AchievementsButton-module__progressBarContainer]');
if ($achievementsProgress) {
TrueAchievements.getInstance().injectAchievementsProgress($achievementsProgress as HTMLElement);
}
}
// Find the element to add buttons to // Find the element to add buttons to
let $target: HTMLElement | null = null; let $target: HTMLElement | null = null;
if (isPlaying) { if (isPlaying) {
@ -123,7 +145,7 @@ export class GuideMenu {
$target = $root.querySelector('a[class*=QuitGameButton]'); $target = $root.querySelector('a[class*=QuitGameButton]');
// Hide xCloud's Home button // Hide xCloud's Home button
const $btnXcloudHome = $root.querySelector('div[class^=HomeButtonWithDivider]') as HTMLElement; const $btnXcloudHome = $root.querySelector<HTMLElement>('div[class^=HomeButtonWithDivider]');
$btnXcloudHome && ($btnXcloudHome.style.display = 'none'); $btnXcloudHome && ($btnXcloudHome.style.display = 'none');
} else { } else {
// Last divider // Last divider
@ -137,26 +159,42 @@ export class GuideMenu {
return false; return false;
} }
const $buttons = GuideMenu.#renderButtons(); const $buttons = this.renderButtons();
$buttons.dataset.isPlaying = isPlaying.toString(); $buttons.dataset.isPlaying = isPlaying.toString();
$target.insertAdjacentElement('afterend', $buttons); $target.insertAdjacentElement('afterend', $buttons);
} }
static async #onShown(e: Event) { async onShown(e: Event) {
const where = (e as any).where as GuideMenuTab; const where = (e as any).where as GuideMenuTab;
if (where === GuideMenuTab.HOME) { if (where === GuideMenuTab.HOME) {
const $root = document.querySelector('#gamepass-dialog-root div[role=dialog] div[role=tabpanel] div[class*=HomeLandingPage]') as HTMLElement; const $root = document.querySelector<HTMLElement>('#gamepass-dialog-root div[role=dialog] div[role=tabpanel] div[class*=HomeLandingPage]');
$root && GuideMenu.#injectHome($root, STATES.isPlaying); $root && this.injectHome($root, STATES.isPlaying);
} }
} }
static addEventListeners() { addEventListeners() {
window.addEventListener(BxEvent.XCLOUD_GUIDE_MENU_SHOWN, GuideMenu.#onShown); window.addEventListener(BxEvent.XCLOUD_GUIDE_MENU_SHOWN, this.onShown.bind(this));
} }
static observe($addedElm: HTMLElement) { observe($addedElm: HTMLElement) {
const className = $addedElm.className; let className = $addedElm.className;
// Fix custom buttons disappearing in Guide Menu (#551)
if (!className) {
className = $addedElm.firstElementChild?.className ?? '';
}
if (!className || className.startsWith('bx-')) {
return;
}
// TrueAchievements
if (isFullVersion() && className.includes('AchievementsButton-module__progressBarContainer')) {
TrueAchievements.getInstance().injectAchievementsProgress($addedElm);
return;
}
if (!className.startsWith('NavigationAnimation') && if (!className.startsWith('NavigationAnimation') &&
!className.startsWith('DialogRoutes') && !className.startsWith('DialogRoutes') &&
!className.startsWith('Dialog-module__container')) { !className.startsWith('Dialog-module__container')) {
@ -164,10 +202,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.getInstance().injectAchievementDetailPage($achievDetailPage as HTMLElement);
return;
}
} }
// Find navigation bar // Find navigation bar

View File

@ -2,41 +2,50 @@ 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";
import { getPref } from "@/utils/settings-storages/global-settings-storage"; import { getPref } from "@/utils/settings-storages/global-settings-storage";
import { BxLogger } from "@/utils/bx-logger";
export class HeaderSection { export class HeaderSection {
static #$remotePlayBtn = createButton({ private static instance: HeaderSection;
classes: ['bx-header-remote-play-button', 'bx-gone'], public static getInstance = () => HeaderSection.instance ?? (HeaderSection.instance = new HeaderSection());
icon: BxIcon.REMOTE_PLAY, private readonly LOG_TAG = 'HeaderSection';
title: t('remote-play'),
style: ButtonStyle.GHOST | ButtonStyle.FOCUSABLE | ButtonStyle.CIRCULAR,
onClick: e => {
RemotePlay.togglePopup();
},
});
static #$settingsBtn = createButton({ private $btnRemotePlay: HTMLElement;
classes: ['bx-header-settings-button'], private $btnSettings: HTMLElement;
label: '???', private $buttonsWrapper: HTMLElement;
style: ButtonStyle.FROSTED | ButtonStyle.DROP_SHADOW | ButtonStyle.FOCUSABLE | ButtonStyle.FULL_HEIGHT,
onClick: e => {
SettingsNavigationDialog.getInstance().show();
},
});
static #$buttonsWrapper = CE('div', {}, private observer?: MutationObserver;
getPref(PrefKey.REMOTE_PLAY_ENABLED) ? HeaderSection.#$remotePlayBtn : null, private timeoutId?: number | null;
HeaderSection.#$settingsBtn,
);
static #observer: MutationObserver; constructor() {
static #timeout: number | null; BxLogger.info(this.LOG_TAG, 'constructor()');
static #injectSettingsButton($parent?: HTMLElement) { this.$btnRemotePlay = createButton({
classes: ['bx-header-remote-play-button', 'bx-gone'],
icon: BxIcon.REMOTE_PLAY,
title: t('remote-play'),
style: ButtonStyle.GHOST | ButtonStyle.FOCUSABLE | ButtonStyle.CIRCULAR,
onClick: e => RemotePlayManager.getInstance().togglePopup(),
});
this.$btnSettings = createButton({
classes: ['bx-header-settings-button'],
label: '???',
style: ButtonStyle.FROSTED | ButtonStyle.DROP_SHADOW | ButtonStyle.FOCUSABLE | ButtonStyle.FULL_HEIGHT,
onClick: e => SettingsNavigationDialog.getInstance().show(),
});
this.$buttonsWrapper = CE('div', {},
getPref(PrefKey.REMOTE_PLAY_ENABLED) ? this.$btnRemotePlay : null,
this.$btnSettings,
);
}
private injectSettingsButton($parent?: HTMLElement) {
if (!$parent) { if (!$parent) {
return; return;
} }
@ -44,8 +53,8 @@ export class HeaderSection {
const PREF_LATEST_VERSION = getPref(PrefKey.LATEST_VERSION); const PREF_LATEST_VERSION = getPref(PrefKey.LATEST_VERSION);
// Setup Settings button // Setup Settings button
const $btnSettings = HeaderSection.#$settingsBtn; const $btnSettings = this.$btnSettings;
if (isElementVisible(HeaderSection.#$buttonsWrapper)) { if (isElementVisible(this.$buttonsWrapper)) {
return; return;
} }
@ -57,38 +66,42 @@ export class HeaderSection {
} }
// Add the Settings button to the web page // Add the Settings button to the web page
$parent.appendChild(HeaderSection.#$buttonsWrapper); $parent.appendChild(this.$buttonsWrapper);
} }
static checkHeader() { private 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 && this.injectSettingsButton($target as HTMLElement);
} }
static showRemotePlayButton() { private watchHeader() {
HeaderSection.#$remotePlayBtn.classList.remove('bx-gone');
}
static watchHeader() {
const $root = document.querySelector('#PageContent header') || document.querySelector('#root'); const $root = document.querySelector('#PageContent header') || document.querySelector('#root');
if (!$root) { if (!$root) {
return; return;
} }
HeaderSection.#timeout && clearTimeout(HeaderSection.#timeout); this.timeoutId && clearTimeout(this.timeoutId);
HeaderSection.#timeout = null; this.timeoutId = null;
HeaderSection.#observer && HeaderSection.#observer.disconnect(); this.observer && this.observer.disconnect();
HeaderSection.#observer = new MutationObserver(mutationList => { this.observer = new MutationObserver(mutationList => {
HeaderSection.#timeout && clearTimeout(HeaderSection.#timeout); this.timeoutId && clearTimeout(this.timeoutId);
HeaderSection.#timeout = window.setTimeout(HeaderSection.checkHeader, 2000); this.timeoutId = window.setTimeout(this.checkHeader.bind(this), 2000);
}); });
HeaderSection.#observer.observe($root, {subtree: true, childList: true}); this.observer.observe($root, {subtree: true, childList: true});
HeaderSection.checkHeader(); this.checkHeader();
}
showRemotePlayButton() {
this.$btnRemotePlay.classList.remove('bx-gone');
}
static watchHeader() {
HeaderSection.getInstance().watchHeader();
} }
} }

View File

@ -1,12 +1,12 @@
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";
import { parseDetailsPath } from "@/utils/utils";
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,22 +17,13 @@ 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,
tabIndex: 0, tabIndex: 0,
onClick: async e => { onClick: e => {
try { const details = parseDetailsPath(window.location.pathname);
const matches = /\/games\/(?<titleSlug>[^\/]+)\/(?<productId>\w+)/.exec(window.location.pathname); details && AppInterface.downloadWallpapers(details.titleSlug, details.productId);
if (!matches?.groups) {
return;
}
const titleSlug = matches.groups.titleSlug.replaceAll('\%' + '7C', '-');
const productId = matches.groups.productId;
AppInterface.downloadWallpapers(titleSlug, productId);
} catch (e) {}
}, },
}); });
@ -48,17 +39,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);
} }

23
src/types/index.d.ts vendored
View File

@ -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;
@ -10,6 +12,7 @@ interface Window {
BX_EXPOSED: any; BX_EXPOSED: any;
BX_VIBRATION_INTENSITY: number; BX_VIBRATION_INTENSITY: number;
BX_CONTROLLER_POLLING_RATE: number;
BX_ENABLE_CONTROLLER_VIBRATION: boolean; BX_ENABLE_CONTROLLER_VIBRATION: boolean;
BX_ENABLE_DEVICE_VIBRATION: boolean; BX_ENABLE_DEVICE_VIBRATION: boolean;
@ -18,20 +21,29 @@ interface Window {
interface NavigatorBattery extends Navigator { interface NavigatorBattery extends Navigator {
getBattery: () => Promise<{ getBattery: () => Promise<{
charging: boolean, charging: boolean;
level: float, level: float;
}>, }>,
} }
type ServerContinent = 'america-north' | 'america-south' | 'asia' | 'australia' | 'europe' | 'other';
type ServerRegion = {
baseUri: string;
isDefault: boolean;
name: string;
shortName: string;
contintent: ServerContinent;
};
type BxStates = { type BxStates = {
supportedRegion: boolean; supportedRegion: boolean;
serverRegions: any; serverRegions: Record<string, ServerRegion>;
selectedRegion: any; selectedRegion: any;
gsToken: string; gsToken: string;
isSignedIn: boolean; isSignedIn: boolean;
isPlaying: boolean; isPlaying: boolean;
appContext: any | null;
browser: { browser: {
capabilities: { capabilities: {
@ -44,6 +56,7 @@ type BxStates = {
isTv: boolean; isTv: boolean;
capabilities: { capabilities: {
touch: boolean; touch: boolean;
mkb: boolean;
}; };
}; };

View File

@ -3,8 +3,9 @@ export type PreferenceSetting = {
optionsGroup?: string; optionsGroup?: string;
options?: {[index: string]: string}; options?: {[index: string]: string};
multipleOptions?: {[index: string]: string}; multipleOptions?: {[index: string]: string};
unsupported?: string | boolean; unsupported?: boolean;
note?: string | HTMLElement; unsupportedNote?: string | (() => HTMLElement);
note?: string | (() => HTMLElement);
type?: SettingElementType; type?: SettingElementType;
ready?: (setting: PreferenceSetting) => void; ready?: (setting: PreferenceSetting) => void;
migrate?: (this: Preferences, savedPrefs: any, value: any) => void; migrate?: (this: Preferences, savedPrefs: any, value: any) => void;

View File

@ -18,12 +18,14 @@ export type SettingDefinition = {
default: any; default: any;
} & Partial<{ } & Partial<{
label: string; label: string;
note: string | HTMLElement; note: string | (() => HTMLElement);
experimental: boolean; experimental: boolean;
unsupported: string | boolean; unsupported: boolean;
unsupportedNote: string | (() => HTMLElement);
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;
}> & ( }> & (
{} | { {} | {
@ -57,4 +59,5 @@ export type NumberStepperParams = Partial<{
exactTicks: number; exactTicks: number;
customTextValue: (value: any) => string | null; customTextValue: (value: any) => string | null;
reverse: boolean;
}> }>

View File

@ -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,

View File

@ -37,6 +37,7 @@ export namespace BxEvent {
export const GAME_BAR_ACTION_ACTIVATED = 'bx-game-bar-action-activated'; export const GAME_BAR_ACTION_ACTIVATED = 'bx-game-bar-action-activated';
export const MICROPHONE_STATE_CHANGED = 'bx-microphone-state-changed'; export const MICROPHONE_STATE_CHANGED = 'bx-microphone-state-changed';
export const SPEAKER_STATE_CHANGED = 'bx-speaker-state-changed';
export const CAPTURE_SCREENSHOT = 'bx-capture-screenshot'; export const CAPTURE_SCREENSHOT = 'bx-capture-screenshot';

View File

@ -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";
@ -6,6 +8,8 @@ import { BX_FLAGS } from "./bx-flags";
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 "./settings-storages/global-settings-storage"; import { getPref, StreamTouchController } from "./settings-storages/global-settings-storage";
import { GamePassCloudGallery } from "@/enums/game-pass-gallery";
import { TouchController } from "@/modules/touch-controller";
export enum SupportedInputType { export enum SupportedInputType {
CONTROLLER = 'Controller', CONTROLLER = 'Controller',
@ -20,7 +24,61 @@ 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 => { modifyPreloadedState: isFullVersion() && ((state: any) => {
let LOG_TAG = 'PreloadState';
// Override User-Agent
try {
state.appContext.requestInfo.userAgent = window.navigator.userAgent;
} catch (e) {
BxLogger.error(LOG_TAG, e);
}
// Add list of games with custom layouts to the official list
try {
const sigls = state.xcloud.sigls;
if (STATES.userAgent.capabilities.touch) {
// The list of custom touch controls
let customList = TouchController.getCustomList();
// Remove non-cloud games from the official list
const allGames = sigls[GamePassCloudGallery.ALL].data.products;
customList = customList.filter(id => allGames.includes(id));
// Add to the official touchlist
sigls[GamePassCloudGallery.TOUCH]?.data.products.push(...customList);
}
} catch (e) {
BxLogger.error(LOG_TAG, e);
}
// Add forced Native MKB titles to the official list
try {
const sigls = state.xcloud.sigls;
if (BX_FLAGS.ForceNativeMkbTitles) {
// Add to the official list
sigls[GamePassCloudGallery.NATIVE_MKB]?.data.products.push(...BX_FLAGS.ForceNativeMkbTitles);
}
} catch (e) {
BxLogger.error(LOG_TAG, e);
}
// Redirect to /en-US/play if visiting from an unsupported region
try {
const xCloud = state.xcloud.authentication.authStatusByStrategy.XCloud;
if (xCloud.type === 3 && xCloud.error.type === 'UnsupportedMarketError') {
// Redirect to /en-US/play
window.stop();
window.location.href = 'https://www.xbox.com/en-US/play';
}
} catch (e) {
BxLogger.error(LOG_TAG, e);
}
return state;
}),
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 +168,8 @@ export const BxExposed = {
} }
}, },
handleControllerShortcut: ControllerShortcut.handle, handleControllerShortcut: isFullVersion() && ControllerShortcut.handle,
resetControllerShortcut: ControllerShortcut.reset, resetControllerShortcut: isFullVersion() && ControllerShortcut.reset,
overrideSettings: { overrideSettings: {
'Tv_settings': { 'Tv_settings': {
@ -142,4 +200,10 @@ export const BxExposed = {
return false; return false;
}, },
GameSlugRegexes: [
/[;,/?:@&=+_`~$%#^*()!^\u2122\xae\xa9]/g,
/ {2,}/g,
/ /g,
],
}; };

View File

@ -1,6 +1,6 @@
import { BxLogger } from "./bx-logger"; import { BxLogger } from "./bx-logger";
type BxFlags = { export type BxFlags = {
Debug: boolean; Debug: boolean;
CheckForUpdate: boolean; CheckForUpdate: boolean;
@ -15,7 +15,10 @@ type BxFlags = {
userAgent?: string, userAgent?: string,
androidInfo?: { androidInfo?: {
manufacturer: string,
brand: string,
board: string, board: string,
model: string,
}, },
} }
} }

View File

@ -1,3 +1,4 @@
// Credit: https://phosphoricons.com
import iconBetterXcloud from "@assets/svg/better-xcloud.svg" with { type: "text" }; import iconBetterXcloud from "@assets/svg/better-xcloud.svg" with { type: "text" };
import iconTrueAchievements from "@assets/svg/true-achievements.svg" with { type: "text" }; import iconTrueAchievements from "@assets/svg/true-achievements.svg" with { type: "text" };
import iconClose from "@assets/svg/close.svg" with { type: "text" }; import iconClose from "@assets/svg/close.svg" with { type: "text" };
@ -7,6 +8,8 @@ import iconCopy from "@assets/svg/copy.svg" with { type: "text" };
import iconCreateShortcut from "@assets/svg/create-shortcut.svg" with { type: "text" }; import iconCreateShortcut from "@assets/svg/create-shortcut.svg" with { type: "text" };
import iconCursorText from "@assets/svg/cursor-text.svg" with { type: "text" }; import iconCursorText from "@assets/svg/cursor-text.svg" with { type: "text" };
import iconDisplay from "@assets/svg/display.svg" with { type: "text" }; import iconDisplay from "@assets/svg/display.svg" with { type: "text" };
import iconEye from "@assets/svg/eye.svg" with { type: "text" };
import iconEyeSlash from "@assets/svg/eye-slash.svg" with { type: "text" };
import iconHome from "@assets/svg/home.svg" with { type: "text" }; import iconHome from "@assets/svg/home.svg" with { type: "text" };
import iconNativeMkb from "@assets/svg/native-mkb.svg" with { type: "text" }; import iconNativeMkb from "@assets/svg/native-mkb.svg" with { type: "text" };
import iconNew from "@assets/svg/new.svg" with { type: "text" }; import iconNew from "@assets/svg/new.svg" with { type: "text" };
@ -14,6 +17,7 @@ import iconPower from "@assets/svg/power.svg" with { type: "text" };
import iconQuestion from "@assets/svg/question.svg" with { type: "text" }; import iconQuestion from "@assets/svg/question.svg" with { type: "text" };
import iconRefresh from "@assets/svg/refresh.svg" with { type: "text" }; import iconRefresh from "@assets/svg/refresh.svg" with { type: "text" };
import iconRemotePlay from "@assets/svg/remote-play.svg" with { type: "text" }; import iconRemotePlay from "@assets/svg/remote-play.svg" with { type: "text" };
import iconSpeakerSlash from "@assets/svg/speaker-slash.svg" with { type: "text" };
import iconStreamSettings from "@assets/svg/stream-settings.svg" with { type: "text" }; import iconStreamSettings from "@assets/svg/stream-settings.svg" with { type: "text" };
import iconStreamStats from "@assets/svg/stream-stats.svg" with { type: "text" }; import iconStreamStats from "@assets/svg/stream-stats.svg" with { type: "text" };
import iconTouchControlDisable from "@assets/svg/touch-control-disable.svg" with { type: "text" }; import iconTouchControlDisable from "@assets/svg/touch-control-disable.svg" with { type: "text" };
@ -47,6 +51,8 @@ export const BxIcon = {
CONTROLLER: iconController, CONTROLLER: iconController,
CREATE_SHORTCUT: iconCreateShortcut, CREATE_SHORTCUT: iconCreateShortcut,
DISPLAY: iconDisplay, DISPLAY: iconDisplay,
EYE: iconEye,
EYE_SLASH: iconEyeSlash,
HOME: iconHome, HOME: iconHome,
NATIVE_MKB: iconNativeMkb, NATIVE_MKB: iconNativeMkb,
NEW: iconNew, NEW: iconNew,
@ -64,6 +70,7 @@ export const BxIcon = {
CARET_LEFT: iconCaretLeft, CARET_LEFT: iconCaretLeft,
CARET_RIGHT: iconCaretRight, CARET_RIGHT: iconCaretRight,
SCREENSHOT: iconCamera, SCREENSHOT: iconCamera,
SPEAKER_MUTED: iconSpeakerSlash,
TOUCH_CONTROL_ENABLE: iconTouchControlEnable, TOUCH_CONTROL_ENABLE: iconTouchControlEnable,
TOUCH_CONTROL_DISABLE: iconTouchControlDisable, TOUCH_CONTROL_DISABLE: iconTouchControlDisable,

View File

@ -1,3 +1,5 @@
import { BX_FLAGS } from "./bx-flags";
const enum TextColor { const enum TextColor {
INFO = '#008746', INFO = '#008746',
WARNING = '#c1a404', WARNING = '#c1a404',
@ -5,22 +7,12 @@ const enum TextColor {
} }
export class BxLogger { export class BxLogger {
static #PREFIX = '[BxC]'; static info = (tag: string, ...args: any[]) => BxLogger.log(TextColor.INFO, tag, ...args);
static warning = (tag: string, ...args: any[]) => BxLogger.log(TextColor.WARNING, tag, ...args);
static error = (tag: string, ...args: any[]) => BxLogger.log(TextColor.ERROR, tag, ...args);
static info(tag: string, ...args: any[]) { private static log(color: string, tag: string, ...args: any) {
BxLogger.#log(TextColor.INFO, tag, ...args); BX_FLAGS.Debug && console.log(`%c[BxC]`, `color:${color};font-weight:bold;`, tag, '//', ...args);
}
static warning(tag: string, ...args: any[]) {
BxLogger.#log(TextColor.WARNING, tag, ...args);
}
static error(tag: string, ...args: any[]) {
BxLogger.#log(TextColor.ERROR, tag, ...args);
}
static #log(color: string, tag: string, ...args: any) {
console.log(`%c${BxLogger.#PREFIX}`, `color:${color};font-weight:bold;`, tag, '//', ...args);
} }
} }

View File

@ -9,11 +9,6 @@ export let FeatureGates: {[key: string]: boolean} = {
'ShowForcedUpdateScreen': false, 'ShowForcedUpdateScreen': false,
}; };
// Disable context menu in Home page
if (getPref(PrefKey.UI_HOME_CONTEXT_MENU_DISABLED)) {
FeatureGates['EnableHomeContextMenu'] = false;
}
// Disable chat feature // Disable chat feature
if (getPref(PrefKey.BLOCK_SOCIAL_FEATURES)) { if (getPref(PrefKey.BLOCK_SOCIAL_FEATURES)) {
FeatureGates['EnableGuideChatTab'] = false; FeatureGates['EnableGuideChatTab'] = false;

View File

@ -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;
} }
@ -33,3 +33,7 @@ export function showGamepadToast(gamepad: Gamepad) {
Toast.show(text, status, {instant: false}); Toast.show(text, status, {instant: false});
} }
export function updatePollingRate() {
window.BX_CONTROLLER_POLLING_RATE = getPref(PrefKey.CONTROLLER_POLLING_RATE);
}

View File

@ -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;
@ -12,6 +13,7 @@ const isTv = userAgent.includes('smart-tv') || userAgent.includes('smarttv') ||
const isVr = window.navigator.userAgent.includes('VR') && window.navigator.userAgent.includes('OculusBrowser'); const isVr = window.navigator.userAgent.includes('VR') && window.navigator.userAgent.includes('OculusBrowser');
const browserHasTouchSupport = 'ontouchstart' in window || navigator.maxTouchPoints > 0; const browserHasTouchSupport = 'ontouchstart' in window || navigator.maxTouchPoints > 0;
const userAgentHasTouchSupport = !isTv && !isVr && browserHasTouchSupport; const userAgentHasTouchSupport = !isTv && !isVr && browserHasTouchSupport;
const supportMkb = AppInterface || !userAgent.match(/(android|iphone|ipad)/);
export const STATES: BxStates = { export const STATES: BxStates = {
supportedRegion: true, supportedRegion: true,
@ -21,7 +23,6 @@ export const STATES: BxStates = {
isSignedIn: false, isSignedIn: false,
isPlaying: false, isPlaying: false,
appContext: {},
browser: { browser: {
capabilities: { capabilities: {
@ -34,6 +35,7 @@ export const STATES: BxStates = {
isTv: isTv, isTv: isTv,
capabilities: { capabilities: {
touch: userAgentHasTouchSupport, touch: userAgentHasTouchSupport,
mkb: supportMkb,
} }
}, },

View File

@ -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);

View File

@ -32,7 +32,7 @@ const ButtonStyleClass = {
[ButtonStyle.NORMAL_LINK]: 'bx-normal-link', [ButtonStyle.NORMAL_LINK]: 'bx-normal-link',
} }
type BxButton = { export type BxButton = {
style?: ButtonStyle; style?: ButtonStyle;
url?: string; url?: string;
classes?: string[]; classes?: string[];
@ -56,6 +56,8 @@ function createElement<T=HTMLElement>(elmName: string, props: CreateElementOptio
let $elm; let $elm;
const hasNs = 'xmlns' in props; const hasNs = 'xmlns' in props;
// console.trace('createElement', elmName, props);
if (hasNs) { if (hasNs) {
$elm = document.createElementNS(props.xmlns, elmName); $elm = document.createElementNS(props.xmlns, elmName);
delete props.xmlns; delete props.xmlns;
@ -101,29 +103,30 @@ function createElement<T=HTMLElement>(elmName: string, props: CreateElementOptio
export const CE = createElement; export const CE = createElement;
// Credit: https://phosphoricons.com const domParser = new DOMParser();
const svgParser = (svg: string) => new DOMParser().parseFromString(svg, 'image/svg+xml').documentElement; export function createSvgIcon(icon: typeof BxIcon) {
return domParser.parseFromString(icon.toString(), 'image/svg+xml').documentElement;
export const createSvgIcon = (icon: typeof BxIcon) => {
return svgParser(icon.toString());
} }
const ButtonStyleIndices = Object.keys(ButtonStyleClass).map(i => parseInt(i)); const ButtonStyleIndices = Object.keys(ButtonStyleClass).map(i => parseInt(i));
export const createButton = <T=HTMLButtonElement>(options: BxButton): T => { export function createButton<T=HTMLButtonElement>(options: BxButton): T {
let $btn; let $btn;
if (options.url) { if (options.url) {
$btn = CE('a', {'class': 'bx-button'}) as HTMLAnchorElement; $btn = CE<HTMLAnchorElement>('a', {'class': 'bx-button'});
$btn.href = options.url; $btn.href = options.url;
$btn.target = '_blank'; $btn.target = '_blank';
} else { } else {
$btn = CE('button', {'class': 'bx-button', type: 'button'}) as HTMLButtonElement; $btn = CE<HTMLButtonElement>('button', {'class': 'bx-button', type: 'button'});
} }
const style = (options.style || 0) as number; const style = (options.style || 0) as number;
style && ButtonStyleIndices.forEach((index: keyof typeof ButtonStyleClass) => { if (style) {
let index: keyof typeof ButtonStyleClass;
for (index of ButtonStyleIndices) {
(style & index) && $btn.classList.add(ButtonStyleClass[index] as string); (style & index) && $btn.classList.add(ButtonStyleClass[index] as string);
}); }
}
options.classes && $btn.classList.add(...options.classes); options.classes && $btn.classList.add(...options.classes);
@ -163,7 +166,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);
@ -174,3 +177,54 @@ export function removeChildElements($parent: HTMLElement) {
$parent.firstElementChild.remove(); $parent.firstElementChild.remove();
} }
} }
export function clearFocus() {
if (document.activeElement instanceof HTMLElement) {
document.activeElement.blur();
}
}
export function clearDataSet($elm: HTMLElement) {
Object.keys($elm.dataset).forEach(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(' ');
}

View File

@ -1,165 +0,0 @@
import { MkbPreset } from "@modules/mkb/mkb-preset";
import { t } from "@utils/translation";
import type { MkbStoredPreset, MkbStoredPresets } from "@/types/mkb";
import { PrefKey } from "@/enums/pref-keys";
import { setPref } from "./settings-storages/global-settings-storage";
export class LocalDb {
static #instance: LocalDb;
static get INSTANCE() {
if (!LocalDb.#instance) {
LocalDb.#instance = new LocalDb();
}
return LocalDb.#instance;
}
static readonly DB_NAME = 'BetterXcloud';
static readonly DB_VERSION = 1;
static readonly TABLE_PRESETS = 'mkb_presets';
#DB: any;
#open() {
return new Promise<void>((resolve, reject) => {
if (this.#DB) {
resolve();
return;
}
const request = window.indexedDB.open(LocalDb.DB_NAME, LocalDb.DB_VERSION);
request.onupgradeneeded = (e: IDBVersionChangeEvent) => {
const db = (e.target! as any).result;
switch (e.oldVersion) {
case 0: {
const presets = db.createObjectStore(LocalDb.TABLE_PRESETS, {keyPath: 'id', autoIncrement: true});
presets.createIndex('name_idx', 'name');
break;
}
}
};
request.onerror = e => {
console.log(e);
alert((e.target as any).error.message);
reject && reject();
};
request.onsuccess = e => {
this.#DB = (e.target as any).result;
resolve();
};
});
}
#table(name: string, type: string): Promise<IDBObjectStore> {
const transaction = this.#DB.transaction(name, type || 'readonly');
const table = transaction.objectStore(name);
return new Promise(resolve => resolve(table));
}
// Convert IndexDB method to Promise
#call(method: any) {
const table = arguments[1];
return new Promise(resolve => {
const request = method.call(table, ...Array.from(arguments).slice(2));
request.onsuccess = (e: Event) => {
resolve([table, (e.target as any).result]);
};
});
}
#count(table: IDBObjectStore): Promise<[IDBObjectStore, number]> {
// @ts-ignore
return this.#call(table.count, ...arguments);
}
#add(table: IDBObjectStore, data: any): Promise<[IDBObjectStore, number]> {
// @ts-ignore
return this.#call(table.add, ...arguments);
}
#put(table: IDBObjectStore, data: any): Promise<[IDBObjectStore, number]> {
// @ts-ignore
return this.#call(table.put, ...arguments);
}
#delete(table: IDBObjectStore, data: any): Promise<[IDBObjectStore, number]> {
// @ts-ignore
return this.#call(table.delete, ...arguments);
}
#get(table: IDBObjectStore, id: number): Promise<any> {
// @ts-ignore
return this.#call(table.get, ...arguments);
}
#getAll(table: IDBObjectStore): Promise<[IDBObjectStore, any]> {
// @ts-ignore
return this.#call(table.getAll, ...arguments);
}
newPreset(name: string, data: any) {
return this.#open()
.then(() => this.#table(LocalDb.TABLE_PRESETS, 'readwrite'))
.then(table => this.#add(table, {name, data}))
.then(([table, id]) => new Promise<number>(resolve => resolve(id)));
}
updatePreset(preset: MkbStoredPreset) {
return this.#open()
.then(() => this.#table(LocalDb.TABLE_PRESETS, 'readwrite'))
.then(table => this.#put(table, preset))
.then(([table, id]) => new Promise(resolve => resolve(id)));
}
deletePreset(id: number) {
return this.#open()
.then(() => this.#table(LocalDb.TABLE_PRESETS, 'readwrite'))
.then(table => this.#delete(table, id))
.then(([table, id]) => new Promise(resolve => resolve(id)));
}
getPreset(id: number): Promise<MkbStoredPreset> {
return this.#open()
.then(() => this.#table(LocalDb.TABLE_PRESETS, 'readwrite'))
.then(table => this.#get(table, id))
.then(([table, preset]) => new Promise(resolve => resolve(preset)));
}
getPresets(): Promise<MkbStoredPresets> {
return this.#open()
.then(() => this.#table(LocalDb.TABLE_PRESETS, 'readwrite'))
.then(table => this.#count(table))
.then(([table, count]) => {
if (count > 0) {
return new Promise(resolve => {
this.#getAll(table)
.then(([table, items]) => {
const presets: MkbStoredPresets = {};
items.forEach((item: MkbStoredPreset) => (presets[item.id!] = item));
resolve(presets);
});
});
}
// Create "Default" preset when the table is empty
const preset: MkbStoredPreset = {
name: t('default'),
data: MkbPreset.DEFAULT_PRESET,
}
return new Promise<MkbStoredPresets>(resolve => {
this.#add(table, preset)
.then(([table, id]) => {
preset.id = id;
setPref(PrefKey.MKB_DEFAULT_PRESET_ID, id);
resolve({[id]: preset});
});
});
});
}
}

View File

@ -0,0 +1,79 @@
export abstract class LocalDb {
static readonly DB_NAME = 'BetterXcloud';
static readonly DB_VERSION = 2;
protected db!: IDBDatabase;
protected open() {
return new Promise<void>((resolve, reject) => {
if (this.db) {
resolve();
return;
}
const request = window.indexedDB.open(LocalDb.DB_NAME, LocalDb.DB_VERSION);
request.onupgradeneeded = this.onUpgradeNeeded.bind(this);
request.onerror = e => {
console.log(e);
alert((e.target as any).error.message);
reject && reject();
};
request.onsuccess = e => {
this.db = (e.target as any).result;
resolve();
};
});
}
protected abstract onUpgradeNeeded(e: IDBVersionChangeEvent): void;
protected table(name: string, type: IDBTransactionMode): Promise<IDBObjectStore> {
const transaction = this.db.transaction(name, type || 'readonly');
const table = transaction.objectStore(name);
return new Promise(resolve => resolve(table));
}
// Convert IndexDB method to Promise
protected call(method: any) {
const table = arguments[1];
return new Promise(resolve => {
const request = method.call(table, ...Array.from(arguments).slice(2));
request.onsuccess = (e: Event) => {
resolve([table, (e.target as any).result]);
};
});
}
protected count(table: IDBObjectStore): Promise<[IDBObjectStore, number]> {
// @ts-ignore
return this.call(table.count, ...arguments);
}
protected add(table: IDBObjectStore, data: any): Promise<[IDBObjectStore, number]> {
// @ts-ignore
return this.call(table.add, ...arguments);
}
protected put(table: IDBObjectStore, data: any): Promise<[IDBObjectStore, number]> {
// @ts-ignore
return this.call(table.put, ...arguments);
}
protected delete(table: IDBObjectStore, data: any): Promise<[IDBObjectStore, number]> {
// @ts-ignore
return this.call(table.delete, ...arguments);
}
protected get(table: IDBObjectStore, id: number): Promise<any> {
// @ts-ignore
return this.call(table.get, ...arguments);
}
protected getAll(table: IDBObjectStore): Promise<[IDBObjectStore, any]> {
// @ts-ignore
return this.call(table.getAll, ...arguments);
}
}

View File

@ -0,0 +1,102 @@
import { PrefKey } from "@/enums/pref-keys";
import { MkbPreset } from "@/modules/mkb/mkb-preset";
import type { MkbStoredPreset, MkbStoredPresets } from "@/types/mkb";
import { setPref } from "../settings-storages/global-settings-storage";
import { t } from "../translation";
import { LocalDb } from "./local-db";
import { BxLogger } from "../bx-logger";
export class MkbPresetsDb extends LocalDb {
private static instance: MkbPresetsDb;
public static getInstance = () => MkbPresetsDb.instance ?? (MkbPresetsDb.instance = new MkbPresetsDb());
private readonly LOG_TAG = 'MkbPresetsDb';
private readonly TABLE_PRESETS = 'mkb_presets';
private constructor() {
super();
BxLogger.info(this.LOG_TAG, 'constructor()');
}
private createTable(db: IDBDatabase) {
const presets = db.createObjectStore(this.TABLE_PRESETS, {
keyPath: 'id',
autoIncrement: true,
});
presets.createIndex('name_idx', 'name');
}
protected onUpgradeNeeded(e: IDBVersionChangeEvent): void {
const db = (e.target! as any).result as IDBDatabase;
if (db.objectStoreNames.contains('undefined')) {
db.deleteObjectStore('undefined');
}
if (!db.objectStoreNames.contains(this.TABLE_PRESETS)) {
this.createTable(db);
}
}
private async presetsTable() {
await this.open();
return await this.table(this.TABLE_PRESETS, 'readwrite');
}
async newPreset(name: string, data: any) {
const table = await this.presetsTable();
const [, id] = await this.add(table, { name, data });
return id;
}
async updatePreset(preset: MkbStoredPreset) {
const table = await this.presetsTable();
const [, id] = await this.put(table, preset);
return id;
}
async deletePreset(id: number) {
const table = await this.presetsTable();
await this.delete(table, id);
return id;
}
async getPreset(id: number): Promise<MkbStoredPreset> {
const table = await this.presetsTable();
const [, preset] = await this.get(table, id);
return preset;
}
async getPresets(): Promise<MkbStoredPresets> {
const table = await this.presetsTable();
const [, count] = await this.count(table);
// Return stored presets
if (count > 0) {
const [, items] = await this.getAll(table);
const presets: MkbStoredPresets = {};
items.forEach((item: MkbStoredPreset) => (presets[item.id!] = item));
return presets;
}
// Create "Default" preset when the table is empty
const preset: MkbStoredPreset = {
name: t('default'),
data: MkbPreset.DEFAULT_PRESET,
};
const [, id] = await this.add(table, preset);
preset.id = id;
setPref(PrefKey.MKB_DEFAULT_PRESET_ID, id);
return {
[id]: preset,
};
}
}

View File

@ -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));

View File

@ -1,4 +1,5 @@
import { BxEvent } from "@utils/bx-event"; import { isFullVersion } from "@macros/build" with {type: "macro"};
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";
import { STATES } from "@utils/global"; import { STATES } from "@utils/global";
@ -27,9 +28,7 @@ function clearDbLogs(dbName: string, table: string) {
const objectStore = db.transaction(table, 'readwrite').objectStore(table); const objectStore = db.transaction(table, 'readwrite').objectStore(table);
const objectStoreRequest = objectStore.clear(); const objectStoreRequest = objectStore.clear();
objectStoreRequest.onsuccess = function() { objectStoreRequest.onsuccess = () => BxLogger.info('clearDbLogs', `Cleared ${dbName}.${table}`);
console.log(`[Better xCloud] Cleared ${dbName}.${table}`);
};
} catch (ex) {} } catch (ex) {}
} }
} }
@ -132,6 +131,7 @@ export function interceptHttpRequests() {
'https://browser.events.data.microsoft.com', 'https://browser.events.data.microsoft.com',
'https://dc.services.visualstudio.com', 'https://dc.services.visualstudio.com',
'https://2c06dea3f26c40c69b8456d319791fd0@o427368.ingest.sentry.io', 'https://2c06dea3f26c40c69b8456d319791fd0@o427368.ingest.sentry.io',
'https://mscom.demdex.net',
]); ]);
} }
@ -170,29 +170,42 @@ export function interceptHttpRequests() {
}; };
let gamepassAllGames: string[] = []; let gamepassAllGames: string[] = [];
const IGNORED_DOMAINS = [
'accounts.xboxlive.com',
'chat.xboxlive.com',
'notificationinbox.xboxlive.com',
'peoplehub.xboxlive.com',
'rta.xboxlive.com',
'userpresence.xboxlive.com',
'xblmessaging.xboxlive.com',
'consent.config.office.com',
'arc.msn.com',
'browser.events.data.microsoft.com',
'dc.services.visualstudio.com',
'2c06dea3f26c40c69b8456d319791fd0@o427368.ingest.sentry.io',
];
(window as any).BX_FETCH = window.fetch = async (request: RequestInfo | URL, init?: RequestInit): Promise<Response> => { (window as any).BX_FETCH = window.fetch = async (request: RequestInfo | URL, init?: RequestInit): Promise<Response> => {
let url = (typeof request === 'string') ? request : (request as Request).url; let url = (typeof request === 'string') ? request : (request as Request).url;
// Check blocked URLs // Check blocked URLs
for (let blocked of BLOCKED_URLS) { for (let blocked of BLOCKED_URLS) {
if (!url.startsWith(blocked)) { if (url.startsWith(blocked)) {
continue; return new Response('{"acc":1,"webResult":{}}', {
status: 200,
statusText: '200 OK',
});
} }
return new Response('{"acc":1,"webResult":{}}', {
status: 200,
statusText: '200 OK',
});
} }
if (url.endsWith('/play')) { // Ignore URLs
BxEvent.dispatch(window, BxEvent.STREAM_LOADING); const domain = (new URL(url)).hostname;
if (IGNORED_DOMAINS.includes(domain)) {
return NATIVE_FETCH(request, init);
} }
if (url.endsWith('/configuration')) { // BxLogger.info('fetch', url);
BxEvent.dispatch(window, BxEvent.STREAM_STARTING);
}
// Override experimentals // Override experimentals
if (url.startsWith('https://emerald.xboxservices.com/xboxcomfd/experimentation')) { if (url.startsWith('https://emerald.xboxservices.com/xboxcomfd/experimentation')) {
@ -210,6 +223,7 @@ export function interceptHttpRequests() {
return response; return response;
} catch (e) { } catch (e) {
console.log(e); console.log(e);
return NATIVE_FETCH(request, init);
} }
} }
@ -222,7 +236,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 +276,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);
} }

View File

@ -1,66 +0,0 @@
import { deepClone, STATES } from "@utils/global";
import { BxLogger } from "./bx-logger";
import { TouchController } from "@modules/touch-controller";
import { GamePassCloudGallery } from "../enums/game-pass-gallery";
import { BX_FLAGS } from "./bx-flags";
import { PrefKey } from "@/enums/pref-keys";
import { getPref } from "./settings-storages/global-settings-storage";
const LOG_TAG = 'PreloadState';
export function overridePreloadState() {
let _state: any;
Object.defineProperty(window, '__PRELOADED_STATE__', {
configurable: true,
get: () => {
return _state;
},
set: state => {
// Override User-Agent
try {
state.appContext.requestInfo.userAgent = window.navigator.userAgent;
} catch (e) {
BxLogger.error(LOG_TAG, e);
}
// Add list of games with custom layouts to the official list
if (STATES.userAgent.capabilities.touch) {
try {
const sigls = state.xcloud.sigls;
if (GamePassCloudGallery.TOUCH in sigls) {
let customList = TouchController.getCustomList();
const allGames = sigls[GamePassCloudGallery.ALL].data.products;
// Remove non-cloud games from the list
customList = customList.filter(id => allGames.includes(id));
// Add to the official list
sigls[GamePassCloudGallery.TOUCH]?.data.products.push(...customList);
}
if (BX_FLAGS.ForceNativeMkbTitles && GamePassCloudGallery.NATIVE_MKB in sigls) {
// Add to the official list
sigls[GamePassCloudGallery.NATIVE_MKB]?.data.products.push(...BX_FLAGS.ForceNativeMkbTitles);
}
} catch (e) {
BxLogger.error(LOG_TAG, e);
}
}
if (getPref(PrefKey.UI_HOME_CONTEXT_MENU_DISABLED)) {
try {
state.experiments.experimentationInfo.data.treatments.EnableHomeContextMenu = false;
} catch (e) {
BxLogger.error(LOG_TAG, e);
}
}
// @ts-ignore
_state = state;
STATES.appContext = deepClone(state.appContext);
}
});
}

View File

@ -0,0 +1,114 @@
import { GuideMenu } from "@/modules/ui/guide-menu";
import { BxEvent } from "./bx-event";
import { BX_FLAGS } from "./bx-flags";
import { BxLogger } from "./bx-logger";
import { BxIcon } from "./bx-icon";
import { AppInterface } from "./global";
import { createButton, ButtonStyle } from "./html";
import { t } from "./translation";
import { parseDetailsPath } from "./utils";
export class RootDialogObserver {
private static $btnShortcut = AppInterface && createButton({
icon: BxIcon.CREATE_SHORTCUT,
label: t('create-shortcut'),
style: ButtonStyle.FOCUSABLE | ButtonStyle.GHOST | ButtonStyle.FULL_WIDTH | ButtonStyle.NORMAL_CASE | ButtonStyle.NORMAL_LINK,
tabIndex: 0,
onClick: e => {
window.BX_EXPOSED.dialogRoutes?.closeAll();
const $btn = (e.target as HTMLElement).closest('button');
AppInterface.createShortcut($btn?.dataset.path);
},
});
private static $btnWallpaper = AppInterface && createButton({
icon: BxIcon.DOWNLOAD,
label: t('wallpaper'),
style: ButtonStyle.FOCUSABLE | ButtonStyle.GHOST | ButtonStyle.FULL_WIDTH | ButtonStyle.NORMAL_CASE | ButtonStyle.NORMAL_LINK,
tabIndex: 0,
onClick: e => {
window.BX_EXPOSED.dialogRoutes?.closeAll();
const $btn = (e.target as HTMLElement).closest('button');
const details = parseDetailsPath($btn!.dataset.path!);
details && AppInterface.downloadWallpapers(details.titleSlug, details.productId);
},
});
private static handleGameCardMenu($root: HTMLElement) {
const $detail = $root.querySelector('a[href^="/play/"]') as HTMLAnchorElement;
if (!$detail) {
return;
}
const path = $detail.getAttribute('href')!;
RootDialogObserver.$btnShortcut.dataset.path = path;
RootDialogObserver.$btnWallpaper.dataset.path = path;
$root.append(RootDialogObserver.$btnShortcut, RootDialogObserver.$btnWallpaper);
}
private static handleAddedElement($root: HTMLElement, $addedElm: HTMLElement): boolean {
if (AppInterface && $addedElm.className.startsWith('SlideSheet-module__container')) {
// Game card's context menu
const $gameCardMenu = $addedElm.querySelector<HTMLElement>('div[class^=MruContextMenu],div[class^=GameCardContextMenu]');
if ($gameCardMenu) {
RootDialogObserver.handleGameCardMenu($gameCardMenu);
return true;
}
} else if ($root.querySelector('div[class*=GuideDialog]')) {
// Guide menu
GuideMenu.getInstance().observe($addedElm);
return true;
}
return false;
}
private static observe($root: HTMLElement) {
let beingShown = false;
const observer = new MutationObserver(mutationList => {
for (const mutation of mutationList) {
if (mutation.type !== 'childList') {
continue;
}
BX_FLAGS.Debug && BxLogger.warning('RootDialog', 'added', mutation.addedNodes);
if (mutation.addedNodes.length === 1) {
const $addedElm = mutation.addedNodes[0];
if ($addedElm instanceof HTMLElement) {
RootDialogObserver.handleAddedElement($root, $addedElm);
}
}
const shown = !!($root.firstElementChild && $root.firstElementChild.childElementCount > 0);
if (shown !== beingShown) {
beingShown = shown;
BxEvent.dispatch(window, shown ? BxEvent.XCLOUD_DIALOG_SHOWN : BxEvent.XCLOUD_DIALOG_DISMISSED);
}
}
});
observer.observe($root, {subtree: true, childList: true});
}
public static waitForRootDialog() {
const observer = new MutationObserver(mutationList => {
for (const mutation of mutationList) {
if (mutation.type !== 'childList') {
continue;
}
const $target = mutation.target as HTMLElement;
if ($target.id && $target.id === 'gamepass-dialog-root') {
observer.disconnect();
RootDialogObserver.observe($target);
break;
}
};
});
observer.observe(document.documentElement, {subtree: true, childList: true});
}
}

View File

@ -0,0 +1,105 @@
import { StreamPlayerType } from "@enums/stream-player";
import { AppInterface, STATES } from "./global";
import { CE } from "./html";
import { PrefKey } from "@/enums/pref-keys";
import { getPref } from "./settings-storages/global-settings-storage";
import { BxLogger } from "./bx-logger";
export class ScreenshotManager {
private static instance: ScreenshotManager;
public static getInstance = () => ScreenshotManager.instance ?? (ScreenshotManager.instance = new ScreenshotManager());
private readonly LOG_TAG = 'ScreenshotManager';
private $download: HTMLAnchorElement;
private $canvas: HTMLCanvasElement;
private canvasContext: CanvasRenderingContext2D;
private constructor() {
BxLogger.info(this.LOG_TAG, 'constructor()');
this.$download = CE<HTMLAnchorElement>('a');
this.$canvas = CE<HTMLCanvasElement>('canvas', {'class': 'bx-gone'});
this.canvasContext = this.$canvas.getContext('2d', {
alpha: false,
willReadFrequently: false,
})!;
}
updateCanvasSize(width: number, height: number) {
this.$canvas.width = width;
this.$canvas.height = height;
}
updateCanvasFilters(filters: string) {
this.canvasContext.filter = filters;
}
private onAnimationEnd(e: Event) {
(e.target as HTMLElement).classList.remove('bx-taking-screenshot');
}
takeScreenshot(callback?: any) {
const currentStream = STATES.currentStream;
const streamPlayer = currentStream.streamPlayer;
const $canvas = this.$canvas;
if (!streamPlayer || !$canvas) {
return;
}
let $player;
if (getPref(PrefKey.SCREENSHOT_APPLY_FILTERS)) {
$player = streamPlayer.getPlayerElement();
} else {
$player = streamPlayer.getPlayerElement(StreamPlayerType.VIDEO);
}
if (!$player || !$player.isConnected) {
return;
}
$player.parentElement!.addEventListener('animationend', this.onAnimationEnd, { once: true });
$player.parentElement!.classList.add('bx-taking-screenshot');
const canvasContext = this.canvasContext;
if ($player instanceof HTMLCanvasElement) {
streamPlayer.getWebGL2Player().forceDrawFrame();
}
canvasContext.drawImage($player, 0, 0, $canvas.width, $canvas.height);
// Get data URL and pass to parent app
if (AppInterface) {
const data = $canvas.toDataURL('image/png').split(';base64,')[1];
AppInterface.saveScreenshot(currentStream.titleSlug, data);
// Free screenshot from memory
canvasContext.clearRect(0, 0, $canvas.width, $canvas.height);
callback && callback();
return;
}
$canvas.toBlob(blob => {
if (!blob) {
return;
}
// Download screenshot
const now = +new Date;
const $download = this.$download;
$download.download = `${currentStream.titleSlug}-${now}.png`;
$download.href = URL.createObjectURL(blob);
$download.click();
// Free screenshot from memory
URL.revokeObjectURL($download.href);
$download.href = '';
$download.download = '';
canvasContext.clearRect(0, 0, $canvas.width, $canvas.height);
callback && callback();
}, 'image/png');
}
}

View File

@ -1,99 +0,0 @@
import { StreamPlayerType } from "@enums/stream-player";
import { AppInterface, STATES } from "./global";
import { CE } from "./html";
import { PrefKey } from "@/enums/pref-keys";
import { getPref } from "./settings-storages/global-settings-storage";
export class Screenshot {
static #$canvas: HTMLCanvasElement;
static #canvasContext: CanvasRenderingContext2D;
static setup() {
if (Screenshot.#$canvas) {
return;
}
Screenshot.#$canvas = CE<HTMLCanvasElement>('canvas', {'class': 'bx-gone'});
Screenshot.#canvasContext = Screenshot.#$canvas.getContext('2d', {
alpha: false,
willReadFrequently: false,
})!;
}
static updateCanvasSize(width: number, height: number) {
const $canvas = Screenshot.#$canvas;
if ($canvas) {
$canvas.width = width;
$canvas.height = height;
}
}
static updateCanvasFilters(filters: string) {
Screenshot.#canvasContext.filter = filters;
}
static #onAnimationEnd(e: Event) {
const $target = e.target as HTMLElement;
$target.classList.remove('bx-taking-screenshot');
}
static takeScreenshot(callback?: any) {
const currentStream = STATES.currentStream;
const streamPlayer = currentStream.streamPlayer;
const $canvas = Screenshot.#$canvas;
if (!streamPlayer || !$canvas) {
return;
}
let $player;
if (getPref(PrefKey.SCREENSHOT_APPLY_FILTERS)) {
$player = streamPlayer.getPlayerElement();
} else {
$player = streamPlayer.getPlayerElement(StreamPlayerType.VIDEO);
}
if (!$player || !$player.isConnected) {
return;
}
$player.parentElement!.addEventListener('animationend', this.#onAnimationEnd, { once: true });
$player.parentElement!.classList.add('bx-taking-screenshot');
const canvasContext = Screenshot.#canvasContext;
if ($player instanceof HTMLCanvasElement) {
streamPlayer.getWebGL2Player().drawFrame();
}
canvasContext.drawImage($player, 0, 0, $canvas.width, $canvas.height);
// Get data URL and pass to parent app
if (AppInterface) {
const data = $canvas.toDataURL('image/png').split(';base64,')[1];
AppInterface.saveScreenshot(currentStream.titleSlug, data);
// Free screenshot from memory
canvasContext.clearRect(0, 0, $canvas.width, $canvas.height);
callback && callback();
return;
}
$canvas && $canvas.toBlob(blob => {
// Download screenshot
const now = +new Date;
const $anchor = CE<HTMLAnchorElement>('a', {
'download': `${currentStream.titleSlug}-${now}.png`,
'href': URL.createObjectURL(blob!),
});
$anchor.click();
// Free screenshot from memory
URL.revokeObjectURL($anchor.href);
canvasContext.clearRect(0, 0, $canvas.width, $canvas.height);
callback && callback();
}, 'image/png');
}
}

View File

@ -25,9 +25,9 @@ export interface BxSelectSettingElement extends HTMLSelectElement, BxBaseSetting
export class SettingElement { export class SettingElement {
static #renderOptions(key: string, setting: PreferenceSetting, currentValue: any, onChange: any): BxSelectSettingElement { static #renderOptions(key: string, setting: PreferenceSetting, currentValue: any, onChange: any): BxSelectSettingElement {
const $control = CE<BxSelectSettingElement>('select', { const $control = CE<BxSelectSettingElement>('select', {
// title: setting.label, // title: setting.label,
tabindex: 0, tabindex: 0,
}); });
let $parent: HTMLElement; let $parent: HTMLElement;
if (setting.optionsGroup) { if (setting.optionsGroup) {
@ -64,10 +64,10 @@ export class SettingElement {
static #renderMultipleOptions(key: string, setting: PreferenceSetting, currentValue: any, onChange: any, params: MultipleOptionsParams={}): BxSelectSettingElement { static #renderMultipleOptions(key: string, setting: PreferenceSetting, currentValue: any, onChange: any, params: MultipleOptionsParams={}): BxSelectSettingElement {
const $control = CE<BxSelectSettingElement>('select', { const $control = CE<BxSelectSettingElement>('select', {
// title: setting.label, // title: setting.label,
multiple: true, multiple: true,
tabindex: 0, tabindex: 0,
}); });
if (params && params.size) { if (params && params.size) {
$control.setAttribute('size', params.size.toString()); $control.setAttribute('size', params.size.toString());
@ -160,10 +160,18 @@ export class SettingElement {
let controlValue = value; let controlValue = value;
const MIN = setting.min!; const MIN = options.reverse ? -setting.max! : setting.min!;
const MAX = setting.max!; const MAX = options.reverse ? -setting.min! : setting.max!;
const STEPS = Math.max(setting.steps || 1, 1); const STEPS = Math.max(setting.steps || 1, 1);
let intervalId: number | null;
let isHolding = false;
const clearIntervalId = () => {
intervalId && clearInterval(intervalId);
intervalId = null;
}
const renderTextValue = (value: any) => { const renderTextValue = (value: any) => {
value = parseInt(value as string); value = parseInt(value as string);
@ -182,6 +190,10 @@ export class SettingElement {
const updateButtonsVisibility = () => { const updateButtonsVisibility = () => {
$btnDec.classList.toggle('bx-inactive', controlValue === MIN); $btnDec.classList.toggle('bx-inactive', controlValue === MIN);
$btnInc.classList.toggle('bx-inactive', controlValue === MAX); $btnInc.classList.toggle('bx-inactive', controlValue === MAX);
if (controlValue === MIN || controlValue === MAX) {
clearIntervalId();
}
} }
const $wrapper = CE<BxHtmlSettingElement>('div', {'class': 'bx-number-stepper', id: `bx_setting_${key}`}, const $wrapper = CE<BxHtmlSettingElement>('div', {'class': 'bx-number-stepper', id: `bx_setting_${key}`},
@ -212,11 +224,11 @@ export class SettingElement {
} }
$range = CE<HTMLInputElement>('input', { $range = CE<HTMLInputElement>('input', {
id: `bx_setting_${key}`, id: `bx_inp_setting_${key}`,
type: 'range', type: 'range',
min: MIN, min: MIN,
max: MAX, max: MAX,
value: value, value: options.reverse ? -value : value,
step: STEPS, step: STEPS,
tabindex: 0, tabindex: 0,
}); });
@ -225,13 +237,16 @@ export class SettingElement {
$range.addEventListener('input', e => { $range.addEventListener('input', e => {
value = parseInt((e.target as HTMLInputElement).value); value = parseInt((e.target as HTMLInputElement).value);
const valueChanged = controlValue !== value; if (options.reverse) {
value *= -1;
}
const valueChanged = controlValue !== value;
if (!valueChanged) { if (!valueChanged) {
return; return;
} }
controlValue = value; controlValue = options.reverse ? -value : value;
updateButtonsVisibility(); updateButtonsVisibility();
$text.textContent = renderTextValue(value); $text.textContent = renderTextValue(value);
@ -245,22 +260,24 @@ export class SettingElement {
if (options.ticks || options.exactTicks) { if (options.ticks || options.exactTicks) {
const markersId = `markers-${key}`; const markersId = `markers-${key}`;
const $markers = CE('datalist', {'id': markersId}); const $markers = CE('datalist', {id: markersId});
$range.setAttribute('list', markersId); $range.setAttribute('list', markersId);
if (options.exactTicks) { if (options.exactTicks) {
let start = Math.max(Math.floor(MIN / options.exactTicks), 1) * options.exactTicks; let start = Math.max(Math.floor(setting.min! / options.exactTicks), 1) * options.exactTicks;
if (start === MIN) { if (start === setting.min!) {
start += options.exactTicks; start += options.exactTicks;
} }
for (let i = start; i < MAX; i += options.exactTicks) { for (let i = start; i < setting.max!; i += options.exactTicks) {
$markers.appendChild(CE<HTMLOptionElement>('option', {'value': i})); $markers.appendChild(CE<HTMLOptionElement>('option', {
value: options.reverse ? -i : i,
}));
} }
} else { } else {
for (let i = MIN + options.ticks!; i < MAX; i += options.ticks!) { for (let i = MIN + options.ticks!; i < MAX; i += options.ticks!) {
$markers.appendChild(CE<HTMLOptionElement>('option', {'value': i})); $markers.appendChild(CE<HTMLOptionElement>('option', {value: i}));
} }
} }
$wrapper.appendChild($markers); $wrapper.appendChild($markers);
@ -268,18 +285,7 @@ export class SettingElement {
updateButtonsVisibility(); updateButtonsVisibility();
let interval: number; const buttonPressed = (e: Event, $btn: HTMLElement) => {
let isHolding = false;
const onClick = (e: Event) => {
if (isHolding) {
e.preventDefault();
isHolding = false;
return;
}
const $btn = e.target as HTMLElement;
let value = parseInt(controlValue); let value = parseInt(controlValue);
const btnType = $btn.dataset.type; const btnType = $btn.dataset.type;
@ -295,27 +301,43 @@ export class SettingElement {
$text.textContent = renderTextValue(value); $text.textContent = renderTextValue(value);
$range && ($range.value = value.toString()); $range && ($range.value = value.toString());
isHolding = false; onChange && onChange(e, value);
!(e as any).ignoreOnChange && onChange && onChange(e, value);
}
const onMouseDown = (e: PointerEvent) => {
e.preventDefault();
isHolding = true;
const args = arguments;
interval && clearInterval(interval);
interval = window.setInterval(() => {
e.target && BxEvent.dispatch(e.target as HTMLElement, 'click', {
arguments: args,
});
}, 200);
}; };
const onMouseUp = (e: PointerEvent) => { const onClick = (e: Event) => {
e.preventDefault();
if (isHolding) {
return;
}
const $btn = (e.target as HTMLElement).closest('button') as HTMLElement;
$btn && buttonPressed(e, $btn);
clearIntervalId();
isHolding = false;
};
const onPointerDown = (e: PointerEvent) => {
clearIntervalId();
const $btn = (e.target as HTMLElement).closest('button') as HTMLElement;
if (!$btn) {
return;
}
isHolding = true;
e.preventDefault(); e.preventDefault();
interval && clearInterval(interval); intervalId = window.setInterval((e: Event) => {
buttonPressed(e, $btn);
}, 200);
window.addEventListener('pointerup', onPointerUp, {once: true});
window.addEventListener('pointercancel', onPointerUp, {once: true});
};
const onPointerUp = (e: PointerEvent) => {
clearIntervalId();
isHolding = false; isHolding = false;
}; };
@ -324,21 +346,14 @@ export class SettingElement {
// Custom method // Custom method
$wrapper.setValue = (value: any) => { $wrapper.setValue = (value: any) => {
$text.textContent = renderTextValue(value); $text.textContent = renderTextValue(value);
$range.value = value; $range.value = options.reverse ? -value : value;
}; };
$btnDec.addEventListener('click', onClick); $wrapper.addEventListener('click', onClick);
$btnDec.addEventListener('pointerdown', onMouseDown); $wrapper.addEventListener('pointerdown', onPointerDown);
$btnDec.addEventListener('pointerup', onMouseUp); $wrapper.addEventListener('contextmenu', onContextMenu);
$btnDec.addEventListener('contextmenu', onContextMenu);
$btnInc.addEventListener('click', onClick);
$btnInc.addEventListener('pointerdown', onMouseDown);
$btnInc.addEventListener('pointerup', onMouseUp);
$btnInc.addEventListener('contextmenu', onContextMenu);
setNearby($wrapper, { setNearby($wrapper, {
focus: $range || $btnInc, focus: options.hideSlider ? $btnInc : $range,
}) })
return $wrapper; return $wrapper;

View File

@ -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)) {

View File

@ -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 {
@ -39,6 +39,10 @@ export const enum ControllerDeviceVibration {
} }
export type GameBarPosition = 'bottom-left' | 'bottom-right' | 'off';
export type GameBarPositionOptions = Record<GameBarPosition, string>;
function getSupportedCodecProfiles() { function getSupportedCodecProfiles() {
const options: PartialRecord<CodecProfile, string> = { const options: PartialRecord<CodecProfile, string> = {
default: t('default'), default: t('default'),
@ -96,7 +100,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,
}, },
@ -113,6 +117,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
}, },
[PrefKey.SERVER_REGION]: { [PrefKey.SERVER_REGION]: {
label: t('region'), label: t('region'),
note: CE('a', {target: '_blank', href: 'https://umap.openstreetmap.fr/en/map/xbox-cloud-gaming-servers_1135022'}, t('server-locations')),
default: 'default', default: 'default',
}, },
[PrefKey.SERVER_BYPASS_RESTRICTION]: { [PrefKey.SERVER_BYPASS_RESTRICTION]: {
@ -131,12 +136,13 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
options: { options: {
default: t('default'), default: t('default'),
'ar-SA': 'العربية', 'ar-SA': 'العربية',
'bg-BG': 'Български',
'cs-CZ': 'čeština', 'cs-CZ': 'čeština',
'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',
@ -151,9 +157,11 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
'pl-PL': 'polski', 'pl-PL': 'polski',
'pt-BR': 'português (Brasil)', 'pt-BR': 'português (Brasil)',
'pt-PT': 'português (Portugal)', 'pt-PT': 'português (Portugal)',
'ro-RO': 'Română',
'ru-RU': 'русский', 'ru-RU': 'русский',
'sk-SK': 'slovenčina', 'sk-SK': 'slovenčina',
'sv-SE': 'svenska', 'sv-SE': 'svenska',
'th-TH': 'ไทย',
'tr-TR': 'Türkçe', 'tr-TR': 'Türkçe',
'zh-CN': '中文(简体)', 'zh-CN': '中文(简体)',
'zh-TW': '中文 (繁體)', 'zh-TW': '中文 (繁體)',
@ -182,7 +190,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
if (keys.length <= 1) { // Unsupported if (keys.length <= 1) { // Unsupported
setting.unsupported = true; setting.unsupported = true;
setting.note = '⚠️ ' + t('browser-unsupported-feature'); setting.unsupportedNote = '⚠️ ' + t('browser-unsupported-feature');
} }
setting.suggest = { setting.suggest = {
@ -197,6 +205,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 +220,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 +229,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 +245,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 +266,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 +277,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 +292,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,26 +325,28 @@ 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' satisfies GameBarPosition,
options: { options: {
'bottom-left': t('bottom-left'), 'bottom-left': t('bottom-left'),
'bottom-right': t('bottom-right'), 'bottom-right': t('bottom-right'),
'off': t('off'), 'off': t('off'),
}, } satisfies GameBarPositionOptions,
}, },
[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', {
href: 'https://github.com/redphx/better-xcloud/discussions/275', href: 'https://github.com/redphx/better-xcloud/discussions/275',
target: '_blank', target: '_blank',
}, t('enable-local-co-op-support-note')), }, t('enable-local-co-op-support-note')),
}, },
/* /*
@ -340,16 +361,14 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
default: true, default: true,
}, },
[PrefKey.CONTROLLER_ENABLE_SHORTCUTS]: {
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 +379,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,
@ -372,13 +392,35 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
}, },
}, },
[PrefKey.CONTROLLER_POLLING_RATE]: {
requiredVariants: 'full',
label: t('polling-rate'),
type: SettingElementType.NUMBER_STEPPER,
default: 4,
min: 4,
max: 60,
steps: 4,
params: {
exactTicks: 20,
reverse: true,
customTextValue(value: any) {
value = parseInt(value);
let text = +(1000 / value).toFixed(2) + ' Hz';
if (value === 4) {
text = `${text} (${t('default')})`;
}
return text;
},
},
},
[PrefKey.MKB_ENABLED]: { [PrefKey.MKB_ENABLED]: {
requiredVariants: 'full',
label: t('enable-mkb'), label: t('enable-mkb'),
default: false, default: false,
unsupported: ((): string | boolean => { unsupported: !STATES.userAgent.capabilities.mkb,
const userAgent = ((window.navigator as any).orgUserAgent || window.navigator.userAgent || '').toLowerCase();
return !AppInterface && userAgent.match(/(android|iphone|ipad)/) ? t('browser-unsupported-feature') : false;
})(),
ready: (setting: SettingDefinition) => { ready: (setting: SettingDefinition) => {
let note; let note;
let url; let url;
@ -390,14 +432,15 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
url = 'https://better-xcloud.github.io/mouse-and-keyboard/#disclaimer'; url = 'https://better-xcloud.github.io/mouse-and-keyboard/#disclaimer';
} }
setting.note = CE('a', { setting.unsupportedNote = () => CE<HTMLAnchorElement>('a', {
href: url, href: url,
target: '_blank', target: '_blank',
}, '⚠️ ' + note); }, '⚠️ ' + note);
}, },
}, },
[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 +462,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 +482,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 +502,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 +517,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 +541,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: {
@ -507,12 +556,8 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
default: false, default: false,
}, },
[PrefKey.UI_HOME_CONTEXT_MENU_DISABLED]: {
label: t('disable-home-context-menu'),
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 +574,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,
}, },
@ -584,13 +630,28 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
default: 'default', default: 'default',
options: { options: {
'default': t('default'), 'default': t('default'),
'low-power': t('low-power'), 'low-power': t('battery-saving'),
'high-performance': t('high-performance'), 'high-performance': t('high-performance'),
}, },
suggest: { suggest: {
highest: 'low-power', highest: 'low-power',
}, },
}, },
[PrefKey.VIDEO_MAX_FPS]: {
label: t('max-fps'),
type: SettingElementType.NUMBER_STEPPER,
default: 60,
min: 10,
max: 60,
steps: 10,
params: {
exactTicks: 10,
customTextValue: (value: any) => {
value = parseInt(value);
return value === 60 ? t('unlimited') : value + 'fps';
},
},
},
[PrefKey.VIDEO_SHARPNESS]: { [PrefKey.VIDEO_SHARPNESS]: {
label: t('sharpness'), label: t('sharpness'),
type: SettingElementType.NUMBER_STEPPER, type: SettingElementType.NUMBER_STEPPER,
@ -606,7 +667,7 @@ export class GlobalSettingsStorage extends BaseSettingsStorage {
}, },
suggest: { suggest: {
lowest: 0, lowest: 0,
highest: 4, highest: 2,
}, },
}, },
[PrefKey.VIDEO_RATIO]: { [PrefKey.VIDEO_RATIO]: {
@ -663,6 +724,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 +746,29 @@ 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.JITTER]: `${StreamStat.JITTER.toUpperCase()}: ${t('jitter')}`,
[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('downloaded')}`,
[StreamStat.UPLOAD]: `${StreamStat.UPLOAD.toUpperCase()}: ${t('uploaded')}`,
}, },
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 +818,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 +833,19 @@ 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'),
}, },
};
[PrefKey.GAME_MSFS2020_FORCE_NATIVE_MKB]: {
requiredVariants: 'full',
label: '✈️ ' + t('msfs2020-force-native-mkb'),
default: false,
note: t('may-not-work-properly'),
},
} satisfies SettingDefinitions;
constructor() { constructor() {
super(StorageKey.GLOBAL, GlobalSettingsStorage.DEFINITIONS); super(StorageKey.GLOBAL, GlobalSettingsStorage.DEFINITIONS);

View File

@ -0,0 +1,332 @@
import { PrefKey } from "@/enums/pref-keys";
import { BxEvent } from "./bx-event";
import { STATES } from "./global";
import { humanFileSize, secondsToHm } from "./html";
import { getPref } from "./settings-storages/global-settings-storage";
import { BxLogger } from "./bx-logger";
export enum StreamStat {
PING = 'ping',
JITTER = 'jit',
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;
grades: [number, number, number];
toString: () => string;
};
[StreamStat.JITTER]: {
current: number;
grades: [number, number, number];
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;
grades: [number, number, number];
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.instance ?? (StreamStatsCollector.instance = new StreamStatsCollector());
private readonly LOG_TAG = 'StreamStatsCollector';
// Collect in background - 60 seconds
static readonly INTERVAL_BACKGROUND = 60 * 1000;
public calculateGrade(value: number, grades: [number, number, number]): StreamStatGrade {
return (value > grades[2]) ? 'bad' : (value > grades[1]) ? 'ok' : (value > grades[0]) ? 'good' : '';
}
private currentStats: CurrentStats = {
[StreamStat.PING]: {
current: -1,
grades: [40, 75, 100],
toString() {
return this.current === -1 ? '???' : this.current.toString();
},
},
[StreamStat.JITTER]: {
current: 0,
grades: [30, 40, 60],
toString() {
return `${this.current.toFixed(2)}ms`;
},
},
[StreamStat.FPS]: {
current: 0,
toString() {
const maxFps = getPref(PrefKey.VIDEO_MAX_FPS);
return maxFps < 60 ? `${maxFps}/${this.current}` : 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,
grades: [6, 9, 12],
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?: RTCInboundRtpStreamStats | null;
private constructor() {
BxLogger.info(this.LOG_TAG, 'constructor()');
}
async collect() {
const stats = await STATES.currentStream.peerConnection?.getStats();
if (!stats) {
return;
}
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;
// Jitter
const jit = this.currentStats[StreamStat.JITTER];
const bufferDelayDiff = (stat as RTCInboundRtpStreamStats).jitterBufferDelay! - lastStat.jitterBufferDelay!;
const emittedCountDiff = (stat as RTCInboundRtpStreamStats).jitterBufferEmittedCount! - lastStat.jitterBufferEmittedCount!;
if (emittedCountDiff > 0) {
jit.current = bufferDelayDiff / emittedCountDiff * 1000;
}
// 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();
});
}
}

View File

@ -1,4 +1,5 @@
import { CE } from "@utils/html"; import { CE } from "@utils/html";
import { BxLogger } from "./bx-logger";
type ToastOptions = { type ToastOptions = {
instant?: boolean; instant?: boolean;
@ -6,84 +7,100 @@ type ToastOptions = {
} }
export class Toast { export class Toast {
static #$wrapper: HTMLElement; private static instance: Toast;
static #$msg: HTMLElement; public static getInstance = () => Toast.instance ?? (Toast.instance = new Toast());
static #$status: HTMLElement; private readonly LOG_TAG = 'Toast';
static #stack: Array<[string, string, ToastOptions]> = [];
static #isShowing = false;
static #timeout?: number | null; private $wrapper: HTMLElement;
static #DURATION = 3000; private $msg: HTMLElement;
private $status: HTMLElement;
static show(msg: string, status?: string, options: Partial<ToastOptions> = {}) { private stack: Array<[string, string, ToastOptions]> = [];
private isShowing = false;
private timeoutId?: number | null;
private DURATION = 3000;
private constructor() {
BxLogger.info(this.LOG_TAG, 'constructor()');
this.$wrapper = CE('div', {class: 'bx-toast bx-offscreen'},
this.$msg = CE('span', {class: 'bx-toast-msg'}),
this.$status = CE('span', {class: 'bx-toast-status'}),
);
this.$wrapper.addEventListener('transitionend', e => {
const classList = this.$wrapper.classList;
if (classList.contains('bx-hide')) {
classList.remove('bx-offscreen', 'bx-hide');
classList.add('bx-offscreen');
this.showNext();
}
});
document.documentElement.appendChild(this.$wrapper);
}
private show(msg: string, status?: string, options: Partial<ToastOptions> = {}) {
options = options || {}; options = options || {};
const args = Array.from(arguments) as [string, string, ToastOptions]; const args = Array.from(arguments) as [string, string, ToastOptions];
if (options.instant) { if (options.instant) {
// Clear stack // Clear stack
Toast.#stack = [args]; this.stack = [args];
Toast.#showNext(); this.showNext();
} else { } else {
Toast.#stack.push(args); this.stack.push(args);
!Toast.#isShowing && Toast.#showNext(); !this.isShowing && this.showNext();
} }
} }
static #showNext() { private showNext() {
if (!Toast.#stack.length) { if (!this.stack.length) {
Toast.#isShowing = false; this.isShowing = false;
return; return;
} }
Toast.#isShowing = true; this.isShowing = true;
Toast.#timeout && clearTimeout(Toast.#timeout); this.timeoutId && clearTimeout(this.timeoutId);
Toast.#timeout = window.setTimeout(Toast.#hide, Toast.#DURATION); this.timeoutId = window.setTimeout(this.hide.bind(this), this.DURATION);
// Get values from item // Get values from item
const [msg, status, options] = Toast.#stack.shift()!; const [msg, status, options] = this.stack.shift()!;
if (options && options.html) { if (options && options.html) {
Toast.#$msg.innerHTML = msg; this.$msg.innerHTML = msg;
} else { } else {
Toast.#$msg.textContent = msg; this.$msg.textContent = msg;
} }
if (status) { if (status) {
Toast.#$status.classList.remove('bx-gone'); this.$status.classList.remove('bx-gone');
Toast.#$status.textContent = status; this.$status.textContent = status;
} else { } else {
Toast.#$status.classList.add('bx-gone'); this.$status.classList.add('bx-gone');
} }
const classList = Toast.#$wrapper.classList; const classList = this.$wrapper.classList;
classList.remove('bx-offscreen', 'bx-hide'); classList.remove('bx-offscreen', 'bx-hide');
classList.add('bx-show'); classList.add('bx-show');
} }
static #hide() { private hide() {
Toast.#timeout = null; this.timeoutId = null;
const classList = Toast.#$wrapper.classList; const classList = this.$wrapper.classList;
classList.remove('bx-show'); classList.remove('bx-show');
classList.add('bx-hide'); classList.add('bx-hide');
} }
static setup() { static show(msg: string, status?: string, options: Partial<ToastOptions> = {}) {
Toast.#$wrapper = CE('div', {'class': 'bx-toast bx-offscreen'}, Toast.getInstance().show(msg, status, options);
Toast.#$msg = CE('span', {'class': 'bx-toast-msg'}), }
Toast.#$status = CE('span', {'class': 'bx-toast-status'}));
Toast.#$wrapper.addEventListener('transitionend', e => { static showNext() {
const classList = Toast.#$wrapper.classList; Toast.getInstance().showNext();
if (classList.contains('bx-hide')) {
classList.remove('bx-offscreen', 'bx-hide');
classList.add('bx-offscreen');
Toast.#showNext();
}
});
document.documentElement.appendChild(Toast.#$wrapper);
} }
} }

View File

@ -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,8 @@ 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", "battery-saving": "Battery saving",
"badge-in": "In",
"badge-out": "Out",
"badge-playtime": "Playtime",
"badge-server": "Server",
"badge-video": "Video",
"better-xcloud": "Better xCloud", "better-xcloud": "Better xCloud",
"bitrate-audio-maximum": "Maximum audio bitrate", "bitrate-audio-maximum": "Maximum audio bitrate",
"bitrate-video-maximum": "Maximum video bitrate", "bitrate-video-maximum": "Maximum video bitrate",
@ -62,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",
@ -71,6 +67,11 @@ const Texts = {
"confirm-reload-stream": "Do you want to refresh the stream?", "confirm-reload-stream": "Do you want to refresh the stream?",
"connected": "Connected", "connected": "Connected",
"console-connect": "Connect", "console-connect": "Connect",
"continent-asia": "Asia",
"continent-australia": "Australia",
"continent-europe": "Europe",
"continent-north-america": "North America",
"continent-south-america": "South America",
"contrast": "Contrast", "contrast": "Contrast",
"controller": "Controller", "controller": "Controller",
"controller-friendly-ui": "Controller-friendly UI", "controller-friendly-ui": "Controller-friendly UI",
@ -96,10 +97,12 @@ const Texts = {
"disable-xcloud-analytics": "Disable xCloud analytics", "disable-xcloud-analytics": "Disable xCloud analytics",
"disabled": "Disabled", "disabled": "Disabled",
"disconnected": "Disconnected", "disconnected": "Disconnected",
"download": "Download",
"downloaded": "Downloaded",
"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",
"enable-local-co-op-support-note": "Only works if the game doesn't require a different profile", "enable-local-co-op-support-note": "Only works with some games",
"enable-mic-on-startup": "Enable microphone on game launch", "enable-mic-on-startup": "Enable microphone on game launch",
"enable-mkb": "Emulate controller with Mouse & Keyboard", "enable-mkb": "Emulate controller with Mouse & Keyboard",
"enable-quick-glance-mode": "Enable \"Quick Glance\" mode", "enable-quick-glance-mode": "Enable \"Quick Glance\" mode",
@ -109,7 +112,7 @@ const Texts = {
"experimental": "Experimental", "experimental": "Experimental",
"export": "Export", "export": "Export",
"fast": "Fast", "fast": "Fast",
"fortnite-allow-stw-mode": "Allows playing STW mode on mobile", "fortnite-allow-stw-mode": "Allows playing \"Save the World\" mode on mobile",
"fortnite-force-console-version": "Fortnite: force console version", "fortnite-force-console-version": "Fortnite: force console version",
"game-bar": "Game Bar", "game-bar": "Game Bar",
"getting-consoles-list": "Getting the list of consoles...", "getting-consoles-list": "Getting the list of consoles...",
@ -133,6 +136,7 @@ const Texts = {
"increase": "Increase", "increase": "Increase",
"install-android": "Better xCloud app for Android", "install-android": "Better xCloud app for Android",
"japan": "Japan", "japan": "Japan",
"jitter": "Jitter",
"keyboard-shortcuts": "Keyboard shortcuts", "keyboard-shortcuts": "Keyboard shortcuts",
"korea": "Korea", "korea": "Korea",
"language": "Language", "language": "Language",
@ -142,9 +146,9 @@ const Texts = {
"load-failed-message": "Failed to run Better xCloud", "load-failed-message": "Failed to run Better xCloud",
"loading-screen": "Loading screen", "loading-screen": "Loading screen",
"local-co-op": "Local co-op", "local-co-op": "Local co-op",
"low-power": "Low power",
"lowest-quality": "Lowest quality", "lowest-quality": "Lowest quality",
"map-mouse-to": "Map mouse to", "map-mouse-to": "Map mouse to",
"max-fps": "Max FPS",
"may-not-work-properly": "May not work properly!", "may-not-work-properly": "May not work properly!",
"menu": "Menu", "menu": "Menu",
"microphone": "Microphone", "microphone": "Microphone",
@ -153,10 +157,32 @@ const Texts = {
"mkb-disclaimer": "Using this feature when playing online could be viewed as cheating", "mkb-disclaimer": "Using this feature when playing online could be viewed as cheating",
"mouse-and-keyboard": "Mouse & Keyboard", "mouse-and-keyboard": "Mouse & Keyboard",
"mouse-wheel": "Mouse wheel", "mouse-wheel": "Mouse wheel",
"msfs2020-force-native-mkb": "MSFS2020: force native M&KB support",
"muted": "Muted", "muted": "Muted",
"name": "Name", "name": "Name",
"native-mkb": "Native Mouse & Keyboard", "native-mkb": "Native Mouse & Keyboard",
"new": "New", "new": "New",
"new-version-available": [
(e: any) => `Version ${e.version} available`,
,
,
(e: any) => `Version ${e.version} verfügbar`,
(e: any) => `Versi ${e.version} tersedia`,
(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) => `${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} พร้อมใช้งานแล้ว`,
(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) => `版本 ${e.version} 可供更新`,
(e: any) => `已可更新為 ${e.version}`,
],
"no-consoles-found": "No consoles found", "no-consoles-found": "No consoles found",
"normal": "Normal", "normal": "Normal",
"off": "Off", "off": "Off",
@ -165,7 +191,9 @@ const Texts = {
"opacity": "Opacity", "opacity": "Opacity",
"other": "Other", "other": "Other",
"playing": "Playing", "playing": "Playing",
"playtime": "Playtime",
"poland": "Poland", "poland": "Poland",
"polling-rate": "Polling rate",
"position": "Position", "position": "Position",
"powered-off": "Powered off", "powered-off": "Powered off",
"powered-on": "Powered on", "powered-on": "Powered on",
@ -199,19 +227,19 @@ const Texts = {
"recommended": "Recommended", "recommended": "Recommended",
"recommended-settings-for-device": [ "recommended-settings-for-device": [
(e: any) => `Recommended settings for ${e.device}`, (e: any) => `Recommended settings for ${e.device}`,
, (e: any) => `Configuració recomanada per a ${e.device}`,
, ,
(e: any) => `Empfohlene Einstellungen für ${e.device}`, (e: any) => `Empfohlene Einstellungen für ${e.device}`,
, (e: any) => `Rekomendasi pengaturan untuk ${e.device}`,
(e: any) => `Ajustes recomendados para ${e.device}`, (e: any) => `Ajustes recomendados para ${e.device}`,
(e: any) => `Paramètres recommandés pour ${e.device}`, (e: any) => `Paramètres recommandés pour ${e.device}`,
(e: any) => `Configurazioni consigliate per ${e.device}`, (e: any) => `Configurazioni consigliate per ${e.device}`,
(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}`,
@ -244,6 +272,7 @@ const Texts = {
"separate-touch-controller": "Separate Touch controller & Controller #1", "separate-touch-controller": "Separate Touch controller & Controller #1",
"separate-touch-controller-note": "Touch controller is Player 1, Controller #1 is Player 2", "separate-touch-controller-note": "Touch controller is Player 1, Controller #1 is Player 2",
"server": "Server", "server": "Server",
"server-locations": "Server locations",
"settings": "Settings", "settings": "Settings",
"settings-reload": "Reload page to reflect changes", "settings-reload": "Reload page to reflect changes",
"settings-reload-note": "Settings in this tab only go into effect on the next page load", "settings-reload-note": "Settings in this tab only go into effect on the next page load",
@ -329,6 +358,8 @@ const Texts = {
"unlimited": "Unlimited", "unlimited": "Unlimited",
"unmuted": "Unmuted", "unmuted": "Unmuted",
"unsharp-masking": "Unsharp masking", "unsharp-masking": "Unsharp masking",
"upload": "Upload",
"uploaded": "Uploaded",
"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",

View File

@ -1,45 +1,117 @@
import { BxIcon } from "./bx-icon"; import { BxIcon } from "./bx-icon";
import { AppInterface, STATES } from "./global"; import { BxLogger } from "./bx-logger";
import { ButtonStyle, CE, createButton, getReactProps } from "./html"; import { AppInterface, SCRIPT_VARIANT, STATES } from "./global";
import { ButtonStyle, CE, clearDataSet, createButton, getReactProps } from "./html";
import { t } from "./translation"; import { t } from "./translation";
export class TrueAchievements { export class TrueAchievements {
private static $link = createButton({ private static instance: TrueAchievements;
label: t('true-achievements'), public static getInstance = () => TrueAchievements.instance ?? (TrueAchievements.instance = new TrueAchievements());
url: '#', private readonly LOG_TAG = 'TrueAchievements';
icon: BxIcon.TRUE_ACHIEVEMENTS,
style: ButtonStyle.FOCUSABLE | ButtonStyle.GHOST | ButtonStyle.FULL_WIDTH | ButtonStyle.NORMAL_LINK,
onClick: TrueAchievements.onClick,
}) as HTMLAnchorElement;
static $button = createButton({ private $link: HTMLElement;
label: t('true-achievements'), private $button: HTMLElement;
title: t('true-achievements'), private $hiddenLink: HTMLAnchorElement;
icon: BxIcon.TRUE_ACHIEVEMENTS,
style: ButtonStyle.FOCUSABLE | ButtonStyle.FULL_WIDTH,
onClick: TrueAchievements.onClick,
}) as HTMLAnchorElement;
private static onClick(e: Event) { constructor() {
BxLogger.info(this.LOG_TAG, 'constructor()');
this.$link = createButton<HTMLAnchorElement>({
label: t('true-achievements'),
url: '#',
icon: BxIcon.TRUE_ACHIEVEMENTS,
style: ButtonStyle.FOCUSABLE | ButtonStyle.GHOST | ButtonStyle.FULL_WIDTH | ButtonStyle.NORMAL_LINK,
onClick: this.onClick.bind(this),
});
this.$button = createButton<HTMLAnchorElement>({
label: t('true-achievements'),
title: t('true-achievements'),
icon: BxIcon.TRUE_ACHIEVEMENTS,
style: ButtonStyle.FOCUSABLE,
onClick: this.onClick.bind(this),
});
this.$hiddenLink = CE<HTMLAnchorElement>('a', {
target: '_blank',
});
}
private onClick(e: Event) {
e.preventDefault(); e.preventDefault();
const dataset = TrueAchievements.$link.dataset; // Close all xCloud's dialogs
TrueAchievements.open(true, dataset.xboxTitleId, dataset.id); window.BX_EXPOSED.dialogRoutes?.closeAll();
const dataset = this.$link.dataset;
this.open(true, dataset.xboxTitleId, dataset.id);
} }
private static $hiddenLink = CE<HTMLAnchorElement>('a', { private updateIds(xboxTitleId?: string, id?: string) {
target: '_blank', const $link = this.$link;
}); const $button = this.$button;
private static updateLinks(xboxTitleId?: string, id?: string) { clearDataSet($link);
TrueAchievements.$link.dataset.xboxTitleId = xboxTitleId; clearDataSet($button);
TrueAchievements.$link.dataset.id = id;
TrueAchievements.$button.dataset.xboxTitleId = xboxTitleId; if (xboxTitleId) {
TrueAchievements.$button.dataset.id = id; $link.dataset.xboxTitleId = xboxTitleId;
$button.dataset.xboxTitleId = xboxTitleId;
}
if (id) {
$link.dataset.id = id;
$button.dataset.id = id;
}
} }
static injectAchievementDetailPage($parent: HTMLElement) { injectAchievementsProgress($elm: HTMLElement) {
// Only do this in Full version
if (SCRIPT_VARIANT !== 'full') {
return;
}
const $parent = $elm.parentElement!;
// Wrap xCloud's element with our own
const $div = CE('div', {
class: 'bx-guide-home-achievements-progress',
}, $elm);
// Get xboxTitleId of the game
let xboxTitleId: string | number | undefined;
try {
const $container = $parent.closest<HTMLElement>('div[class*=AchievementsPreview-module__container]');
if ($container) {
const props = getReactProps($container);
xboxTitleId = props.children.props.data.data.xboxTitleId;
}
} catch (e) {}
if (!xboxTitleId) {
xboxTitleId = this.getStreamXboxTitleId();
}
if (typeof xboxTitleId !== 'undefined') {
xboxTitleId = xboxTitleId.toString();
}
this.updateIds(xboxTitleId);
if (document.documentElement.dataset.xdsPlatform === 'tv') {
$div.appendChild(this.$link);
} else {
$div.appendChild(this.$button);
}
$parent.appendChild($div);
}
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;
@ -50,7 +122,7 @@ export class TrueAchievements {
const achievementList: XboxAchievement[] = props.children.props.data.data; const achievementList: XboxAchievement[] = props.children.props.data.data;
// Get current achievement name // Get current achievement name
const $header = $parent.querySelector('div[class*=AchievementDetailHeader]') as HTMLElement; const $header = $parent.querySelector<HTMLElement>('div[class*=AchievementDetailHeader]')!;
const achievementName = getReactProps($header).children[0].props.achievementName; const achievementName = getReactProps($header).children[0].props.achievementName;
// Find achievement based on name // Find achievement based on name
@ -66,15 +138,19 @@ export class TrueAchievements {
// Found achievement -> add TrueAchievements button // Found achievement -> add TrueAchievements button
if (id) { if (id) {
TrueAchievements.updateLinks(xboxTitleId, id); this.updateIds(xboxTitleId, id);
$parent.appendChild(TrueAchievements.$link); $parent.appendChild(this.$link);
} }
} catch (e) {}; } catch (e) {};
} }
static open(override: boolean, xboxTitleId?: number | string, id?: number | string) { private getStreamXboxTitleId() : number | undefined {
return STATES.currentStream.xboxTitleId || STATES.currentStream.titleInfo?.details.xboxTitleId;
}
open(override: boolean, xboxTitleId?: number | string, id?: number | string) {
if (!xboxTitleId || xboxTitleId === 'undefined') { if (!xboxTitleId || xboxTitleId === 'undefined') {
xboxTitleId = STATES.currentStream.xboxTitleId || STATES.currentStream.titleInfo?.details.xboxTitleId; xboxTitleId = this.getStreamXboxTitleId();
} }
if (AppInterface && AppInterface.openTrueAchievementsLink) { if (AppInterface && AppInterface.openTrueAchievementsLink) {
@ -84,14 +160,14 @@ export class TrueAchievements {
let url = 'https://www.trueachievements.com'; let url = 'https://www.trueachievements.com';
if (xboxTitleId) { if (xboxTitleId) {
if (id && id !== 'undefined') { url += `/deeplink/${xboxTitleId}`;
url += `/deeplink/${xboxTitleId}/${id}`;
} else { if (id) {
url += `/deeplink/${xboxTitleId}`; url += `/${id}`;
} }
} }
TrueAchievements.$hiddenLink.href = url; this.$hiddenLink.href = url;
TrueAchievements.$hiddenLink.click(); this.$hiddenLink.click();
} }
} }

View File

@ -28,7 +28,7 @@ export class UserAgent {
static #USER_AGENTS: PartialRecord<UserAgentProfile, string> = { static #USER_AGENTS: PartialRecord<UserAgentProfile, string> = {
[UserAgentProfile.WINDOWS_EDGE]: `Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${CHROMIUM_VERSION} Safari/537.36 Edg/${CHROMIUM_VERSION}`, [UserAgentProfile.WINDOWS_EDGE]: `Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/${CHROMIUM_VERSION} Safari/537.36 Edg/${CHROMIUM_VERSION}`,
[UserAgentProfile.MACOS_SAFARI]: 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.5.2 Safari/605.1.1', [UserAgentProfile.MACOS_SAFARI]: 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/16.5.2 Safari/605.1.1',
[UserAgentProfile.SMART_TV_GENERIC]: `${window.navigator.userAgent} SmartTV`, [UserAgentProfile.SMART_TV_GENERIC]: `${window.navigator.userAgent} Smart-TV`,
[UserAgentProfile.SMART_TV_TIZEN]: `Mozilla/5.0 (SMART-TV; LINUX; Tizen 7.0) AppleWebKit/537.36 (KHTML, like Gecko) ${CHROMIUM_VERSION}/7.0 TV Safari/537.36 ${SMART_TV_UNIQUE_ID}`, [UserAgentProfile.SMART_TV_TIZEN]: `Mozilla/5.0 (SMART-TV; LINUX; Tizen 7.0) AppleWebKit/537.36 (KHTML, like Gecko) ${CHROMIUM_VERSION}/7.0 TV Safari/537.36 ${SMART_TV_UNIQUE_ID}`,
[UserAgentProfile.VR_OCULUS]: window.navigator.userAgent + ' OculusBrowser VR', [UserAgentProfile.VR_OCULUS]: window.navigator.userAgent + ' OculusBrowser VR',
} }

View File

@ -120,3 +120,15 @@ export function productTitleToSlug(title: string): string {
.replace(/ /g, '-') .replace(/ /g, '-')
.toLowerCase(); .toLowerCase();
} }
export function parseDetailsPath(path: string) {
const matches = /\/games\/(?<titleSlug>[^\/]+)\/(?<productId>\w+)/.exec(path);
if (!matches?.groups) {
return;
}
const titleSlug = matches.groups.titleSlug.replaceAll('\%' + '7C', '-');
const productId = matches.groups.productId;
return {titleSlug, productId};
}

View File

@ -1,23 +1,22 @@
import { NATIVE_FETCH } from "./bx-flags"; import { NATIVE_FETCH } from "./bx-flags";
import { BxLogger } from "./bx-logger";
import { STATES } from "./global"; import { STATES } from "./global";
export class XcloudApi { export class XcloudApi {
private static instance: XcloudApi; private static instance: XcloudApi;
public static getInstance = () => XcloudApi.instance ?? (XcloudApi.instance = new XcloudApi());
private readonly LOG_TAG = 'XcloudApi';
public static getInstance(): XcloudApi { private CACHE_TITLES: {[key: string]: XcloudTitleInfo} = {};
if (!XcloudApi.instance) { private CACHE_WAIT_TIME: {[key: string]: XcloudWaitTimeInfo} = {};
XcloudApi.instance = new XcloudApi();
}
return XcloudApi.instance; private constructor() {
BxLogger.info(this.LOG_TAG, 'constructor()');
} }
#CACHE_TITLES: {[key: string]: XcloudTitleInfo} = {};
#CACHE_WAIT_TIME: {[key: string]: XcloudWaitTimeInfo} = {};
async getTitleInfo(id: string): Promise<XcloudTitleInfo | null> { async getTitleInfo(id: string): Promise<XcloudTitleInfo | null> {
if (id in this.#CACHE_TITLES) { if (id in this.CACHE_TITLES) {
return this.#CACHE_TITLES[id]; return this.CACHE_TITLES[id];
} }
const baseUri = STATES.selectedRegion.baseUri; const baseUri = STATES.selectedRegion.baseUri;
@ -45,13 +44,13 @@ export class XcloudApi {
} catch (e) { } catch (e) {
json = {} json = {}
} }
this.#CACHE_TITLES[id] = json; this.CACHE_TITLES[id] = json;
return json; return json;
} }
async getWaitTime(id: string): Promise<XcloudWaitTimeInfo | null> { async getWaitTime(id: string): Promise<XcloudWaitTimeInfo | null> {
if (id in this.#CACHE_WAIT_TIME) { if (id in this.CACHE_WAIT_TIME) {
return this.#CACHE_WAIT_TIME[id]; return this.CACHE_WAIT_TIME[id];
} }
const baseUri = STATES.selectedRegion.baseUri; const baseUri = STATES.selectedRegion.baseUri;
@ -73,7 +72,7 @@ export class XcloudApi {
json = {}; json = {};
} }
this.#CACHE_WAIT_TIME[id] = json; this.CACHE_WAIT_TIME[id] = json;
return json; return json;
} }
} }

View File

@ -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";
@ -11,9 +13,35 @@ import { BypassServerIps } from "@/enums/bypass-servers";
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";
export export class XcloudInterceptor {
class XcloudInterceptor { private static readonly SERVER_EXTRA_INFO: Record<string, [string, ServerContinent]> = {
static async #handleLogin(request: RequestInfo | URL, init?: RequestInit) { // North America
EastUS: ['🇺🇸', 'america-north'],
EastUS2: ['🇺🇸', 'america-north'],
NorthCentralUs: ['🇺🇸', 'america-north'],
SouthCentralUS: ['🇺🇸', 'america-north'],
WestUS: ['🇺🇸', 'america-north'],
WestUS2: ['🇺🇸', 'america-north'],
MexicoCentral: ['🇲🇽', 'america-north'],
// South America
BrazilSouth: ['🇧🇷', 'america-south'],
// Asia
JapanEast: ['🇯🇵', 'asia'],
KoreaCentral: ['🇰🇷', 'asia'],
// Australia
AustraliaEast: ['🇦🇺', 'australia'],
AustraliaSouthEast: ['🇦🇺', 'australia'],
// Europe
SwedenCentral: ['🇸🇪', 'europe'],
UKSouth: ['🇬🇧', 'europe'],
WestEurope: ['🇪🇺', 'europe'],
};
private static async handleLogin(request: RequestInfo | URL, init?: RequestInit) {
const bypassServer = getPref(PrefKey.SERVER_BYPASS_RESTRICTION); const bypassServer = getPref(PrefKey.SERVER_BYPASS_RESTRICTION);
if (bypassServer !== 'off') { if (bypassServer !== 'off') {
const ip = BypassServerIps[bypassServer as keyof typeof BypassServerIps]; const ip = BypassServerIps[bypassServer as keyof typeof BypassServerIps];
@ -30,30 +58,15 @@ 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 = {
AustraliaEast: '🇦🇺',
AustraliaSouthEast: '🇦🇺',
BrazilSouth: '🇧🇷',
EastUS: '🇺🇸',
EastUS2: '🇺🇸',
JapanEast: '🇯🇵',
KoreaCentral: '🇰🇷',
MexicoCentral: '🇲🇽',
NorthCentralUs: '🇺🇸',
SouthCentralUS: '🇺🇸',
UKSouth: '🇬🇧',
WestEurope: '🇪🇺',
WestUS: '🇺🇸',
WestUS2: '🇺🇸',
};
const serverRegex = /\/\/(\w+)\./; const serverRegex = /\/\/(\w+)\./;
const serverExtra = XcloudInterceptor.SERVER_EXTRA_INFO;
for (let region of obj.offeringSettings.regions) { let region: ServerRegion;
const regionName = region.name as keyof typeof serverEmojis; for (region of obj.offeringSettings.regions) {
const regionName = region.name as keyof typeof serverExtra;
let shortName = region.name; let shortName = region.name;
if (region.isDefault) { if (region.isDefault) {
@ -63,8 +76,11 @@ class XcloudInterceptor {
let match = serverRegex.exec(region.baseUri); let match = serverRegex.exec(region.baseUri);
if (match) { if (match) {
shortName = match[1]; shortName = match[1];
if (serverEmojis[regionName]) { if (serverExtra[regionName]) {
shortName = serverEmojis[regionName] + ' ' + shortName; shortName = serverExtra[regionName][0] + ' ' + shortName;
region.contintent = serverExtra[regionName][1];
} else {
region.contintent = 'other';
} }
} }
@ -89,7 +105,9 @@ class XcloudInterceptor {
return response; return response;
} }
static async #handlePlay(request: RequestInfo | URL, init?: RequestInit) { private static async handlePlay(request: RequestInfo | URL, init?: RequestInit) {
BxEvent.dispatch(window, BxEvent.STREAM_LOADING);
const PREF_STREAM_TARGET_RESOLUTION = getPref(PrefKey.STREAM_TARGET_RESOLUTION); const PREF_STREAM_TARGET_RESOLUTION = getPref(PrefKey.STREAM_TARGET_RESOLUTION);
const PREF_STREAM_PREFERRED_LOCALE = getPref(PrefKey.STREAM_PREFERRED_LOCALE); const PREF_STREAM_PREFERRED_LOCALE = getPref(PrefKey.STREAM_PREFERRED_LOCALE);
@ -127,7 +145,7 @@ class XcloudInterceptor {
return NATIVE_FETCH(newRequest); return NATIVE_FETCH(newRequest);
} }
static async #handleWaitTime(request: RequestInfo | URL, init?: RequestInit) { private static async handleWaitTime(request: RequestInfo | URL, init?: RequestInit) {
const response = await NATIVE_FETCH(request, init); const response = await NATIVE_FETCH(request, init);
if (getPref(PrefKey.UI_LOADING_SCREEN_WAIT_TIME)) { if (getPref(PrefKey.UI_LOADING_SCREEN_WAIT_TIME)) {
@ -141,13 +159,13 @@ class XcloudInterceptor {
return response; return response;
} }
static async #handleConfiguration(request: RequestInfo | URL, init?: RequestInit) { private static async handleConfiguration(request: RequestInfo | URL, init?: RequestInit) {
if ((request as Request).method !== 'GET') { if ((request as Request).method !== 'GET') {
return NATIVE_FETCH(request, init); return NATIVE_FETCH(request, init);
} }
// 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();
@ -163,6 +181,8 @@ class XcloudInterceptor {
return response; return response;
} }
BxEvent.dispatch(window, BxEvent.STREAM_STARTING);
const obj = JSON.parse(text); const obj = JSON.parse(text);
let overrides = JSON.parse(obj.clientStreamingConfigOverrides || '{}') || {}; let overrides = JSON.parse(obj.clientStreamingConfigOverrides || '{}') || {};
@ -187,7 +207,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;
} }
@ -211,13 +231,13 @@ class XcloudInterceptor {
// Server list // Server list
if (url.endsWith('/v2/login/user')) { if (url.endsWith('/v2/login/user')) {
return XcloudInterceptor.#handleLogin(request, init); return XcloudInterceptor.handleLogin(request, init);
} else if (url.endsWith('/sessions/cloud/play')) { // Get session } else if (url.endsWith('/sessions/cloud/play')) { // Get session
return XcloudInterceptor.#handlePlay(request, init); return XcloudInterceptor.handlePlay(request, init);
} else if (url.includes('xboxlive.com') && url.includes('/waittime/')) { } else if (url.includes('xboxlive.com') && url.includes('/waittime/')) {
return XcloudInterceptor.#handleWaitTime(request, init); return XcloudInterceptor.handleWaitTime(request, init);
} else if (url.endsWith('/configuration')) { } else if (url.endsWith('/configuration')) {
return XcloudInterceptor.#handleConfiguration(request, init); return XcloudInterceptor.handleConfiguration(request, init);
} else if (url && url.endsWith('/ice') && url.includes('/sessions/') && (request as Request).method === 'GET') { } else if (url && url.endsWith('/ice') && url.includes('/sessions/') && (request as Request).method === 'GET') {
return patchIceCandidates(request as Request); return patchIceCandidates(request as Request);
} }

View File

@ -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,14 +7,54 @@ 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 = {}; private static consoleAddrs: RemotePlayConsoleAddresses = {};
static async #handleLogin(request: Request) { 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',
},
},
};
private static async handleLogin(request: Request) {
try { try {
const clone = (request as Request).clone(); const clone = request.clone();
const obj = await clone.json(); const obj = await clone.json();
obj.offeringId = 'xhome'; obj.offeringId = 'xhome';
@ -34,31 +73,31 @@ export class XhomeInterceptor {
return NATIVE_FETCH(request); return NATIVE_FETCH(request);
} }
static async #handleConfiguration(request: Request | URL) { private static async handleConfiguration(request: Request | URL) {
BxEvent.dispatch(window, BxEvent.STREAM_STARTING);
const response = await NATIVE_FETCH(request); const response = await NATIVE_FETCH(request);
const obj = await response.clone().json();
const obj = await response.clone().json()
console.log(obj);
const processPorts = (port: number): number[] => {
const ports = new Set<number>();
ports.add(port);
ports.add(9002);
return Array.from(ports);
};
const serverDetails = obj.serverDetails; const serverDetails = obj.serverDetails;
if (serverDetails.ipAddress) { const pairs = [
XhomeInterceptor.#consoleAddrs[serverDetails.ipAddress] = processPorts(serverDetails.port); ['ipAddress', 'port'],
} ['ipV4Address', 'ipV4Port'],
['ipV6Address', 'ipV6Port'],
];
if (serverDetails.ipV4Address) { XhomeInterceptor.consoleAddrs = {};
XhomeInterceptor.#consoleAddrs[serverDetails.ipV4Address] = processPorts(serverDetails.ipV4Port); for (const pair of pairs) {
} const [keyAddr, keyPort] = pair;
if (serverDetails[keyAddr]) {
if (serverDetails.ipV6Address) { const port = serverDetails[keyPort];
XhomeInterceptor.#consoleAddrs[serverDetails.ipV6Address] = processPorts(serverDetails.ipV6Port); // Add port 9002 to the list of ports
const ports = new Set<number>();
port && ports.add(port);
ports.add(9002);
// Save it
XhomeInterceptor.consoleAddrs[serverDetails[keyAddr]] = Array.from(ports);
}
} }
response.json = () => Promise.resolve(obj); response.json = () => Promise.resolve(obj);
@ -67,7 +106,7 @@ export class XhomeInterceptor {
return response; return response;
} }
static async #handleInputConfigs(request: Request | URL, opts: {[index: string]: any}) { private static async handleInputConfigs(request: Request | URL, opts: {[index: string]: any}) {
const response = await NATIVE_FETCH(request); const response = await NATIVE_FETCH(request);
if (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) !== StreamTouchController.ALL) { if (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) !== StreamTouchController.ALL) {
@ -104,14 +143,14 @@ export class XhomeInterceptor {
return response; return response;
} }
static async #handleTitles(request: Request) { private static async handleTitles(request: Request) {
const clone = request.clone(); const clone = request.clone();
const headers: {[index: string]: any} = {}; const headers: {[index: string]: any} = {};
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), {
@ -123,7 +162,9 @@ export class XhomeInterceptor {
return NATIVE_FETCH(request); return NATIVE_FETCH(request);
} }
static async #handlePlay(request: RequestInfo | URL) { private static async handlePlay(request: RequestInfo | URL) {
BxEvent.dispatch(window, BxEvent.STREAM_LOADING);
const clone = (request as Request).clone(); const clone = (request as Request).clone();
const body = await clone.json(); const body = await clone.json();
@ -146,47 +187,49 @@ 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';
} }
headers['x-ms-device-info'] = JSON.stringify(deviceInfo); headers['x-ms-device-info'] = JSON.stringify(deviceInfo);
const opts: {[index: string]: any} = { const opts: Record<string, any> = {
method: clone.method, method: clone.method,
headers: headers, headers: headers,
}; };
// Copy body
if (clone.method === 'POST') { if (clone.method === 'POST') {
opts.body = await clone.text(); opts.body = await clone.text();
} }
let newUrl = request.url; // Replace xCloud domain with xHome domain
if (!newUrl.includes('/servers/home')) { let url = request.url;
const index = request.url.indexOf('.xboxlive.com'); if (!url.includes('/servers/home')) {
newUrl = STATES.remotePlay.server + request.url.substring(index + 13); const parsed = new URL(url);
url = STATES.remotePlay.server + parsed.pathname;
} }
request = new Request(newUrl, opts); // Create new Request instance
let url = (typeof request === 'string') ? request : request.url; request = new Request(url, opts);
// Get console IP // Get console IP
if (url.includes('/configuration')) { if (url.includes('/configuration')) {
return XhomeInterceptor.#handleConfiguration(request); return XhomeInterceptor.handleConfiguration(request);
} else if (url.endsWith('/sessions/home/play')) { } else if (url.endsWith('/sessions/home/play')) {
return XhomeInterceptor.#handlePlay(request); return XhomeInterceptor.handlePlay(request);
} else if (url.includes('inputconfigs')) { } else if (url.includes('inputconfigs')) {
return XhomeInterceptor.#handleInputConfigs(request, opts); return XhomeInterceptor.handleInputConfigs(request, opts);
} else if (url.includes('/login/user')) { } else if (url.includes('/login/user')) {
return XhomeInterceptor.#handleLogin(request); return XhomeInterceptor.handleLogin(request);
} else if (url.endsWith('/titles')) { } else if (url.endsWith('/titles')) {
return XhomeInterceptor.#handleTitles(request); return XhomeInterceptor.handleTitles(request);
} else if (url && url.endsWith('/ice') && url.includes('/sessions/') && (request as Request).method === 'GET') { } else if (url && url.endsWith('/ice') && url.includes('/sessions/') && request.method === 'GET') {
return patchIceCandidates(request, XhomeInterceptor.#consoleAddrs); return patchIceCandidates(request, XhomeInterceptor.consoleAddrs);
} }
return await NATIVE_FETCH(request); return await NATIVE_FETCH(request);

Some files were not shown because too many files have changed in this diff Show More