From b2e932cc4c438827b920fdcfb60ae026e714490f Mon Sep 17 00:00:00 2001
From: redphx <96280+redphx@users.noreply.github.com>
Date: Fri, 10 May 2024 18:35:40 +0700
Subject: [PATCH] Game bar (#392)
* Fix games with custom touch control sometimes not showing touch icon
* Create game-bar with screenshot button
* Disable Game bar when opening the Guide
* Remove SCREENSHOT_BUTTON_POSITION pref
* Make the touch control action functional
* Show game bar when the game starts
* Fix 720p/High not working (#387)
* Update icons
* Update game bar's animations
* Reset states of Game bar actions before playing
* Don't show Touch control action on non-touch-supported devices
* Clean up
* Update translations
* Update actions' texts
* Clean up
---
src/assets/css/game-bar.styl | 91 ++++++++++
src/assets/css/root.styl | 3 +-
src/assets/css/stream-actions.styl | 46 -----
src/assets/css/styles.styl | 2 +-
src/assets/svg/camera.svg | 6 +
src/assets/svg/caret-right.svg | 5 +
src/assets/svg/touch-control-disable.svg | 9 +
src/assets/svg/touch-control-enable.svg | 6 +
src/index.ts | 41 +----
src/modules/game-bar/action-base.ts | 6 +
src/modules/game-bar/action-screenshot.ts | 78 +++++++++
src/modules/game-bar/action-touch-control.ts | 51 ++++++
src/modules/game-bar/game-bar.ts | 116 +++++++++++++
src/modules/screenshot.ts | 100 -----------
src/modules/touch-controller.ts | 59 ++-----
src/modules/ui/global-settings.ts | 1 -
src/modules/ui/ui.ts | 7 +-
src/utils/bx-event.ts | 2 +
src/utils/bx-exposed.ts | 22 +--
src/utils/bx-icon.ts | 9 +
src/utils/network.ts | 7 +-
src/utils/preferences.ts | 10 --
src/utils/translation.ts | 171 ++++++++++++-------
23 files changed, 533 insertions(+), 315 deletions(-)
create mode 100644 src/assets/css/game-bar.styl
delete mode 100644 src/assets/css/stream-actions.styl
create mode 100644 src/assets/svg/camera.svg
create mode 100644 src/assets/svg/caret-right.svg
create mode 100644 src/assets/svg/touch-control-disable.svg
create mode 100644 src/assets/svg/touch-control-enable.svg
create mode 100644 src/modules/game-bar/action-base.ts
create mode 100644 src/modules/game-bar/action-screenshot.ts
create mode 100644 src/modules/game-bar/action-touch-control.ts
create mode 100644 src/modules/game-bar/game-bar.ts
delete mode 100644 src/modules/screenshot.ts
diff --git a/src/assets/css/game-bar.styl b/src/assets/css/game-bar.styl
new file mode 100644
index 0000000..84cd586
--- /dev/null
+++ b/src/assets/css/game-bar.styl
@@ -0,0 +1,91 @@
+#bx-game-bar {
+ z-index: var(--bx-game-bar-z-index);
+ position: fixed;
+ left: 0;
+ bottom: 0;
+ width: 40px;
+ height: 90px;
+ overflow: visible;
+ cursor: pointer;
+
+ > svg {
+ display: none;
+ pointer-events: none;
+ position: absolute;
+ height: 28px;
+ margin-top: 16px;
+ }
+
+ @media (hover: hover) {
+ &:hover {
+ > svg {
+ display: block;
+ }
+ }
+ }
+
+ .bx-game-bar-container {
+ opacity: 0;
+ position absolute;
+ display: flex;
+ overflow: hidden;
+ background: #1a1b1ee8;
+ border-radius: 0 10px 10px 0;
+ box-shadow: 0px 0px 6px #1c1c1c;
+ transition: opacity 0.1s ease-in;
+
+ &.bx-show {
+ opacity: 1;
+
+ + svg {
+ display: none !important;
+ }
+ }
+
+ &.bx-hide {
+ opacity: 0;
+ }
+
+ button {
+ width: 60px;
+ height: 60px;
+
+ svg {
+ width: 28px;
+ height: 28px;
+ transition: transform 0.08s ease 0s;
+ }
+
+ &:hover {
+ border-radius: 0;
+ }
+
+ &:active {
+ svg {
+ transform: scale(0.75);
+ }
+ }
+ }
+
+ /* Touch controller buttons */
+ div[data-enabled] {
+ button {
+ display: none;
+ }
+ }
+
+ /* Show disable button */
+ div[data-enabled='true'] {
+ button:last-of-type {
+ display: block;
+ }
+ }
+
+ /* Show enable button */
+ div[data-enabled='false'] {
+ button:first-of-type {
+ display: block;
+ }
+ }
+ }
+}
diff --git a/src/assets/css/root.styl b/src/assets/css/root.styl
index 4ac4a7e..56f5e13 100644
--- a/src/assets/css/root.styl
+++ b/src/assets/css/root.styl
@@ -27,8 +27,7 @@
--bx-stats-bar-z-index: 9001;
--bx-stream-settings-z-index: 9000;
--bx-mkb-pointer-lock-msg-z-index: 8999;
- --bx-screenshot-z-index: 8888;
- --bx-touch-controller-bar-z-index: 5555;
+ --bx-game-bar-z-index: 8888;
--bx-wait-time-box-z-index: 100;
}
diff --git a/src/assets/css/stream-actions.styl b/src/assets/css/stream-actions.styl
deleted file mode 100644
index db6d1de..0000000
--- a/src/assets/css/stream-actions.styl
+++ /dev/null
@@ -1,46 +0,0 @@
-.bx-screenshot-button {
- display: none;
- opacity: 0;
- position: fixed;
- bottom: 0;
- box-sizing: border-box;
- width: 60px;
- height: 90px;
- padding: 16px 16px 46px 16px;
- background-size: cover;
- background-repeat: no-repeat;
- background-origin: content-box;
- filter: drop-shadow(0 0 2px #000000B0);
- transition: opacity 0.1s ease-in-out 0s, padding 0.1s ease-in 0s;
- z-index: var(--bx-screenshot-z-index);
-
- /* Credit: https://phosphoricons.com */
- background-image: url('');
-
- &[data-showing=true] {
- opacity: 0.9;
- }
-
- &[data-capturing=true] {
- padding: 8px 8px 38px 8px;
- }
-}
-
-.bx-screenshot-canvas {
- display: none;
-}
-
-#bx-touch-controller-bar {
- display: none;
- opacity: 0;
- position: fixed;
- bottom: 0;
- left: 0;
- right: 0;
- height: 6vh;
- z-index: var(--bx-touch-controller-bar-z-index);
-
- &[data-showing=true] {
- display: block;
- }
-}
diff --git a/src/assets/css/styles.styl b/src/assets/css/styles.styl
index 475bea2..d9c6780 100644
--- a/src/assets/css/styles.styl
+++ b/src/assets/css/styles.styl
@@ -10,7 +10,7 @@
@import 'stream.styl';
@import 'number-stepper.styl';
-@import 'stream-actions.styl';
+@import 'game-bar.styl';
@import 'stream-stats.styl';
@import 'stream-settings.styl';
@import 'mkb.styl';
diff --git a/src/assets/svg/camera.svg b/src/assets/svg/camera.svg
new file mode 100644
index 0000000..5be4147
--- /dev/null
+++ b/src/assets/svg/camera.svg
@@ -0,0 +1,6 @@
+
diff --git a/src/assets/svg/caret-right.svg b/src/assets/svg/caret-right.svg
new file mode 100644
index 0000000..804de29
--- /dev/null
+++ b/src/assets/svg/caret-right.svg
@@ -0,0 +1,5 @@
+
diff --git a/src/assets/svg/touch-control-disable.svg b/src/assets/svg/touch-control-disable.svg
new file mode 100644
index 0000000..a3a29d7
--- /dev/null
+++ b/src/assets/svg/touch-control-disable.svg
@@ -0,0 +1,9 @@
+
diff --git a/src/assets/svg/touch-control-enable.svg b/src/assets/svg/touch-control-enable.svg
new file mode 100644
index 0000000..2764d91
--- /dev/null
+++ b/src/assets/svg/touch-control-enable.svg
@@ -0,0 +1,6 @@
+
diff --git a/src/index.ts b/src/index.ts
index 899008f..be9ad9f 100644
--- a/src/index.ts
+++ b/src/index.ts
@@ -11,7 +11,7 @@ import { StreamBadges } from "@modules/stream/stream-badges";
import { StreamStats } from "@modules/stream/stream-stats";
import { addCss } from "@utils/css";
import { Toast } from "@utils/toast";
-import { setupBxUi, updateVideoPlayerCss } from "@modules/ui/ui";
+import { setupStreamUi, updateVideoPlayerCss } from "@modules/ui/ui";
import { PrefKey, getPref } from "@utils/preferences";
import { LoadingScreen } from "@modules/loading-screen";
import { MouseCursorHider } from "@modules/mkb/mouse-cursor-hider";
@@ -27,6 +27,7 @@ import { patchAudioContext, patchCanvasContext, patchMeControl, patchRtcCodecs,
import { STATES } from "@utils/global";
import { injectStreamMenuButtons } from "@modules/stream/stream-ui";
import { BxLogger } from "@utils/bx-logger";
+import { GameBar } from "./modules/game-bar/game-bar";
// Handle login page
if (window.location.pathname.includes('/auth/msa')) {
@@ -123,9 +124,7 @@ window.addEventListener(BxEvent.STREAM_LOADING, e => {
}
// Setup UI
- setupBxUi();
-
-
+ setupStreamUi();
});
// Setup loading screen
@@ -148,32 +147,14 @@ window.addEventListener(BxEvent.STREAM_PLAYING, e => {
STATES.isPlaying = true;
injectStreamMenuButtons();
- /*
- if (getPref(Preferences.CONTROLLER_ENABLE_SHORTCUTS)) {
- GamepadHandler.startPolling();
- }
- */
- const PREF_SCREENSHOT_BUTTON_POSITION = getPref(PrefKey.SCREENSHOT_BUTTON_POSITION);
+ GameBar.reset();
+ GameBar.enable();
+ GameBar.showBar();
+
STATES.currentStream.$screenshotCanvas!.width = $video.videoWidth;
STATES.currentStream.$screenshotCanvas!.height = $video.videoHeight;
updateVideoPlayerCss();
-
- // Setup screenshot button
- if (PREF_SCREENSHOT_BUTTON_POSITION !== 'none') {
- const $btn = document.querySelector('.bx-screenshot-button')! as HTMLElement;
- $btn.classList.remove('bx-gone');
- $btn.style.display = 'block';
-
- if (PREF_SCREENSHOT_BUTTON_POSITION === 'bottom-right') {
- $btn.style.right = '0';
- } else {
- $btn.style.left = '0';
- }
- }
-
- const $touchControllerBar = document.getElementById('bx-touch-controller-bar');
- $touchControllerBar && $touchControllerBar.classList.remove('bx-gone');
});
window.addEventListener(BxEvent.STREAM_ERROR_PAGE, e => {
@@ -199,13 +180,9 @@ window.addEventListener(BxEvent.STREAM_STOPPED, e => {
STATES.currentStream.$video = null;
StreamStats.onStoppedPlaying();
- const $screenshotBtn = document.querySelector('.bx-screenshot-button');
- if ($screenshotBtn) {
- $screenshotBtn.removeAttribute('style');
- }
-
MouseCursorHider.stop();
TouchController.reset();
+ GameBar.disable();
});
@@ -231,7 +208,7 @@ function main() {
// Setup UI
addCss();
Toast.setup();
- BX_FLAGS.PreloadUi && setupBxUi();
+ BX_FLAGS.PreloadUi && setupStreamUi();
StreamBadges.setupEvents();
StreamStats.setupEvents();
diff --git a/src/modules/game-bar/action-base.ts b/src/modules/game-bar/action-base.ts
new file mode 100644
index 0000000..ecd678e
--- /dev/null
+++ b/src/modules/game-bar/action-base.ts
@@ -0,0 +1,6 @@
+export abstract class BaseGameBarAction {
+ constructor() {}
+ reset() {}
+
+ abstract render(): HTMLElement;
+}
diff --git a/src/modules/game-bar/action-screenshot.ts b/src/modules/game-bar/action-screenshot.ts
new file mode 100644
index 0000000..3d6121a
--- /dev/null
+++ b/src/modules/game-bar/action-screenshot.ts
@@ -0,0 +1,78 @@
+import { BxEvent } from "@utils/bx-event";
+import { AppInterface, STATES } from "@utils/global";
+import { BxIcon } from "@utils/bx-icon";
+import { createButton, ButtonStyle, CE } from "@utils/html";
+import { BaseGameBarAction } from "./action-base";
+import { t } from "@utils/translation";
+
+export class ScreenshotAction extends BaseGameBarAction {
+ $content: HTMLElement;
+
+ constructor() {
+ super();
+
+ const currentStream = STATES.currentStream;
+ currentStream.$screenshotCanvas = CE('canvas', {'class': 'bx-gone'});
+ document.documentElement.appendChild(currentStream.$screenshotCanvas!);
+
+ const onClick = (e: Event) => {
+ BxEvent.dispatch(window, BxEvent.GAME_BAR_ACTION_ACTIVATED);
+ this.takeScreenshot();
+ };
+
+ this.$content = createButton({
+ style: ButtonStyle.GHOST,
+ icon: BxIcon.SCREENSHOT,
+ title: t('take-screenshot'),
+ onClick: onClick,
+ });
+ }
+
+ render(): HTMLElement {
+ return this.$content;
+ }
+
+ takeScreenshot(callback?: any) {
+ const currentStream = STATES.currentStream;
+ const $video = currentStream.$video;
+ const $canvas = currentStream.$screenshotCanvas;
+ if (!$video || !$canvas) {
+ return;
+ }
+
+ const $canvasContext = $canvas.getContext('2d', {
+ alpha: false,
+ willReadFrequently: false,
+ })!;
+
+ $canvasContext.drawImage($video, 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.titleId, 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('a', {
+ 'download': `${currentStream.titleId}-${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');
+ }
+}
diff --git a/src/modules/game-bar/action-touch-control.ts b/src/modules/game-bar/action-touch-control.ts
new file mode 100644
index 0000000..03bb13c
--- /dev/null
+++ b/src/modules/game-bar/action-touch-control.ts
@@ -0,0 +1,51 @@
+import { BxEvent } from "@utils/bx-event";
+import { BxIcon } from "@utils/bx-icon";
+import { createButton, ButtonStyle, CE } from "@utils/html";
+import { TouchController } from "@modules/touch-controller";
+import { BaseGameBarAction } from "./action-base";
+import { t } from "@utils/translation";
+
+export class TouchControlAction extends BaseGameBarAction {
+ $content: HTMLElement;
+
+ constructor() {
+ 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({
+ style: ButtonStyle.GHOST,
+ icon: BxIcon.TOUCH_CONTROL_ENABLE,
+ title: t('show-touch-controller'),
+ onClick: onClick,
+ });
+
+ const $btnDisable = createButton({
+ style: ButtonStyle.GHOST,
+ icon: BxIcon.TOUCH_CONTROL_DISABLE,
+ title: t('hide-touch-controller'),
+ onClick: onClick,
+ });
+
+ this.$content = CE('div', {'data-enabled': 'true'},
+ $btnEnable,
+ $btnDisable,
+ );
+ }
+
+ render(): HTMLElement {
+ return this.$content;
+ }
+
+ reset(): void {
+ this.$content.setAttribute('data-enabled', 'true');
+ }
+}
diff --git a/src/modules/game-bar/game-bar.ts b/src/modules/game-bar/game-bar.ts
new file mode 100644
index 0000000..11e7ef4
--- /dev/null
+++ b/src/modules/game-bar/game-bar.ts
@@ -0,0 +1,116 @@
+import { CE, createSvgIcon } from "@utils/html";
+import { ScreenshotAction } from "./action-screenshot";
+import { TouchControlAction } from "./action-touch-control";
+import { BxEvent } from "@utils/bx-event";
+import { BxIcon } from "@utils/bx-icon";
+import type { BaseGameBarAction } from "./action-base";
+import { STATES } from "@utils/global";
+import { PrefKey, getPref } from "@utils/preferences";
+
+
+export class GameBar {
+ static readonly #VISIBLE_DURATION = 2000;
+ static #timeout: number | null;
+
+ static #$gameBar: HTMLElement;
+ static #$container: HTMLElement;
+
+ static #$actions: BaseGameBarAction[] = [];
+
+ static #beginHideTimeout() {
+ GameBar.#clearHideTimeout();
+
+ GameBar.#timeout = window.setTimeout(() => {
+ GameBar.#timeout = null;
+ GameBar.hideBar();
+ }, GameBar.#VISIBLE_DURATION);
+ }
+
+ static #clearHideTimeout() {
+ GameBar.#timeout && clearTimeout(GameBar.#timeout);
+ GameBar.#timeout = null;
+ }
+
+ static enable() {
+ GameBar.#$gameBar && GameBar.#$gameBar.classList.remove('bx-gone');
+ }
+
+ static disable() {
+ GameBar.#$gameBar && GameBar.#$gameBar.classList.add('bx-gone');
+ GameBar.hideBar();
+ }
+
+ static showBar() {
+ if (!GameBar.#$container) {
+ return;
+ }
+
+ GameBar.#$container.classList.remove('bx-offscreen', 'bx-hide');
+ GameBar.#$container.classList.add('bx-show');
+
+ GameBar.#beginHideTimeout();
+ }
+
+ static hideBar() {
+ if (!GameBar.#$container) {
+ return;
+ }
+
+ GameBar.#$container.classList.remove('bx-show');
+ GameBar.#$container.classList.add('bx-hide');
+ }
+
+ // Reset all states
+ static reset() {
+ for (const action of GameBar.#$actions) {
+ action.reset();
+ }
+ }
+
+ static setup() {
+ let $container;
+ const $gameBar = CE('div', {id: 'bx-game-bar', class: 'bx-gone'},
+ $container = CE('div', {class: 'bx-game-bar-container bx-offscreen'}),
+ createSvgIcon(BxIcon.CARET_RIGHT),
+ );
+
+ GameBar.#$actions = [
+ new ScreenshotAction(),
+ ...(STATES.hasTouchSupport && (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) !== 'off') ? [new TouchControlAction()] : []),
+ ];
+
+ for (const action of GameBar.#$actions) {
+ $container.appendChild(action.render());
+ }
+
+ // Toggle game bar when clicking on the game bar box
+ $gameBar.addEventListener('click', e => {
+ if (e.target === $gameBar) {
+ if ($container.classList.contains('bx-show')) {
+ GameBar.hideBar();
+ } else {
+ GameBar.showBar();
+ }
+ }
+ });
+
+ // Hide game bar after clicking on an action
+ window.addEventListener(BxEvent.GAME_BAR_ACTION_ACTIVATED, GameBar.hideBar);
+
+ $container.addEventListener('pointerover', GameBar.#clearHideTimeout);
+ $container.addEventListener('pointerout', GameBar.#beginHideTimeout);
+
+ // Add animation when hiding game bar
+ $container.addEventListener('transitionend', e => {
+ const classList = $container.classList;
+ if (classList.contains('bx-hide')) {
+ classList.remove('bx-offscreen', 'bx-hide');
+ classList.add('bx-offscreen');
+ }
+ });
+
+ document.documentElement.appendChild($gameBar);
+ GameBar.#$gameBar = $gameBar;
+ GameBar.#$container = $container;
+ }
+}
diff --git a/src/modules/screenshot.ts b/src/modules/screenshot.ts
deleted file mode 100644
index a02cb37..0000000
--- a/src/modules/screenshot.ts
+++ /dev/null
@@ -1,100 +0,0 @@
-import { STATES, AppInterface } from "@utils/global";
-import { CE } from "@utils/html";
-
-export function takeScreenshot(callback: any) {
- const currentStream = STATES.currentStream!;
- const $video = currentStream.$video;
- const $canvas = currentStream.$screenshotCanvas;
- if (!$video || !$canvas) {
- return;
- }
-
- const $canvasContext = $canvas.getContext('2d', {
- alpha: false,
- willReadFrequently: false,
- })!;
-
- $canvasContext.drawImage($video, 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.titleId, 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('a', {
- 'download': `${currentStream.titleId}-${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');
-}
-
-
-export function setupScreenshotButton() {
- const currentStream = STATES.currentStream!
- currentStream.$screenshotCanvas = CE('canvas', {'class': 'bx-screenshot-canvas'});
- document.documentElement.appendChild(currentStream.$screenshotCanvas!);
-
- const delay = 2000;
- const $btn = CE('div', {'class': 'bx-screenshot-button', 'data-showing': false});
-
- let timeout: number | null;
- const detectDbClick = (e: MouseEvent) => {
- if (!currentStream.$video) {
- timeout = null;
- $btn.style.display = 'none';
- return;
- }
-
- if (timeout) {
- clearTimeout(timeout);
- timeout = null;
- $btn.setAttribute('data-capturing', 'true');
-
- takeScreenshot(() => {
- // Hide button
- $btn.setAttribute('data-showing', 'false');
- window.setTimeout(() => {
- if (!timeout) {
- $btn.setAttribute('data-capturing', 'false');
- }
- }, 100);
- });
-
- return;
- }
-
- const isShowing = $btn.getAttribute('data-showing') === 'true';
- if (!isShowing) {
- // Show button
- $btn.setAttribute('data-showing', 'true');
- $btn.setAttribute('data-capturing', 'false');
-
- timeout && clearTimeout(timeout);
- timeout = window.setTimeout(() => {
- timeout = null;
- $btn.setAttribute('data-showing', 'false');
- $btn.setAttribute('data-capturing', 'false');
- }, delay);
- }
- }
-
- $btn.addEventListener('mousedown', detectDbClick);
- document.documentElement.appendChild($btn);
-}
diff --git a/src/modules/touch-controller.ts b/src/modules/touch-controller.ts
index 466de00..b9ec819 100644
--- a/src/modules/touch-controller.ts
+++ b/src/modules/touch-controller.ts
@@ -1,5 +1,5 @@
import { STATES } from "@utils/global";
-import { CE, escapeHtml } from "@utils/html";
+import { escapeHtml } from "@utils/html";
import { Toast } from "@utils/toast";
import { BxEvent } from "@utils/bx-event";
import { BX_FLAGS } from "@utils/bx-flags";
@@ -12,7 +12,11 @@ const LOG_TAG = 'TouchController';
export class TouchController {
static readonly #EVENT_SHOW_DEFAULT_CONTROLLER = new MessageEvent('message', {
- data: '{"content":"{\\"layoutId\\":\\"\\"}","target":"/streaming/touchcontrols/showlayoutv2","type":"Message"}',
+ data: JSON.stringify({
+ content: '{"layoutId":""}',
+ target: '/streaming/touchcontrols/showlayoutv2',
+ type: 'Message',
+ }),
origin: 'better-xcloud',
});
@@ -23,17 +27,17 @@ export class TouchController {
});
*/
- static #$bar: HTMLElement;
static #$style: HTMLStyleElement;
static #enable = false;
- static #showing = false;
static #dataChannel: RTCDataChannel | null;
static #customLayouts: {[index: string]: any} = {};
static #baseCustomLayouts: {[index: string]: any} = {};
static #currentLayoutId: string;
+ static #customList: string[];
+
static enable() {
TouchController.#enable = true;
}
@@ -48,37 +52,28 @@ export class TouchController {
static #showDefault() {
TouchController.#dispatchMessage(TouchController.#EVENT_SHOW_DEFAULT_CONTROLLER);
- TouchController.#showing = true;
}
static #show() {
document.querySelector('#BabylonCanvasContainer-main')?.parentElement?.classList.remove('bx-offscreen');
- TouchController.#showing = true;
}
static #hide() {
document.querySelector('#BabylonCanvasContainer-main')?.parentElement?.classList.add('bx-offscreen');
- TouchController.#showing = false;
}
- static #toggleVisibility() {
+ static toggleVisibility(status: boolean) {
if (!TouchController.#dataChannel) {
return;
}
- TouchController.#showing ? TouchController.#hide() : TouchController.#show();
- }
-
- static #toggleBar(value: boolean) {
- TouchController.#$bar && TouchController.#$bar.setAttribute('data-showing', value.toString());
+ status ? TouchController.#hide() : TouchController.#show();
}
static reset() {
TouchController.#enable = false;
- TouchController.#showing = false;
TouchController.#dataChannel = null;
- TouchController.#$bar && TouchController.#$bar.removeAttribute('data-showing');
TouchController.#$style && (TouchController.#$style.textContent = '');
}
@@ -195,15 +190,19 @@ export class TouchController {
}
static updateCustomList() {
+ const key = 'better_xcloud_custom_touch_layouts';
+ TouchController.#customList = JSON.parse(window.localStorage.getItem(key) || '[]');
+
NATIVE_FETCH('https://raw.githubusercontent.com/redphx/better-xcloud/gh-pages/touch-layouts/ids.json')
.then(response => response.json())
.then(json => {
- window.localStorage.setItem('better_xcloud_custom_touch_layouts', JSON.stringify(json));
+ TouchController.#customList = json;
+ window.localStorage.setItem(key, JSON.stringify(json));
});
}
static getCustomList(): string[] {
- return JSON.parse(window.localStorage.getItem('better_xcloud_custom_touch_layouts') || '[]');
+ return TouchController.#customList;
}
static setup() {
@@ -223,32 +222,9 @@ export class TouchController {
});
};
- const $fragment = document.createDocumentFragment();
const $style = document.createElement('style');
- $fragment.appendChild($style);
+ document.documentElement.appendChild($style);
- const $bar = CE('div', {'id': 'bx-touch-controller-bar'});
- $fragment.appendChild($bar);
-
- document.documentElement.appendChild($fragment);
-
- // Setup double-tap event
- let clickTimeout: number | null;
- $bar.addEventListener('mousedown', (e: MouseEvent) => {
- clickTimeout && clearTimeout(clickTimeout);
- if (clickTimeout) {
- // Double-clicked
- clickTimeout = null;
- TouchController.#toggleVisibility();
- return;
- }
-
- clickTimeout = window.setTimeout(() => {
- clickTimeout = null;
- }, 400);
- });
-
- TouchController.#$bar = $bar;
TouchController.#$style = $style;
const PREF_STYLE_STANDARD = getPref(PrefKey.STREAM_TOUCH_CONTROLLER_STYLE_STANDARD);
@@ -307,7 +283,6 @@ export class TouchController {
try {
if (msg.data.includes('/titleinfo')) {
const json = JSON.parse(JSON.parse(msg.data).content);
- TouchController.#toggleBar(json.focused);
focused = json.focused;
if (!json.focused) {
diff --git a/src/modules/ui/global-settings.ts b/src/modules/ui/global-settings.ts
index d3eec15..9fcf8f9 100644
--- a/src/modules/ui/global-settings.ts
+++ b/src/modules/ui/global-settings.ts
@@ -31,7 +31,6 @@ const SETTINGS_UI = {
PrefKey.AUDIO_MIC_ON_PLAYING,
PrefKey.STREAM_DISABLE_FEEDBACK_DIALOG,
- PrefKey.SCREENSHOT_BUTTON_POSITION,
PrefKey.SCREENSHOT_APPLY_FILTERS,
PrefKey.AUDIO_ENABLE_VOLUME_CONTROL,
diff --git a/src/modules/ui/ui.ts b/src/modules/ui/ui.ts
index dfd527a..49218a7 100644
--- a/src/modules/ui/ui.ts
+++ b/src/modules/ui/ui.ts
@@ -5,11 +5,11 @@ import { UserAgent } from "@utils/user-agent";
import { BxEvent } from "@utils/bx-event";
import { MkbRemapper } from "@modules/mkb/mkb-remapper";
import { getPref, PrefKey, toPrefElement } from "@utils/preferences";
-import { setupScreenshotButton } from "@modules/screenshot";
import { StreamStats } from "@modules/stream/stream-stats";
import { TouchController } from "@modules/touch-controller";
import { t } from "@utils/translation";
import { VibrationManager } from "@modules/vibration-manager";
+import { GameBar } from "../game-bar/game-bar";
export function localRedirect(path: string) {
@@ -468,13 +468,14 @@ div[data-testid="media-container"] {
$elm.textContent = css;
}
-export function setupBxUi() {
+export function setupStreamUi() {
// Prevent initializing multiple times
if (!document.querySelector('.bx-quick-settings-bar')) {
window.addEventListener('resize', updateVideoPlayerCss);
setupQuickSettingsBar();
- setupScreenshotButton();
StreamStats.render();
+
+ GameBar.setup();
}
updateVideoPlayerCss();
diff --git a/src/utils/bx-event.ts b/src/utils/bx-event.ts
index 6264dbe..023c34f 100644
--- a/src/utils/bx-event.ts
+++ b/src/utils/bx-event.ts
@@ -27,6 +27,8 @@ export enum BxEvent {
XCLOUD_SERVERS_READY = 'bx-servers-ready',
DATA_CHANNEL_CREATED = 'bx-data-channel-created',
+
+ GAME_BAR_ACTION_ACTIVATED = 'bx-game-bar-action-activated',
}
export namespace BxEvent {
diff --git a/src/utils/bx-exposed.ts b/src/utils/bx-exposed.ts
index e63c35f..29a2f04 100644
--- a/src/utils/bx-exposed.ts
+++ b/src/utils/bx-exposed.ts
@@ -1,3 +1,4 @@
+import { GameBar } from "@modules/game-bar/game-bar";
import { BxEvent } from "@utils/bx-event";
import { STATES } from "@utils/global";
import { getPref, PrefKey } from "@utils/preferences";
@@ -15,25 +16,12 @@ enum InputType {
export const BxExposed = {
onPollingModeChanged: (mode: 'All' | 'None') => {
if (!STATES.isPlaying) {
- return false;
+ GameBar.disable();
+ return;
}
- const $screenshotBtn = document.querySelector('.bx-screenshot-button');
- const $touchControllerBar = document.getElementById('bx-touch-controller-bar');
-
- if (mode !== 'None') {
- // Hide screenshot button
- $screenshotBtn && $screenshotBtn.classList.add('bx-gone');
-
- // Hide touch controller bar
- $touchControllerBar && $touchControllerBar.classList.add('bx-gone');
- } else {
- // Show screenshot button
- $screenshotBtn && $screenshotBtn.classList.remove('bx-gone');
-
- // Show touch controller bar
- $touchControllerBar && $touchControllerBar.classList.remove('bx-gone');
- }
+ // Toggle Game bar
+ mode !== 'None' ? GameBar.disable() : GameBar.enable();
},
getTitleInfo: () => STATES.currentStream.titleInfo,
diff --git a/src/utils/bx-icon.ts b/src/utils/bx-icon.ts
index 862dfab..2e16174 100644
--- a/src/utils/bx-icon.ts
+++ b/src/utils/bx-icon.ts
@@ -1,3 +1,5 @@
+import iconCadetRight from "@assets/svg/caret-right.svg" with { type: "text" };
+import iconCamera from "@assets/svg/camera.svg" with { type: "text" };
import iconController from "@assets/svg/controller.svg" with { type: "text" };
import iconCopy from "@assets/svg/copy.svg" with { type: "text" };
import iconCursorText from "@assets/svg/cursor-text.svg" with { type: "text" };
@@ -11,6 +13,8 @@ import iconRemotePlay from "@assets/svg/remote-play.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 iconTrash from "@assets/svg/trash.svg" with { type: "text" };
+import iconTouchControlEnable from "@assets/svg/touch-control-enable.svg" with { type: "text" };
+import iconTouchControlDisable from "@assets/svg/touch-control-disable.svg" with { type: "text" };
export const BxIcon = {
STREAM_SETTINGS: iconStreamSettings,
@@ -28,5 +32,10 @@ export const BxIcon = {
REMOTE_PLAY: iconRemotePlay,
+ CARET_RIGHT: iconCadetRight,
+ SCREENSHOT: iconCamera,
+ TOUCH_CONTROL_ENABLE: iconTouchControlEnable,
+ TOUCH_CONTROL_DISABLE: iconTouchControlDisable,
+
// HAND_TAP = '',
} as const;
diff --git a/src/utils/network.ts b/src/utils/network.ts
index 1c7f23e..01f92ca 100644
--- a/src/utils/network.ts
+++ b/src/utils/network.ts
@@ -438,6 +438,9 @@ class XcloudInterceptor {
overrides.inputConfiguration = overrides.inputConfiguration || {};
overrides.inputConfiguration.enableVibration = true;
+ overrides.videoConfiguration = overrides.videoConfiguration || {};
+ overrides.videoConfiguration.setCodecPreferences = true;
+
// Enable touch controller
if (TouchController.isEnabled()) {
overrides.inputConfiguration.enableTouchInput = true;
@@ -570,7 +573,9 @@ export function interceptHttpRequests() {
const newCustomList = customList.map(item => ({ id: item }));
obj.push(...newCustomList);
- } catch (e) {}
+ } catch (e) {
+ console.log(e);
+ }
}
response.json = () => Promise.resolve(obj);
diff --git a/src/utils/preferences.ts b/src/utils/preferences.ts
index d86c1da..58717d3 100644
--- a/src/utils/preferences.ts
+++ b/src/utils/preferences.ts
@@ -46,7 +46,6 @@ export enum PrefKey {
MKB_ABSOLUTE_MOUSE = 'mkb_absolute_mouse',
MKB_DEFAULT_PRESET_ID = 'mkb_default_preset_id',
- SCREENSHOT_BUTTON_POSITION = 'screenshot_button_position',
SCREENSHOT_APPLY_FILTERS = 'screenshot_apply_filters',
BLOCK_TRACKING = 'block_tracking',
@@ -227,15 +226,6 @@ export class Preferences {
default: false,
},
- [PrefKey.SCREENSHOT_BUTTON_POSITION]: {
- label: t('screenshot-button-position'),
- default: 'bottom-left',
- options: {
- 'bottom-left': t('bottom-left'),
- 'bottom-right': t('bottom-right'),
- 'none': t('disable'),
- },
- },
[PrefKey.SCREENSHOT_APPLY_FILTERS]: {
label: t('screenshot-apply-filters'),
default: false,
diff --git a/src/utils/translation.ts b/src/utils/translation.ts
index 2ed7624..f43bad5 100644
--- a/src/utils/translation.ts
+++ b/src/utils/translation.ts
@@ -23,7 +23,7 @@ const Texts = {
"Activate",
"Activo",
,
- ,
+ "Abilita",
"設定する",
"활성화",
"Aktywuj",
@@ -40,7 +40,7 @@ const Texts = {
"Activated",
"Activado",
,
- ,
+ "In uso",
"設定中",
"활성화 됨",
"Aktywowane",
@@ -57,7 +57,7 @@ const Texts = {
"Active",
"Activo",
,
- ,
+ "Attivo",
"有効",
"활성화",
"Aktywny",
@@ -91,7 +91,7 @@ const Texts = {
"Apply",
"Aplicar",
,
- ,
+ "Applica",
"適用",
,
"Zastosuj",
@@ -303,7 +303,7 @@ const Texts = {
"Яркость",
"Aydınlık",
"Яскравість",
- "Độ sáng",
+ "Độ sáng",
"亮度",
],
"browser-unsupported-feature": [
@@ -414,7 +414,7 @@ const Texts = {
"Clear",
"Borrar",
,
- ,
+ "Pulisci",
"消去",
"비우기",
"Wyczyść",
@@ -448,7 +448,7 @@ const Texts = {
"Combine audio & video streams",
"Combinar flujos de audio y vídeo",
,
- ,
+ "Combinare i flussi audio e video",
"音声を映像ストリーミングと統合",
,
"Połącz strumienie audio i wideo",
@@ -465,7 +465,7 @@ const Texts = {
"May fix the laggy audio problem",
"Puede arreglar el problema de audio con retraso",
,
- ,
+ "Potrebbe risolvere il problema dell'audio irregolare",
"音声の遅延を改善できる可能性があります",
,
"Może rozwiązać problem z zacinającym dźwiękiem",
@@ -499,7 +499,7 @@ const Texts = {
"Do you want to delete this preset?",
"¿Desea eliminar este preajuste?",
"Voulez-vous supprimer ce préréglage?",
- ,
+ "Vuoi eliminare questo profilo?",
"このプリセットを削除しますか?",
"이 프리셋을 삭제하시겠습니까?",
"Czy na pewno chcesz usunąć ten szablon?",
@@ -533,7 +533,7 @@ const Texts = {
"Connected",
"Conectado",
,
- ,
+ "Connesso",
"接続済み",
,
"Połączony",
@@ -618,7 +618,7 @@ const Texts = {
"Controller vibration",
"Vibración del mando",
,
- ,
+ "Vibrazione del controller",
"コントローラーの振動",
"컨트롤러 진동",
"Wibracje kontrolera",
@@ -635,7 +635,7 @@ const Texts = {
"Copy",
"Copiar",
,
- ,
+ "Duplica",
"コピー",
"복사",
"Kopiuj",
@@ -669,7 +669,7 @@ const Texts = {
"Deadzone counterweight",
"Contrapeso de la zona muerta",
,
- ,
+ "Compensazione della zona morta",
"デッドゾーンのカウンターウエイト",
,
"Przeciwwaga martwej strefy",
@@ -737,7 +737,7 @@ const Texts = {
"Device vibration",
"Vibración del dispositivo",
,
- ,
+ "Vibrazione del dispositivo",
"デバイスの振動",
"기기 진동",
"Wibracje urządzenia",
@@ -754,7 +754,7 @@ const Texts = {
"On when not using gamepad",
"Activado cuando no se utiliza el mando",
,
- ,
+ "Abilita quando non si usa un gamepad",
"ゲームパッド未使用時にオン",
"게임패드를 사용하지 않을 때",
"Włączone, gdy nie używasz kontrolera",
@@ -839,7 +839,7 @@ const Texts = {
"Disabled",
"Desactivado",
,
- ,
+ "Disattivato",
"無効",
"비활성화됨",
"Wyłączony",
@@ -856,7 +856,7 @@ const Texts = {
"Disconnected",
"Desconectado",
,
- ,
+ "Disconnesso",
"切断",
,
"Rozłączony",
@@ -907,7 +907,7 @@ const Texts = {
"Enable local co-op support",
"Habilitar soporte co-op local",
,
- ,
+ "Abilita supporto cooperativo locale",
"ローカルマルチプレイのサポートを有効化",
,
"Włącz lokalny co-op",
@@ -924,7 +924,7 @@ const Texts = {
"Only works if the game doesn't require a different profile",
"Solo funciona si el juego no requiere un perfil diferente",
,
- ,
+ "Funziona quando il gioco non richiede un profilo differente",
"別アカウントでのサインインを必要としないゲームのみ動作します",
,
"Działa tylko wtedy, gdy gra nie wymaga innego profilu",
@@ -1026,7 +1026,7 @@ const Texts = {
"Enabled",
"Activado",
,
- ,
+ "Attivato",
"有効",
"활성화됨",
"Włączony",
@@ -1043,7 +1043,7 @@ const Texts = {
"Experimental",
"Experimental",
,
- ,
+ "Sperimentale",
"実験的機能",
,
"Eksperymentalne",
@@ -1060,7 +1060,7 @@ const Texts = {
"Export",
"Exportar",
,
- ,
+ "Esporta",
"エクスポート(書出し)",
"내보내기",
"Eksportuj",
@@ -1094,7 +1094,7 @@ const Texts = {
"Allows playing STW mode on mobile",
"Permitir jugar al modo STW en el móvil",
,
- ,
+ "Consente di riprodurre la modalità Salva il Mondo sul cellulare",
"モバイル版で「世界を救え」をプレイできるようになります",
,
"Zezwól na granie w tryb STW na urządzeniu mobilnym",
@@ -1111,7 +1111,7 @@ const Texts = {
"Fortnite: force console version",
"Fortnite: forzar versión de consola",
,
- "Fortnite: Foza la versione console",
+ "Fortnite: forza la versione console",
"Fortnite: 強制的にコンソール版を起動する",
,
"Fortnite: wymuś wersję konsolową",
@@ -1145,7 +1145,7 @@ const Texts = {
"Help",
"Ayuda",
,
- ,
+ "Guida",
"ヘルプ",
,
"Pomoc",
@@ -1179,7 +1179,7 @@ const Texts = {
"Hide web page's scrollbar",
"Oculta la barra de desplazamiento de la página",
,
- ,
+ "Nascondi la barra di scorrimento della pagina web",
"Webページのスクロールバーを隠す",
,
"Ukryj pasek przewijania strony",
@@ -1207,13 +1207,30 @@ const Texts = {
"Ẩn biểu tượng của menu Hệ thống",
"隐藏系统菜单图标",
],
+ "hide-touch-controller": [
+ "Touch-Controller ausblenden",
+ ,
+ "Hide touch controller",
+ "Ocultar controles táctiles",
+ ,
+ ,
+ "タッチコントローラーを隠す",
+ ,
+ ,
+ ,
+ "Скрыть сенсорный контроллер",
+ ,
+ "Приховати сенсорний контролер",
+ "Ẩn bộ điều khiển cảm ứng",
+ ,
+ ],
"horizontal-sensitivity": [
"Horizontale Empfindlichkeit",
"Sensitifitas horizontal",
"Horizontal sensitivity",
"Sensibilidad horizontal",
,
- ,
+ "Sensibilità orizzontale",
"左右方向の感度",
,
"Czułość pozioma",
@@ -1230,7 +1247,7 @@ const Texts = {
"Import",
"Importar",
,
- ,
+ "Importa",
"インポート(読込み)",
"가져오기",
"Importuj",
@@ -1247,7 +1264,7 @@ const Texts = {
"Install Better xCloud app for Android",
"Instale la aplicación Better xCloud para Android",
,
- ,
+ "Installa l'applicazione Better xCloud per Android",
"Android用のBetter xCloudをインストール",
,
"Zainstaluj aplikację Better xCloud na Androida",
@@ -1264,7 +1281,7 @@ const Texts = {
"Keyboard shortcuts",
"Atajos del teclado",
,
- ,
+ "Scorciatoie da tastiera",
"キーボードショートカット",
,
"Skróty klawiszowe",
@@ -1332,7 +1349,7 @@ const Texts = {
"Left stick",
"Joystick izquierdo",
,
- ,
+ "Levetta sinistra",
"左スティック",
"왼쪽 스틱",
"Lewy drążek analogowy",
@@ -1366,7 +1383,7 @@ const Texts = {
"Local co-op",
"Co-op local",
,
- ,
+ "Cooperativa locale",
"ローカルマルチプレイ",
,
"Lokalna kooperacja",
@@ -1383,7 +1400,7 @@ const Texts = {
"Map mouse to",
"Mapear ratón a",
,
- ,
+ "Usa il mouse come",
"マウスの割り当て",
,
"Przypisz myszkę do",
@@ -1553,7 +1570,7 @@ const Texts = {
"Name",
"Nombre",
,
- ,
+ "Nome",
"名前",
"이름",
"Nazwa",
@@ -1570,7 +1587,7 @@ const Texts = {
"New",
"Nuevo",
,
- ,
+ "Nuovo",
"新しい",
"새로 만들기",
"Nowy",
@@ -1808,7 +1825,7 @@ const Texts = {
"Preset",
"Preajuste",
,
- ,
+ "Profilo",
"プリセット",
"프리셋",
"Szablon",
@@ -1825,7 +1842,7 @@ const Texts = {
"Press Esc to cancel",
"Presione Esc para cancelar",
,
- ,
+ "Premi Esc per annullare",
"Escを押してキャンセル",
"ESC를 눌러 취소",
"Naciśnij Esc, aby anulować",
@@ -1842,7 +1859,7 @@ const Texts = {
(e: any) => `Press ${e.key} to toggle the Mouse and Keyboard feature`,
(e: any) => `Pulsa ${e.key} para activar la función de ratón y teclado`,
,
- ,
+ (e: any) => `Premi ${e.key} per attivare o disattivare la funzione Mouse e Tastiera`,
(e: any) => `${e.key} キーでマウスとキーボードの機能を切り替える`,
(e: any) => `${e.key} 키를 눌러 마우스와 키보드 기능을 활성화 하십시오`,
(e: any) => `Naciśnij ${e.key}, aby przełączyć funkcję myszy i klawiatury`,
@@ -1859,7 +1876,7 @@ const Texts = {
"Press a key or do a mouse click to bind...",
"Presione una tecla o haga un clic del ratón para enlazar...",
,
- ,
+ "Premi un tasto o fai un clic del mouse per associare...",
"キーを押すかマウスをクリックして割り当て...",
"정지하려면 아무키나 마우스를 클릭해주세요...",
"Naciśnij klawisz lub kliknij myszą, aby przypisać...",
@@ -1876,7 +1893,7 @@ const Texts = {
"Preset's name:",
"Nombre del preajuste:",
,
- ,
+ "Nome del profilo:",
"プリセット名:",
"프리셋 이름:",
"Nazwa szablonu:",
@@ -1961,7 +1978,7 @@ const Texts = {
"Rename",
"Renombrar",
,
- ,
+ "Rinomina",
"名前変更",
"이름 바꾸기",
"Zmień nazwę",
@@ -1978,7 +1995,7 @@ const Texts = {
"Right-click on a key to unbind it",
"Clic derecho en una tecla para desvincularla",
,
- ,
+ "Clic col tasto destro su una assegnazione per dissociarla",
"右クリックで割り当て解除",
"할당 해제하려면 키를 오른쪽 클릭하세요",
"Kliknij prawym przyciskiem myszy na klawisz, aby anulować przypisanie",
@@ -1995,7 +2012,7 @@ const Texts = {
"Right stick",
"Joystick derecho",
,
- ,
+ "Levetta destra",
"右スティック",
"오른쪽 스틱",
"Prawy drążek analogowy",
@@ -2114,7 +2131,7 @@ const Texts = {
"Save",
"Guardar",
,
- ,
+ "Conferma",
"保存",
"저장",
"Zapisz",
@@ -2131,7 +2148,7 @@ const Texts = {
"Applies video filters to screenshots",
"Aplica filtros de vídeo a las capturas de pantalla",
,
- ,
+ "Applica filtri video agli screenshot",
"スクリーンショットにビデオフィルターを適用",
,
"Stosuje filtry wideo do zrzutów ekranu",
@@ -2165,7 +2182,7 @@ const Texts = {
"Separate Touch controller & Controller #1",
"Separar controlador táctil y controlador #1",
,
- ,
+ "Controller su schermo e Controller #1 separati",
"タッチコントローラーとコントローラー#1を分ける",
,
"Oddziel Kontroler dotykowy i Kontroler #1",
@@ -2182,7 +2199,7 @@ const Texts = {
"Touch controller is Player 1, Controller #1 is Player 2",
"El controlador táctil es Jugador 1, Controlador #1 es Jugador 2",
,
- ,
+ "Il Giocatore 1 userà il Controller su schermo, il Giocatore 2 userà il Controller #1",
"タッチコントローラーがプレイヤー1、コントローラー#1がプレイヤー2に割り当てられます",
,
"Kontroler dotykowy to Gracz 1, Kontroler #1 to Gracz 2",
@@ -2250,7 +2267,7 @@ const Texts = {
"Shortcut keys",
"Teclas de atajo",
,
- ,
+ "Tasti di scelta rapida",
"ショートカットキー",
,
"Skróty klawiszowe",
@@ -2295,6 +2312,23 @@ const Texts = {
"Hiển thị thông số khi vào game",
"开始游戏时显示统计信息",
],
+ "show-touch-controller": [
+ "Touch-Controller anzeigen",
+ ,
+ "Show touch controller",
+ "Mostrar controles táctiles",
+ ,
+ ,
+ "タッチコントローラーを表示",
+ ,
+ ,
+ ,
+ "Показать сенсорный контроллер",
+ ,
+ "Показати сенсорний контролер",
+ "Hiện bộ điều khiển cảm ứng",
+ ,
+ ],
"show-wait-time": [
"Geschätzte Wartezeit anzeigen",
"Tampilkan waktu antrian",
@@ -2556,7 +2590,7 @@ const Texts = {
"Stick decay minimum",
"Disminuir mínimamente el analógico",
,
- ,
+ "Tempo minimo di rilascio dello stick",
"スティックの減衰の最小値",
,
"Minimalne opóźnienie drążka",
@@ -2573,7 +2607,7 @@ const Texts = {
"Stick decay strength",
"Intensidad de decaimiento del analógico",
,
- ,
+ "Velocità di rilascio dello stick",
"スティックの減衰の強さ",
,
"Siła opóźnienia drążka",
@@ -2624,7 +2658,7 @@ const Texts = {
"Support Better xCloud",
"Apoyar a Better xCloud",
,
- ,
+ "Sostieni Better xCloud",
"Better xCloudをサポート",
,
"Wesprzyj Better xCloud",
@@ -2652,6 +2686,23 @@ const Texts = {
"Hoán đổi nút",
"交换按钮",
],
+ "take-screenshot": [
+ "Screenshot aufnehmen",
+ ,
+ "Take screenshot",
+ "Capturar pantalla",
+ ,
+ ,
+ "スクリーンショットを撮影",
+ ,
+ ,
+ ,
+ "Сделать снимок экрана",
+ ,
+ "Зробити знімок екрану",
+ "Lưu ảnh màn hình",
+ ,
+ ],
"target-resolution": [
"Festgelegte Auflösung",
"Resolusi",
@@ -2709,7 +2760,7 @@ const Texts = {
"Off when controller found",
"Desactivar cuando se encuentra el controlador",
,
- ,
+ "Disabilitata quando un controllor viene rilevato",
"コントローラー接続時に無効化",
,
"Wyłącz, gdy kontroler zostanie znaleziony",
@@ -2756,11 +2807,11 @@ const Texts = {
],
"tc-default-opacity": [
"Standard Deckkraft",
- ,
+ "Opasitas bawaan",
"Default opacity",
"Opacidad por defecto",
,
- ,
+ "Opacità predefinita",
"既定の透過度",
,
"Domyślna przezroczystość",
@@ -2894,14 +2945,14 @@ const Texts = {
(e: any) => `Touch-Steuerungslayout von ${e.name}`,
,
(e: any) => `Touch control layout by ${e.name}`,
+ (e: any) => `Disposición del control táctil por ${e.nombre}`,
,
- ,
- ,
+ (e: any) => `Configurazione dei comandi su schermo creata da ${e.name}`,
(e: any) => `タッチ操作レイアウト作成者: ${e.name}`,
,
(e: any) => `Układ sterowania dotykowego stworzony przez ${e.name}`,
,
- ,
+ (e: any) => `Сенсорная раскладка по ${e.name}`,
(e: any) => `${e.name} kişisinin dokunmatik kontrolcü tuş şeması`,
(e: any) => `Розташування сенсорного керування від ${e.name}`,
(e: any) => `Bố cục điều khiển cảm ứng tạo bởi ${e.name}`,
@@ -3015,7 +3066,7 @@ const Texts = {
"Use mouse's absolute position",
"Usar la posición absoluta del ratón",
,
- ,
+ "Usa la posizione assoluta del mouse",
"マウスの絶対座標を使用",
"마우스 절대위치 사용",
"Użyj pozycji bezwzględnej myszy",
@@ -3066,7 +3117,7 @@ const Texts = {
"Vibration intensity",
"Intensidad de la vibración",
,
- ,
+ "Intensità della vibrazione",
"振動の強さ",
"진동 세기",
"Siła wibracji",
@@ -3083,7 +3134,7 @@ const Texts = {
"Vibration",
"Vibración",
,
- ,
+ "Vibrazione",
"振動",
,
"Wibracje",