mirror of
https://github.com/redphx/better-xcloud.git
synced 2025-07-05 05:41:43 +02:00
Compare commits
11 Commits
Author | SHA1 | Date | |
---|---|---|---|
109cd63a7b | |||
8ea6b7f81a | |||
e7c10d43f5 | |||
2f7a57e084 | |||
c99e38b097 | |||
f6ec6d7c9b | |||
e69fa19ef3 | |||
cc422b31a4 | |||
9609d0ae7b | |||
506fd71433 | |||
f40b8cb0b2 |
39
build.ts
39
build.ts
@ -35,13 +35,42 @@ const postProcess = (str: string): string => {
|
|||||||
// Add ADDITIONAL CODE block
|
// Add ADDITIONAL CODE block
|
||||||
str = str.replace('var DEFAULT_FLAGS', '\n/* ADDITIONAL CODE */\n\nvar DEFAULT_FLAGS');
|
str = str.replace('var DEFAULT_FLAGS', '\n/* ADDITIONAL CODE */\n\nvar DEFAULT_FLAGS');
|
||||||
|
|
||||||
// Minify SVG
|
str = str.replaceAll('(e) => `', 'e => `');
|
||||||
str = str.replaceAll(/= "(<svg.*)";/g, function(match) {
|
|
||||||
match = match.replaceAll(/\\n*\s*/g, '');
|
// Simplify object definitions
|
||||||
return match;
|
// {[1]: "a"} => {1: "a"}
|
||||||
|
str = str.replaceAll(/\[(\d+)\]: /g, '$1: ');
|
||||||
|
// {["a"]: 1, ["b-c"]: 2} => {a: 1, "b-c": 2}
|
||||||
|
str = str.replaceAll(/\["([^"]+)"\]: /g, function(match, p1) {
|
||||||
|
if (p1.includes('-') || p1.match(/^\d/)) {
|
||||||
|
p1 = `"${p1}"`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return p1 + ': ';
|
||||||
});
|
});
|
||||||
|
|
||||||
str = str.replaceAll('(e) => `', 'e => `');
|
// Minify SVG import code
|
||||||
|
const svgMap = {}
|
||||||
|
str = str.replaceAll(/var ([\w_]+) = ("<svg.*?");\n\n/g, function(match, p1, p2) {
|
||||||
|
// Remove new lines in SVG
|
||||||
|
p2 = p2.replaceAll(/\\n*\s*/g, '');
|
||||||
|
|
||||||
|
svgMap[p1] = p2;
|
||||||
|
return '';
|
||||||
|
});
|
||||||
|
|
||||||
|
for (const name in svgMap) {
|
||||||
|
str = str.replace(`: ${name}`, `: ${svgMap[name]}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Collapse empty brackets
|
||||||
|
str = str.replaceAll(/\{[\s\n]+\}/g, '{}');
|
||||||
|
|
||||||
|
// Collapse if/else blocks without curly braces
|
||||||
|
str = str.replaceAll(/((if \(.*?\)|else)\n\s+)/g, '$2 ');
|
||||||
|
|
||||||
|
// Remove blank lines
|
||||||
|
str = str.replaceAll(/\n([\s]*)\n/g, "\n");
|
||||||
|
|
||||||
assert(str.includes('/* ADDITIONAL CODE */'));
|
assert(str.includes('/* ADDITIONAL CODE */'));
|
||||||
assert(str.includes('window.BX_EXPOSED = BxExposed'));
|
assert(str.includes('window.BX_EXPOSED = BxExposed'));
|
||||||
|
2
dist/better-xcloud.meta.js
vendored
2
dist/better-xcloud.meta.js
vendored
@ -1,5 +1,5 @@
|
|||||||
// ==UserScript==
|
// ==UserScript==
|
||||||
// @name Better xCloud
|
// @name Better xCloud
|
||||||
// @namespace https://github.com/redphx
|
// @namespace https://github.com/redphx
|
||||||
// @version 5.7.6
|
// @version 5.7.7
|
||||||
// ==/UserScript==
|
// ==/UserScript==
|
||||||
|
2955
dist/better-xcloud.user.js
vendored
2955
dist/better-xcloud.user.js
vendored
File diff suppressed because one or more lines are too long
@ -179,15 +179,3 @@ button.bx-inactive {
|
|||||||
opacity: 0.2;
|
opacity: 0.2;
|
||||||
background: transparent !important;
|
background: transparent !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
.bx-button-shortcut {
|
|
||||||
max-width: max-content;
|
|
||||||
margin: 10px 0 0 0;
|
|
||||||
flex: 1 0 auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (min-width: 568px) and (max-height: 480px) {
|
|
||||||
.bx-button-shortcut {
|
|
||||||
margin: 8px 0 0 10px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
21
src/assets/css/misc.styl
Normal file
21
src/assets/css/misc.styl
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
.bx-product-details-buttons {
|
||||||
|
display: flex;
|
||||||
|
gap: 10px;
|
||||||
|
flex-direction: row;
|
||||||
|
|
||||||
|
button {
|
||||||
|
max-width: max-content;
|
||||||
|
margin: 10px 0 0 0;
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 568px) and (max-height: 480px) {
|
||||||
|
.bx-product-details-buttons {
|
||||||
|
flex-direction: column;
|
||||||
|
|
||||||
|
button {
|
||||||
|
margin: 8px 0 0 10px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -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';
|
||||||
|
@ -67,8 +67,10 @@ if (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',{
|
||||||
|
@ -314,6 +314,7 @@ export class ControllerShortcut {
|
|||||||
const $selectProfile = CE<HTMLSelectElement>('select', {class: 'bx-shortcut-profile', autocomplete: 'off'});
|
const $selectProfile = CE<HTMLSelectElement>('select', {class: 'bx-shortcut-profile', autocomplete: 'off'});
|
||||||
|
|
||||||
const $profile = PREF_CONTROLLER_FRIENDLY_UI ? BxSelectElement.wrap($selectProfile) : $selectProfile;
|
const $profile = PREF_CONTROLLER_FRIENDLY_UI ? BxSelectElement.wrap($selectProfile) : $selectProfile;
|
||||||
|
$profile.classList.add('bx-full-width');
|
||||||
|
|
||||||
const $container = CE('div', {
|
const $container = CE('div', {
|
||||||
'data-has-gamepad': 'false',
|
'data-has-gamepad': 'false',
|
||||||
@ -390,6 +391,8 @@ export class ControllerShortcut {
|
|||||||
|
|
||||||
if (PREF_CONTROLLER_FRIENDLY_UI) {
|
if (PREF_CONTROLLER_FRIENDLY_UI) {
|
||||||
const $bxSelect = BxSelectElement.wrap($select);
|
const $bxSelect = BxSelectElement.wrap($select);
|
||||||
|
$bxSelect.classList.add('bx-full-width');
|
||||||
|
|
||||||
$div.appendChild($bxSelect);
|
$div.appendChild($bxSelect);
|
||||||
setNearby($row, {
|
setNearby($row, {
|
||||||
focus: $bxSelect,
|
focus: $bxSelect,
|
||||||
|
@ -44,7 +44,7 @@ export class LoadingScreen {
|
|||||||
static #hideRocket() {
|
static #hideRocket() {
|
||||||
let $bgStyle = LoadingScreen.#$bgStyle;
|
let $bgStyle = LoadingScreen.#$bgStyle;
|
||||||
|
|
||||||
const css = compressCss(`
|
$bgStyle.textContent! += compressCss(`
|
||||||
#game-stream div[class*=RocketAnimation-module__container] > svg {
|
#game-stream div[class*=RocketAnimation-module__container] > svg {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
@ -53,7 +53,6 @@ export class LoadingScreen {
|
|||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
`);
|
`);
|
||||||
$bgStyle.textContent! += css;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static #setBackground(imageUrl: string) {
|
static #setBackground(imageUrl: string) {
|
||||||
@ -63,7 +62,7 @@ export class LoadingScreen {
|
|||||||
// Limit max width to reduce image size
|
// Limit max width to reduce image size
|
||||||
imageUrl = imageUrl + '?w=1920';
|
imageUrl = imageUrl + '?w=1920';
|
||||||
|
|
||||||
const css = compressCss(`
|
$bgStyle.textContent! += compressCss(`
|
||||||
#game-stream {
|
#game-stream {
|
||||||
background-color: transparent !important;
|
background-color: transparent !important;
|
||||||
background-position: center center !important;
|
background-position: center center !important;
|
||||||
@ -75,7 +74,6 @@ export class LoadingScreen {
|
|||||||
transition: opacity 0.3s ease-in-out !important;
|
transition: opacity 0.3s ease-in-out !important;
|
||||||
}
|
}
|
||||||
`) + `#game-stream {background-image: linear-gradient(#00000033, #000000e6), url(${imageUrl}) !important;}`;
|
`) + `#game-stream {background-image: linear-gradient(#00000033, #000000e6), url(${imageUrl}) !important;}`;
|
||||||
$bgStyle.textContent! += css;
|
|
||||||
|
|
||||||
const bg = new Image();
|
const bg = new Image();
|
||||||
bg.onload = e => {
|
bg.onload = e => {
|
||||||
|
@ -171,28 +171,42 @@ export class NavigationDialogManager {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Find un-calculated <select> elements
|
// Find un-calculated <select> elements
|
||||||
const $selects = $dialog.querySelectorAll('.bx-select:not([data-calculated]) select');
|
this.calculateSelectBoxes($dialog);
|
||||||
|
});
|
||||||
|
observer.observe(this.$container, {childList: true});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
calculateSelectBoxes($root: HTMLElement) {
|
||||||
|
const $selects = $root.querySelectorAll('.bx-select:not([data-calculated]) select');
|
||||||
$selects.forEach($select => {
|
$selects.forEach($select => {
|
||||||
const rect = $select.getBoundingClientRect();
|
|
||||||
const $parent = $select.parentElement! as HTMLElement;
|
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';
|
$parent.dataset.calculated = 'true';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const rect = $select.getBoundingClientRect();
|
||||||
|
|
||||||
let $label;
|
let $label;
|
||||||
let width = Math.ceil(rect.width);
|
let width = Math.ceil(rect.width);
|
||||||
|
if (!width) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (($select as HTMLSelectElement).multiple) {
|
if (($select as HTMLSelectElement).multiple) {
|
||||||
$label = $parent.querySelector('.bx-select-value') as HTMLElement;
|
$label = $parent.querySelector('.bx-select-value') as HTMLElement;
|
||||||
width += 15; // Add checkbox's width
|
width += 20; // Add checkbox's width
|
||||||
} else {
|
} else {
|
||||||
$label = $parent.querySelector('div') as HTMLElement;
|
$label = $parent.querySelector('div') as HTMLElement;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set min-width
|
// Set min-width
|
||||||
$label.style.minWidth = width + 'px';
|
$label.style.minWidth = width + 'px';
|
||||||
|
$parent.dataset.calculated = 'true';
|
||||||
});
|
});
|
||||||
});
|
|
||||||
observer.observe(this.$container, {childList: true});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
handleEvent(event: Event) {
|
handleEvent(event: Event) {
|
||||||
|
@ -943,6 +943,11 @@ export class SettingsNavigationDialog extends NavigationDialog {
|
|||||||
for (const $child of Array.from(this.$settings.children)) {
|
for (const $child of Array.from(this.$settings.children)) {
|
||||||
if ($child.getAttribute('data-tab-group') === settingTab.group) {
|
if ($child.getAttribute('data-tab-group') === settingTab.group) {
|
||||||
$child.classList.remove('bx-gone');
|
$child.classList.remove('bx-gone');
|
||||||
|
|
||||||
|
// Calculate size of controller-friendly select boxes
|
||||||
|
if (getPref(PrefKey.UI_CONTROLLER_FRIENDLY)) {
|
||||||
|
this.dialogManager.calculateSelectBoxes($child as HTMLElement);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
$child.classList.add('bx-gone');
|
$child.classList.add('bx-gone');
|
||||||
}
|
}
|
||||||
|
@ -1,12 +1,11 @@
|
|||||||
import { BX_FLAGS } from "@/utils/bx-flags";
|
import { BX_FLAGS } from "@/utils/bx-flags";
|
||||||
import { BxIcon } from "@/utils/bx-icon";
|
import { BxIcon } from "@/utils/bx-icon";
|
||||||
import { AppInterface } from "@/utils/global";
|
import { AppInterface } from "@/utils/global";
|
||||||
import { ButtonStyle, createButton } from "@/utils/html";
|
import { ButtonStyle, CE, createButton } from "@/utils/html";
|
||||||
import { t } from "@/utils/translation";
|
import { t } from "@/utils/translation";
|
||||||
|
|
||||||
export class ProductDetailsPage {
|
export class ProductDetailsPage {
|
||||||
private static $btnShortcut = AppInterface && createButton({
|
private static $btnShortcut = AppInterface && createButton({
|
||||||
classes: ['bx-button-shortcut'],
|
|
||||||
icon: BxIcon.CREATE_SHORTCUT,
|
icon: BxIcon.CREATE_SHORTCUT,
|
||||||
label: t('create-shortcut'),
|
label: t('create-shortcut'),
|
||||||
style: ButtonStyle.FOCUSABLE,
|
style: ButtonStyle.FOCUSABLE,
|
||||||
@ -17,7 +16,6 @@ export class ProductDetailsPage {
|
|||||||
});
|
});
|
||||||
|
|
||||||
private static $btnWallpaper = AppInterface && createButton({
|
private static $btnWallpaper = AppInterface && createButton({
|
||||||
classes: ['bx-button-shortcut'],
|
|
||||||
icon: BxIcon.DOWNLOAD,
|
icon: BxIcon.DOWNLOAD,
|
||||||
label: t('wallpaper'),
|
label: t('wallpaper'),
|
||||||
style: ButtonStyle.FOCUSABLE,
|
style: ButtonStyle.FOCUSABLE,
|
||||||
@ -48,17 +46,12 @@ export class ProductDetailsPage {
|
|||||||
// Find action buttons container
|
// Find action buttons container
|
||||||
const $container = document.querySelector('div[class*=ActionButtons-module__container]');
|
const $container = document.querySelector('div[class*=ActionButtons-module__container]');
|
||||||
if ($container && $container.parentElement) {
|
if ($container && $container.parentElement) {
|
||||||
const fragment = document.createDocumentFragment();
|
$container.parentElement.appendChild(CE('div', {
|
||||||
|
class: 'bx-product-details-buttons',
|
||||||
// Shortcut button
|
},
|
||||||
if (BX_FLAGS.DeviceInfo.deviceType === 'android') {
|
BX_FLAGS.DeviceInfo.deviceType === 'android' && ProductDetailsPage.$btnShortcut,
|
||||||
fragment.appendChild(ProductDetailsPage.$btnShortcut);
|
ProductDetailsPage.$btnWallpaper,
|
||||||
}
|
));
|
||||||
|
|
||||||
// Wallpaper button
|
|
||||||
fragment.appendChild(ProductDetailsPage.$btnWallpaper);
|
|
||||||
|
|
||||||
$container.parentElement.appendChild(fragment);
|
|
||||||
}
|
}
|
||||||
}, 500);
|
}, 500);
|
||||||
}
|
}
|
||||||
|
@ -171,7 +171,7 @@ const Texts = {
|
|||||||
(e: any) => `Dostępna jest nowa wersja ${e.version}`,
|
(e: any) => `Dostępna jest nowa wersja ${e.version}`,
|
||||||
(e: any) => `Versão ${e.version} disponível`,
|
(e: any) => `Versão ${e.version} disponível`,
|
||||||
,
|
,
|
||||||
,
|
(e: any) => `เวอร์ชัน ${e.version} พร้อมใช้งานแล้ว`,
|
||||||
,
|
,
|
||||||
(e: any) => `Доступна версія ${e.version}`,
|
(e: any) => `Доступна версія ${e.version}`,
|
||||||
(e: any) => `Đã có phiên bản ${e.version}`,
|
(e: any) => `Đã có phiên bản ${e.version}`,
|
||||||
@ -232,7 +232,7 @@ const Texts = {
|
|||||||
(e: any) => `Zalecane ustawienia dla ${e.device}`,
|
(e: any) => `Zalecane ustawienia dla ${e.device}`,
|
||||||
(e: any) => `Configurações recomendadas para ${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}`,
|
||||||
|
Reference in New Issue
Block a user