From 75d4e6f65c0f5384369a40204d47ce532bd4c6aa Mon Sep 17 00:00:00 2001
From: redphx <96280+redphx@users.noreply.github.com>
Date: Sun, 7 Jan 2024 16:59:42 +0700
Subject: [PATCH] Add a "Remote Play" button to the "Jump back in" list
---
better-xcloud.user.js | 98 +++++++++++++++++++++++++++++++++++++++----
1 file changed, 90 insertions(+), 8 deletions(-)
diff --git a/better-xcloud.user.js b/better-xcloud.user.js
index a946f15..bd41040 100644
--- a/better-xcloud.user.js
+++ b/better-xcloud.user.js
@@ -39,6 +39,14 @@ window.NATIVE_MKB_TITLES = [
console.log(`[Better xCloud] readyState: ${document.readyState}`);
+const BxEvent = {
+ JUMP_BACK_IN_READY: 'bx-jump-back-in-ready',
+ POPSTATE: 'bx-popstate',
+
+ STREAM_STARTING: 'bx-stream-starting',
+ STREAM_STARTED: 'bx-stream-started',
+ STREAM_STOPPED: 'bx-stream-stopped',
+};
// Quickly create a tree of elements without having to use innerHTML
function createElement(elmName, props = {}) {
@@ -82,6 +90,7 @@ function createElement(elmName, props = {}) {
}
const CE = createElement;
+window.BX_CE = CE;
const CTN = document.createTextNode.bind(document);
@@ -2719,6 +2728,8 @@ const Icon = {
CURSOR_TEXT: '',
INFO: '',
+ REMOTE_PLAY: '',
+
SCREENSHOT_B64: '',
};
@@ -6609,6 +6620,56 @@ class Patcher {
return funcStr.replace(text, `connectMode:window.BX_REMOTE_PLAY_CONFIG?"xhome-connect":"cloud-connect",remotePlayServerId:(window.BX_REMOTE_PLAY_CONFIG&&window.BX_REMOTE_PLAY_CONFIG.serverId)||''`);
},
+ // Add a "Remote Play" button to the "Jump back in" list
+ remotePlayPatchHomeJumpBackIn: PREFS.get(Preferences.REMOTE_PLAY_ENABLED) && function(funcStr) {
+ const index = funcStr.indexOf('MruTitlesGamesRow-module__childContainer');
+ if (index === -1) {
+ return false;
+ }
+
+ const startIndex = funcStr.indexOf('return(', index);
+ const newCode = `
+setTimeout(() => { window.dispatchEvent(new Event('${BxEvent.JUMP_BACK_IN_READY}')); }, 1000);
+`;
+
+ // Add event listener
+ window.addEventListener(BxEvent.JUMP_BACK_IN_READY, e => {
+ const $list = document.querySelector('ol[class^=ItemRow-module__list]');
+ if (!$list) {
+ return;
+ }
+
+ if ($list.querySelector('.bx-jump-in-li')) {
+ return;
+ }
+
+ const $li = $list.firstElementChild.cloneNode(true);
+ $li.classList.add('bx-jump-in-li');
+
+ $li.addEventListener('click', e => { document.getElementById('bxRemotePlayBtn').click(); });
+
+ const $button = $li.querySelector('button');
+ $button.removeAttribute('id');
+ $button.removeAttribute('aria-labelledby');
+ $button.setAttribute('aria-label', __('remote-play'));
+
+ // Change card's name
+ $button.querySelector('div[class^=GameCard-module__gameTitle___]').textContent = __('remote-play');
+
+ // Rmove icons
+ $button.querySelector('div[class^=GameCard-module__children]').innerHTML = '';
+
+ const $images = $button.querySelector('div[class^=WrappedResponsiveImage]');
+ $images.innerHTML = '';
+ $images.appendChild(createSvgIcon(Icon.REMOTE_PLAY), 2);
+
+ $list.insertBefore($li, $list.firstElementChild);
+ });
+
+ funcStr = funcStr.substring(0, startIndex) + newCode + funcStr.substring(startIndex);
+ return funcStr;
+ },
+
// Disable trackEvent() function
disableTrackEvent: PREFS.get(Preferences.BLOCK_TRACKING) && function(funcStr) {
const text = 'this.trackEvent=';
@@ -6772,13 +6833,15 @@ if (window.BX_VIBRATION_INTENSITY && window.BX_VIBRATION_INTENSITY < 1) {
};
static #PATCH_ORDERS = [
- ['disableStreamGate'],
-
[
'disableAiTrack',
'disableTelemetry',
],
+ ['disableStreamGate'],
+
+ ['remotePlayPatchHomeJumpBackIn'],
+
['tvLayout'],
['enableXcloudLogger'],
@@ -7035,7 +7098,6 @@ function addCss() {
--bx-danger-button-hover-color: #e61d1d;
--bx-danger-button-disabled-color: #a26c6c;
-
--bx-toast-z-index: 9999;
--bx-dialog-z-index: 9101;
--bx-dialog-overlay-z-index: 9100;
@@ -7061,6 +7123,24 @@ div[class^=HUDButton-module__hiddenContainer] ~ div:not([class^=HUDButton-module
left: -9999px;
}
+.bx-jump-in-li {
+ display: inline-block;
+}
+
+.bx-jump-in-li button {
+ width: 140px;
+}
+
+.bx-jump-in-li div[class^=WrappedResponsiveImage-module__imageWrapper] {
+ background: linear-gradient(#005b2f, #151515);
+}
+
+.bx-jump-in-li div[class^=WrappedResponsiveImage-module__imageWrapper] svg {
+ padding: 40px;
+ height: 100%;
+ opacity: 80%;
+}
+
.bx-button {
background-color: var(--bx-default-button-color);
user-select: none;
@@ -8818,7 +8898,7 @@ function injectSettingsButton($parent) {
'href': SCRIPT_HOME,
'target': '_blank',
}, 'Better xCloud ' + SCRIPT_VERSION),
- $remotePlayBtn = CE('button', {'class': 'bx-primary-button bx-no-margin'}, __('remote-play')),
+ $remotePlayBtn = CE('button', {'id': 'bxRemotePlayBtn', 'class': 'bx-primary-button bx-no-margin'}, __('remote-play')),
)
);
$updateAvailable = CE('a', {
@@ -9806,9 +9886,10 @@ function setupScreenshotButton() {
function patchHistoryMethod(type) {
- var orig = window.history[type];
+ const orig = window.history[type];
+
return function(...args) {
- const event = new Event('xcloud_popstate');
+ const event = new Event(BxEvent.POPSTATE);
event.arguments = args;
window.dispatchEvent(event);
@@ -10025,9 +10106,10 @@ function setupBxUi() {
// Hide Settings UI when navigate to another page
-window.addEventListener('xcloud_popstate', onHistoryChanged);
+window.addEventListener(BxEvent.POPSTATE, onHistoryChanged);
window.addEventListener('popstate', onHistoryChanged);
-// Make pushState/replaceState methods dispatch "xcloud_popstate" event
+
+// Make pushState/replaceState methods dispatch BxEvent.POPSTATE event
window.history.pushState = patchHistoryMethod('pushState');
window.history.replaceState = patchHistoryMethod('replaceState');