diff --git a/better-xcloud.user.js b/better-xcloud.user.js index a478845..028655a 100644 --- a/better-xcloud.user.js +++ b/better-xcloud.user.js @@ -47,6 +47,8 @@ const BxEvent = { STREAM_STARTING: 'bx-stream-starting', STREAM_STARTED: 'bx-stream-started', STREAM_STOPPED: 'bx-stream-stopped', + + CUSTOM_TOUCH_LAYOUTS_LOADED: 'bx-custom-touch-layouts-loaded', }; // Quickly create a tree of elements without having to use innerHTML @@ -2796,6 +2798,8 @@ const Icon = { REMOTE_PLAY: '', + HAND_TAP: '', + SCREENSHOT_B64: '', }; @@ -3362,6 +3366,7 @@ class TouchController { static #dataChannel; static #customLayouts = {}; + static #currentLayoutId; static enable() { TouchController.#enable = true; @@ -3381,7 +3386,7 @@ class TouchController { } static #show() { - TouchController.loadCustomLayout(GAME_XBOX_TITLE_ID, 0); + TouchController.loadCustomLayout(GAME_XBOX_TITLE_ID, TouchController.#currentLayoutId, 0); TouchController.#showing = true; } @@ -3417,10 +3422,16 @@ class TouchController { }, 10); } - static #getCustomLayout(xboxTitleId, callback) { + static getCustomLayouts(xboxTitleId) { + const dispatchLayouts = data => { + const event = new Event(BxEvent.CUSTOM_TOUCH_LAYOUTS_LOADED); + event.data = data; + window.dispatchEvent(event); + }; + xboxTitleId = '' + xboxTitleId; if (xboxTitleId in TouchController.#customLayouts) { - callback(TouchController.#customLayouts[xboxTitleId]); + dispatchLayouts(TouchController.#customLayouts[xboxTitleId]); return; } @@ -3430,43 +3441,65 @@ class TouchController { } else { url += `${xboxTitleId}.json`; } - window.BX_EXPOSED.touch_layout_manager && NATIVE_FETCH(url) + NATIVE_FETCH(url) .then(resp => resp.json(), () => { TouchController.#customLayouts[xboxTitleId] = null; - callback(null); + // Wait for BX_EXPOSED.touch_layout_manager + setTimeout(() => dispatchLayouts(null), 1000); }) .then(json => { + // Normalize data + const schema_version = json.schema_version || 1; + let layout; + try { + if (schema_version === 1) { + json.layouts = { + default: { + name: 'Default', + content: json.layout, + }, + }; + json.default_layout = 'default'; + delete json.layout; + } + } catch (e) {} + TouchController.#customLayouts[xboxTitleId] = json; - callback(json); + + // Wait for BX_EXPOSED.touch_layout_manager + setTimeout(() => dispatchLayouts(json), 1000); }); } - static loadCustomLayout(xboxTitleId, delay) { + static loadCustomLayout(xboxTitleId, layoutId, delay) { if (!window.BX_EXPOSED.touch_layout_manager) { return; } + TouchController.#currentLayoutId = layoutId; xboxTitleId = '' + xboxTitleId; - TouchController.#getCustomLayout(xboxTitleId, json => { - if (!json) { - return; - } - setTimeout(() => { - window.BX_EXPOSED.touch_layout_manager.changeLayoutForScope({ - type: 'showLayout', - scope: xboxTitleId, - subscope: 'base', - layout: { - id: 'System.Standard', - displayName: 'System', - layoutFile: { - content: json.layout, - }, - } - }); - }, delay); + const layoutData = TouchController.#customLayouts[xboxTitleId]; + if (!xboxTitleId || !layoutId || !layoutData) { + TouchController.#enable && TouchController.#showDefault(); + return; + } + + const layout = (layoutData.layouts[layoutId] || layoutData.layouts[layoutData.default_layout]); + layout && setTimeout(() => { + window.BX_EXPOSED.touch_layout_manager.changeLayoutForScope({ + type: 'showLayout', + scope: xboxTitleId, + subscope: 'base', + layout: { + id: 'System.Standard', + displayName: 'System', + layoutFile: { + content: layout.content, + }, + } }); + }, delay); } static setup() { @@ -3501,7 +3534,7 @@ class TouchController { const nativeCreateDataChannel = RTCPeerConnection.prototype.createDataChannel; RTCPeerConnection.prototype.createDataChannel = function() { const dataChannel = nativeCreateDataChannel.apply(this, arguments); - if (!TouchController.#enable || dataChannel.label !== 'message') { + if (dataChannel.label !== 'message') { return dataChannel; } @@ -3533,27 +3566,20 @@ class TouchController { return; } + // Dispatch a message to display generic touch controller + if (msg.data.includes('touchcontrols/showtitledefault')) { + TouchController.#enable && TouchController.getCustomLayouts(GAME_XBOX_TITLE_ID); + return; + } + // Load custom touch layout try { if (msg.data.includes('/titleinfo')) { const json = JSON.parse(JSON.parse(msg.data).content); - if (json.focused) { - const xboxTitleId = parseInt(json.titleid, 16); - GAME_XBOX_TITLE_ID = xboxTitleId; - TouchController.loadCustomLayout(xboxTitleId, 1000); - } else { - GAME_XBOX_TITLE_ID = null; - } - - return; + const xboxTitleId = parseInt(json.titleid, 16); + GAME_XBOX_TITLE_ID = xboxTitleId; } } catch (e) { console.log(e) } - - - // Dispatch a message to display generic touch controller - if (msg.data.includes('touchcontrols/showtitledefault')) { - TouchController.#show(); - } }); return dataChannel; @@ -8764,7 +8790,7 @@ function interceptHttpRequests() { } if (IS_REMOTE_PLAYING && (url.includes('/sessions/home') || url.includes('inputconfigs'))) { - TouchController.enable(); + TouchController.disable(); const clone = request.clone(); @@ -8818,6 +8844,31 @@ function interceptHttpRequests() { response.json = () => Promise.resolve(obj); response.text = () => Promise.resolve(JSON.stringify(obj)); + return response; + }); + }); + } else if (PREF_STREAM_TOUCH_CONTROLLER === 'all' && url.includes('inputconfigs')) { + const promise = NATIVE_FETCH(...arg); + + return promise.then(response => { + return response.clone().json().then(obj => { + if (obj[0].supportedTabs.length > 0) { + TouchController.disable(); + + const event = new Event(BxEvent.CUSTOM_TOUCH_LAYOUTS_LOADED); + event.data = null; + window.dispatchEvent(event); + } else { + TouchController.enable(); + + const xboxTitleId = JSON.parse(opts.body).titleIds[0]; + GAME_XBOX_TITLE_ID = xboxTitleId; + TouchController.getCustomLayouts(xboxTitleId); + } + + response.json = () => Promise.resolve(obj); + response.text = () => Promise.resolve(JSON.stringify(obj)); + return response; }); }); @@ -9041,7 +9092,7 @@ function interceptHttpRequests() { }); } - if (PREF_STREAM_TOUCH_CONTROLLER === 'all' && (url.endsWith('/titles') || url.endsWith('/mru'))) { + if (PREF_STREAM_TOUCH_CONTROLLER === 'all' && (url.includes('/titles') || url.includes('/mru'))) { const promise = NATIVE_FETCH(...arg); return promise.then(response => { return response.clone().json().then(json => { @@ -9837,8 +9888,9 @@ function setupQuickSettingsBar() { group: 'audio', label: __('audio'), help_url: 'https://better-xcloud.github.io/ingame-features/#audio', - items: { - [Preferences.AUDIO_VOLUME]: { + items: [ + { + pref: Preferences.AUDIO_VOLUME, label: __('volume'), onChange: (e, value) => { STREAM_AUDIO_GAIN_NODE && (STREAM_AUDIO_GAIN_NODE.gain.value = (value / 100).toFixed(2)); @@ -9847,7 +9899,7 @@ function setupQuickSettingsBar() { disabled: !getPref(Preferences.AUDIO_ENABLE_VOLUME_CONTROL), }, }, - }, + ], }, { @@ -9855,33 +9907,38 @@ function setupQuickSettingsBar() { label: __('video'), help_url: 'https://better-xcloud.github.io/ingame-features/#video', note: CE('div', {'class': 'bx-quick-settings-bar-note bx-clarity-boost-warning'}, `⚠️ ${__('clarity-boost-warning')}`), - items: { - [Preferences.VIDEO_RATIO]: { + items: [ + { + pref: Preferences.VIDEO_RATIO, label: __('ratio'), onChange: updateVideoPlayerCss, }, - [Preferences.VIDEO_CLARITY]: { + { + pref: Preferences.VIDEO_CLARITY, label: __('clarity'), onChange: updateVideoPlayerCss, unsupported: isSafari, }, - [Preferences.VIDEO_SATURATION]: { + { + pref: Preferences.VIDEO_SATURATION, label: __('saturation'), onChange: updateVideoPlayerCss, }, - [Preferences.VIDEO_CONTRAST]: { + { + pref: Preferences.VIDEO_CONTRAST, label: __('contrast'), onChange: updateVideoPlayerCss, }, - [Preferences.VIDEO_BRIGHTNESS]: { + { + pref: Preferences.VIDEO_BRIGHTNESS, label: __('brightness'), onChange: updateVideoPlayerCss, }, - }, + ], }, ], }, @@ -9894,29 +9951,91 @@ function setupQuickSettingsBar() { group: 'controller', label: __('controller'), help_url: 'https://better-xcloud.github.io/ingame-features/#controller', - items: { - [Preferences.CONTROLLER_ENABLE_VIBRATION]: { + items: [ + { + pref: Preferences.CONTROLLER_ENABLE_VIBRATION, label: __('controller-vibration'), unsupported: !VibrationManager.supportControllerVibration(), onChange: VibrationManager.updateGlobalVars, }, - [Preferences.CONTROLLER_DEVICE_VIBRATION]: { + { + pref: Preferences.CONTROLLER_DEVICE_VIBRATION, label: __('device-vibration'), unsupported: !VibrationManager.supportDeviceVibration(), onChange: VibrationManager.updateGlobalVars, }, - [Preferences.CONTROLLER_VIBRATION_INTENSITY]: (VibrationManager.supportControllerVibration() || VibrationManager.supportDeviceVibration()) && { + (VibrationManager.supportControllerVibration() || VibrationManager.supportDeviceVibration()) && { + pref: Preferences.CONTROLLER_VIBRATION_INTENSITY, label: __('vibration-intensity'), unsupported: !VibrationManager.supportDeviceVibration(), onChange: VibrationManager.updateGlobalVars, }, - }, + ], }, ], }, + HAS_TOUCH_SUPPORT && { + icon: Icon.HAND_TAP, + group: 'touch-controller', + items: [ + { + group: 'touch-controller', + label: __('touch-controller'), + items: [ + { + label: __('layout'), + content: CE('select', {disabled: true}, CE('option', {}, __('default'))), + onMounted: $elm => { + $elm.addEventListener('change', e => { + TouchController.loadCustomLayout(GAME_XBOX_TITLE_ID, $elm.value, 1000); + }); + + window.addEventListener(BxEvent.CUSTOM_TOUCH_LAYOUTS_LOADED, e => { + const data = e.data; + + if (GAME_XBOX_TITLE_ID && $elm.xboxTitleId === GAME_XBOX_TITLE_ID) { + $elm.dispatchEvent(new Event('change')); + return; + } + + $elm.xboxTitleId = GAME_XBOX_TITLE_ID; + + // Clear options + while ($elm.firstChild) { + $elm.removeChild($elm.firstChild); + } + + $elm.disabled = !data; + if (!data) { + $elm.appendChild(CE('option', {value: ''}, __('default'))); + $elm.value = ''; + $elm.dispatchEvent(new Event('change')); + return; + } + + // Add options + const $fragment = document.createDocumentFragment(); + for (const key in data.layouts) { + const layout = data.layouts[key]; + + const $option = CE('option', {value: key}, layout.name); + $fragment.appendChild($option); + } + + $elm.appendChild($fragment); + $elm.value = data.default_layout; + $elm.dispatchEvent(new Event('change')); + }); + }, + }, + ], + } + ], + }, + { icon: Icon.STREAM_STATS, group: 'stats', @@ -9925,41 +10044,49 @@ function setupQuickSettingsBar() { group: 'stats', label: __('menu-stream-stats'), help_url: 'https://better-xcloud.github.io/stream-stats/', - items: { - [Preferences.STATS_SHOW_WHEN_PLAYING]: { + items: [ + { + pref: Preferences.STATS_SHOW_WHEN_PLAYING, label: __('show-stats-on-startup'), }, - [Preferences.STATS_QUICK_GLANCE]: { + { + pref: Preferences.STATS_QUICK_GLANCE, label: __('enable-quick-glance-mode'), onChange: e => { e.target.checked ? StreamStats.quickGlanceSetup() : StreamStats.quickGlanceStop(); }, }, - [Preferences.STATS_ITEMS]: { + { + pref: Preferences.STATS_ITEMS, label: __('stats'), onChange: StreamStats.refreshStyles, }, - [Preferences.STATS_POSITION]: { + { + pref: Preferences.STATS_POSITION, label: __('position'), onChange: StreamStats.refreshStyles, }, - [Preferences.STATS_TEXT_SIZE]: { + { + pref: Preferences.STATS_TEXT_SIZE, label: __('text-size'), onChange: StreamStats.refreshStyles, }, - [Preferences.STATS_OPACITY]: { + { + pref: Preferences.STATS_OPACITY, label: __('opacity'), onChange: StreamStats.refreshStyles, }, - [Preferences.STATS_TRANSPARENT]: { + { + pref: Preferences.STATS_TRANSPARENT, label: __('transparent-background'), onChange: StreamStats.refreshStyles, }, - [Preferences.STATS_CONDITIONAL_FORMATTING]: { + { + pref: Preferences.STATS_CONDITIONAL_FORMATTING, label: __('conditional-formatting'), onChange: StreamStats.refreshStyles, }, - }, + ], }, ], }, @@ -10034,14 +10161,21 @@ function setupQuickSettingsBar() { continue; } - for (const pref in settingGroup.items) { - const setting = settingGroup.items[pref]; + if (!settingGroup.items) { + settingGroup.items = []; + } + + for (const setting of settingGroup.items) { if (!setting) { continue; } + const pref = setting.pref; + let $control; - if (!setting.unsupported) { + if (setting.content) { + $control = setting.content; + } else if (!setting.unsupported) { $control = PREFS.toElement(pref, setting.onChange, setting.params); } @@ -10054,6 +10188,8 @@ function setupQuickSettingsBar() { ); $group.appendChild($content); + + setting.onMounted && setting.onMounted($control); } }