mirror of
https://github.com/redphx/better-xcloud.git
synced 2025-08-07 21:58:27 +02:00
Add native MKB support for Android app
This commit is contained in:
@@ -3,7 +3,7 @@ import { GamepadKey } from "./mkb/definitions";
|
||||
import { PrompFont } from "@utils/prompt-font";
|
||||
import { CE } from "@utils/html";
|
||||
import { t } from "@utils/translation";
|
||||
import { MkbHandler } from "./mkb/mkb-handler";
|
||||
import { EmulatedMkbHandler } from "./mkb/mkb-handler";
|
||||
import { StreamStats } from "./stream/stream-stats";
|
||||
import { MicrophoneShortcut } from "./shortcuts/shortcut-microphone";
|
||||
import { StreamUiShortcut } from "./shortcuts/shortcut-stream-ui";
|
||||
@@ -172,7 +172,7 @@ export class ControllerShortcut {
|
||||
}
|
||||
|
||||
// Ignore emulated gamepad
|
||||
if (gamepad.id === MkbHandler.VIRTUAL_GAMEPAD_ID) {
|
||||
if (gamepad.id === EmulatedMkbHandler.VIRTUAL_GAMEPAD_ID) {
|
||||
continue;
|
||||
}
|
||||
|
||||
|
23
src/modules/mkb/base-mkb-handler.ts
Normal file
23
src/modules/mkb/base-mkb-handler.ts
Normal file
@@ -0,0 +1,23 @@
|
||||
export abstract class MouseDataProvider {
|
||||
protected mkbHandler: MkbHandler;
|
||||
constructor(handler: MkbHandler) {
|
||||
this.mkbHandler = handler;
|
||||
}
|
||||
|
||||
abstract init(): void;
|
||||
abstract start(): void;
|
||||
abstract stop(): void;
|
||||
abstract destroy(): void;
|
||||
}
|
||||
|
||||
export abstract class MkbHandler {
|
||||
abstract init(): void;
|
||||
abstract start(): void;
|
||||
abstract stop(): void;
|
||||
abstract destroy(): void;
|
||||
abstract handleMouseMove(data: MkbMouseMove): void;
|
||||
abstract handleMouseClick(data: MkbMouseClick): void;
|
||||
abstract handleMouseWheel(data: MkbMouseWheel): boolean;
|
||||
abstract waitForMouseData(enabled: boolean): void;
|
||||
abstract isEnabled(): boolean;
|
||||
}
|
@@ -1,5 +1,5 @@
|
||||
import { MkbPreset } from "./mkb-preset";
|
||||
import { GamepadKey, MkbPresetKey, GamepadStick, MouseMapTo } from "./definitions";
|
||||
import { GamepadKey, MkbPresetKey, GamepadStick, MouseMapTo, WheelCode } from "./definitions";
|
||||
import { createButton, ButtonStyle, CE } from "@utils/html";
|
||||
import { BxEvent } from "@utils/bx-event";
|
||||
import { PrefKey, getPref } from "@utils/preferences";
|
||||
@@ -12,25 +12,19 @@ import { showStreamSettings } from "@modules/stream/stream-ui";
|
||||
import { AppInterface, STATES } from "@utils/global";
|
||||
import { UserAgent } from "@utils/user-agent";
|
||||
import { BxLogger } from "@utils/bx-logger";
|
||||
import { BxIcon } from "@utils/bx-icon";
|
||||
import { PointerClient } from "./pointer-client";
|
||||
import { NativeMkbHandler } from "./native-mkb-handler";
|
||||
import { MkbHandler, MouseDataProvider } from "./base-mkb-handler";
|
||||
|
||||
const LOG_TAG = 'MkbHandler';
|
||||
|
||||
|
||||
abstract class MouseDataProvider {
|
||||
protected mkbHandler: MkbHandler;
|
||||
constructor(handler: MkbHandler) {
|
||||
this.mkbHandler = handler;
|
||||
}
|
||||
|
||||
abstract init(): void;
|
||||
abstract start(): void;
|
||||
abstract stop(): void;
|
||||
abstract destroy(): void;
|
||||
abstract toggle(enabled: boolean): void;
|
||||
const PointerToMouseButton = {
|
||||
1: 0,
|
||||
2: 2,
|
||||
4: 1,
|
||||
}
|
||||
|
||||
|
||||
class WebSocketMouseDataProvider extends MouseDataProvider {
|
||||
#pointerClient: PointerClient | undefined
|
||||
#connected = false
|
||||
@@ -57,36 +51,22 @@ class WebSocketMouseDataProvider extends MouseDataProvider {
|
||||
destroy(): void {
|
||||
this.#connected && this.#pointerClient?.stop();
|
||||
}
|
||||
|
||||
toggle(enabled: boolean): void {
|
||||
if (!this.#connected) {
|
||||
enabled = false;
|
||||
}
|
||||
|
||||
enabled ? this.mkbHandler.start() : this.mkbHandler.stop();
|
||||
this.mkbHandler.waitForMouseData(!enabled);
|
||||
}
|
||||
}
|
||||
|
||||
class PointerLockMouseDataProvider extends MouseDataProvider {
|
||||
init(): void {
|
||||
document.addEventListener('pointerlockchange', this.#onPointerLockChange);
|
||||
document.addEventListener('pointerlockerror', this.#onPointerLockError);
|
||||
}
|
||||
init(): void {}
|
||||
|
||||
start(): void {
|
||||
if (!document.pointerLockElement) {
|
||||
document.body.requestPointerLock();
|
||||
}
|
||||
|
||||
window.addEventListener('mousemove', this.#onMouseMoveEvent);
|
||||
window.addEventListener('mousedown', this.#onMouseEvent);
|
||||
window.addEventListener('mouseup', this.#onMouseEvent);
|
||||
window.addEventListener('wheel', this.#onWheelEvent);
|
||||
window.addEventListener('wheel', this.#onWheelEvent, {passive: false});
|
||||
window.addEventListener('contextmenu', this.#disableContextMenu);
|
||||
}
|
||||
|
||||
stop(): void {
|
||||
document.pointerLockElement && document.exitPointerLock();
|
||||
|
||||
window.removeEventListener('mousemove', this.#onMouseMoveEvent);
|
||||
window.removeEventListener('mousedown', this.#onMouseEvent);
|
||||
window.removeEventListener('mouseup', this.#onMouseEvent);
|
||||
@@ -94,32 +74,7 @@ class PointerLockMouseDataProvider extends MouseDataProvider {
|
||||
window.removeEventListener('contextmenu', this.#disableContextMenu);
|
||||
}
|
||||
|
||||
destroy(): void {
|
||||
document.removeEventListener('pointerlockchange', this.#onPointerLockChange);
|
||||
document.removeEventListener('pointerlockerror', this.#onPointerLockError);
|
||||
}
|
||||
|
||||
toggle(enabled: boolean): void {
|
||||
enabled ? document.pointerLockElement && this.mkbHandler.start() : this.mkbHandler.stop();
|
||||
|
||||
if (enabled) {
|
||||
!document.pointerLockElement && this.mkbHandler.waitForMouseData(true);
|
||||
} else {
|
||||
this.mkbHandler.waitForMouseData(false);
|
||||
document.pointerLockElement && document.exitPointerLock();
|
||||
}
|
||||
}
|
||||
|
||||
#onPointerLockChange = () => {
|
||||
if (this.mkbHandler.isEnabled() && !document.pointerLockElement) {
|
||||
this.mkbHandler.stop();
|
||||
}
|
||||
}
|
||||
|
||||
#onPointerLockError = (e: Event) => {
|
||||
console.log(e);
|
||||
this.stop();
|
||||
}
|
||||
destroy(): void {}
|
||||
|
||||
#onMouseMoveEvent = (e: MouseEvent) => {
|
||||
this.mkbHandler.handleMouseMove({
|
||||
@@ -132,10 +87,9 @@ class PointerLockMouseDataProvider extends MouseDataProvider {
|
||||
e.preventDefault();
|
||||
|
||||
const isMouseDown = e.type === 'mousedown';
|
||||
const key = KeyHelper.getKeyFromEvent(e);
|
||||
const data: MkbMouseClick = {
|
||||
key: key,
|
||||
pressed: isMouseDown
|
||||
mouseButton: e.button,
|
||||
pressed: isMouseDown,
|
||||
};
|
||||
|
||||
this.mkbHandler.handleMouseClick(data);
|
||||
@@ -147,7 +101,12 @@ class PointerLockMouseDataProvider extends MouseDataProvider {
|
||||
return;
|
||||
}
|
||||
|
||||
if (this.mkbHandler.handleMouseWheel({key})) {
|
||||
const data: MkbMouseWheel = {
|
||||
vertical: e.deltaY,
|
||||
horizontal: e.deltaX,
|
||||
};
|
||||
|
||||
if (this.mkbHandler.handleMouseWheel(data)) {
|
||||
e.preventDefault();
|
||||
}
|
||||
}
|
||||
@@ -159,14 +118,14 @@ class PointerLockMouseDataProvider extends MouseDataProvider {
|
||||
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
|
||||
*/
|
||||
export class MkbHandler {
|
||||
static #instance: MkbHandler;
|
||||
static get INSTANCE() {
|
||||
if (!MkbHandler.#instance) {
|
||||
MkbHandler.#instance = new MkbHandler();
|
||||
export class EmulatedMkbHandler extends MkbHandler {
|
||||
static #instance: EmulatedMkbHandler;
|
||||
public static getInstance(): EmulatedMkbHandler {
|
||||
if (!EmulatedMkbHandler.#instance) {
|
||||
EmulatedMkbHandler.#instance = new EmulatedMkbHandler();
|
||||
}
|
||||
|
||||
return MkbHandler.#instance;
|
||||
return EmulatedMkbHandler.#instance;
|
||||
}
|
||||
|
||||
#CURRENT_PRESET_DATA = MkbPreset.convert(MkbPreset.DEFAULT_PRESET);
|
||||
@@ -178,7 +137,7 @@ export class MkbHandler {
|
||||
static VIRTUAL_GAMEPAD_ID = 'Xbox 360 Controller';
|
||||
|
||||
#VIRTUAL_GAMEPAD = {
|
||||
id: MkbHandler.VIRTUAL_GAMEPAD_ID,
|
||||
id: EmulatedMkbHandler.VIRTUAL_GAMEPAD_ID,
|
||||
index: 3,
|
||||
connected: false,
|
||||
hapticActuators: null,
|
||||
@@ -203,6 +162,8 @@ export class MkbHandler {
|
||||
|
||||
#$message?: HTMLElement;
|
||||
|
||||
#escKeyDownTime: number = -1;
|
||||
|
||||
#STICK_MAP: {[key in GamepadKey]?: [GamepadKey[], number, number]};
|
||||
#LEFT_STICK_X: GamepadKey[] = [];
|
||||
#LEFT_STICK_Y: GamepadKey[] = [];
|
||||
@@ -210,6 +171,8 @@ export class MkbHandler {
|
||||
#RIGHT_STICK_Y: GamepadKey[] = [];
|
||||
|
||||
constructor() {
|
||||
super();
|
||||
|
||||
this.#STICK_MAP = {
|
||||
[GamepadKey.LS_LEFT]: [this.#LEFT_STICK_X, 0, -1],
|
||||
[GamepadKey.LS_RIGHT]: [this.#LEFT_STICK_X, 0, 1],
|
||||
@@ -307,20 +270,34 @@ export class MkbHandler {
|
||||
const isKeyDown = e.type === 'keydown';
|
||||
|
||||
// Toggle MKB feature
|
||||
if (isKeyDown) {
|
||||
if (e.code === 'F8') {
|
||||
if (e.code === 'F8') {
|
||||
if (!isKeyDown) {
|
||||
e.preventDefault();
|
||||
this.toggle();
|
||||
return;
|
||||
} else if (e.code === 'Escape') {
|
||||
e.preventDefault();
|
||||
this.#enabled && this.stop();
|
||||
return;
|
||||
}
|
||||
|
||||
if (!this.#isPolling) {
|
||||
return;
|
||||
return;
|
||||
}
|
||||
|
||||
// Hijack the Esc button
|
||||
if (e.code === 'Escape') {
|
||||
e.preventDefault();
|
||||
|
||||
// Hold the Esc for 1 second to disable MKB
|
||||
if (this.#enabled && isKeyDown) {
|
||||
if (this.#escKeyDownTime === -1) {
|
||||
this.#escKeyDownTime = performance.now();
|
||||
} else if (performance.now() - this.#escKeyDownTime >= 1000) {
|
||||
this.stop();
|
||||
}
|
||||
} else {
|
||||
this.#escKeyDownTime = -1;
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
if (!this.#isPolling) {
|
||||
return;
|
||||
}
|
||||
|
||||
const buttonIndex = this.#CURRENT_PRESET_DATA.mapping[e.code || e.key]!;
|
||||
@@ -347,11 +324,24 @@ export class MkbHandler {
|
||||
}
|
||||
|
||||
handleMouseClick = (data: MkbMouseClick) => {
|
||||
if (!data || !data.key) {
|
||||
let mouseButton;
|
||||
if (typeof data.mouseButton !== 'undefined') {
|
||||
mouseButton = data.mouseButton;
|
||||
} else if (typeof data.pointerButton !== 'undefined') {
|
||||
mouseButton = PointerToMouseButton[data.pointerButton as keyof typeof PointerToMouseButton];
|
||||
}
|
||||
|
||||
const keyCode = 'Mouse' + mouseButton;
|
||||
const key = {
|
||||
code: keyCode,
|
||||
name: KeyHelper.codeToKeyName(keyCode),
|
||||
};
|
||||
|
||||
if (!key.name) {
|
||||
return;
|
||||
}
|
||||
|
||||
const buttonIndex = this.#CURRENT_PRESET_DATA.mapping[data.key.code]!;
|
||||
const buttonIndex = this.#CURRENT_PRESET_DATA.mapping[key.code]!;
|
||||
if (typeof buttonIndex === 'undefined') {
|
||||
return;
|
||||
}
|
||||
@@ -379,9 +369,9 @@ export class MkbHandler {
|
||||
if (length !== 0 && length < deadzoneCounterweight) {
|
||||
x *= deadzoneCounterweight / length;
|
||||
y *= deadzoneCounterweight / length;
|
||||
} else if (length > MkbHandler.MAXIMUM_STICK_RANGE) {
|
||||
x *= MkbHandler.MAXIMUM_STICK_RANGE / length;
|
||||
y *= MkbHandler.MAXIMUM_STICK_RANGE / length;
|
||||
} else if (length > EmulatedMkbHandler.MAXIMUM_STICK_RANGE) {
|
||||
x *= EmulatedMkbHandler.MAXIMUM_STICK_RANGE / length;
|
||||
y *= EmulatedMkbHandler.MAXIMUM_STICK_RANGE / length;
|
||||
}
|
||||
|
||||
const analog = mouseMapTo === MouseMapTo.LS ? GamepadStick.LEFT : GamepadStick.RIGHT;
|
||||
@@ -389,16 +379,32 @@ export class MkbHandler {
|
||||
}
|
||||
|
||||
handleMouseWheel = (data: MkbMouseWheel): boolean => {
|
||||
if (!data || !data.key) {
|
||||
let code = '';
|
||||
if (data.vertical < 0) {
|
||||
code = WheelCode.SCROLL_UP;
|
||||
} else if (data.vertical > 0) {
|
||||
code = WheelCode.SCROLL_DOWN;
|
||||
} else if (data.horizontal < 0) {
|
||||
code = WheelCode.SCROLL_LEFT;
|
||||
} else if (data.horizontal > 0) {
|
||||
code = WheelCode.SCROLL_RIGHT;
|
||||
}
|
||||
|
||||
if (!code) {
|
||||
return false;
|
||||
}
|
||||
|
||||
const buttonIndex = this.#CURRENT_PRESET_DATA.mapping[data.key.code]!;
|
||||
const key = {
|
||||
code: code,
|
||||
name: KeyHelper.codeToKeyName(code),
|
||||
};
|
||||
|
||||
const buttonIndex = this.#CURRENT_PRESET_DATA.mapping[key.code]!;
|
||||
if (typeof buttonIndex === 'undefined') {
|
||||
return false;
|
||||
}
|
||||
|
||||
if (this.#prevWheelCode === null || this.#prevWheelCode === data.key.code) {
|
||||
if (this.#prevWheelCode === null || this.#prevWheelCode === key.code) {
|
||||
this.#wheelStoppedTimeout && clearTimeout(this.#wheelStoppedTimeout);
|
||||
this.#pressButton(buttonIndex, true);
|
||||
}
|
||||
@@ -418,8 +424,11 @@ export class MkbHandler {
|
||||
this.#enabled = !this.#enabled;
|
||||
}
|
||||
|
||||
Toast.show(t('mouse-and-keyboard'), t(this.#enabled ? 'enabled' : 'disabled'), {instant: true});
|
||||
this.#mouseDataProvider?.toggle(this.#enabled);
|
||||
if (this.#enabled) {
|
||||
document.body.requestPointerLock();
|
||||
} else {
|
||||
document.pointerLockElement && document.exitPointerLock();
|
||||
}
|
||||
}
|
||||
|
||||
#getCurrentPreset = (): Promise<MkbStoredPreset> => {
|
||||
@@ -455,9 +464,97 @@ export class MkbHandler {
|
||||
}
|
||||
}
|
||||
|
||||
#onDialogShown = () => {
|
||||
document.pointerLockElement && document.exitPointerLock();
|
||||
}
|
||||
|
||||
#initMessage = () => {
|
||||
if (!this.#$message) {
|
||||
this.#$message = CE('div', {'class': 'bx-mkb-pointer-lock-msg bx-gone'},
|
||||
CE('div', {},
|
||||
CE('p', {}, t('virtual-controller')),
|
||||
CE('p', {}, t('press-key-to-toggle-mkb', {key: 'F8'})),
|
||||
),
|
||||
|
||||
CE('div', {'data-type': 'virtual'},
|
||||
createButton({
|
||||
style: ButtonStyle.PRIMARY | ButtonStyle.TALL | ButtonStyle.FULL_WIDTH,
|
||||
label: t('activate'),
|
||||
onClick: ((e: Event) => {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
|
||||
this.toggle(true);
|
||||
}).bind(this),
|
||||
}),
|
||||
|
||||
CE('div', {},
|
||||
createButton({
|
||||
label: t('ignore'),
|
||||
style: ButtonStyle.GHOST,
|
||||
onClick: e => {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
|
||||
this.toggle(false);
|
||||
this.waitForMouseData(false);
|
||||
},
|
||||
}),
|
||||
|
||||
createButton({
|
||||
label: t('edit'),
|
||||
onClick: e => {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
|
||||
showStreamSettings('mkb');
|
||||
},
|
||||
}),
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
if (!this.#$message.isConnected) {
|
||||
document.documentElement.appendChild(this.#$message);
|
||||
}
|
||||
}
|
||||
|
||||
#onPointerLockChange = () => {
|
||||
if (document.pointerLockElement) {
|
||||
this.start();
|
||||
} else {
|
||||
this.stop();
|
||||
}
|
||||
}
|
||||
|
||||
#onPointerLockError = (e: Event) => {
|
||||
console.log(e);
|
||||
this.stop();
|
||||
}
|
||||
|
||||
#onPointerLockRequested = () => {
|
||||
this.start();
|
||||
}
|
||||
|
||||
#onPointerLockExited = () => {
|
||||
this.#mouseDataProvider?.stop();
|
||||
}
|
||||
|
||||
handleEvent(event: Event) {
|
||||
switch (event.type) {
|
||||
case BxEvent.POINTER_LOCK_REQUESTED:
|
||||
this.#onPointerLockRequested();
|
||||
break;
|
||||
case BxEvent.POINTER_LOCK_EXITED:
|
||||
this.#onPointerLockExited();
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
init = () => {
|
||||
this.refreshPresetData();
|
||||
this.#enabled = true;
|
||||
this.#enabled = false;
|
||||
|
||||
if (AppInterface) {
|
||||
this.#mouseDataProvider = new WebSocketMouseDataProvider(this);
|
||||
@@ -467,48 +564,29 @@ export class MkbHandler {
|
||||
this.#mouseDataProvider.init();
|
||||
|
||||
window.addEventListener('keydown', this.#onKeyboardEvent);
|
||||
|
||||
if (!this.#$message) {
|
||||
this.#$message = CE('div', {'class': 'bx-mkb-pointer-lock-msg'},
|
||||
CE('div', {},
|
||||
CE('p', {}, t('mkb-click-to-activate')),
|
||||
CE('p', {}, t('press-key-to-toggle-mkb', {key: 'F8'})),
|
||||
),
|
||||
|
||||
CE('div', {},
|
||||
createButton({
|
||||
icon: BxIcon.MOUSE_SETTINGS,
|
||||
label: t('edit'),
|
||||
style: ButtonStyle.PRIMARY,
|
||||
onClick: e => {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
|
||||
showStreamSettings('mkb');
|
||||
},
|
||||
}),
|
||||
|
||||
createButton({
|
||||
label: t('disable'),
|
||||
onClick: e => {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
|
||||
this.toggle(false);
|
||||
this.waitForMouseData(false);
|
||||
},
|
||||
}),
|
||||
),
|
||||
);
|
||||
|
||||
this.#$message.addEventListener('click', this.start.bind(this));
|
||||
document.documentElement.appendChild(this.#$message);
|
||||
}
|
||||
window.addEventListener('keyup', this.#onKeyboardEvent);
|
||||
|
||||
window.addEventListener(BxEvent.XCLOUD_POLLING_MODE_CHANGED, this.#onPollingModeChanged);
|
||||
window.addEventListener(BxEvent.XCLOUD_DIALOG_SHOWN, this.#onDialogShown);
|
||||
|
||||
this.#$message.classList.add('bx-gone');
|
||||
this.waitForMouseData(true);
|
||||
if (AppInterface) {
|
||||
// Android app doesn't support PointerLock API so we need to use a different method
|
||||
window.addEventListener(BxEvent.POINTER_LOCK_REQUESTED, this);
|
||||
window.addEventListener(BxEvent.POINTER_LOCK_EXITED, this);
|
||||
} else {
|
||||
document.addEventListener('pointerlockchange', this.#onPointerLockChange);
|
||||
document.addEventListener('pointerlockerror', this.#onPointerLockError);
|
||||
}
|
||||
|
||||
this.#initMessage();
|
||||
this.#$message?.classList.add('bx-gone');
|
||||
|
||||
if (AppInterface) {
|
||||
Toast.show(t('press-key-to-toggle-mkb', {key: `<b>F8</b>`}), t('virtual-controller'), {html: true});
|
||||
this.waitForMouseData(false);
|
||||
} else {
|
||||
this.waitForMouseData(true);
|
||||
}
|
||||
}
|
||||
|
||||
destroy = () => {
|
||||
@@ -520,6 +598,18 @@ export class MkbHandler {
|
||||
document.pointerLockElement && document.exitPointerLock();
|
||||
|
||||
window.removeEventListener('keydown', this.#onKeyboardEvent);
|
||||
window.removeEventListener('keyup', this.#onKeyboardEvent);
|
||||
|
||||
if (AppInterface) {
|
||||
window.removeEventListener(BxEvent.POINTER_LOCK_REQUESTED, this);
|
||||
window.removeEventListener(BxEvent.POINTER_LOCK_EXITED, this);
|
||||
} else {
|
||||
document.removeEventListener('pointerlockchange', this.#onPointerLockChange);
|
||||
document.removeEventListener('pointerlockerror', this.#onPointerLockError);
|
||||
}
|
||||
|
||||
window.removeEventListener(BxEvent.XCLOUD_POLLING_MODE_CHANGED, this.#onPollingModeChanged);
|
||||
window.removeEventListener(BxEvent.XCLOUD_DIALOG_SHOWN, this.#onDialogShown);
|
||||
|
||||
this.#mouseDataProvider?.destroy();
|
||||
|
||||
@@ -529,17 +619,17 @@ export class MkbHandler {
|
||||
start = () => {
|
||||
if (!this.#enabled) {
|
||||
this.#enabled = true;
|
||||
Toast.show(t('mouse-and-keyboard'), t('enabled'), {instant: true});
|
||||
Toast.show(t('virtual-controller'), t('enabled'), {instant: true});
|
||||
}
|
||||
|
||||
this.#isPolling = true;
|
||||
this.#escKeyDownTime = -1;
|
||||
|
||||
this.#resetGamepad();
|
||||
window.navigator.getGamepads = this.#patchedGetGamepads;
|
||||
|
||||
this.waitForMouseData(false);
|
||||
|
||||
window.addEventListener('keyup', this.#onKeyboardEvent);
|
||||
this.#mouseDataProvider?.start();
|
||||
|
||||
// Dispatch "gamepadconnected" event
|
||||
@@ -550,36 +640,48 @@ export class MkbHandler {
|
||||
BxEvent.dispatch(window, 'gamepadconnected', {
|
||||
gamepad: virtualGamepad,
|
||||
});
|
||||
|
||||
window.BX_EXPOSED.stopTakRendering = true;
|
||||
|
||||
Toast.show(t('virtual-controller'), t('enabled'), {instant: true});
|
||||
}
|
||||
|
||||
stop = () => {
|
||||
this.#enabled = false;
|
||||
this.#isPolling = false;
|
||||
|
||||
// Dispatch "gamepaddisconnected" event
|
||||
this.#resetGamepad();
|
||||
this.#escKeyDownTime = -1;
|
||||
|
||||
const virtualGamepad = this.#getVirtualGamepad();
|
||||
virtualGamepad.connected = false;
|
||||
virtualGamepad.timestamp = performance.now();
|
||||
if (virtualGamepad.connected) {
|
||||
// Dispatch "gamepaddisconnected" event
|
||||
this.#resetGamepad();
|
||||
|
||||
BxEvent.dispatch(window, 'gamepaddisconnected', {
|
||||
gamepad: virtualGamepad,
|
||||
});
|
||||
virtualGamepad.connected = false;
|
||||
virtualGamepad.timestamp = performance.now();
|
||||
|
||||
window.navigator.getGamepads = this.#nativeGetGamepads;
|
||||
BxEvent.dispatch(window, 'gamepaddisconnected', {
|
||||
gamepad: virtualGamepad,
|
||||
});
|
||||
|
||||
window.removeEventListener('keyup', this.#onKeyboardEvent);
|
||||
window.navigator.getGamepads = this.#nativeGetGamepads;
|
||||
}
|
||||
|
||||
this.waitForMouseData(true);
|
||||
this.#mouseDataProvider?.stop();
|
||||
|
||||
// Toast.show(t('virtual-controller'), t('disabled'), {instant: true});
|
||||
}
|
||||
|
||||
static setupEvents() {
|
||||
getPref(PrefKey.MKB_ENABLED) && (AppInterface || !UserAgent.isMobile()) && window.addEventListener(BxEvent.STREAM_PLAYING, () => {
|
||||
// Enable MKB
|
||||
if (!STATES.currentStream.titleInfo?.details.hasMkbSupport) {
|
||||
BxLogger.info(LOG_TAG, 'Emulate MKB');
|
||||
MkbHandler.INSTANCE.init();
|
||||
window.addEventListener(BxEvent.STREAM_PLAYING, () => {
|
||||
if (STATES.currentStream.titleInfo?.details.hasMkbSupport) {
|
||||
// Enable native MKB in Android app
|
||||
if (AppInterface && getPref(PrefKey.NATIVE_MKB_ENABLED) === 'on') {
|
||||
AppInterface && NativeMkbHandler.getInstance().init();
|
||||
}
|
||||
} else if (getPref(PrefKey.MKB_ENABLED) && (AppInterface || !UserAgent.isMobile())) {
|
||||
BxLogger.info(LOG_TAG, 'Emulate MKB');
|
||||
EmulatedMkbHandler.getInstance().init();
|
||||
}
|
||||
});
|
||||
}
|
||||
|
@@ -1,7 +1,7 @@
|
||||
import { t } from "@utils/translation";
|
||||
import { SettingElementType } from "@utils/settings";
|
||||
import { GamepadKey, MouseButtonCode, MouseMapTo, MkbPresetKey } from "./definitions";
|
||||
import { MkbHandler } from "./mkb-handler";
|
||||
import { EmulatedMkbHandler } from "./mkb-handler";
|
||||
import type { MkbPresetData, MkbConvertedPresetData } from "@/types/mkb";
|
||||
import type { PreferenceSettings } from "@/types/preferences";
|
||||
|
||||
@@ -119,9 +119,9 @@ export class MkbPreset {
|
||||
|
||||
// Pre-calculate mouse's sensitivities
|
||||
const mouse = obj.mouse;
|
||||
mouse[MkbPresetKey.MOUSE_SENSITIVITY_X] *= MkbHandler.DEFAULT_PANNING_SENSITIVITY;
|
||||
mouse[MkbPresetKey.MOUSE_SENSITIVITY_Y] *= MkbHandler.DEFAULT_PANNING_SENSITIVITY;
|
||||
mouse[MkbPresetKey.MOUSE_DEADZONE_COUNTERWEIGHT] *= MkbHandler.DEFAULT_DEADZONE_COUNTERWEIGHT;
|
||||
mouse[MkbPresetKey.MOUSE_SENSITIVITY_X] *= EmulatedMkbHandler.DEFAULT_PANNING_SENSITIVITY;
|
||||
mouse[MkbPresetKey.MOUSE_SENSITIVITY_Y] *= EmulatedMkbHandler.DEFAULT_PANNING_SENSITIVITY;
|
||||
mouse[MkbPresetKey.MOUSE_DEADZONE_COUNTERWEIGHT] *= EmulatedMkbHandler.DEFAULT_DEADZONE_COUNTERWEIGHT;
|
||||
|
||||
const mouseMapTo = MouseMapTo[mouse[MkbPresetKey.MOUSE_MAP_TO]!];
|
||||
if (typeof mouseMapTo !== 'undefined') {
|
||||
|
@@ -6,7 +6,7 @@ import { getPref, setPref, PrefKey } from "@utils/preferences";
|
||||
import { MkbPresetKey, GamepadKeyName } from "./definitions";
|
||||
import { KeyHelper } from "./key-helper";
|
||||
import { MkbPreset } from "./mkb-preset";
|
||||
import { MkbHandler } from "./mkb-handler";
|
||||
import { EmulatedMkbHandler } from "./mkb-handler";
|
||||
import { LocalDb } from "@utils/local-db";
|
||||
import { BxIcon } from "@utils/bx-icon";
|
||||
import { SettingElement } from "@utils/settings";
|
||||
@@ -258,7 +258,7 @@ export class MkbRemapper {
|
||||
|
||||
defaultPresetId = this.#STATE.currentPresetId;
|
||||
setPref(PrefKey.MKB_DEFAULT_PRESET_ID, defaultPresetId);
|
||||
MkbHandler.INSTANCE.refreshPresetData();
|
||||
EmulatedMkbHandler.getInstance().refreshPresetData();
|
||||
} else {
|
||||
defaultPresetId = getPref(PrefKey.MKB_DEFAULT_PRESET_ID);
|
||||
}
|
||||
@@ -487,7 +487,7 @@ export class MkbRemapper {
|
||||
style: ButtonStyle.PRIMARY,
|
||||
onClick: e => {
|
||||
setPref(PrefKey.MKB_DEFAULT_PRESET_ID, this.#STATE.currentPresetId);
|
||||
MkbHandler.INSTANCE.refreshPresetData();
|
||||
EmulatedMkbHandler.getInstance().refreshPresetData();
|
||||
|
||||
this.#refresh();
|
||||
},
|
||||
@@ -517,7 +517,7 @@ export class MkbRemapper {
|
||||
LocalDb.INSTANCE.updatePreset(updatedPreset).then(id => {
|
||||
// If this is the default preset => refresh preset data
|
||||
if (id === getPref(PrefKey.MKB_DEFAULT_PRESET_ID)) {
|
||||
MkbHandler.INSTANCE.refreshPresetData();
|
||||
EmulatedMkbHandler.getInstance().refreshPresetData();
|
||||
}
|
||||
|
||||
this.#toggleEditing(false);
|
||||
|
319
src/modules/mkb/native-mkb-handler.ts
Normal file
319
src/modules/mkb/native-mkb-handler.ts
Normal file
@@ -0,0 +1,319 @@
|
||||
import { Toast } from "@/utils/toast";
|
||||
import { PointerClient } from "./pointer-client";
|
||||
import { AppInterface } from "@/utils/global";
|
||||
import { MkbHandler } from "./base-mkb-handler";
|
||||
import { t } from "@/utils/translation";
|
||||
import { BxEvent } from "@/utils/bx-event";
|
||||
import { ButtonStyle, CE, createButton } from "@/utils/html";
|
||||
import { PrefKey, getPref } from "@/utils/preferences";
|
||||
|
||||
type NativeMouseData = {
|
||||
X: number,
|
||||
Y: number,
|
||||
Buttons: number,
|
||||
WheelX: number,
|
||||
WheelY: number,
|
||||
Type? : 0, // 0: Relative, 1: Absolute
|
||||
}
|
||||
|
||||
type XcloudInputSink = {
|
||||
onMouseInput: (data: NativeMouseData) => void;
|
||||
}
|
||||
|
||||
export class NativeMkbHandler extends MkbHandler {
|
||||
private static instance: NativeMkbHandler;
|
||||
#pointerClient: PointerClient | undefined;
|
||||
#enabled: boolean = false;
|
||||
|
||||
#mouseButtonsPressed = 0;
|
||||
#mouseWheelX = 0;
|
||||
#mouseWheelY = 0;
|
||||
|
||||
#mouseVerticalMultiply = 0;
|
||||
#mouseHorizontalMultiply = 0;
|
||||
|
||||
#inputSink: XcloudInputSink | undefined;
|
||||
|
||||
#$message?: HTMLElement;
|
||||
|
||||
public static getInstance(): NativeMkbHandler {
|
||||
if (!NativeMkbHandler.instance) {
|
||||
NativeMkbHandler.instance = new NativeMkbHandler();
|
||||
}
|
||||
|
||||
return NativeMkbHandler.instance;
|
||||
}
|
||||
|
||||
#onKeyboardEvent(e: KeyboardEvent) {
|
||||
if (e.type === 'keyup' && e.code === 'F8') {
|
||||
e.preventDefault();
|
||||
this.toggle();
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
#onPointerLockRequested(e: Event) {
|
||||
AppInterface.requestPointerCapture();
|
||||
this.start();
|
||||
}
|
||||
|
||||
#onPointerLockExited(e: Event) {
|
||||
AppInterface.releasePointerCapture();
|
||||
this.stop();
|
||||
}
|
||||
|
||||
#onPollingModeChanged = (e: Event) => {
|
||||
if (!this.#$message) {
|
||||
return;
|
||||
}
|
||||
|
||||
const mode = (e as any).mode;
|
||||
if (mode === 'None') {
|
||||
this.#$message.classList.remove('bx-offscreen');
|
||||
} else {
|
||||
this.#$message.classList.add('bx-offscreen');
|
||||
}
|
||||
}
|
||||
|
||||
#onDialogShown = () => {
|
||||
document.pointerLockElement && document.exitPointerLock();
|
||||
}
|
||||
|
||||
#initMessage() {
|
||||
if (!this.#$message) {
|
||||
this.#$message = CE('div', {'class': 'bx-mkb-pointer-lock-msg'},
|
||||
CE('div', {},
|
||||
CE('p', {}, t('native-mkb')),
|
||||
CE('p', {}, t('press-key-to-toggle-mkb', {key: 'F8'})),
|
||||
),
|
||||
|
||||
CE('div', {'data-type': 'native'},
|
||||
createButton({
|
||||
style: ButtonStyle.PRIMARY | ButtonStyle.FULL_WIDTH | ButtonStyle.TALL,
|
||||
label: t('activate'),
|
||||
onClick: ((e: Event) => {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
|
||||
this.toggle(true);
|
||||
}).bind(this),
|
||||
}),
|
||||
|
||||
createButton({
|
||||
style: ButtonStyle.GHOST | ButtonStyle.FULL_WIDTH,
|
||||
label: t('ignore'),
|
||||
onClick: e => {
|
||||
e.preventDefault();
|
||||
e.stopPropagation();
|
||||
|
||||
this.#$message?.classList.add('bx-gone');
|
||||
},
|
||||
}),
|
||||
),
|
||||
);
|
||||
}
|
||||
|
||||
if (!this.#$message.isConnected) {
|
||||
document.documentElement.appendChild(this.#$message);
|
||||
}
|
||||
}
|
||||
|
||||
handleEvent(event: Event) {
|
||||
switch (event.type) {
|
||||
case 'keyup':
|
||||
this.#onKeyboardEvent(event as KeyboardEvent);
|
||||
break;
|
||||
|
||||
case BxEvent.XCLOUD_DIALOG_SHOWN:
|
||||
this.#onDialogShown();
|
||||
break;
|
||||
|
||||
case BxEvent.POINTER_LOCK_REQUESTED:
|
||||
this.#onPointerLockRequested(event);
|
||||
break;
|
||||
case BxEvent.POINTER_LOCK_EXITED:
|
||||
this.#onPointerLockExited(event);
|
||||
break;
|
||||
|
||||
case BxEvent.XCLOUD_POLLING_MODE_CHANGED:
|
||||
this.#onPollingModeChanged(event);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
init() {
|
||||
this.#pointerClient = PointerClient.getInstance();
|
||||
this.#inputSink = window.BX_EXPOSED.inputSink;
|
||||
|
||||
// Stop keyboard input at startup
|
||||
this.#updateInputConfigurationAsync(false);
|
||||
|
||||
try {
|
||||
this.#pointerClient.start(this);
|
||||
} catch (e) {
|
||||
Toast.show('Cannot enable Mouse & Keyboard feature');
|
||||
}
|
||||
|
||||
this.#mouseVerticalMultiply = getPref(PrefKey.NATIVE_MKB_SCROLL_VERTICAL_SENSITIVITY);
|
||||
this.#mouseHorizontalMultiply = getPref(PrefKey.NATIVE_MKB_SCROLL_HORIZONTAL_SENSITIVITY);
|
||||
|
||||
window.addEventListener('keyup', this);
|
||||
|
||||
window.addEventListener(BxEvent.XCLOUD_DIALOG_SHOWN, this);
|
||||
window.addEventListener(BxEvent.POINTER_LOCK_REQUESTED, this);
|
||||
window.addEventListener(BxEvent.POINTER_LOCK_EXITED, this);
|
||||
window.addEventListener(BxEvent.XCLOUD_POLLING_MODE_CHANGED, this);
|
||||
|
||||
this.#initMessage();
|
||||
|
||||
if (AppInterface) {
|
||||
Toast.show(t('press-key-to-toggle-mkb', {key: `<b>F8</b>`}), t('native-mkb'), {html: true});
|
||||
this.#$message?.classList.add('bx-gone');
|
||||
} else {
|
||||
this.#$message?.classList.remove('bx-gone');
|
||||
}
|
||||
}
|
||||
|
||||
toggle(force?: boolean) {
|
||||
let setEnable: boolean;
|
||||
if (typeof force !== 'undefined') {
|
||||
setEnable = force;
|
||||
} else {
|
||||
setEnable = !this.#enabled;
|
||||
}
|
||||
|
||||
if (setEnable) {
|
||||
document.documentElement.requestPointerLock();
|
||||
} else {
|
||||
document.exitPointerLock();
|
||||
}
|
||||
}
|
||||
|
||||
#updateInputConfigurationAsync(enabled: boolean) {
|
||||
window.BX_EXPOSED.streamSession.updateInputConfigurationAsync({
|
||||
enableKeyboardInput: enabled,
|
||||
enableMouseInput: enabled,
|
||||
enableAbsoluteMouse: false,
|
||||
enableTouchInput: false,
|
||||
});
|
||||
}
|
||||
|
||||
start() {
|
||||
this.#resetMouseInput();
|
||||
this.#enabled = true;
|
||||
|
||||
this.#updateInputConfigurationAsync(true);
|
||||
|
||||
window.BX_EXPOSED.stopTakRendering = true;
|
||||
this.#$message?.classList.add('bx-gone');
|
||||
|
||||
Toast.show(t('native-mkb'), t('enabled'), {instant: true});
|
||||
}
|
||||
|
||||
stop() {
|
||||
this.#resetMouseInput();
|
||||
this.#enabled = false;
|
||||
this.#updateInputConfigurationAsync(false);
|
||||
|
||||
this.#$message?.classList.remove('bx-gone');
|
||||
}
|
||||
|
||||
destroy(): void {
|
||||
this.#pointerClient?.stop();
|
||||
window.removeEventListener('keyup', this);
|
||||
|
||||
window.removeEventListener(BxEvent.XCLOUD_DIALOG_SHOWN, this);
|
||||
window.removeEventListener(BxEvent.POINTER_LOCK_REQUESTED, this);
|
||||
window.removeEventListener(BxEvent.POINTER_LOCK_EXITED, this);
|
||||
window.removeEventListener(BxEvent.XCLOUD_POLLING_MODE_CHANGED, this);
|
||||
|
||||
this.#$message?.classList.add('bx-gone');
|
||||
}
|
||||
|
||||
handleMouseMove(data: MkbMouseMove): void {
|
||||
this.#sendMouseInput({
|
||||
X: data.movementX,
|
||||
Y: data.movementY,
|
||||
Buttons: this.#mouseButtonsPressed,
|
||||
WheelX: this.#mouseWheelX,
|
||||
WheelY: this.#mouseWheelY,
|
||||
});
|
||||
}
|
||||
|
||||
handleMouseClick(data: MkbMouseClick): void {
|
||||
const { pointerButton, pressed } = data;
|
||||
|
||||
if (pressed) {
|
||||
this.#mouseButtonsPressed |= pointerButton!;
|
||||
} else {
|
||||
this.#mouseButtonsPressed ^= pointerButton!;
|
||||
}
|
||||
this.#mouseButtonsPressed = Math.max(0, this.#mouseButtonsPressed);
|
||||
|
||||
this.#sendMouseInput({
|
||||
X: 0,
|
||||
Y: 0,
|
||||
Buttons: this.#mouseButtonsPressed,
|
||||
WheelX: this.#mouseWheelX,
|
||||
WheelY: this.#mouseWheelY,
|
||||
});
|
||||
}
|
||||
|
||||
handleMouseWheel(data: MkbMouseWheel): boolean {
|
||||
const { vertical, horizontal } = data;
|
||||
|
||||
this.#mouseWheelX = horizontal;
|
||||
if (this.#mouseHorizontalMultiply && this.#mouseHorizontalMultiply !== 1) {
|
||||
this.#mouseWheelX *= this.#mouseHorizontalMultiply;
|
||||
}
|
||||
|
||||
this.#mouseWheelY = vertical;
|
||||
if (this.#mouseVerticalMultiply && this.#mouseVerticalMultiply !== 1) {
|
||||
this.#mouseWheelY *= this.#mouseVerticalMultiply;
|
||||
}
|
||||
|
||||
this.#sendMouseInput({
|
||||
X: 0,
|
||||
Y: 0,
|
||||
Buttons: this.#mouseButtonsPressed,
|
||||
WheelX: this.#mouseWheelX,
|
||||
WheelY: this.#mouseWheelY,
|
||||
});
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
setVerticalScrollMultiplier(vertical: number) {
|
||||
this.#mouseVerticalMultiply = vertical;
|
||||
}
|
||||
|
||||
setHorizontalScrollMultiplier(horizontal: number) {
|
||||
this.#mouseHorizontalMultiply = horizontal;
|
||||
}
|
||||
|
||||
waitForMouseData(enabled: boolean): void {
|
||||
}
|
||||
|
||||
isEnabled(): boolean {
|
||||
return this.#enabled;
|
||||
}
|
||||
|
||||
#sendMouseInput(data: NativeMouseData) {
|
||||
data.Type = 0; // Relative
|
||||
this.#inputSink?.onMouseInput(data);
|
||||
}
|
||||
|
||||
#resetMouseInput() {
|
||||
this.#mouseButtonsPressed = 0;
|
||||
this.#mouseWheelX = 0;
|
||||
this.#mouseWheelY = 0;
|
||||
|
||||
this.#sendMouseInput({
|
||||
X: 0,
|
||||
Y: 0,
|
||||
Buttons: 0,
|
||||
WheelX: 0,
|
||||
WheelY: 0,
|
||||
});
|
||||
}
|
||||
}
|
@@ -1,8 +1,6 @@
|
||||
import { BxLogger } from "@/utils/bx-logger";
|
||||
import type { MkbHandler } from "./mkb-handler";
|
||||
import { KeyHelper } from "./key-helper";
|
||||
import { WheelCode } from "./definitions";
|
||||
import { Toast } from "@/utils/toast";
|
||||
import type { MkbHandler } from "./base-mkb-handler";
|
||||
|
||||
const LOG_TAG = 'PointerClient';
|
||||
|
||||
@@ -14,11 +12,6 @@ enum PointerAction {
|
||||
POINTER_CAPTURE_CHANGED = 5,
|
||||
}
|
||||
|
||||
const FixedMouseIndex = {
|
||||
1: 0,
|
||||
2: 2,
|
||||
4: 1,
|
||||
}
|
||||
|
||||
export class PointerClient {
|
||||
static #PORT = 9269;
|
||||
@@ -97,15 +90,10 @@ export class PointerClient {
|
||||
}
|
||||
|
||||
onPress(messageType: PointerAction, dataView: DataView, offset: number) {
|
||||
const buttonIndex = dataView.getInt8(offset);
|
||||
const fixedIndex = FixedMouseIndex[buttonIndex as keyof typeof FixedMouseIndex];
|
||||
const keyCode = 'Mouse' + fixedIndex;
|
||||
const button = dataView.getUint8(offset);
|
||||
|
||||
this.#mkbHandler?.handleMouseClick({
|
||||
key: {
|
||||
code: keyCode,
|
||||
name: KeyHelper.codeToKeyName(keyCode),
|
||||
},
|
||||
pointerButton: button,
|
||||
pressed: messageType === PointerAction.BUTTON_PRESS,
|
||||
});
|
||||
|
||||
@@ -114,26 +102,13 @@ export class PointerClient {
|
||||
|
||||
onScroll(dataView: DataView, offset: number) {
|
||||
// [V_SCROLL, H_SCROLL]
|
||||
const vScroll = dataView.getInt8(offset);
|
||||
offset += Int8Array.BYTES_PER_ELEMENT;
|
||||
const hScroll = dataView.getInt8(offset);
|
||||
const vScroll = dataView.getInt16(offset);
|
||||
offset += Int16Array.BYTES_PER_ELEMENT;
|
||||
const hScroll = dataView.getInt16(offset);
|
||||
|
||||
let code = '';
|
||||
if (vScroll < 0) {
|
||||
code = WheelCode.SCROLL_UP;
|
||||
} else if (vScroll > 0) {
|
||||
code = WheelCode.SCROLL_DOWN;
|
||||
} else if (hScroll < 0) {
|
||||
code = WheelCode.SCROLL_LEFT;
|
||||
} else if (hScroll > 0) {
|
||||
code = WheelCode.SCROLL_RIGHT;
|
||||
}
|
||||
|
||||
code && this.#mkbHandler?.handleMouseWheel({
|
||||
key: {
|
||||
code: code,
|
||||
name: KeyHelper.codeToKeyName(code),
|
||||
},
|
||||
this.#mkbHandler?.handleMouseWheel({
|
||||
vertical: vScroll,
|
||||
horizontal: hScroll,
|
||||
});
|
||||
|
||||
// BxLogger.info(LOG_TAG, 'scroll', vScroll, hScroll);
|
||||
@@ -148,5 +123,6 @@ export class PointerClient {
|
||||
try {
|
||||
this.#socket?.close();
|
||||
} catch (e) {}
|
||||
this.#socket = null;
|
||||
}
|
||||
}
|
||||
|
@@ -304,6 +304,37 @@ window.dispatchEvent(new Event("${BxEvent.TOUCH_LAYOUT_MANAGER_READY}"));
|
||||
return str;
|
||||
},
|
||||
|
||||
|
||||
patchBabylonRendererClass(str: string) {
|
||||
// ()=>{a.current.render(),h.current=window.requestAnimationFrame(l)
|
||||
let index = str.indexOf('.current.render(),');
|
||||
if (index === -1) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// Move back a character
|
||||
index -= 1;
|
||||
|
||||
// Get variable of the "BabylonRendererClass" object
|
||||
const rendererVar = str[index];
|
||||
|
||||
const newCode = `
|
||||
if (window.BX_EXPOSED.stopTakRendering) {
|
||||
try {
|
||||
document.getElementById('BabylonCanvasContainer-main')?.parentElement.classList.add('bx-offscreen');
|
||||
|
||||
${rendererVar}.current.dispose();
|
||||
} catch (e) {}
|
||||
|
||||
window.BX_EXPOSED.stopTakRendering = false;
|
||||
return;
|
||||
}
|
||||
`;
|
||||
|
||||
str = str.substring(0, index) + newCode + str.substring(index);
|
||||
return str;
|
||||
},
|
||||
|
||||
supportLocalCoOp(str: string) {
|
||||
const text = 'this.gamepadMappingsToSend=[],';
|
||||
if (!str.includes(text)) {
|
||||
@@ -564,9 +595,58 @@ true` + text;
|
||||
str = str.replace(text, '&& false ' + text);
|
||||
return str;
|
||||
},
|
||||
|
||||
enableNativeMkb(str: string) {
|
||||
const text = 'e.mouseSupported&&e.keyboardSupported&&e.fullscreenSupported;';
|
||||
if ((!str.includes(text))) {
|
||||
return false;
|
||||
}
|
||||
|
||||
str = str.replace(text, text + 'return true;');
|
||||
return str;
|
||||
},
|
||||
|
||||
patchMouseAndKeyboardEnabled(str: string) {
|
||||
const text = 'get mouseAndKeyboardEnabled(){';
|
||||
if (!str.includes(text)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
str = str.replace(text, text + 'return true;');
|
||||
return str;
|
||||
},
|
||||
|
||||
exposeInputSink(str: string) {
|
||||
const text = 'this.controlChannel=null,this.inputChannel=null';
|
||||
if (!str.includes(text)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
const newCode = 'window.BX_EXPOSED.inputSink = this;';
|
||||
|
||||
str = str.replace(text, newCode + text);
|
||||
return str;
|
||||
},
|
||||
|
||||
disableNativeRequestPointerLock(str: string) {
|
||||
const text = 'async requestPointerLock(){';
|
||||
if (!str.includes(text)) {
|
||||
return false;
|
||||
}
|
||||
|
||||
str = str.replace(text, text + 'return;');
|
||||
return str;
|
||||
}
|
||||
};
|
||||
|
||||
let PATCH_ORDERS: PatchArray = [
|
||||
...(getPref(PrefKey.NATIVE_MKB_ENABLED) === 'on' ? [
|
||||
'enableNativeMkb',
|
||||
'patchMouseAndKeyboardEnabled',
|
||||
'disableNativeRequestPointerLock',
|
||||
'exposeInputSink',
|
||||
] : []),
|
||||
|
||||
'disableStreamGate',
|
||||
'overrideSettings',
|
||||
'broadcastPollingMode',
|
||||
@@ -618,11 +698,13 @@ let PLAYING_PATCH_ORDERS: PatchArray = [
|
||||
// Skip feedback dialog
|
||||
getPref(PrefKey.STREAM_DISABLE_FEEDBACK_DIALOG) && 'skipFeedbackDialog',
|
||||
|
||||
|
||||
STATES.hasTouchSupport && getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === 'all' && 'patchShowSensorControls',
|
||||
STATES.hasTouchSupport && getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === 'all' && 'exposeTouchLayoutManager',
|
||||
STATES.hasTouchSupport && (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === 'off' || getPref(PrefKey.STREAM_TOUCH_CONTROLLER_AUTO_OFF)) && 'disableTakRenderer',
|
||||
STATES.hasTouchSupport && getPref(PrefKey.STREAM_TOUCH_CONTROLLER_DEFAULT_OPACITY) !== 100 && 'patchTouchControlDefaultOpacity',
|
||||
...(STATES.hasTouchSupport ? [
|
||||
getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === 'all' && 'patchShowSensorControls',
|
||||
getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === 'all' && 'exposeTouchLayoutManager',
|
||||
(getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === 'off' || getPref(PrefKey.STREAM_TOUCH_CONTROLLER_AUTO_OFF)) && 'disableTakRenderer',
|
||||
getPref(PrefKey.STREAM_TOUCH_CONTROLLER_DEFAULT_OPACITY) !== 100 && 'patchTouchControlDefaultOpacity',
|
||||
'patchBabylonRendererClass',
|
||||
] : []),
|
||||
|
||||
BX_FLAGS.EnableXcloudLogging && 'enableConsoleLogging',
|
||||
|
||||
|
@@ -1,5 +1,5 @@
|
||||
import { t } from "@utils/translation";
|
||||
import { BxEvent, XcloudGuideWhere } from "@utils/bx-event";
|
||||
import { BxEvent } from "@utils/bx-event";
|
||||
import { CE, createSvgIcon } from "@utils/html";
|
||||
import { STATES } from "@utils/global";
|
||||
import { BxLogger } from "@/utils/bx-logger";
|
||||
|
@@ -1,7 +1,7 @@
|
||||
import { STATES } from "@utils/global.ts";
|
||||
import { ButtonStyle, createButton, createSvgIcon } from "@utils/html.ts";
|
||||
import { createSvgIcon } from "@utils/html.ts";
|
||||
import { BxIcon } from "@utils/bx-icon";
|
||||
import { BxEvent, XcloudGuideWhere } from "@utils/bx-event.ts";
|
||||
import { BxEvent } from "@utils/bx-event.ts";
|
||||
import { t } from "@utils/translation.ts";
|
||||
import { StreamBadges } from "./stream-badges.ts";
|
||||
import { StreamStats } from "./stream-stats.ts";
|
||||
@@ -283,43 +283,3 @@ export function showStreamSettings(tabId: string) {
|
||||
$parent.addEventListener('click', onClick);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
export function setupStreamUiEvents() {
|
||||
window.addEventListener(BxEvent.XCLOUD_GUIDE_SHOWN, async e => {
|
||||
const where = (e as any).where as XcloudGuideWhere;
|
||||
|
||||
if (where !== XcloudGuideWhere.HOME || !STATES.isPlaying) {
|
||||
return;
|
||||
}
|
||||
|
||||
const $btnQuit = document.querySelector('#gamepass-dialog-root a[class*=QuitGameButton]');
|
||||
if (!$btnQuit) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Add buttons
|
||||
const $btnReload = createButton({
|
||||
label: t('reload-stream'),
|
||||
style: ButtonStyle.FULL_WIDTH | ButtonStyle.FOCUSABLE,
|
||||
onClick: e => {
|
||||
confirm(t('confirm-reload-stream')) && window.location.reload();
|
||||
},
|
||||
});
|
||||
|
||||
const $btnHome = createButton({
|
||||
label: 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));
|
||||
},
|
||||
});
|
||||
|
||||
$btnQuit.insertAdjacentElement('afterend', $btnReload);
|
||||
$btnReload.insertAdjacentElement('afterend', $btnHome);
|
||||
|
||||
// Hide xCloud's Home button
|
||||
const $btnXcloudHome = document.querySelector('#gamepass-dialog-root div[class^=HomeButtonWithDivider]') as HTMLElement;
|
||||
$btnXcloudHome && ($btnXcloudHome.style.display = 'none');
|
||||
});
|
||||
}
|
||||
|
@@ -55,7 +55,7 @@ const SETTINGS_UI = {
|
||||
|
||||
[t('mouse-and-keyboard')]: {
|
||||
items: [
|
||||
PrefKey.NATIVE_MKB_DISABLED,
|
||||
PrefKey.NATIVE_MKB_ENABLED,
|
||||
PrefKey.MKB_ENABLED,
|
||||
PrefKey.MKB_HIDE_IDLE_CURSOR,
|
||||
],
|
||||
@@ -375,7 +375,7 @@ export function setupSettingsUi() {
|
||||
$btnReload = createButton({
|
||||
label: t('settings-reload'),
|
||||
classes: ['bx-settings-reload-button'],
|
||||
style: ButtonStyle.FOCUSABLE | ButtonStyle.FULL_WIDTH,
|
||||
style: ButtonStyle.FOCUSABLE | ButtonStyle.FULL_WIDTH | ButtonStyle.TALL,
|
||||
onClick: e => {
|
||||
window.location.reload();
|
||||
$btnReload.disabled = true;
|
||||
|
80
src/modules/ui/guide-menu.ts
Normal file
80
src/modules/ui/guide-menu.ts
Normal file
@@ -0,0 +1,80 @@
|
||||
import { BxEvent } from "@/utils/bx-event";
|
||||
import { AppInterface, STATES } from "@/utils/global";
|
||||
import { createButton, ButtonStyle } from "@/utils/html";
|
||||
import { t } from "@/utils/translation";
|
||||
|
||||
export enum GuideMenuTab {
|
||||
HOME,
|
||||
}
|
||||
|
||||
export class GuideMenu {
|
||||
static #injectHome($root: HTMLElement) {
|
||||
// Find the last divider
|
||||
const $dividers = $root.querySelectorAll('div[class*=Divider-module__divider]');
|
||||
if (!$dividers) {
|
||||
return;
|
||||
}
|
||||
const $lastDivider = $dividers[$dividers.length - 1];
|
||||
|
||||
// Add "Close app" button
|
||||
if (AppInterface) {
|
||||
const $btnQuit = createButton({
|
||||
label: t('close-app'),
|
||||
style: ButtonStyle.FULL_WIDTH | ButtonStyle.FOCUSABLE | ButtonStyle.DANGER,
|
||||
onClick: e => {
|
||||
AppInterface.closeApp();
|
||||
},
|
||||
});
|
||||
|
||||
$lastDivider.insertAdjacentElement('afterend', $btnQuit);
|
||||
}
|
||||
}
|
||||
|
||||
static #injectHomePlaying($root: HTMLElement) {
|
||||
const $btnQuit = $root.querySelector('a[class*=QuitGameButton]');
|
||||
if (!$btnQuit) {
|
||||
return;
|
||||
}
|
||||
|
||||
// Add buttons
|
||||
const $btnReload = createButton({
|
||||
label: t('reload-stream'),
|
||||
style: ButtonStyle.FULL_WIDTH | ButtonStyle.FOCUSABLE,
|
||||
onClick: e => {
|
||||
confirm(t('confirm-reload-stream')) && window.location.reload();
|
||||
},
|
||||
});
|
||||
|
||||
const $btnHome = createButton({
|
||||
label: 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));
|
||||
},
|
||||
});
|
||||
|
||||
$btnQuit.insertAdjacentElement('afterend', $btnReload);
|
||||
$btnReload.insertAdjacentElement('afterend', $btnHome);
|
||||
|
||||
// Hide xCloud's Home button
|
||||
const $btnXcloudHome = $root.querySelector('div[class^=HomeButtonWithDivider]') as HTMLElement;
|
||||
$btnXcloudHome && ($btnXcloudHome.style.display = 'none');
|
||||
}
|
||||
|
||||
static async #onShown(e: Event) {
|
||||
const where = (e as any).where as GuideMenuTab;
|
||||
|
||||
if (where === GuideMenuTab.HOME) {
|
||||
const $root = document.querySelector('#gamepass-dialog-root div[role=dialog]') as HTMLElement;
|
||||
if (STATES.isPlaying) {
|
||||
GuideMenu.#injectHomePlaying($root);
|
||||
} else {
|
||||
GuideMenu.#injectHome($root);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static observe() {
|
||||
window.addEventListener(BxEvent.XCLOUD_GUIDE_MENU_SHOWN, GuideMenu.#onShown);
|
||||
}
|
||||
}
|
@@ -1,4 +1,4 @@
|
||||
import { STATES } from "@utils/global";
|
||||
import { AppInterface, STATES } from "@utils/global";
|
||||
import { CE, createButton, ButtonStyle, createSvgIcon } from "@utils/html";
|
||||
import { BxIcon } from "@utils/bx-icon";
|
||||
import { UserAgent } from "@utils/user-agent";
|
||||
@@ -12,6 +12,7 @@ import { VibrationManager } from "@modules/vibration-manager";
|
||||
import { Screenshot } from "@/utils/screenshot";
|
||||
import { ControllerShortcut } from "../controller-shortcut";
|
||||
import { SoundShortcut } from "../shortcuts/shortcut-sound";
|
||||
import { NativeMkbHandler } from "../mkb/native-mkb-handler";
|
||||
|
||||
|
||||
export function localRedirect(path: string) {
|
||||
@@ -72,19 +73,6 @@ function setupStreamSettingsDialog() {
|
||||
const isSafari = UserAgent.isSafari();
|
||||
|
||||
const SETTINGS_UI = [
|
||||
getPref(PrefKey.MKB_ENABLED) && {
|
||||
icon: BxIcon.MOUSE,
|
||||
group: 'mkb',
|
||||
items: [
|
||||
{
|
||||
group: 'mkb',
|
||||
label: t('mouse-and-keyboard'),
|
||||
help_url: 'https://better-xcloud.github.io/mouse-and-keyboard/',
|
||||
content: MkbRemapper.INSTANCE.render(),
|
||||
},
|
||||
],
|
||||
},
|
||||
|
||||
{
|
||||
icon: BxIcon.DISPLAY,
|
||||
group: 'stream',
|
||||
@@ -241,6 +229,44 @@ function setupStreamSettingsDialog() {
|
||||
],
|
||||
},
|
||||
|
||||
getPref(PrefKey.MKB_ENABLED) && {
|
||||
icon: BxIcon.VIRTUAL_CONTROLLER,
|
||||
group: 'mkb',
|
||||
items: [
|
||||
{
|
||||
group: 'mkb',
|
||||
label: t('virtual-controller'),
|
||||
help_url: 'https://better-xcloud.github.io/mouse-and-keyboard/',
|
||||
content: MkbRemapper.INSTANCE.render(),
|
||||
},
|
||||
],
|
||||
},
|
||||
|
||||
AppInterface && getPref(PrefKey.NATIVE_MKB_ENABLED) === 'on' && {
|
||||
icon: BxIcon.NATIVE_MKB,
|
||||
group: 'native-mkb',
|
||||
items: [
|
||||
{
|
||||
group: 'native-mkb',
|
||||
label: t('native-mkb'),
|
||||
items: [
|
||||
{
|
||||
pref: PrefKey.NATIVE_MKB_SCROLL_VERTICAL_SENSITIVITY,
|
||||
onChange: (e: any, value: number) => {
|
||||
NativeMkbHandler.getInstance().setVerticalScrollMultiplier(value / 100);
|
||||
},
|
||||
},
|
||||
{
|
||||
pref: PrefKey.NATIVE_MKB_SCROLL_HORIZONTAL_SENSITIVITY,
|
||||
onChange: (e: any, value: number) => {
|
||||
NativeMkbHandler.getInstance().setHorizontalScrollMultiplier(value / 100);
|
||||
},
|
||||
},
|
||||
],
|
||||
},
|
||||
],
|
||||
},
|
||||
|
||||
{
|
||||
icon: BxIcon.COMMAND,
|
||||
group: 'shortcuts',
|
||||
|
Reference in New Issue
Block a user