mirror of
https://github.com/redphx/better-xcloud.git
synced 2025-06-30 03:11:43 +02:00
Compare commits
95 Commits
Author | SHA1 | Date | |
---|---|---|---|
1f632db6b4 | |||
c07e3297ca | |||
5e43915ff7 | |||
e21375821d | |||
6438e533d6 | |||
e9671cbe5d | |||
b99ec65cc9 | |||
addcf56abf | |||
db17bda673 | |||
0a60119c3b | |||
ef14c78941 | |||
f2dc102996 | |||
02db103a72 | |||
f291047b64 | |||
5866644673 | |||
5baad2d89a | |||
381f3fb679 | |||
0f48cb891f | |||
228c2ad008 | |||
5604664b66 | |||
beb02796b3 | |||
9041f70dbd | |||
c13845ffe1 | |||
0d0ecca155 | |||
c09bd9be83 | |||
15a2c67703 | |||
9166761780 | |||
ac37fe05bc | |||
030791d9c4 | |||
5523be1b7f | |||
2a9b070373 | |||
8ba305af2b | |||
29813fbaf2 | |||
02f33875e4 | |||
474f655707 | |||
78021020ce | |||
7c206bd079 | |||
298a40d156 | |||
498123af85 | |||
579dc6bf40 | |||
17e02e5b32 | |||
bf135d34d1 | |||
9fec033173 | |||
78d74cfd23 | |||
3418cdd666 | |||
567770c86e | |||
18027ed1c5 | |||
dcbae39042 | |||
90df5d655f | |||
774a822e69 | |||
5623f3f02f | |||
4eda413da6 | |||
f5b4bd2f40 | |||
a702d29f22 | |||
71576439fd | |||
07c1757237 | |||
22e29e1d92 | |||
e18e05589a | |||
88df490c50 | |||
e2e2322d94 | |||
a4a1743062 | |||
a3600dfd75 | |||
c4ad50906e | |||
a87b26b077 | |||
6874d64ceb | |||
a376f443ef | |||
3bfe11280e | |||
b6a3e56d9f | |||
a6f06fe0f1 | |||
229df61f53 | |||
0c712b6a31 | |||
f06e36e46b | |||
1db19f69ac | |||
dc62c13c21 | |||
d5f02550c7 | |||
88b63a5518 | |||
afd851861a | |||
378f186ee2 | |||
423b171964 | |||
4acf9eba11 | |||
0f5c4f004b | |||
c7dfacf5c4 | |||
0e724b0e4f | |||
47078da413 | |||
e52a296872 | |||
4c593a298e | |||
962b57f0a6 | |||
22fc730fa1 | |||
5bd25bf31c | |||
aba9340e91 | |||
d07d6127df | |||
e45ed6f9ea | |||
07b477a738 | |||
fcaab4ce77 | |||
3954a5d934 |
73
.github/ISSUE_TEMPLATE/01-bug-report.yml
vendored
73
.github/ISSUE_TEMPLATE/01-bug-report.yml
vendored
@ -4,12 +4,19 @@ title: "[Bug] "
|
|||||||
labels:
|
labels:
|
||||||
- bug
|
- bug
|
||||||
body:
|
body:
|
||||||
- type: markdown
|
- type: checkboxes
|
||||||
|
id: checklist
|
||||||
attributes:
|
attributes:
|
||||||
value: |
|
label: Checklist
|
||||||
Please fill out the following information to help us resolve the issue.
|
options:
|
||||||
> [!warning]
|
- label: I will only use English in my report.
|
||||||
> Only use English. Any other languages will be deleted.
|
required: true
|
||||||
|
- label: "The bug doesn't happen when I disable Better xCloud script."
|
||||||
|
required: true
|
||||||
|
- label: I have used the search function for [**open and closed issues**](https://github.com/redphx/better-xcloud/issues?q=is%3Aissue) to see if someone else has already submitted the same bug report.
|
||||||
|
required: true
|
||||||
|
- label: I will describe the problem with as much detail as possible.
|
||||||
|
required: true
|
||||||
- type: dropdown
|
- type: dropdown
|
||||||
id: device_type
|
id: device_type
|
||||||
attributes:
|
attributes:
|
||||||
@ -24,40 +31,28 @@ body:
|
|||||||
multiple: false
|
multiple: false
|
||||||
validations:
|
validations:
|
||||||
required: true
|
required: true
|
||||||
- type: dropdown
|
- type: input
|
||||||
|
id: device_name
|
||||||
|
attributes:
|
||||||
|
label: "Device"
|
||||||
|
description: "Name of the device"
|
||||||
|
placeholder: "e.g., Google Pixel 8"
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
- type: input
|
||||||
id: os
|
id: os
|
||||||
attributes:
|
attributes:
|
||||||
label: "Operating System"
|
label: "Operating System"
|
||||||
description: "Which operating system is it running?"
|
description: "Which operating system is it running?"
|
||||||
options:
|
placeholder: "e.g., Android 14"
|
||||||
- Windows
|
|
||||||
- macOS
|
|
||||||
- Linux
|
|
||||||
- Android
|
|
||||||
- iOS/iPadOS
|
|
||||||
- Other
|
|
||||||
multiple: false
|
|
||||||
validations:
|
|
||||||
required: true
|
|
||||||
- type: dropdown
|
|
||||||
id: browser
|
|
||||||
attributes:
|
|
||||||
label: "Browser"
|
|
||||||
description: "Which browser are you using?"
|
|
||||||
options:
|
|
||||||
- Chrome/Edge/Chromium
|
|
||||||
- Kiwi Browser
|
|
||||||
- Safari
|
|
||||||
- Other
|
|
||||||
multiple: false
|
|
||||||
validations:
|
validations:
|
||||||
required: true
|
required: true
|
||||||
- type: input
|
- type: input
|
||||||
id: browser_version
|
id: browser_version
|
||||||
attributes:
|
attributes:
|
||||||
label: "Browser Version"
|
label: "Browser Version"
|
||||||
description: "What is the version of the browser?"
|
description: "What is the name and version of the browser?"
|
||||||
placeholder: "e.g., 122.0"
|
placeholder: "e.g., Chrome 124.0"
|
||||||
validations:
|
validations:
|
||||||
required: true
|
required: true
|
||||||
- type: input
|
- type: input
|
||||||
@ -68,12 +63,20 @@ body:
|
|||||||
placeholder: "e.g., 3.5.0"
|
placeholder: "e.g., 3.5.0"
|
||||||
validations:
|
validations:
|
||||||
required: true
|
required: true
|
||||||
|
- type: input
|
||||||
|
id: game_list
|
||||||
|
attributes:
|
||||||
|
label: "Game list"
|
||||||
|
description: "Name the game(s) where you saw this bug"
|
||||||
|
placeholder: "e.g., Halo"
|
||||||
|
validations:
|
||||||
|
required: false
|
||||||
- type: textarea
|
- type: textarea
|
||||||
id: repro
|
id: reproduction
|
||||||
attributes:
|
attributes:
|
||||||
label: "Reproduction Steps"
|
label: "Reproduction Steps"
|
||||||
description: |
|
description: |
|
||||||
How did you trigger this bug? Please provide screenshot/video if possible.
|
How did you trigger this bug?
|
||||||
placeholder: |
|
placeholder: |
|
||||||
Example:
|
Example:
|
||||||
1. Open game X
|
1. Open game X
|
||||||
@ -81,3 +84,11 @@ body:
|
|||||||
3. Error
|
3. Error
|
||||||
validations:
|
validations:
|
||||||
required: true
|
required: true
|
||||||
|
- type: textarea
|
||||||
|
id: media
|
||||||
|
attributes:
|
||||||
|
label: "Screenshot/video"
|
||||||
|
description: |
|
||||||
|
Please provide screenshot/video if possible.
|
||||||
|
validations:
|
||||||
|
required: false
|
||||||
|
1
.github/ISSUE_TEMPLATE/config.yml
vendored
Normal file
1
.github/ISSUE_TEMPLATE/config.yml
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
blank_issues_enabled: false
|
2
dist/better-xcloud.meta.js
vendored
2
dist/better-xcloud.meta.js
vendored
@ -1,5 +1,5 @@
|
|||||||
// ==UserScript==
|
// ==UserScript==
|
||||||
// @name Better xCloud
|
// @name Better xCloud
|
||||||
// @namespace https://github.com/redphx
|
// @namespace https://github.com/redphx
|
||||||
// @version 4.3.0
|
// @version 4.6.2
|
||||||
// ==/UserScript==
|
// ==/UserScript==
|
||||||
|
7598
dist/better-xcloud.user.js
vendored
7598
dist/better-xcloud.user.js
vendored
File diff suppressed because it is too large
Load Diff
@ -71,9 +71,10 @@
|
|||||||
|
|
||||||
span {
|
span {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
height: calc(var(--bx-button-height) - 2px);
|
height: var(--bx-button-height);
|
||||||
line-height: var(--bx-button-height);
|
line-height: var(--bx-button-height);
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
|
/* vertical-align: -webkit-baseline-middle; */
|
||||||
color: #fff;
|
color: #fff;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
|
@ -1,17 +1,6 @@
|
|||||||
.bx-settings-reload-button-wrapper {
|
.bx-settings-reload-button {
|
||||||
z-index: var(--bx-reload-button-z-index);
|
margin-top: 10px;
|
||||||
position: fixed;
|
height: calc(var(--bx-button-height) * 1.5);
|
||||||
bottom: 0;
|
|
||||||
left: 0;
|
|
||||||
right: 0;
|
|
||||||
text-align: center;
|
|
||||||
background: #000000cf;
|
|
||||||
padding: 10px;
|
|
||||||
|
|
||||||
button {
|
|
||||||
max-width: 450px;
|
|
||||||
margin: 0 !important;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.bx-settings-container {
|
.bx-settings-container {
|
||||||
@ -98,34 +87,56 @@
|
|||||||
|
|
||||||
.bx-settings-row {
|
.bx-settings-row {
|
||||||
display: flex;
|
display: flex;
|
||||||
margin-bottom: 8px;
|
padding: 6px 12px;
|
||||||
padding: 2px 4px;
|
position: relative;
|
||||||
|
|
||||||
label {
|
label {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
align-self: center;
|
align-self: center;
|
||||||
margin-bottom: 0;
|
margin-bottom: 0;
|
||||||
padding-left: 10px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
&:focus-within {
|
&:hover, &:focus-within {
|
||||||
@media (hover: none) {
|
background-color: #242424;
|
||||||
background-color: #242424;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
input {
|
input {
|
||||||
align-self: center;
|
align-self: center;
|
||||||
accent-color: var(--bx-primary-button-color);
|
accent-color: var(--bx-primary-button-color);
|
||||||
|
|
||||||
|
&:focus {
|
||||||
|
accent-color: var(--bx-danger-button-color);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
select:disabled {
|
select {
|
||||||
-webkit-appearance: none;
|
&:disabled {
|
||||||
background: transparent;
|
-webkit-appearance: none;
|
||||||
text-align-last: right;
|
background: transparent;
|
||||||
border: none;
|
text-align-last: right;
|
||||||
color: #fff;
|
border: none;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
input[type=checkbox], select {
|
||||||
|
&:focus {
|
||||||
|
filter: drop-shadow(1px 0 0 #fff) drop-shadow(-1px 0 0 #fff) drop-shadow(0 1px 0 #fff) drop-shadow(0 -1px 0 #fff);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
&:has(input:focus), &:has(select:focus) {
|
||||||
|
&::before {
|
||||||
|
content: ' ';
|
||||||
|
border-radius: 4px;
|
||||||
|
border: 2px solid #fff;
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
bottom: 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -161,6 +172,10 @@
|
|||||||
&:hover {
|
&:hover {
|
||||||
color: #6dd72b;
|
color: #6dd72b;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&:focus {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.bx-settings-custom-user-agent {
|
.bx-settings-custom-user-agent {
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.bx-mkb-pointer-lock-msg {
|
.bx-mkb-pointer-lock-msg {
|
||||||
display: flex;
|
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
user-select: none;
|
user-select: none;
|
||||||
-webkit-user-select: none;
|
-webkit-user-select: none;
|
||||||
@ -25,7 +24,7 @@
|
|||||||
top: 50%;
|
top: 50%;
|
||||||
transform: translateX(-50%) translateY(-50%);
|
transform: translateX(-50%) translateY(-50%);
|
||||||
margin: auto;
|
margin: auto;
|
||||||
background: #000000e5;
|
background: #000000b3;
|
||||||
z-index: var(--bx-mkb-pointer-lock-msg-z-index);
|
z-index: var(--bx-mkb-pointer-lock-msg-z-index);
|
||||||
color: #fff;
|
color: #fff;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
@ -41,16 +40,7 @@
|
|||||||
background: #151515;
|
background: #151515;
|
||||||
}
|
}
|
||||||
|
|
||||||
button {
|
> div:first-of-type {
|
||||||
margin-right: 12px;
|
|
||||||
height: 60px;
|
|
||||||
}
|
|
||||||
|
|
||||||
svg {
|
|
||||||
width: 32px;
|
|
||||||
}
|
|
||||||
|
|
||||||
div {
|
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
@ -69,6 +59,26 @@
|
|||||||
font-style: italic;
|
font-style: italic;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
> div:last-of-type {
|
||||||
|
display: flex;
|
||||||
|
flex-flow: row;
|
||||||
|
margin-top: 10px;
|
||||||
|
|
||||||
|
button {
|
||||||
|
flex: 1;
|
||||||
|
|
||||||
|
&:first-of-type {
|
||||||
|
margin-right: 5px;
|
||||||
|
}
|
||||||
|
|
||||||
|
&:last-of-type {
|
||||||
|
margin-left: 5px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
button
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.bx-mkb-preset-tools {
|
.bx-mkb-preset-tools {
|
||||||
|
@ -20,7 +20,6 @@
|
|||||||
--bx-danger-button-disabled-color: #a26c6c;
|
--bx-danger-button-disabled-color: #a26c6c;
|
||||||
|
|
||||||
--bx-toast-z-index: 9999;
|
--bx-toast-z-index: 9999;
|
||||||
--bx-reload-button-z-index: 9200;
|
|
||||||
--bx-dialog-z-index: 9101;
|
--bx-dialog-z-index: 9101;
|
||||||
--bx-dialog-overlay-z-index: 9100;
|
--bx-dialog-overlay-z-index: 9100;
|
||||||
--bx-remote-play-popup-z-index: 9090;
|
--bx-remote-play-popup-z-index: 9090;
|
||||||
@ -88,6 +87,10 @@ div[class^=HUDButton-module__hiddenContainer] ~ div:not([class^=HUDButton-module
|
|||||||
padding: 0 !important;
|
padding: 0 !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.bx-prompt {
|
||||||
|
font-family: var(--bx-promptfont-font);
|
||||||
|
}
|
||||||
|
|
||||||
/* Hide UI elements */
|
/* Hide UI elements */
|
||||||
#headerArea, #uhfSkipToMain, .uhf-footer {
|
#headerArea, #uhfSkipToMain, .uhf-footer {
|
||||||
display: none;
|
display: none;
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
.bx-quick-settings-bar {
|
.bx-stream-settings-dialog {
|
||||||
display: flex;
|
display: flex;
|
||||||
position: fixed;
|
position: fixed;
|
||||||
z-index: var(--bx-stream-settings-z-index);
|
z-index: var(--bx-stream-settings-z-index);
|
||||||
@ -7,7 +7,7 @@
|
|||||||
-webkit-user-select: none;
|
-webkit-user-select: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.bx-quick-settings-tabs {
|
.bx-stream-settings-tabs {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
top: 0;
|
top: 0;
|
||||||
right: 420px;
|
right: 420px;
|
||||||
@ -39,7 +39,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
.bx-quick-settings-tab-contents {
|
.bx-stream-settings-tab-contents {
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
position: fixed;
|
position: fixed;
|
||||||
right: 0;
|
right: 0;
|
||||||
@ -89,7 +89,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
.bx-quick-settings-row {
|
.bx-stream-settings-row {
|
||||||
display: flex;
|
display: flex;
|
||||||
border-bottom: 1px solid #40404080;
|
border-bottom: 1px solid #40404080;
|
||||||
margin-bottom: 16px;
|
margin-bottom: 16px;
|
||||||
@ -116,11 +116,74 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.bx-quick-settings-bar-note {
|
.bx-stream-settings-dialog-note {
|
||||||
display: block;
|
display: block;
|
||||||
text-align: center;
|
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
font-weight: lighter;
|
font-weight: lighter;
|
||||||
font-style: italic;
|
font-style: italic;
|
||||||
padding-top: 16px;
|
}
|
||||||
|
|
||||||
|
.bx-stream-settings-tab-contents {
|
||||||
|
div[data-group="shortcuts"] {
|
||||||
|
> div {
|
||||||
|
&[data-has-gamepad=true] {
|
||||||
|
> div:first-of-type {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
> div:last-of-type {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
&[data-has-gamepad=false] {
|
||||||
|
> div:first-of-type {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
> div:last-of-type {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.bx-shortcut-profile {
|
||||||
|
width: 100%;
|
||||||
|
height: 36px;
|
||||||
|
display: block;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bx-shortcut-note {
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bx-shortcut-row {
|
||||||
|
display: flex;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
|
||||||
|
label.bx-prompt {
|
||||||
|
flex: 1;
|
||||||
|
font-size: 26px;
|
||||||
|
margin-bottom: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bx-shortcut-actions {
|
||||||
|
flex: 2;
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
select {
|
||||||
|
position: absolute;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
display: block;
|
||||||
|
|
||||||
|
&:last-of-type {
|
||||||
|
opacity: 0;
|
||||||
|
z-index: calc(var(--bx-stream-settings-z-index) + 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -34,11 +34,22 @@ body[data-media-type=tv] .bx-stream-refresh-button {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
div[data-testid=media-container].bx-taking-screenshot:before {
|
div[data-testid=media-container] {
|
||||||
animation: bx-anim-taking-screenshot 0.5s ease;
|
display: flex;
|
||||||
content: ' ';
|
|
||||||
position: absolute;
|
&.bx-taking-screenshot:before {
|
||||||
width: 100%;
|
animation: bx-anim-taking-screenshot 0.5s ease;
|
||||||
height: 100%;
|
content: ' ';
|
||||||
z-index: var(--bx-screenshot-animation-z-index);
|
position: absolute;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
z-index: var(--bx-screenshot-animation-z-index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#game-stream video {
|
||||||
|
margin: auto;
|
||||||
|
align-self: center;
|
||||||
|
background: #000;
|
||||||
}
|
}
|
||||||
|
4
src/assets/svg/command.svg
Normal file
4
src/assets/svg/command.svg
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
<svg xmlns='http://www.w3.org/2000/svg' fill='none' stroke='#fff' fill-rule='evenodd' stroke-linecap='round' stroke-linejoin='round' stroke-width='2' viewBox='0 0 32 32'>
|
||||||
|
<path d="M25.425 1.5c2.784 0 5.075 2.291 5.075 5.075s-2.291 5.075-5.075 5.075H20.35V6.575c0-2.784 2.291-5.075 5.075-5.075zM11.65 11.65H6.575C3.791 11.65 1.5 9.359 1.5 6.575S3.791 1.5 6.575 1.5s5.075 2.291 5.075 5.075v5.075zm8.7 8.7h5.075c2.784 0 5.075 2.291 5.075 5.075S28.209 30.5 25.425 30.5s-5.075-2.291-5.075-5.075V20.35zM6.575 30.5c-2.784 0-5.075-2.291-5.075-5.075s2.291-5.075 5.075-5.075h5.075v5.075c0 2.784-2.291 5.075-5.075 5.075z"/>
|
||||||
|
<path d="M11.65 11.65h8.7v8.7h-8.7z"/>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 667 B |
55
src/index.ts
55
src/index.ts
@ -24,7 +24,7 @@ import { onHistoryChanged, patchHistoryMethod } from "@utils/history";
|
|||||||
import { VibrationManager } from "@modules/vibration-manager";
|
import { VibrationManager } from "@modules/vibration-manager";
|
||||||
import { overridePreloadState } from "@utils/preload-state";
|
import { overridePreloadState } from "@utils/preload-state";
|
||||||
import { patchAudioContext, patchCanvasContext, patchMeControl, patchRtcCodecs, patchRtcPeerConnection, patchVideoApi } from "@utils/monkey-patches";
|
import { patchAudioContext, patchCanvasContext, patchMeControl, patchRtcCodecs, patchRtcPeerConnection, patchVideoApi } from "@utils/monkey-patches";
|
||||||
import { STATES } from "@utils/global";
|
import { AppInterface, STATES } from "@utils/global";
|
||||||
import { injectStreamMenuButtons } from "@modules/stream/stream-ui";
|
import { injectStreamMenuButtons } from "@modules/stream/stream-ui";
|
||||||
import { BxLogger } from "@utils/bx-logger";
|
import { BxLogger } from "@utils/bx-logger";
|
||||||
import { GameBar } from "./modules/game-bar/game-bar";
|
import { GameBar } from "./modules/game-bar/game-bar";
|
||||||
@ -178,9 +178,9 @@ window.addEventListener(BxEvent.STREAM_STOPPED, e => {
|
|||||||
// Stop MKB listeners
|
// Stop MKB listeners
|
||||||
getPref(PrefKey.MKB_ENABLED) && MkbHandler.INSTANCE.destroy();
|
getPref(PrefKey.MKB_ENABLED) && MkbHandler.INSTANCE.destroy();
|
||||||
|
|
||||||
const $quickBar = document.querySelector('.bx-quick-settings-bar');
|
const $streamSettingsDialog = document.querySelector('.bx-stream-settings-dialog');
|
||||||
if ($quickBar) {
|
if ($streamSettingsDialog) {
|
||||||
$quickBar.classList.add('bx-gone');
|
$streamSettingsDialog.classList.add('bx-gone');
|
||||||
}
|
}
|
||||||
|
|
||||||
STATES.currentStream.audioGainNode = null;
|
STATES.currentStream.audioGainNode = null;
|
||||||
@ -192,8 +192,52 @@ window.addEventListener(BxEvent.STREAM_STOPPED, e => {
|
|||||||
GameBar.getInstance().disable();
|
GameBar.getInstance().disable();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
window.addEventListener(BxEvent.CAPTURE_SCREENSHOT, e => {
|
||||||
|
Screenshot.takeScreenshot();
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
function observeRootDialog($root: HTMLElement) {
|
||||||
|
let currentShown = false;
|
||||||
|
|
||||||
|
const observer = new MutationObserver(mutationList => {
|
||||||
|
for (const mutation of mutationList) {
|
||||||
|
if (mutation.type !== 'childList') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const shown = ($root.firstElementChild && $root.firstElementChild.childElementCount > 0) || false;
|
||||||
|
if (shown !== currentShown) {
|
||||||
|
currentShown = shown;
|
||||||
|
BxEvent.dispatch(window, shown ? BxEvent.XCLOUD_DIALOG_SHOWN : BxEvent.XCLOUD_DIALOG_DISMISSED);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
observer.observe($root, {subtree: true, childList: true});
|
||||||
|
}
|
||||||
|
|
||||||
|
function waitForRootDialog() {
|
||||||
|
const observer = new MutationObserver(mutationList => {
|
||||||
|
for (const mutation of mutationList) {
|
||||||
|
if (mutation.type !== 'childList') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const $target = mutation.target as HTMLElement;
|
||||||
|
if ($target.id && $target.id === 'gamepass-dialog-root') {
|
||||||
|
observer.disconnect();
|
||||||
|
observeRootDialog($target);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
||||||
|
observer.observe(document.documentElement, {subtree: true, childList: true});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
function main() {
|
function main() {
|
||||||
|
waitForRootDialog();
|
||||||
|
|
||||||
// Monkey patches
|
// Monkey patches
|
||||||
patchRtcPeerConnection();
|
patchRtcPeerConnection();
|
||||||
patchRtcCodecs();
|
patchRtcCodecs();
|
||||||
@ -238,6 +282,9 @@ function main() {
|
|||||||
if (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === 'all') {
|
if (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === 'all') {
|
||||||
TouchController.setup();
|
TouchController.setup();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Start PointerProviderServer
|
||||||
|
(getPref(PrefKey.MKB_ENABLED)) && AppInterface && AppInterface.startPointerServer();
|
||||||
}
|
}
|
||||||
|
|
||||||
main();
|
main();
|
||||||
|
369
src/modules/controller-shortcut.ts
Normal file
369
src/modules/controller-shortcut.ts
Normal file
@ -0,0 +1,369 @@
|
|||||||
|
import { Screenshot } from "@utils/screenshot";
|
||||||
|
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 { StreamStats } from "./stream/stream-stats";
|
||||||
|
import { MicrophoneShortcut } from "./shortcuts/shortcut-microphone";
|
||||||
|
import { StreamUiShortcut } from "./shortcuts/shortcut-stream-ui";
|
||||||
|
import { PrefKey, getPref } from "@utils/preferences";
|
||||||
|
import { SoundShortcut } from "./shortcuts/shortcut-sound";
|
||||||
|
import { BxEvent } from "@/utils/bx-event";
|
||||||
|
import { AppInterface } from "@/utils/global";
|
||||||
|
|
||||||
|
enum ShortcutAction {
|
||||||
|
STREAM_SCREENSHOT_CAPTURE = 'stream-screenshot-capture',
|
||||||
|
|
||||||
|
STREAM_MENU_SHOW = 'stream-menu-show',
|
||||||
|
STREAM_STATS_TOGGLE = 'stream-stats-toggle',
|
||||||
|
STREAM_SOUND_TOGGLE = 'stream-sound-toggle',
|
||||||
|
STREAM_MICROPHONE_TOGGLE = 'stream-microphone-toggle',
|
||||||
|
|
||||||
|
STREAM_VOLUME_INC = 'stream-volume-inc',
|
||||||
|
STREAM_VOLUME_DEC = 'stream-volume-dec',
|
||||||
|
|
||||||
|
DEVICE_SOUND_TOGGLE = 'device-sound-toggle',
|
||||||
|
DEVICE_VOLUME_INC = 'device-volume-inc',
|
||||||
|
DEVICE_VOLUME_DEC = 'device-volume-dec',
|
||||||
|
|
||||||
|
DEVICE_BRIGHTNESS_INC = 'device-brightness-inc',
|
||||||
|
DEVICE_BRIGHTNESS_DEC = 'device-brightness-dec',
|
||||||
|
}
|
||||||
|
|
||||||
|
export class ControllerShortcut {
|
||||||
|
static readonly #STORAGE_KEY = 'better_xcloud_controller_shortcuts';
|
||||||
|
|
||||||
|
static #buttonsCache: {[key: string]: boolean[]} = {};
|
||||||
|
static #buttonsStatus: {[key: string]: boolean[]} = {};
|
||||||
|
|
||||||
|
static #$selectProfile: HTMLSelectElement;
|
||||||
|
static #$selectActions: Partial<{[key in GamepadKey]: HTMLSelectElement}> = {};
|
||||||
|
static #$container: HTMLElement;
|
||||||
|
|
||||||
|
static #ACTIONS: {[key: string]: (ShortcutAction | null)[]} = {};
|
||||||
|
|
||||||
|
static reset(index: number) {
|
||||||
|
ControllerShortcut.#buttonsCache[index] = [];
|
||||||
|
ControllerShortcut.#buttonsStatus[index] = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
static handle(gamepad: Gamepad): boolean {
|
||||||
|
const gamepadIndex = gamepad.index;
|
||||||
|
const actions = ControllerShortcut.#ACTIONS[gamepad.id];
|
||||||
|
if (!actions) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Move the buttons status from the previous frame to the cache
|
||||||
|
ControllerShortcut.#buttonsCache[gamepadIndex] = ControllerShortcut.#buttonsStatus[gamepadIndex].slice(0);
|
||||||
|
// Clear the buttons status
|
||||||
|
ControllerShortcut.#buttonsStatus[gamepadIndex] = [];
|
||||||
|
|
||||||
|
const pressed: boolean[] = [];
|
||||||
|
let otherButtonPressed = false;
|
||||||
|
|
||||||
|
gamepad.buttons.forEach((button, index) => {
|
||||||
|
// Only add the newly pressed button to the array (holding doesn't count)
|
||||||
|
if (button.pressed && index !== GamepadKey.HOME) {
|
||||||
|
otherButtonPressed = true;
|
||||||
|
pressed[index] = true;
|
||||||
|
|
||||||
|
// If this is newly pressed button -> run action
|
||||||
|
if (actions[index] && !ControllerShortcut.#buttonsCache[gamepadIndex][index]) {
|
||||||
|
setTimeout(() => ControllerShortcut.#runAction(actions[index]!), 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
ControllerShortcut.#buttonsStatus[gamepadIndex] = pressed;
|
||||||
|
return otherButtonPressed;
|
||||||
|
}
|
||||||
|
|
||||||
|
static #runAction(action: ShortcutAction) {
|
||||||
|
switch (action) {
|
||||||
|
case ShortcutAction.STREAM_SCREENSHOT_CAPTURE:
|
||||||
|
Screenshot.takeScreenshot();
|
||||||
|
break;
|
||||||
|
|
||||||
|
case ShortcutAction.STREAM_STATS_TOGGLE:
|
||||||
|
StreamStats.toggle();
|
||||||
|
break;
|
||||||
|
|
||||||
|
case ShortcutAction.STREAM_MICROPHONE_TOGGLE:
|
||||||
|
MicrophoneShortcut.toggle();
|
||||||
|
break;
|
||||||
|
|
||||||
|
case ShortcutAction.STREAM_MENU_SHOW:
|
||||||
|
StreamUiShortcut.showHideStreamMenu();
|
||||||
|
break;
|
||||||
|
|
||||||
|
case ShortcutAction.STREAM_SOUND_TOGGLE:
|
||||||
|
SoundShortcut.muteUnmute();
|
||||||
|
break;
|
||||||
|
|
||||||
|
case ShortcutAction.STREAM_VOLUME_INC:
|
||||||
|
SoundShortcut.adjustGainNodeVolume(10);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case ShortcutAction.STREAM_VOLUME_DEC:
|
||||||
|
SoundShortcut.adjustGainNodeVolume(-10);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case ShortcutAction.DEVICE_BRIGHTNESS_INC:
|
||||||
|
case ShortcutAction.DEVICE_BRIGHTNESS_DEC:
|
||||||
|
case ShortcutAction.DEVICE_SOUND_TOGGLE:
|
||||||
|
case ShortcutAction.DEVICE_VOLUME_INC:
|
||||||
|
case ShortcutAction.DEVICE_VOLUME_DEC:
|
||||||
|
AppInterface && AppInterface.runShortcut && AppInterface.runShortcut(action);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static #updateAction(profile: string, button: GamepadKey, action: ShortcutAction | null) {
|
||||||
|
if (!(profile in ControllerShortcut.#ACTIONS)) {
|
||||||
|
ControllerShortcut.#ACTIONS[profile] = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!action) {
|
||||||
|
action = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
ControllerShortcut.#ACTIONS[profile][button] = action;
|
||||||
|
|
||||||
|
// Remove empty profiles
|
||||||
|
for (const key in ControllerShortcut.#ACTIONS) {
|
||||||
|
let empty = true;
|
||||||
|
for (const value of ControllerShortcut.#ACTIONS[key]) {
|
||||||
|
if (!!value) {
|
||||||
|
empty = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (empty) {
|
||||||
|
delete ControllerShortcut.#ACTIONS[key];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save to storage
|
||||||
|
window.localStorage.setItem(ControllerShortcut.#STORAGE_KEY, JSON.stringify(ControllerShortcut.#ACTIONS));
|
||||||
|
|
||||||
|
console.log(ControllerShortcut.#ACTIONS);
|
||||||
|
}
|
||||||
|
|
||||||
|
static #updateProfileList(e?: GamepadEvent) {
|
||||||
|
const $select = ControllerShortcut.#$selectProfile;
|
||||||
|
const $container = ControllerShortcut.#$container;
|
||||||
|
|
||||||
|
const $fragment = document.createDocumentFragment();
|
||||||
|
|
||||||
|
// Remove old profiles
|
||||||
|
while ($select.firstElementChild) {
|
||||||
|
$select.firstElementChild.remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
const gamepads = navigator.getGamepads();
|
||||||
|
let hasGamepad = false;
|
||||||
|
|
||||||
|
for (const gamepad of gamepads) {
|
||||||
|
if (!gamepad || !gamepad.connected) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ignore emulated gamepad
|
||||||
|
if (gamepad.id === MkbHandler.VIRTUAL_GAMEPAD_ID) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
hasGamepad = true;
|
||||||
|
|
||||||
|
const $option = CE<HTMLOptionElement>('option', {value: gamepad.id}, gamepad.id);
|
||||||
|
$fragment.appendChild($option);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hasGamepad) {
|
||||||
|
$select.appendChild($fragment);
|
||||||
|
|
||||||
|
$select.selectedIndex = 0;
|
||||||
|
$select.dispatchEvent(new Event('change'));
|
||||||
|
}
|
||||||
|
|
||||||
|
$container.dataset.hasGamepad = hasGamepad.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
static #switchProfile(profile: string) {
|
||||||
|
let actions = ControllerShortcut.#ACTIONS[profile];
|
||||||
|
if (!actions) {
|
||||||
|
actions = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reset selects' values
|
||||||
|
let button: any;
|
||||||
|
for (button in ControllerShortcut.#$selectActions) {
|
||||||
|
const $select = ControllerShortcut.#$selectActions[button as GamepadKey]!;
|
||||||
|
$select.value = actions[button] || '';
|
||||||
|
|
||||||
|
BxEvent.dispatch($select, 'change', {
|
||||||
|
ignoreOnChange: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static renderSettings() {
|
||||||
|
// Read actions from localStorage
|
||||||
|
ControllerShortcut.#ACTIONS = JSON.parse(window.localStorage.getItem(ControllerShortcut.#STORAGE_KEY) || '{}');
|
||||||
|
|
||||||
|
const buttons: Map<GamepadKey, PrompFont> = new Map();
|
||||||
|
buttons.set(GamepadKey.Y, PrompFont.Y);
|
||||||
|
buttons.set(GamepadKey.A, PrompFont.A);
|
||||||
|
buttons.set(GamepadKey.B, PrompFont.B);
|
||||||
|
buttons.set(GamepadKey.X, PrompFont.X);
|
||||||
|
|
||||||
|
buttons.set(GamepadKey.UP, PrompFont.UP);
|
||||||
|
buttons.set(GamepadKey.DOWN, PrompFont.DOWN);
|
||||||
|
buttons.set(GamepadKey.LEFT, PrompFont.LEFT);
|
||||||
|
buttons.set(GamepadKey.RIGHT, PrompFont.RIGHT);
|
||||||
|
|
||||||
|
buttons.set(GamepadKey.SELECT, PrompFont.SELECT);
|
||||||
|
buttons.set(GamepadKey.START, PrompFont.START);
|
||||||
|
|
||||||
|
buttons.set(GamepadKey.LB, PrompFont.LB);
|
||||||
|
buttons.set(GamepadKey.RB, PrompFont.RB);
|
||||||
|
|
||||||
|
buttons.set(GamepadKey.LT, PrompFont.LT);
|
||||||
|
buttons.set(GamepadKey.RT, PrompFont.RT);
|
||||||
|
|
||||||
|
buttons.set(GamepadKey.L3, PrompFont.L3);
|
||||||
|
buttons.set(GamepadKey.R3, PrompFont.R3);
|
||||||
|
|
||||||
|
const actions: {[key: string]: Partial<{[key in ShortcutAction]: string | string[]}>} = {
|
||||||
|
[t('device')]: AppInterface && {
|
||||||
|
[ShortcutAction.DEVICE_SOUND_TOGGLE]: [t('sound'), t('toggle')],
|
||||||
|
[ShortcutAction.DEVICE_VOLUME_INC]: [t('volume'), t('increase')],
|
||||||
|
[ShortcutAction.DEVICE_VOLUME_DEC]: [t('volume'), t('decrease')],
|
||||||
|
|
||||||
|
[ShortcutAction.DEVICE_BRIGHTNESS_INC]: [t('brightness'), t('increase')],
|
||||||
|
[ShortcutAction.DEVICE_BRIGHTNESS_DEC]: [t('brightness'), t('decrease')],
|
||||||
|
},
|
||||||
|
|
||||||
|
[t('stream')]: {
|
||||||
|
[ShortcutAction.STREAM_SCREENSHOT_CAPTURE]: t('take-screenshot'),
|
||||||
|
|
||||||
|
[ShortcutAction.STREAM_SOUND_TOGGLE]: [t('sound'), t('toggle')],
|
||||||
|
[ShortcutAction.STREAM_VOLUME_INC]: getPref(PrefKey.AUDIO_ENABLE_VOLUME_CONTROL) && [t('volume'), t('increase')],
|
||||||
|
[ShortcutAction.STREAM_VOLUME_DEC]: getPref(PrefKey.AUDIO_ENABLE_VOLUME_CONTROL) && [t('volume'), t('decrease')],
|
||||||
|
|
||||||
|
[ShortcutAction.STREAM_MENU_SHOW]: [t('menu'), t('show')],
|
||||||
|
[ShortcutAction.STREAM_STATS_TOGGLE]: [t('stats'), t('show-hide')],
|
||||||
|
[ShortcutAction.STREAM_MICROPHONE_TOGGLE]: [t('microphone'), t('toggle')],
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const $baseSelect = CE<HTMLSelectElement>('select', {autocomplete: 'off'}, CE('option', {value: ''}, '---'));
|
||||||
|
for (const groupLabel in actions) {
|
||||||
|
const items = actions[groupLabel];
|
||||||
|
if (!items) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const $optGroup = CE<HTMLOptGroupElement>('optgroup', {'label': groupLabel});
|
||||||
|
|
||||||
|
for (const action in items) {
|
||||||
|
let label = items[action as keyof typeof items];
|
||||||
|
if (!label) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Array.isArray(label)) {
|
||||||
|
label = label.join(' ❯ ');
|
||||||
|
}
|
||||||
|
|
||||||
|
const $option = CE<HTMLOptionElement>('option', {value: action}, label);
|
||||||
|
$optGroup.appendChild($option);
|
||||||
|
}
|
||||||
|
|
||||||
|
$baseSelect.appendChild($optGroup);
|
||||||
|
}
|
||||||
|
|
||||||
|
let $remap: HTMLElement;
|
||||||
|
let $selectProfile: HTMLSelectElement;
|
||||||
|
|
||||||
|
const $container = CE('div', {'data-has-gamepad': 'false'},
|
||||||
|
CE('div', {},
|
||||||
|
CE('p', {'class': 'bx-shortcut-note'}, t('controller-shortcuts-connect-note')),
|
||||||
|
),
|
||||||
|
|
||||||
|
$remap = CE('div', {},
|
||||||
|
$selectProfile = CE('select', {'class': 'bx-shortcut-profile', autocomplete: 'off'}),
|
||||||
|
CE('p', {'class': 'bx-shortcut-note'},
|
||||||
|
CE('span', {'class': 'bx-prompt'}, PrompFont.HOME),
|
||||||
|
': ' + t('controller-shortcuts-xbox-note'),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
$selectProfile.addEventListener('change', e => {
|
||||||
|
ControllerShortcut.#switchProfile($selectProfile.value);
|
||||||
|
});
|
||||||
|
|
||||||
|
const onActionChanged = (e: Event) => {
|
||||||
|
const $target = e.target as HTMLSelectElement;
|
||||||
|
|
||||||
|
const profile = $selectProfile.value;
|
||||||
|
const button: unknown = $target.dataset.button;
|
||||||
|
const action = $target.value as ShortcutAction;
|
||||||
|
|
||||||
|
const $fakeSelect = $target.previousElementSibling! as HTMLSelectElement;
|
||||||
|
let fakeText = '---';
|
||||||
|
if (action) {
|
||||||
|
const $selectedOption = $target.options[$target.selectedIndex];
|
||||||
|
const $optGroup = $selectedOption.parentElement as HTMLOptGroupElement;
|
||||||
|
fakeText = $optGroup.label + ' ❯ ' + $selectedOption.text;
|
||||||
|
}
|
||||||
|
($fakeSelect.firstElementChild as HTMLOptionElement).text = fakeText;
|
||||||
|
|
||||||
|
!(e as any).ignoreOnChange && ControllerShortcut.#updateAction(profile, button as GamepadKey, action);
|
||||||
|
};
|
||||||
|
|
||||||
|
// @ts-ignore
|
||||||
|
for (const [button, prompt] of buttons) {
|
||||||
|
const $row = CE('div', {'class': 'bx-shortcut-row'});
|
||||||
|
|
||||||
|
const $label = CE('label', {'class': 'bx-prompt'}, `${PrompFont.HOME} + ${prompt}`);
|
||||||
|
|
||||||
|
const $div = CE('div', {'class': 'bx-shortcut-actions'});
|
||||||
|
|
||||||
|
const $fakeSelect = CE<HTMLSelectElement>('select', {autocomplete: 'off'},
|
||||||
|
CE('option', {}, '---'),
|
||||||
|
);
|
||||||
|
$div.appendChild($fakeSelect);
|
||||||
|
|
||||||
|
const $select = $baseSelect.cloneNode(true) as HTMLSelectElement;
|
||||||
|
$select.dataset.button = button.toString();
|
||||||
|
$select.addEventListener('change', onActionChanged);
|
||||||
|
|
||||||
|
ControllerShortcut.#$selectActions[button] = $select;
|
||||||
|
|
||||||
|
$div.appendChild($select);
|
||||||
|
|
||||||
|
$row.appendChild($label);
|
||||||
|
$row.appendChild($div);
|
||||||
|
|
||||||
|
$remap.appendChild($row);
|
||||||
|
}
|
||||||
|
|
||||||
|
$container.appendChild($remap);
|
||||||
|
|
||||||
|
ControllerShortcut.#$selectProfile = $selectProfile;
|
||||||
|
ControllerShortcut.#$container = $container;
|
||||||
|
|
||||||
|
// Detect when gamepad connected/disconnect
|
||||||
|
window.addEventListener('gamepadconnected', ControllerShortcut.#updateProfileList);
|
||||||
|
window.addEventListener('gamepaddisconnected', ControllerShortcut.#updateProfileList);
|
||||||
|
|
||||||
|
ControllerShortcut.#updateProfileList();
|
||||||
|
|
||||||
|
return $container;
|
||||||
|
}
|
||||||
|
}
|
@ -55,7 +55,7 @@ export class Dialog {
|
|||||||
}),
|
}),
|
||||||
),
|
),
|
||||||
this.$content = CE('div', {'class': 'bx-dialog-content'}, content),
|
this.$content = CE('div', {'class': 'bx-dialog-content'}, content),
|
||||||
!hideCloseButton && ($close = CE('button', {}, t('close'))),
|
!hideCloseButton && ($close = CE('button', {type: 'button'}, t('close'))),
|
||||||
);
|
);
|
||||||
|
|
||||||
$close && $close.addEventListener('click', e => {
|
$close && $close.addEventListener('click', e => {
|
||||||
|
@ -3,14 +3,8 @@ import { BxIcon } from "@utils/bx-icon";
|
|||||||
import { createButton, ButtonStyle, CE } from "@utils/html";
|
import { createButton, ButtonStyle, CE } from "@utils/html";
|
||||||
import { t } from "@utils/translation";
|
import { t } from "@utils/translation";
|
||||||
import { BaseGameBarAction } from "./action-base";
|
import { BaseGameBarAction } from "./action-base";
|
||||||
|
import { MicrophoneShortcut, MicrophoneState } from "../shortcuts/shortcut-microphone";
|
||||||
|
|
||||||
enum MicrophoneState {
|
|
||||||
REQUESTED = 'Requested',
|
|
||||||
ENABLED = 'Enabled',
|
|
||||||
MUTED = 'Muted',
|
|
||||||
NOT_ALLOWED = 'NotAllowed',
|
|
||||||
NOT_FOUND = 'NotFound',
|
|
||||||
}
|
|
||||||
|
|
||||||
export class MicrophoneAction extends BaseGameBarAction {
|
export class MicrophoneAction extends BaseGameBarAction {
|
||||||
$content: HTMLElement;
|
$content: HTMLElement;
|
||||||
@ -22,15 +16,9 @@ export class MicrophoneAction extends BaseGameBarAction {
|
|||||||
|
|
||||||
const onClick = (e: Event) => {
|
const onClick = (e: Event) => {
|
||||||
BxEvent.dispatch(window, BxEvent.GAME_BAR_ACTION_ACTIVATED);
|
BxEvent.dispatch(window, BxEvent.GAME_BAR_ACTION_ACTIVATED);
|
||||||
const state = this.$content.getAttribute('data-enabled');
|
|
||||||
const enableMic = state === 'true' ? false : true;
|
|
||||||
|
|
||||||
try {
|
const enabled = MicrophoneShortcut.toggle(false);
|
||||||
window.BX_EXPOSED.streamSession.tryEnableChatAsync(enableMic);
|
this.$content.setAttribute('data-enabled', enabled.toString());
|
||||||
this.$content.setAttribute('data-enabled', enableMic.toString());
|
|
||||||
} catch (e) {
|
|
||||||
console.log(e);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const $btnDefault = createButton({
|
const $btnDefault = createButton({
|
||||||
|
@ -82,6 +82,18 @@ export class GameBar {
|
|||||||
document.documentElement.appendChild($gameBar);
|
document.documentElement.appendChild($gameBar);
|
||||||
this.$gameBar = $gameBar;
|
this.$gameBar = $gameBar;
|
||||||
this.$container = $container;
|
this.$container = $container;
|
||||||
|
|
||||||
|
// Enable/disable Game Bar when playing/pausing
|
||||||
|
getPref(PrefKey.GAME_BAR_POSITION) !== 'off' && window.addEventListener(BxEvent.XCLOUD_POLLING_MODE_CHANGED, ((e: Event) => {
|
||||||
|
if (!STATES.isPlaying) {
|
||||||
|
this.disable();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Toggle Game bar
|
||||||
|
const mode = (e as any).mode;
|
||||||
|
mode !== 'None' ? this.disable() : this.enable();
|
||||||
|
}).bind(this));
|
||||||
}
|
}
|
||||||
|
|
||||||
private beginHideTimeout() {
|
private beginHideTimeout() {
|
||||||
|
@ -46,6 +46,10 @@ export class LoadingScreen {
|
|||||||
#game-stream div[class*=RocketAnimation-module__container] > svg {
|
#game-stream div[class*=RocketAnimation-module__container] > svg {
|
||||||
display: none;
|
display: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#game-stream video[class*=RocketAnimationVideo-module__video] {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
`;
|
`;
|
||||||
$bgStyle.textContent += css;
|
$bgStyle.textContent += css;
|
||||||
}
|
}
|
||||||
@ -163,9 +167,9 @@ export class LoadingScreen {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static reset() {
|
static reset() {
|
||||||
LoadingScreen.#$waitTimeBox && LoadingScreen.#$waitTimeBox.classList.add('bx-gone');
|
LoadingScreen.#$bgStyle && setTimeout(() => LoadingScreen.#$bgStyle.textContent = '', 2000);
|
||||||
LoadingScreen.#$bgStyle && (LoadingScreen.#$bgStyle.textContent = '');
|
|
||||||
|
|
||||||
|
LoadingScreen.#$waitTimeBox && LoadingScreen.#$waitTimeBox.classList.add('bx-gone');
|
||||||
LoadingScreen.#waitTimeInterval && clearInterval(LoadingScreen.#waitTimeInterval);
|
LoadingScreen.#waitTimeInterval && clearInterval(LoadingScreen.#waitTimeInterval);
|
||||||
LoadingScreen.#waitTimeInterval = null;
|
LoadingScreen.#waitTimeInterval = null;
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import type { GamepadKeyNameType } from "@/types/mkb";
|
import type { GamepadKeyNameType } from "@/types/mkb";
|
||||||
|
import { PrompFont } from "@/utils/prompt-font";
|
||||||
|
|
||||||
export enum GamepadKey {
|
export enum GamepadKey {
|
||||||
A = 0,
|
A = 0,
|
||||||
@ -18,6 +19,7 @@ export enum GamepadKey {
|
|||||||
LEFT = 14,
|
LEFT = 14,
|
||||||
RIGHT = 15,
|
RIGHT = 15,
|
||||||
HOME = 16,
|
HOME = 16,
|
||||||
|
SHARE = 17,
|
||||||
|
|
||||||
LS_UP = 100,
|
LS_UP = 100,
|
||||||
LS_DOWN = 101,
|
LS_DOWN = 101,
|
||||||
@ -32,36 +34,36 @@ export enum GamepadKey {
|
|||||||
|
|
||||||
|
|
||||||
export const GamepadKeyName: GamepadKeyNameType = {
|
export const GamepadKeyName: GamepadKeyNameType = {
|
||||||
[GamepadKey.A]: ['A', '⇓'],
|
[GamepadKey.A]: ['A', PrompFont.A],
|
||||||
[GamepadKey.B]: ['B', '⇒'],
|
[GamepadKey.B]: ['B', PrompFont.B],
|
||||||
[GamepadKey.X]: ['X', '⇐'],
|
[GamepadKey.X]: ['X', PrompFont.X],
|
||||||
[GamepadKey.Y]: ['Y', '⇑'],
|
[GamepadKey.Y]: ['Y', PrompFont.Y],
|
||||||
|
|
||||||
[GamepadKey.LB]: ['LB', '↘'],
|
[GamepadKey.LB]: ['LB', PrompFont.LB],
|
||||||
[GamepadKey.RB]: ['RB', '↙'],
|
[GamepadKey.RB]: ['RB', PrompFont.RB],
|
||||||
[GamepadKey.LT]: ['LT', '↖'],
|
[GamepadKey.LT]: ['LT', PrompFont.LT],
|
||||||
[GamepadKey.RT]: ['RT', '↗'],
|
[GamepadKey.RT]: ['RT', PrompFont.RT],
|
||||||
|
|
||||||
[GamepadKey.SELECT]: ['Select', '⇺'],
|
[GamepadKey.SELECT]: ['Select', PrompFont.SELECT],
|
||||||
[GamepadKey.START]: ['Start', '⇻'],
|
[GamepadKey.START]: ['Start', PrompFont.START],
|
||||||
[GamepadKey.HOME]: ['Home', ''],
|
[GamepadKey.HOME]: ['Home', PrompFont.HOME],
|
||||||
|
|
||||||
[GamepadKey.UP]: ['D-Pad Up', '≻'],
|
[GamepadKey.UP]: ['D-Pad Up', PrompFont.UP],
|
||||||
[GamepadKey.DOWN]: ['D-Pad Down', '≽'],
|
[GamepadKey.DOWN]: ['D-Pad Down', PrompFont.DOWN],
|
||||||
[GamepadKey.LEFT]: ['D-Pad Left', '≺'],
|
[GamepadKey.LEFT]: ['D-Pad Left', PrompFont.LEFT],
|
||||||
[GamepadKey.RIGHT]: ['D-Pad Right', '≼'],
|
[GamepadKey.RIGHT]: ['D-Pad Right', PrompFont.RIGHT],
|
||||||
|
|
||||||
[GamepadKey.L3]: ['L3', '↺'],
|
[GamepadKey.L3]: ['L3', PrompFont.L3],
|
||||||
[GamepadKey.LS_UP]: ['Left Stick Up', '↾'],
|
[GamepadKey.LS_UP]: ['Left Stick Up', PrompFont.LS_UP],
|
||||||
[GamepadKey.LS_DOWN]: ['Left Stick Down', '⇂'],
|
[GamepadKey.LS_DOWN]: ['Left Stick Down', PrompFont.LS_DOWN],
|
||||||
[GamepadKey.LS_LEFT]: ['Left Stick Left', '↼'],
|
[GamepadKey.LS_LEFT]: ['Left Stick Left', PrompFont.LS_LEFT],
|
||||||
[GamepadKey.LS_RIGHT]: ['Left Stick Right', '⇀'],
|
[GamepadKey.LS_RIGHT]: ['Left Stick Right', PrompFont.LS_RIGHT],
|
||||||
|
|
||||||
[GamepadKey.R3]: ['R3', '↻'],
|
[GamepadKey.R3]: ['R3', PrompFont.R3],
|
||||||
[GamepadKey.RS_UP]: ['Right Stick Up', '↿'],
|
[GamepadKey.RS_UP]: ['Right Stick Up', PrompFont.RS_UP],
|
||||||
[GamepadKey.RS_DOWN]: ['Right Stick Down', '⇃'],
|
[GamepadKey.RS_DOWN]: ['Right Stick Down', PrompFont.RS_DOWN],
|
||||||
[GamepadKey.RS_LEFT]: ['Right Stick Left', '↽'],
|
[GamepadKey.RS_LEFT]: ['Right Stick Left', PrompFont.RS_LEFT],
|
||||||
[GamepadKey.RS_RIGHT]: ['Right Stick Right', '⇁'],
|
[GamepadKey.RS_RIGHT]: ['Right Stick Right', PrompFont.RS_RIGHT],
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
@ -97,7 +99,4 @@ export enum MkbPresetKey {
|
|||||||
MOUSE_SENSITIVITY_Y = 'sensitivity_y',
|
MOUSE_SENSITIVITY_Y = 'sensitivity_y',
|
||||||
|
|
||||||
MOUSE_DEADZONE_COUNTERWEIGHT = 'deadzone_counterweight',
|
MOUSE_DEADZONE_COUNTERWEIGHT = 'deadzone_counterweight',
|
||||||
|
|
||||||
MOUSE_STICK_DECAY_STRENGTH = 'stick_decay_strength',
|
|
||||||
MOUSE_STICK_DECAY_MIN = 'stick_decay_min',
|
|
||||||
}
|
}
|
||||||
|
@ -20,7 +20,7 @@ export class KeyHelper {
|
|||||||
let name;
|
let name;
|
||||||
|
|
||||||
if (e instanceof KeyboardEvent) {
|
if (e instanceof KeyboardEvent) {
|
||||||
code = e.code;
|
code = e.code || e.key;
|
||||||
} else if (e instanceof WheelEvent) {
|
} else if (e instanceof WheelEvent) {
|
||||||
if (e.deltaY < 0) {
|
if (e.deltaY < 0) {
|
||||||
code = WheelCode.SCROLL_UP;
|
code = WheelCode.SCROLL_UP;
|
||||||
@ -28,7 +28,7 @@ export class KeyHelper {
|
|||||||
code = WheelCode.SCROLL_DOWN;
|
code = WheelCode.SCROLL_DOWN;
|
||||||
} else if (e.deltaX < 0) {
|
} else if (e.deltaX < 0) {
|
||||||
code = WheelCode.SCROLL_LEFT;
|
code = WheelCode.SCROLL_LEFT;
|
||||||
} else {
|
} else if (e.deltaX > 0) {
|
||||||
code = WheelCode.SCROLL_RIGHT;
|
code = WheelCode.SCROLL_RIGHT;
|
||||||
}
|
}
|
||||||
} else if (e instanceof MouseEvent) {
|
} else if (e instanceof MouseEvent) {
|
||||||
|
@ -9,13 +9,152 @@ import { LocalDb } from "@utils/local-db";
|
|||||||
import { KeyHelper } from "./key-helper";
|
import { KeyHelper } from "./key-helper";
|
||||||
import type { MkbStoredPreset } from "@/types/mkb";
|
import type { MkbStoredPreset } from "@/types/mkb";
|
||||||
import { showStreamSettings } from "@modules/stream/stream-ui";
|
import { showStreamSettings } from "@modules/stream/stream-ui";
|
||||||
import { STATES } from "@utils/global";
|
import { AppInterface, STATES } from "@utils/global";
|
||||||
import { UserAgent } from "@utils/user-agent";
|
import { UserAgent } from "@utils/user-agent";
|
||||||
import { BxLogger } from "@utils/bx-logger";
|
import { BxLogger } from "@utils/bx-logger";
|
||||||
import { BxIcon } from "@utils/bx-icon";
|
import { BxIcon } from "@utils/bx-icon";
|
||||||
|
import { PointerClient } from "./pointer-client";
|
||||||
|
|
||||||
const LOG_TAG = 'MkbHandler';
|
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;
|
||||||
|
}
|
||||||
|
|
||||||
|
class WebSocketMouseDataProvider extends MouseDataProvider {
|
||||||
|
#pointerClient: PointerClient | undefined
|
||||||
|
#connected = false
|
||||||
|
|
||||||
|
init(): void {
|
||||||
|
this.#pointerClient = PointerClient.getInstance();
|
||||||
|
this.#connected = false;
|
||||||
|
try {
|
||||||
|
this.#pointerClient.start(this.mkbHandler);
|
||||||
|
this.#connected = true;
|
||||||
|
} catch (e) {
|
||||||
|
Toast.show('Cannot enable Mouse & Keyboard feature');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
start(): void {
|
||||||
|
this.#connected && AppInterface.requestPointerCapture();
|
||||||
|
}
|
||||||
|
|
||||||
|
stop(): void {
|
||||||
|
this.#connected && AppInterface.releasePointerCapture();
|
||||||
|
}
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
||||||
|
|
||||||
|
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('contextmenu', this.#disableContextMenu);
|
||||||
|
}
|
||||||
|
|
||||||
|
stop(): void {
|
||||||
|
window.removeEventListener('mousemove', this.#onMouseMoveEvent);
|
||||||
|
window.removeEventListener('mousedown', this.#onMouseEvent);
|
||||||
|
window.removeEventListener('mouseup', this.#onMouseEvent);
|
||||||
|
window.removeEventListener('wheel', this.#onWheelEvent);
|
||||||
|
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();
|
||||||
|
}
|
||||||
|
|
||||||
|
#onMouseMoveEvent = (e: MouseEvent) => {
|
||||||
|
this.mkbHandler.handleMouseMove({
|
||||||
|
movementX: e.movementX,
|
||||||
|
movementY: e.movementY,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
#onMouseEvent = (e: MouseEvent) => {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
const isMouseDown = e.type === 'mousedown';
|
||||||
|
const key = KeyHelper.getKeyFromEvent(e);
|
||||||
|
const data: MkbMouseClick = {
|
||||||
|
key: key,
|
||||||
|
pressed: isMouseDown
|
||||||
|
};
|
||||||
|
|
||||||
|
this.mkbHandler.handleMouseClick(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
#onWheelEvent = (e: WheelEvent) => {
|
||||||
|
const key = KeyHelper.getKeyFromEvent(e);
|
||||||
|
if (!key) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.mkbHandler.handleMouseWheel({key})) {
|
||||||
|
e.preventDefault();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#disableContextMenu = (e: Event) => e.preventDefault();
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
This class uses some code from Yuzu emulator to handle mouse's movements
|
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
|
Source: https://github.com/yuzu-emu/yuzu-mainline/blob/master/src/input_common/drivers/mouse.cpp
|
||||||
@ -33,7 +172,6 @@ export class MkbHandler {
|
|||||||
#CURRENT_PRESET_DATA = MkbPreset.convert(MkbPreset.DEFAULT_PRESET);
|
#CURRENT_PRESET_DATA = MkbPreset.convert(MkbPreset.DEFAULT_PRESET);
|
||||||
|
|
||||||
static readonly DEFAULT_PANNING_SENSITIVITY = 0.0010;
|
static readonly DEFAULT_PANNING_SENSITIVITY = 0.0010;
|
||||||
static readonly DEFAULT_STICK_SENSITIVITY = 0.0006;
|
|
||||||
static readonly DEFAULT_DEADZONE_COUNTERWEIGHT = 0.01;
|
static readonly DEFAULT_DEADZONE_COUNTERWEIGHT = 0.01;
|
||||||
static readonly MAXIMUM_STICK_RANGE = 1.1;
|
static readonly MAXIMUM_STICK_RANGE = 1.1;
|
||||||
|
|
||||||
@ -55,13 +193,13 @@ export class MkbHandler {
|
|||||||
#nativeGetGamepads = window.navigator.getGamepads.bind(window.navigator);
|
#nativeGetGamepads = window.navigator.getGamepads.bind(window.navigator);
|
||||||
|
|
||||||
#enabled = false;
|
#enabled = false;
|
||||||
|
#mouseDataProvider: MouseDataProvider | undefined;
|
||||||
#isPolling = false;
|
#isPolling = false;
|
||||||
|
|
||||||
#prevWheelCode = null;
|
#prevWheelCode = null;
|
||||||
#wheelStoppedTimeout?: number | null;
|
#wheelStoppedTimeout?: number | null;
|
||||||
|
|
||||||
#detectMouseStoppedTimeout?: number | null;
|
#detectMouseStoppedTimeout?: number | null;
|
||||||
#allowStickDecaying = false;
|
|
||||||
|
|
||||||
#$message?: HTMLElement;
|
#$message?: HTMLElement;
|
||||||
|
|
||||||
@ -85,6 +223,8 @@ export class MkbHandler {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
isEnabled = () => this.#enabled;
|
||||||
|
|
||||||
#patchedGetGamepads = () => {
|
#patchedGetGamepads = () => {
|
||||||
const gamepads = this.#nativeGetGamepads() || [];
|
const gamepads = this.#nativeGetGamepads() || [];
|
||||||
(gamepads as any)[this.#VIRTUAL_GAMEPAD.index] = this.#VIRTUAL_GAMEPAD;
|
(gamepads as any)[this.#VIRTUAL_GAMEPAD.index] = this.#VIRTUAL_GAMEPAD;
|
||||||
@ -102,6 +242,7 @@ export class MkbHandler {
|
|||||||
virtualGamepad.timestamp = performance.now();
|
virtualGamepad.timestamp = performance.now();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
#getStickAxes(stick: GamepadStick) {
|
#getStickAxes(stick: GamepadStick) {
|
||||||
const virtualGamepad = this.#getVirtualGamepad();
|
const virtualGamepad = this.#getVirtualGamepad();
|
||||||
return {
|
return {
|
||||||
@ -109,11 +250,10 @@ export class MkbHandler {
|
|||||||
y: virtualGamepad.axes[stick * 2 + 1],
|
y: virtualGamepad.axes[stick * 2 + 1],
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
#vectorLength = (x: number, y: number): number => Math.sqrt(x ** 2 + y ** 2);
|
#vectorLength = (x: number, y: number): number => Math.sqrt(x ** 2 + y ** 2);
|
||||||
|
|
||||||
#disableContextMenu = (e: Event) => e.preventDefault();
|
|
||||||
|
|
||||||
#resetGamepad = () => {
|
#resetGamepad = () => {
|
||||||
const gamepad = this.#getVirtualGamepad();
|
const gamepad = this.#getVirtualGamepad();
|
||||||
|
|
||||||
@ -172,6 +312,10 @@ export class MkbHandler {
|
|||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
this.toggle();
|
this.toggle();
|
||||||
return;
|
return;
|
||||||
|
} else if (e.code === 'Escape') {
|
||||||
|
e.preventDefault();
|
||||||
|
this.#enabled && this.stop();
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!this.#isPolling) {
|
if (!this.#isPolling) {
|
||||||
@ -179,7 +323,7 @@ export class MkbHandler {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const buttonIndex = this.#CURRENT_PRESET_DATA.mapping[e.code]!;
|
const buttonIndex = this.#CURRENT_PRESET_DATA.mapping[e.code || e.key]!;
|
||||||
if (typeof buttonIndex === 'undefined') {
|
if (typeof buttonIndex === 'undefined') {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -193,89 +337,29 @@ export class MkbHandler {
|
|||||||
this.#pressButton(buttonIndex, isKeyDown);
|
this.#pressButton(buttonIndex, isKeyDown);
|
||||||
}
|
}
|
||||||
|
|
||||||
#onMouseEvent = (e: MouseEvent) => {
|
#onMouseStopped = () => {
|
||||||
const isMouseDown = e.type === 'mousedown';
|
// Reset stick position
|
||||||
const key = KeyHelper.getKeyFromEvent(e);
|
this.#detectMouseStoppedTimeout = null;
|
||||||
if (!key) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const buttonIndex = this.#CURRENT_PRESET_DATA.mapping[key.code]!;
|
|
||||||
if (typeof buttonIndex === 'undefined') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
e.preventDefault();
|
|
||||||
this.#pressButton(buttonIndex, isMouseDown);
|
|
||||||
}
|
|
||||||
|
|
||||||
#onWheelEvent = (e: WheelEvent) => {
|
|
||||||
const key = KeyHelper.getKeyFromEvent(e);
|
|
||||||
if (!key) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const buttonIndex = this.#CURRENT_PRESET_DATA.mapping[key.code]!;
|
|
||||||
if (typeof buttonIndex === 'undefined') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
e.preventDefault();
|
|
||||||
|
|
||||||
if (this.#prevWheelCode === null || this.#prevWheelCode === key.code) {
|
|
||||||
this.#wheelStoppedTimeout && clearTimeout(this.#wheelStoppedTimeout);
|
|
||||||
this.#pressButton(buttonIndex, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.#wheelStoppedTimeout = window.setTimeout(() => {
|
|
||||||
this.#prevWheelCode = null;
|
|
||||||
this.#pressButton(buttonIndex, false);
|
|
||||||
}, 20);
|
|
||||||
}
|
|
||||||
|
|
||||||
#decayStick = () => {
|
|
||||||
if (!this.#allowStickDecaying) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const mouseMapTo = this.#CURRENT_PRESET_DATA.mouse[MkbPresetKey.MOUSE_MAP_TO];
|
const mouseMapTo = this.#CURRENT_PRESET_DATA.mouse[MkbPresetKey.MOUSE_MAP_TO];
|
||||||
if (mouseMapTo === MouseMapTo.OFF) {
|
const analog = mouseMapTo === MouseMapTo.LS ? GamepadStick.LEFT : GamepadStick.RIGHT;
|
||||||
|
this.#updateStick(analog, 0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
handleMouseClick = (data: MkbMouseClick) => {
|
||||||
|
if (!data || !data.key) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const analog = mouseMapTo === MouseMapTo.LS ? GamepadStick.LEFT : GamepadStick.RIGHT;
|
const buttonIndex = this.#CURRENT_PRESET_DATA.mapping[data.key.code]!;
|
||||||
|
if (typeof buttonIndex === 'undefined') {
|
||||||
let { x, y } = this.#getStickAxes(analog);
|
return;
|
||||||
const length = this.#vectorLength(x, y);
|
|
||||||
|
|
||||||
const clampedLength = Math.min(1.0, length);
|
|
||||||
const decayStrength = this.#CURRENT_PRESET_DATA.mouse[MkbPresetKey.MOUSE_STICK_DECAY_STRENGTH];
|
|
||||||
const decay = 1 - clampedLength * clampedLength * decayStrength;
|
|
||||||
const minDecay = this.#CURRENT_PRESET_DATA.mouse[MkbPresetKey.MOUSE_STICK_DECAY_MIN];
|
|
||||||
const clampedDecay = Math.min(1 - minDecay, decay);
|
|
||||||
|
|
||||||
x *= clampedDecay;
|
|
||||||
y *= clampedDecay;
|
|
||||||
|
|
||||||
const deadzoneCounterweight = 20 * MkbHandler.DEFAULT_DEADZONE_COUNTERWEIGHT;
|
|
||||||
if (Math.abs(x) <= deadzoneCounterweight && Math.abs(y) <= deadzoneCounterweight) {
|
|
||||||
x = 0;
|
|
||||||
y = 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.#allowStickDecaying) {
|
this.#pressButton(buttonIndex, data.pressed);
|
||||||
this.#updateStick(analog, x, y);
|
|
||||||
|
|
||||||
(x !== 0 || y !== 0) && requestAnimationFrame(this.#decayStick);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#onMouseStopped = () => {
|
handleMouseMove = (data: MkbMouseMove) => {
|
||||||
this.#allowStickDecaying = true;
|
|
||||||
requestAnimationFrame(this.#decayStick);
|
|
||||||
}
|
|
||||||
|
|
||||||
#onMouseMoveEvent = (e: MouseEvent) => {
|
|
||||||
// TODO: optimize this
|
// TODO: optimize this
|
||||||
const mouseMapTo = this.#CURRENT_PRESET_DATA.mouse[MkbPresetKey.MOUSE_MAP_TO];
|
const mouseMapTo = this.#CURRENT_PRESET_DATA.mouse[MkbPresetKey.MOUSE_MAP_TO];
|
||||||
if (mouseMapTo === MouseMapTo.OFF) {
|
if (mouseMapTo === MouseMapTo.OFF) {
|
||||||
@ -283,17 +367,13 @@ export class MkbHandler {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.#allowStickDecaying = false;
|
|
||||||
this.#detectMouseStoppedTimeout && clearTimeout(this.#detectMouseStoppedTimeout);
|
this.#detectMouseStoppedTimeout && clearTimeout(this.#detectMouseStoppedTimeout);
|
||||||
this.#detectMouseStoppedTimeout = window.setTimeout(this.#onMouseStopped.bind(this), 100);
|
this.#detectMouseStoppedTimeout = window.setTimeout(this.#onMouseStopped.bind(this), 50);
|
||||||
|
|
||||||
const deltaX = e.movementX;
|
|
||||||
const deltaY = e.movementY;
|
|
||||||
|
|
||||||
const deadzoneCounterweight = this.#CURRENT_PRESET_DATA.mouse[MkbPresetKey.MOUSE_DEADZONE_COUNTERWEIGHT];
|
const deadzoneCounterweight = this.#CURRENT_PRESET_DATA.mouse[MkbPresetKey.MOUSE_DEADZONE_COUNTERWEIGHT];
|
||||||
|
|
||||||
let x = deltaX * this.#CURRENT_PRESET_DATA.mouse[MkbPresetKey.MOUSE_SENSITIVITY_X];
|
let x = data.movementX * this.#CURRENT_PRESET_DATA.mouse[MkbPresetKey.MOUSE_SENSITIVITY_X];
|
||||||
let y = deltaY * this.#CURRENT_PRESET_DATA.mouse[MkbPresetKey.MOUSE_SENSITIVITY_Y];
|
let y = data.movementY * this.#CURRENT_PRESET_DATA.mouse[MkbPresetKey.MOUSE_SENSITIVITY_Y];
|
||||||
|
|
||||||
let length = this.#vectorLength(x, y);
|
let length = this.#vectorLength(x, y);
|
||||||
if (length !== 0 && length < deadzoneCounterweight) {
|
if (length !== 0 && length < deadzoneCounterweight) {
|
||||||
@ -308,18 +388,33 @@ export class MkbHandler {
|
|||||||
this.#updateStick(analog, x, y);
|
this.#updateStick(analog, x, y);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
handleMouseWheel = (data: MkbMouseWheel): boolean => {
|
||||||
|
if (!data || !data.key) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const buttonIndex = this.#CURRENT_PRESET_DATA.mapping[data.key.code]!;
|
||||||
|
if (typeof buttonIndex === 'undefined') {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.#prevWheelCode === null || this.#prevWheelCode === data.key.code) {
|
||||||
|
this.#wheelStoppedTimeout && clearTimeout(this.#wheelStoppedTimeout);
|
||||||
|
this.#pressButton(buttonIndex, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.#wheelStoppedTimeout = window.setTimeout(() => {
|
||||||
|
this.#prevWheelCode = null;
|
||||||
|
this.#pressButton(buttonIndex, false);
|
||||||
|
}, 20);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
toggle = () => {
|
toggle = () => {
|
||||||
this.#enabled = !this.#enabled;
|
this.#enabled = !this.#enabled;
|
||||||
this.#enabled ? document.pointerLockElement && this.start() : this.stop();
|
|
||||||
|
|
||||||
Toast.show(t('mouse-and-keyboard'), t(this.#enabled ? 'enabled' : 'disabled'), {instant: true});
|
Toast.show(t('mouse-and-keyboard'), t(this.#enabled ? 'enabled' : 'disabled'), {instant: true});
|
||||||
|
this.#mouseDataProvider?.toggle(this.#enabled);
|
||||||
if (this.#enabled) {
|
|
||||||
!document.pointerLockElement && this.#waitForPointerLock(true);
|
|
||||||
} else {
|
|
||||||
this.#waitForPointerLock(false);
|
|
||||||
document.pointerLockElement && document.exitPointerLock();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#getCurrentPreset = (): Promise<MkbStoredPreset> => {
|
#getCurrentPreset = (): Promise<MkbStoredPreset> => {
|
||||||
@ -338,72 +433,73 @@ export class MkbHandler {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
#onPointerLockChange = () => {
|
waitForMouseData = (wait: boolean) => {
|
||||||
if (this.#enabled && !document.pointerLockElement) {
|
|
||||||
this.stop();
|
|
||||||
this.#waitForPointerLock(true);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#onPointerLockError = (e: Event) => {
|
|
||||||
console.log(e);
|
|
||||||
this.stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
#onActivatePointerLock = () => {
|
|
||||||
if (!document.pointerLockElement) {
|
|
||||||
document.body.requestPointerLock();
|
|
||||||
}
|
|
||||||
|
|
||||||
this.#waitForPointerLock(false);
|
|
||||||
this.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
#waitForPointerLock = (wait: boolean) => {
|
|
||||||
this.#$message && this.#$message.classList.toggle('bx-gone', !wait);
|
this.#$message && this.#$message.classList.toggle('bx-gone', !wait);
|
||||||
}
|
}
|
||||||
|
|
||||||
#onStreamMenuShown = () => {
|
#onPollingModeChanged = (e: Event) => {
|
||||||
this.#enabled && this.#waitForPointerLock(false);
|
if (!this.#$message) {
|
||||||
}
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
#onStreamMenuHidden = () => {
|
const mode = (e as any).mode;
|
||||||
this.#enabled && this.#waitForPointerLock(true);
|
if (mode === 'None') {
|
||||||
|
this.#$message.classList.remove('bx-offscreen');
|
||||||
|
} else {
|
||||||
|
this.#$message.classList.add('bx-offscreen');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
init = () => {
|
init = () => {
|
||||||
this.refreshPresetData();
|
this.refreshPresetData();
|
||||||
this.#enabled = true;
|
this.#enabled = true;
|
||||||
|
|
||||||
|
if (AppInterface) {
|
||||||
|
this.#mouseDataProvider = new WebSocketMouseDataProvider(this);
|
||||||
|
} else {
|
||||||
|
this.#mouseDataProvider = new PointerLockMouseDataProvider(this);
|
||||||
|
}
|
||||||
|
this.#mouseDataProvider.init();
|
||||||
|
|
||||||
window.addEventListener('keydown', this.#onKeyboardEvent);
|
window.addEventListener('keydown', this.#onKeyboardEvent);
|
||||||
|
|
||||||
document.addEventListener('pointerlockchange', this.#onPointerLockChange);
|
|
||||||
document.addEventListener('pointerlockerror', this.#onPointerLockError);
|
|
||||||
|
|
||||||
this.#$message = CE('div', {'class': 'bx-mkb-pointer-lock-msg bx-gone'},
|
this.#$message = CE('div', {'class': 'bx-mkb-pointer-lock-msg bx-gone'},
|
||||||
createButton({
|
|
||||||
icon: BxIcon.MOUSE_SETTINGS,
|
|
||||||
style: ButtonStyle.PRIMARY,
|
|
||||||
onClick: e => {
|
|
||||||
e.preventDefault();
|
|
||||||
e.stopPropagation();
|
|
||||||
|
|
||||||
showStreamSettings('mkb');
|
|
||||||
},
|
|
||||||
}),
|
|
||||||
CE('div', {},
|
CE('div', {},
|
||||||
CE('p', {}, t('mkb-click-to-activate')),
|
CE('p', {}, t('mkb-click-to-activate')),
|
||||||
CE('p', {}, t('press-key-to-toggle-mkb', {key: 'F8'})),
|
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();
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
this.#$message.addEventListener('click', this.#onActivatePointerLock);
|
this.#$message.addEventListener('click', this.start.bind(this));
|
||||||
document.documentElement.appendChild(this.#$message);
|
document.documentElement.appendChild(this.#$message);
|
||||||
|
|
||||||
window.addEventListener(BxEvent.STREAM_MENU_SHOWN, this.#onStreamMenuShown);
|
window.addEventListener(BxEvent.XCLOUD_POLLING_MODE_CHANGED, this.#onPollingModeChanged);
|
||||||
window.addEventListener(BxEvent.STREAM_MENU_HIDDEN, this.#onStreamMenuHidden);
|
|
||||||
|
|
||||||
this.#waitForPointerLock(true);
|
this.waitForMouseData(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
destroy = () => {
|
destroy = () => {
|
||||||
@ -411,31 +507,31 @@ export class MkbHandler {
|
|||||||
this.#enabled = false;
|
this.#enabled = false;
|
||||||
this.stop();
|
this.stop();
|
||||||
|
|
||||||
this.#waitForPointerLock(false);
|
this.waitForMouseData(false);
|
||||||
document.pointerLockElement && document.exitPointerLock();
|
document.pointerLockElement && document.exitPointerLock();
|
||||||
|
|
||||||
window.removeEventListener('keydown', this.#onKeyboardEvent);
|
window.removeEventListener('keydown', this.#onKeyboardEvent);
|
||||||
|
|
||||||
document.removeEventListener('pointerlockchange', this.#onPointerLockChange);
|
this.#mouseDataProvider?.destroy();
|
||||||
document.removeEventListener('pointerlockerror', this.#onPointerLockError);
|
|
||||||
|
|
||||||
window.removeEventListener(BxEvent.STREAM_MENU_SHOWN, this.#onStreamMenuShown);
|
window.removeEventListener(BxEvent.XCLOUD_POLLING_MODE_CHANGED, this.#onPollingModeChanged);
|
||||||
window.removeEventListener(BxEvent.STREAM_MENU_HIDDEN, this.#onStreamMenuHidden);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
start = () => {
|
start = () => {
|
||||||
|
if (!this.#enabled) {
|
||||||
|
this.#enabled = true;
|
||||||
|
Toast.show(t('mouse-and-keyboard'), t('enabled'), {instant: true});
|
||||||
|
}
|
||||||
|
|
||||||
this.#isPolling = true;
|
this.#isPolling = true;
|
||||||
window.navigator.getGamepads = this.#patchedGetGamepads;
|
|
||||||
|
|
||||||
this.#resetGamepad();
|
this.#resetGamepad();
|
||||||
|
window.navigator.getGamepads = this.#patchedGetGamepads;
|
||||||
|
|
||||||
|
this.waitForMouseData(false);
|
||||||
|
|
||||||
window.addEventListener('keyup', this.#onKeyboardEvent);
|
window.addEventListener('keyup', this.#onKeyboardEvent);
|
||||||
|
this.#mouseDataProvider?.start();
|
||||||
window.addEventListener('mousemove', this.#onMouseMoveEvent);
|
|
||||||
window.addEventListener('mousedown', this.#onMouseEvent);
|
|
||||||
window.addEventListener('mouseup', this.#onMouseEvent);
|
|
||||||
window.addEventListener('wheel', this.#onWheelEvent);
|
|
||||||
window.addEventListener('contextmenu', this.#disableContextMenu);
|
|
||||||
|
|
||||||
// Dispatch "gamepadconnected" event
|
// Dispatch "gamepadconnected" event
|
||||||
const virtualGamepad = this.#getVirtualGamepad();
|
const virtualGamepad = this.#getVirtualGamepad();
|
||||||
@ -451,6 +547,8 @@ export class MkbHandler {
|
|||||||
this.#isPolling = false;
|
this.#isPolling = false;
|
||||||
|
|
||||||
// Dispatch "gamepaddisconnected" event
|
// Dispatch "gamepaddisconnected" event
|
||||||
|
this.#resetGamepad();
|
||||||
|
|
||||||
const virtualGamepad = this.#getVirtualGamepad();
|
const virtualGamepad = this.#getVirtualGamepad();
|
||||||
virtualGamepad.connected = false;
|
virtualGamepad.connected = false;
|
||||||
virtualGamepad.timestamp = performance.now();
|
virtualGamepad.timestamp = performance.now();
|
||||||
@ -461,19 +559,14 @@ export class MkbHandler {
|
|||||||
|
|
||||||
window.navigator.getGamepads = this.#nativeGetGamepads;
|
window.navigator.getGamepads = this.#nativeGetGamepads;
|
||||||
|
|
||||||
this.#resetGamepad();
|
|
||||||
|
|
||||||
window.removeEventListener('keyup', this.#onKeyboardEvent);
|
window.removeEventListener('keyup', this.#onKeyboardEvent);
|
||||||
|
|
||||||
window.removeEventListener('mousemove', this.#onMouseMoveEvent);
|
this.waitForMouseData(true);
|
||||||
window.removeEventListener('mousedown', this.#onMouseEvent);
|
this.#mouseDataProvider?.stop();
|
||||||
window.removeEventListener('mouseup', this.#onMouseEvent);
|
|
||||||
window.removeEventListener('wheel', this.#onWheelEvent);
|
|
||||||
window.removeEventListener('contextmenu', this.#disableContextMenu);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static setupEvents() {
|
static setupEvents() {
|
||||||
getPref(PrefKey.MKB_ENABLED) && !UserAgent.isMobile() && window.addEventListener(BxEvent.STREAM_PLAYING, () => {
|
getPref(PrefKey.MKB_ENABLED) && (AppInterface || !UserAgent.isMobile()) && window.addEventListener(BxEvent.STREAM_PLAYING, () => {
|
||||||
// Enable MKB
|
// Enable MKB
|
||||||
if (!STATES.currentStream.titleInfo?.details.hasMkbSupport) {
|
if (!STATES.currentStream.titleInfo?.details.hasMkbSupport) {
|
||||||
BxLogger.info(LOG_TAG, 'Emulate MKB');
|
BxLogger.info(LOG_TAG, 'Emulate MKB');
|
||||||
|
@ -24,11 +24,11 @@ export class MkbPreset {
|
|||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 50,
|
default: 50,
|
||||||
min: 1,
|
min: 1,
|
||||||
max: 200,
|
max: 300,
|
||||||
|
|
||||||
params: {
|
params: {
|
||||||
suffix: '%',
|
suffix: '%',
|
||||||
exactTicks: 20,
|
exactTicks: 50,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -37,11 +37,11 @@ export class MkbPreset {
|
|||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 50,
|
default: 50,
|
||||||
min: 1,
|
min: 1,
|
||||||
max: 200,
|
max: 300,
|
||||||
|
|
||||||
params: {
|
params: {
|
||||||
suffix: '%',
|
suffix: '%',
|
||||||
exactTicks: 20,
|
exactTicks: 50,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -50,38 +50,13 @@ export class MkbPreset {
|
|||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 20,
|
default: 20,
|
||||||
min: 1,
|
min: 1,
|
||||||
max: 100,
|
max: 50,
|
||||||
|
|
||||||
params: {
|
params: {
|
||||||
suffix: '%',
|
suffix: '%',
|
||||||
exactTicks: 10,
|
exactTicks: 10,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
[MkbPresetKey.MOUSE_STICK_DECAY_STRENGTH]: {
|
|
||||||
label: t('stick-decay-strength'),
|
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
|
||||||
default: 100,
|
|
||||||
min: 10,
|
|
||||||
max: 100,
|
|
||||||
|
|
||||||
params: {
|
|
||||||
suffix: '%',
|
|
||||||
exactTicks: 10,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
|
|
||||||
[MkbPresetKey.MOUSE_STICK_DECAY_MIN]: {
|
|
||||||
label: t('stick-decay-minimum'),
|
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
|
||||||
default: 10,
|
|
||||||
min: 1,
|
|
||||||
max: 10,
|
|
||||||
|
|
||||||
params: {
|
|
||||||
suffix: '%',
|
|
||||||
},
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static DEFAULT_PRESET: MkbPresetData = {
|
static DEFAULT_PRESET: MkbPresetData = {
|
||||||
@ -124,11 +99,9 @@ export class MkbPreset {
|
|||||||
|
|
||||||
'mouse': {
|
'mouse': {
|
||||||
[MkbPresetKey.MOUSE_MAP_TO]: MouseMapTo[MouseMapTo.RS],
|
[MkbPresetKey.MOUSE_MAP_TO]: MouseMapTo[MouseMapTo.RS],
|
||||||
[MkbPresetKey.MOUSE_SENSITIVITY_X]: 50,
|
[MkbPresetKey.MOUSE_SENSITIVITY_X]: 100,
|
||||||
[MkbPresetKey.MOUSE_SENSITIVITY_Y]: 50,
|
[MkbPresetKey.MOUSE_SENSITIVITY_Y]: 100,
|
||||||
[MkbPresetKey.MOUSE_DEADZONE_COUNTERWEIGHT]: 20,
|
[MkbPresetKey.MOUSE_DEADZONE_COUNTERWEIGHT]: 20,
|
||||||
[MkbPresetKey.MOUSE_STICK_DECAY_STRENGTH]: 18,
|
|
||||||
[MkbPresetKey.MOUSE_STICK_DECAY_MIN]: 6,
|
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -149,8 +122,6 @@ export class MkbPreset {
|
|||||||
mouse[MkbPresetKey.MOUSE_SENSITIVITY_X] *= MkbHandler.DEFAULT_PANNING_SENSITIVITY;
|
mouse[MkbPresetKey.MOUSE_SENSITIVITY_X] *= MkbHandler.DEFAULT_PANNING_SENSITIVITY;
|
||||||
mouse[MkbPresetKey.MOUSE_SENSITIVITY_Y] *= 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_DEADZONE_COUNTERWEIGHT] *= MkbHandler.DEFAULT_DEADZONE_COUNTERWEIGHT;
|
||||||
mouse[MkbPresetKey.MOUSE_STICK_DECAY_STRENGTH] *= 0.01;
|
|
||||||
mouse[MkbPresetKey.MOUSE_STICK_DECAY_MIN] *= 0.01;
|
|
||||||
|
|
||||||
const mouseMapTo = MouseMapTo[mouse[MkbPresetKey.MOUSE_MAP_TO]!];
|
const mouseMapTo = MouseMapTo[mouse[MkbPresetKey.MOUSE_MAP_TO]!];
|
||||||
if (typeof mouseMapTo !== 'undefined') {
|
if (typeof mouseMapTo !== 'undefined') {
|
||||||
|
@ -426,6 +426,7 @@ export class MkbRemapper {
|
|||||||
const $fragment = document.createDocumentFragment();
|
const $fragment = document.createDocumentFragment();
|
||||||
for (let i = 0; i < keysPerButton; i++) {
|
for (let i = 0; i < keysPerButton; i++) {
|
||||||
$elm = CE('button', {
|
$elm = CE('button', {
|
||||||
|
type: 'button',
|
||||||
'data-prompt': buttonPrompt,
|
'data-prompt': buttonPrompt,
|
||||||
'data-button-index': buttonIndex,
|
'data-button-index': buttonIndex,
|
||||||
'data-key-slot': i,
|
'data-key-slot': i,
|
||||||
@ -459,7 +460,7 @@ export class MkbRemapper {
|
|||||||
const onChange = (e: Event, value: any) => {
|
const onChange = (e: Event, value: any) => {
|
||||||
(this.#STATE.editingPresetData!.mouse as any)[key] = value;
|
(this.#STATE.editingPresetData!.mouse as any)[key] = value;
|
||||||
};
|
};
|
||||||
const $row = CE('div', {'class': 'bx-quick-settings-row'},
|
const $row = CE('div', {'class': 'bx-stream-settings-row'},
|
||||||
CE('label', {'for': `bx_setting_${key}`}, setting.label),
|
CE('label', {'for': `bx_setting_${key}`}, setting.label),
|
||||||
$elm = SettingElement.render(setting.type, key, setting, value, onChange, setting.params),
|
$elm = SettingElement.render(setting.type, key, setting, value, onChange, setting.params),
|
||||||
);
|
);
|
||||||
|
152
src/modules/mkb/pointer-client.ts
Normal file
152
src/modules/mkb/pointer-client.ts
Normal file
@ -0,0 +1,152 @@
|
|||||||
|
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";
|
||||||
|
|
||||||
|
const LOG_TAG = 'PointerClient';
|
||||||
|
|
||||||
|
enum PointerAction {
|
||||||
|
MOVE = 1,
|
||||||
|
BUTTON_PRESS = 2,
|
||||||
|
BUTTON_RELEASE = 3,
|
||||||
|
SCROLL = 4,
|
||||||
|
POINTER_CAPTURE_CHANGED = 5,
|
||||||
|
}
|
||||||
|
|
||||||
|
const FixedMouseIndex = {
|
||||||
|
1: 0,
|
||||||
|
2: 2,
|
||||||
|
4: 1,
|
||||||
|
}
|
||||||
|
|
||||||
|
export class PointerClient {
|
||||||
|
static #PORT = 9269;
|
||||||
|
|
||||||
|
private static instance: PointerClient;
|
||||||
|
public static getInstance(): PointerClient {
|
||||||
|
if (!PointerClient.instance) {
|
||||||
|
PointerClient.instance = new PointerClient();
|
||||||
|
}
|
||||||
|
|
||||||
|
return PointerClient.instance;
|
||||||
|
}
|
||||||
|
|
||||||
|
#socket: WebSocket | undefined | null;
|
||||||
|
#mkbHandler: MkbHandler | undefined;
|
||||||
|
|
||||||
|
start(mkbHandler: MkbHandler) {
|
||||||
|
this.#mkbHandler = mkbHandler;
|
||||||
|
|
||||||
|
// Create WebSocket connection.
|
||||||
|
this.#socket = new WebSocket(`ws://localhost:${PointerClient.#PORT}`);
|
||||||
|
this.#socket.binaryType = 'arraybuffer';
|
||||||
|
|
||||||
|
// Connection opened
|
||||||
|
this.#socket.addEventListener('open', (event) => {
|
||||||
|
BxLogger.info(LOG_TAG, 'connected')
|
||||||
|
});
|
||||||
|
|
||||||
|
// Error
|
||||||
|
this.#socket.addEventListener('error', (event) => {
|
||||||
|
BxLogger.error(LOG_TAG, event);
|
||||||
|
Toast.show('Cannot setup mouse');
|
||||||
|
});
|
||||||
|
|
||||||
|
this.#socket.addEventListener('close', (event) => {
|
||||||
|
this.#socket = null;
|
||||||
|
});
|
||||||
|
|
||||||
|
// Listen for messages
|
||||||
|
this.#socket.addEventListener('message', (event) => {
|
||||||
|
const dataView = new DataView(event.data);
|
||||||
|
|
||||||
|
let messageType = dataView.getInt8(0);
|
||||||
|
let offset = Int8Array.BYTES_PER_ELEMENT;
|
||||||
|
switch (messageType) {
|
||||||
|
case PointerAction.MOVE:
|
||||||
|
this.onMove(dataView, offset);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case PointerAction.BUTTON_PRESS:
|
||||||
|
case PointerAction.BUTTON_RELEASE:
|
||||||
|
this.onPress(messageType, dataView, offset);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case PointerAction.SCROLL:
|
||||||
|
this.onScroll(dataView, offset);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case PointerAction.POINTER_CAPTURE_CHANGED:
|
||||||
|
this.onPointerCaptureChanged(dataView, offset);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
onMove(dataView: DataView, offset: number) {
|
||||||
|
// [X, Y]
|
||||||
|
const x = dataView.getInt16(offset);
|
||||||
|
offset += Int16Array.BYTES_PER_ELEMENT;
|
||||||
|
const y = dataView.getInt16(offset);
|
||||||
|
|
||||||
|
this.#mkbHandler?.handleMouseMove({
|
||||||
|
movementX: x,
|
||||||
|
movementY: y,
|
||||||
|
});
|
||||||
|
// BxLogger.info(LOG_TAG, 'move', x, y);
|
||||||
|
}
|
||||||
|
|
||||||
|
onPress(messageType: PointerAction, dataView: DataView, offset: number) {
|
||||||
|
const buttonIndex = dataView.getInt8(offset);
|
||||||
|
const fixedIndex = FixedMouseIndex[buttonIndex as keyof typeof FixedMouseIndex];
|
||||||
|
const keyCode = 'Mouse' + fixedIndex;
|
||||||
|
|
||||||
|
this.#mkbHandler?.handleMouseClick({
|
||||||
|
key: {
|
||||||
|
code: keyCode,
|
||||||
|
name: KeyHelper.codeToKeyName(keyCode),
|
||||||
|
},
|
||||||
|
pressed: messageType === PointerAction.BUTTON_PRESS,
|
||||||
|
});
|
||||||
|
|
||||||
|
// BxLogger.info(LOG_TAG, 'press', buttonIndex);
|
||||||
|
}
|
||||||
|
|
||||||
|
onScroll(dataView: DataView, offset: number) {
|
||||||
|
// [V_SCROLL, H_SCROLL]
|
||||||
|
const vScroll = dataView.getInt8(offset);
|
||||||
|
offset += Int8Array.BYTES_PER_ELEMENT;
|
||||||
|
const hScroll = dataView.getInt8(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),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// BxLogger.info(LOG_TAG, 'scroll', vScroll, hScroll);
|
||||||
|
}
|
||||||
|
|
||||||
|
onPointerCaptureChanged(dataView: DataView, offset: number) {
|
||||||
|
const hasCapture = dataView.getInt8(offset) === 1;
|
||||||
|
!hasCapture && this.#mkbHandler?.stop();
|
||||||
|
}
|
||||||
|
|
||||||
|
stop() {
|
||||||
|
try {
|
||||||
|
this.#socket?.close();
|
||||||
|
} catch (e) {}
|
||||||
|
}
|
||||||
|
}
|
@ -3,9 +3,15 @@ import { BX_FLAGS } from "@utils/bx-flags";
|
|||||||
import { getPref, PrefKey } from "@utils/preferences";
|
import { getPref, PrefKey } from "@utils/preferences";
|
||||||
import { VibrationManager } from "@modules/vibration-manager";
|
import { VibrationManager } from "@modules/vibration-manager";
|
||||||
import { BxLogger } from "@utils/bx-logger";
|
import { BxLogger } from "@utils/bx-logger";
|
||||||
import { hashCode } from "@utils/utils";
|
import { hashCode, renderString } from "@utils/utils";
|
||||||
import { BxEvent } from "@/utils/bx-event";
|
import { BxEvent } from "@/utils/bx-event";
|
||||||
|
|
||||||
|
import codeControllerShortcuts from "./patches/controller-shortcuts.js" with { type: "text" };
|
||||||
|
import codeLocalCoOpEnable from "./patches/local-co-op-enable.js" with { type: "text" };
|
||||||
|
import codeRemotePlayEnable from "./patches/remote-play-enable.js" with { type: "text" };
|
||||||
|
import codeRemotePlayKeepAlive from "./patches/remote-play-keep-alive.js" with { type: "text" };
|
||||||
|
import codeVibrationAdjust from "./patches/vibration-adjust.js" with { type: "text" };
|
||||||
|
|
||||||
type PatchArray = (keyof typeof PATCHES)[];
|
type PatchArray = (keyof typeof PATCHES)[];
|
||||||
|
|
||||||
const ENDING_CHUNKS_PATCH_NAME = 'loadingEndingChunks';
|
const ENDING_CHUNKS_PATCH_NAME = 'loadingEndingChunks';
|
||||||
@ -92,31 +98,24 @@ const PATCHES = {
|
|||||||
},
|
},
|
||||||
|
|
||||||
remotePlayKeepAlive(str: string) {
|
remotePlayKeepAlive(str: string) {
|
||||||
if (!str.includes('onServerDisconnectMessage(e){')) {
|
const text = 'onServerDisconnectMessage(e){';
|
||||||
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
str = str.replace('onServerDisconnectMessage(e){', `onServerDisconnectMessage(e) {
|
str = str.replace(text, text + codeRemotePlayKeepAlive);
|
||||||
const msg = JSON.parse(e);
|
|
||||||
if (msg.reason === 'WarningForBeingIdle' && !window.location.pathname.includes('/launch/')) {
|
|
||||||
try {
|
|
||||||
this.sendKeepAlive();
|
|
||||||
return;
|
|
||||||
} catch (ex) { console.log(ex); }
|
|
||||||
}
|
|
||||||
`);
|
|
||||||
|
|
||||||
return str;
|
return str;
|
||||||
},
|
},
|
||||||
|
|
||||||
// Enable Remote Play feature
|
// Enable Remote Play feature
|
||||||
remotePlayConnectMode(str: string) {
|
remotePlayConnectMode(str: string) {
|
||||||
const text = 'connectMode:"cloud-connect"';
|
const text = 'connectMode:"cloud-connect",';
|
||||||
if (!str.includes(text)) {
|
if (!str.includes(text)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return str.replace(text, `connectMode:window.BX_REMOTE_PLAY_CONFIG?"xhome-connect":"cloud-connect",remotePlayServerId:(window.BX_REMOTE_PLAY_CONFIG&&window.BX_REMOTE_PLAY_CONFIG.serverId)||''`);
|
return str.replace(text, codeRemotePlayEnable);
|
||||||
},
|
},
|
||||||
|
|
||||||
// Disable achievement toast in Remote Play
|
// Disable achievement toast in Remote Play
|
||||||
@ -155,15 +154,36 @@ if (!!window.BX_REMOTE_PLAY_CONFIG) {
|
|||||||
return str.replace(text, 'this.shouldCollectStats=!1');
|
return str.replace(text, 'this.shouldCollectStats=!1');
|
||||||
},
|
},
|
||||||
|
|
||||||
blockGamepadStatsCollector(str: string) {
|
patchPollGamepads(str: string) {
|
||||||
const text = 'this.inputPollingIntervalStats.addValue';
|
const index = str.indexOf('},this.pollGamepads=()=>{');
|
||||||
if (!str.includes(text)) {
|
if (index === -1) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
str = str.replace('this.inputPollingIntervalStats.addValue', '');
|
const nextIndex = str.indexOf('setTimeout(this.pollGamepads', index);
|
||||||
str = str.replace('this.inputPollingDurationStats.addValue', '');
|
if (nextIndex === -1) {
|
||||||
return str;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
let codeBlock = str.substring(index, nextIndex);
|
||||||
|
|
||||||
|
// Block gamepad stats collecting
|
||||||
|
if (getPref(PrefKey.BLOCK_TRACKING)) {
|
||||||
|
codeBlock = codeBlock.replaceAll('this.inputPollingIntervalStats.addValue', '');
|
||||||
|
}
|
||||||
|
|
||||||
|
// Map the Share button on Xbox Series controller with the capturing screenshot feature
|
||||||
|
const match = codeBlock.match(/this\.gamepadTimestamps\.set\((\w+)\.index/);
|
||||||
|
if (match) {
|
||||||
|
const gamepadVar = match[1];
|
||||||
|
const newCode = renderString(codeControllerShortcuts, {
|
||||||
|
gamepadVar,
|
||||||
|
});
|
||||||
|
|
||||||
|
codeBlock = codeBlock.replace('this.gamepadTimestamps.set', newCode + 'this.gamepadTimestamps.set');
|
||||||
|
}
|
||||||
|
|
||||||
|
return str.substring(0, index) + codeBlock + str.substring(nextIndex);
|
||||||
},
|
},
|
||||||
|
|
||||||
enableXcloudLogger(str: string) {
|
enableXcloudLogger(str: string) {
|
||||||
@ -193,20 +213,8 @@ if (!!window.BX_REMOTE_PLAY_CONFIG) {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
const newCode = `
|
|
||||||
if (!window.BX_ENABLE_CONTROLLER_VIBRATION) {
|
|
||||||
return void(0);
|
|
||||||
}
|
|
||||||
if (window.BX_VIBRATION_INTENSITY && window.BX_VIBRATION_INTENSITY < 1) {
|
|
||||||
e.leftMotorPercent = e.leftMotorPercent * window.BX_VIBRATION_INTENSITY;
|
|
||||||
e.rightMotorPercent = e.rightMotorPercent * window.BX_VIBRATION_INTENSITY;
|
|
||||||
e.leftTriggerMotorPercent = e.leftTriggerMotorPercent * window.BX_VIBRATION_INTENSITY;
|
|
||||||
e.rightTriggerMotorPercent = e.rightTriggerMotorPercent * window.BX_VIBRATION_INTENSITY;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
VibrationManager.updateGlobalVars();
|
VibrationManager.updateGlobalVars();
|
||||||
str = str.replaceAll(text, text + newCode);
|
str = str.replaceAll(text, text + codeVibrationAdjust);
|
||||||
return str;
|
return str;
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -302,27 +310,7 @@ window.dispatchEvent(new Event("${BxEvent.TOUCH_LAYOUT_MANAGER_READY}"));
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
let patchstr = `
|
const newCode = `true; ${codeLocalCoOpEnable}; true,`;
|
||||||
let match;
|
|
||||||
let onGamepadChangedStr = this.onGamepadChanged.toString();
|
|
||||||
|
|
||||||
onGamepadChangedStr = onGamepadChangedStr.replaceAll('0', 'arguments[1]');
|
|
||||||
eval(\`this.onGamepadChanged = function \${onGamepadChangedStr}\`);
|
|
||||||
|
|
||||||
let onGamepadInputStr = this.onGamepadInput.toString();
|
|
||||||
|
|
||||||
match = onGamepadInputStr.match(/(\\w+\\.GamepadIndex)/);
|
|
||||||
if (match) {
|
|
||||||
const gamepadIndexVar = match[0];
|
|
||||||
onGamepadInputStr = onGamepadInputStr.replace('this.gamepadStates.get(', \`this.gamepadStates.get(\${gamepadIndexVar},\`);
|
|
||||||
eval(\`this.onGamepadInput = function \${onGamepadInputStr}\`);
|
|
||||||
BxLogger.info('supportLocalCoOp', '✅ Successfully patched local co-op support');
|
|
||||||
} else {
|
|
||||||
BxLogger.error('supportLocalCoOp', '❌ Unable to patch local co-op support');
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
const newCode = `true; ${patchstr}; true,`;
|
|
||||||
|
|
||||||
str = str.replace(text, text + newCode);
|
str = str.replace(text, text + newCode);
|
||||||
return str;
|
return str;
|
||||||
@ -396,13 +384,19 @@ if (!!window.BX_REMOTE_PLAY_CONFIG) {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Restore the "..." button
|
let newCode = `
|
||||||
str = str.replace(text, 'e.guideUI = null;' + text);
|
// Expose onShowStreamMenu
|
||||||
|
window.BX_EXPOSED.showStreamMenu = e.onShowStreamMenu;
|
||||||
|
// Restore the "..." button
|
||||||
|
e.guideUI = null;
|
||||||
|
`;
|
||||||
|
|
||||||
// Remove the TAK Edit button when the touch controller is disabled
|
// Remove the TAK Edit button when the touch controller is disabled
|
||||||
if (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === 'off') {
|
if (getPref(PrefKey.STREAM_TOUCH_CONTROLLER) === 'off') {
|
||||||
str = str.replace(text, 'e.canShowTakHUD = false;' + text);
|
newCode += 'e.canShowTakHUD = false;';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
str = str.replace(text, newCode + text);
|
||||||
return str;
|
return str;
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -413,7 +407,7 @@ if (!!window.BX_REMOTE_PLAY_CONFIG) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const newCode = `
|
const newCode = `
|
||||||
window.BX_EXPOSED.onPollingModeChanged && window.BX_EXPOSED.onPollingModeChanged(e);
|
BxEvent.dispatch(window, BxEvent.XCLOUD_POLLING_MODE_CHANGED, {mode: e});
|
||||||
`;
|
`;
|
||||||
str = str.replace(text, text + newCode);
|
str = str.replace(text, text + newCode);
|
||||||
return str;
|
return str;
|
||||||
@ -619,7 +613,7 @@ let PLAYING_PATCH_ORDERS: PatchArray = [
|
|||||||
|
|
||||||
BX_FLAGS.EnableXcloudLogging && 'enableConsoleLogging',
|
BX_FLAGS.EnableXcloudLogging && 'enableConsoleLogging',
|
||||||
|
|
||||||
getPref(PrefKey.BLOCK_TRACKING) && 'blockGamepadStatsCollector',
|
'patchPollGamepads',
|
||||||
|
|
||||||
getPref(PrefKey.STREAM_COMBINE_SOURCES) && 'streamCombineSources',
|
getPref(PrefKey.STREAM_COMBINE_SOURCES) && 'streamCombineSources',
|
||||||
|
|
||||||
|
87
src/modules/patches/controller-shortcuts.js
Normal file
87
src/modules/patches/controller-shortcuts.js
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
const currentGamepad = ${gamepadVar};
|
||||||
|
|
||||||
|
// Share button on XS controller
|
||||||
|
if (currentGamepad.buttons[17] && currentGamepad.buttons[17].pressed) {
|
||||||
|
window.dispatchEvent(new Event(BxEvent.CAPTURE_SCREENSHOT));
|
||||||
|
}
|
||||||
|
|
||||||
|
const btnHome = currentGamepad.buttons[16];
|
||||||
|
if (btnHome) {
|
||||||
|
if (!this.bxHomeStates) {
|
||||||
|
this.bxHomeStates = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (btnHome.pressed) {
|
||||||
|
this.gamepadIsIdle.set(currentGamepad.index, false);
|
||||||
|
|
||||||
|
if (this.bxHomeStates[currentGamepad.index]) {
|
||||||
|
const lastTimestamp = this.bxHomeStates[currentGamepad.index].timestamp;
|
||||||
|
|
||||||
|
if (currentGamepad.timestamp !== lastTimestamp) {
|
||||||
|
this.bxHomeStates[currentGamepad.index].timestamp = currentGamepad.timestamp;
|
||||||
|
|
||||||
|
const handled = window.BX_EXPOSED.handleControllerShortcut(currentGamepad);
|
||||||
|
if (handled) {
|
||||||
|
this.bxHomeStates[currentGamepad.index].shortcutPressed += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// First time pressing > save current timestamp
|
||||||
|
window.BX_EXPOSED.resetControllerShortcut(currentGamepad.index);
|
||||||
|
this.bxHomeStates[currentGamepad.index] = {
|
||||||
|
shortcutPressed: 0,
|
||||||
|
timestamp: currentGamepad.timestamp,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Listen to next button press
|
||||||
|
const intervalMs = 16;
|
||||||
|
this.inputConfiguration.useIntervalWorkerThreadForInput && this.intervalWorker ? this.intervalWorker.scheduleTimer(intervalMs) : this.pollGamepadssetTimeoutTimerID = setTimeout(this.pollGamepads, intervalMs);
|
||||||
|
|
||||||
|
// Hijack this button
|
||||||
|
return;
|
||||||
|
} else if (this.bxHomeStates[currentGamepad.index]) {
|
||||||
|
const info = structuredClone(this.bxHomeStates[currentGamepad.index]);
|
||||||
|
|
||||||
|
// Home button released
|
||||||
|
this.bxHomeStates[currentGamepad.index] = null;
|
||||||
|
|
||||||
|
if (info.shortcutPressed === 0) {
|
||||||
|
const fakeGamepadMappings = [{
|
||||||
|
GamepadIndex: currentGamepad.index,
|
||||||
|
A: 0,
|
||||||
|
B: 0,
|
||||||
|
X: 0,
|
||||||
|
Y: 0,
|
||||||
|
LeftShoulder: 0,
|
||||||
|
RightShoulder: 0,
|
||||||
|
LeftTrigger: 0,
|
||||||
|
RightTrigger: 0,
|
||||||
|
View: 0,
|
||||||
|
Menu: 0,
|
||||||
|
LeftThumb: 0,
|
||||||
|
RightThumb: 0,
|
||||||
|
DPadUp: 0,
|
||||||
|
DPadDown: 0,
|
||||||
|
DPadLeft: 0,
|
||||||
|
DPadRight: 0,
|
||||||
|
Nexus: 1,
|
||||||
|
LeftThumbXAxis: 0,
|
||||||
|
LeftThumbYAxis: 0,
|
||||||
|
RightThumbXAxis: 0,
|
||||||
|
RightThumbYAxis: 0,
|
||||||
|
PhysicalPhysicality: 0,
|
||||||
|
VirtualPhysicality: 0,
|
||||||
|
Dirty: true,
|
||||||
|
Virtual: false,
|
||||||
|
}];
|
||||||
|
|
||||||
|
const isLongPress = (currentGamepad.timestamp - info.timestamp) >= 500;
|
||||||
|
const intervalMs = isLongPress ? 500 : 100;
|
||||||
|
|
||||||
|
this.inputSink.onGamepadInput(performance.now() - intervalMs, fakeGamepadMappings);
|
||||||
|
this.inputConfiguration.useIntervalWorkerThreadForInput && this.intervalWorker ? this.intervalWorker.scheduleTimer(intervalMs) : this.pollGamepadssetTimeoutTimerID = setTimeout(this.pollGamepads, intervalMs);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
17
src/modules/patches/local-co-op-enable.js
Normal file
17
src/modules/patches/local-co-op-enable.js
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
let match;
|
||||||
|
let onGamepadChangedStr = this.onGamepadChanged.toString();
|
||||||
|
|
||||||
|
onGamepadChangedStr = onGamepadChangedStr.replaceAll('0', 'arguments[1]');
|
||||||
|
eval(`this.onGamepadChanged = function ${onGamepadChangedStr}`);
|
||||||
|
|
||||||
|
let onGamepadInputStr = this.onGamepadInput.toString();
|
||||||
|
|
||||||
|
match = onGamepadInputStr.match(/(\w+\.GamepadIndex)/);
|
||||||
|
if (match) {
|
||||||
|
const gamepadIndexVar = match[0];
|
||||||
|
onGamepadInputStr = onGamepadInputStr.replace('this.gamepadStates.get(', `this.gamepadStates.get(${gamepadIndexVar},`);
|
||||||
|
eval(`this.onGamepadInput = function ${onGamepadInputStr}`);
|
||||||
|
BxLogger.info('supportLocalCoOp', '✅ Successfully patched local co-op support');
|
||||||
|
} else {
|
||||||
|
BxLogger.error('supportLocalCoOp', '❌ Unable to patch local co-op support');
|
||||||
|
}
|
2
src/modules/patches/remote-play-enable.js
Normal file
2
src/modules/patches/remote-play-enable.js
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
connectMode: window.BX_REMOTE_PLAY_CONFIG ? "xhome-connect" : "cloud-connect",
|
||||||
|
remotePlayServerId: (window.BX_REMOTE_PLAY_CONFIG && window.BX_REMOTE_PLAY_CONFIG.serverId) || '',
|
7
src/modules/patches/remote-play-keep-alive.js
Normal file
7
src/modules/patches/remote-play-keep-alive.js
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
const msg = JSON.parse(e);
|
||||||
|
if (msg.reason === 'WarningForBeingIdle' && !window.location.pathname.includes('/launch/')) {
|
||||||
|
try {
|
||||||
|
this.sendKeepAlive();
|
||||||
|
return;
|
||||||
|
} catch (ex) { console.log(ex); }
|
||||||
|
}
|
11
src/modules/patches/vibration-adjust.js
Normal file
11
src/modules/patches/vibration-adjust.js
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
if (!window.BX_ENABLE_CONTROLLER_VIBRATION) {
|
||||||
|
return void(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
const intensity = window.BX_VIBRATION_INTENSITY;
|
||||||
|
if (intensity && intensity < 1) {
|
||||||
|
e.leftMotorPercent *= intensity;
|
||||||
|
e.rightMotorPercent *= intensity;
|
||||||
|
e.leftTriggerMotorPercent *= intensity;
|
||||||
|
e.rightTriggerMotorPercent *= intensity;
|
||||||
|
}
|
33
src/modules/shortcuts/shortcut-microphone.ts
Normal file
33
src/modules/shortcuts/shortcut-microphone.ts
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
import { t } from "@utils/translation";
|
||||||
|
import { Toast } from "@utils/toast";
|
||||||
|
|
||||||
|
|
||||||
|
export enum MicrophoneState {
|
||||||
|
REQUESTED = 'Requested',
|
||||||
|
ENABLED = 'Enabled',
|
||||||
|
MUTED = 'Muted',
|
||||||
|
NOT_ALLOWED = 'NotAllowed',
|
||||||
|
NOT_FOUND = 'NotFound',
|
||||||
|
}
|
||||||
|
|
||||||
|
export class MicrophoneShortcut {
|
||||||
|
static toggle(showToast: boolean = true): boolean {
|
||||||
|
if (!window.BX_EXPOSED.streamSession) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
const state = window.BX_EXPOSED.streamSession._microphoneState;
|
||||||
|
const enableMic = state === MicrophoneState.ENABLED ? false : true;
|
||||||
|
|
||||||
|
try {
|
||||||
|
window.BX_EXPOSED.streamSession.tryEnableChatAsync(enableMic);
|
||||||
|
showToast && Toast.show(t('microphone'), t(enableMic ? 'unmuted': 'muted'), {instant: true});
|
||||||
|
|
||||||
|
return enableMic;
|
||||||
|
} catch (e) {
|
||||||
|
console.log(e);
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
90
src/modules/shortcuts/shortcut-sound.ts
Normal file
90
src/modules/shortcuts/shortcut-sound.ts
Normal file
@ -0,0 +1,90 @@
|
|||||||
|
import { t } from "@utils/translation";
|
||||||
|
import { STATES } from "@utils/global";
|
||||||
|
import { PrefKey, getPref, setPref } from "@utils/preferences";
|
||||||
|
import { Toast } from "@utils/toast";
|
||||||
|
import { BxEvent } from "@/utils/bx-event";
|
||||||
|
import { ceilToNearest, floorToNearest } from "@/utils/utils";
|
||||||
|
|
||||||
|
export class SoundShortcut {
|
||||||
|
static adjustGainNodeVolume(amount: number): number {
|
||||||
|
if (!getPref(PrefKey.AUDIO_ENABLE_VOLUME_CONTROL)) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
const currentValue = getPref(PrefKey.AUDIO_VOLUME);
|
||||||
|
let nearestValue: number;
|
||||||
|
|
||||||
|
if (amount > 0) { // Increase
|
||||||
|
nearestValue = ceilToNearest(currentValue, amount);
|
||||||
|
} else { // Decrease
|
||||||
|
nearestValue = floorToNearest(currentValue, -1 * amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
let newValue: number;
|
||||||
|
if (currentValue !== nearestValue) {
|
||||||
|
newValue = nearestValue;
|
||||||
|
} else {
|
||||||
|
newValue = currentValue + amount;
|
||||||
|
}
|
||||||
|
|
||||||
|
newValue = setPref(PrefKey.AUDIO_VOLUME, newValue);
|
||||||
|
SoundShortcut.setGainNodeVolume(newValue);
|
||||||
|
|
||||||
|
// Show toast
|
||||||
|
Toast.show(`${t('stream')} ❯ ${t('volume')}`, newValue + '%', {instant: true});
|
||||||
|
BxEvent.dispatch(window, BxEvent.GAINNODE_VOLUME_CHANGED, {
|
||||||
|
volume: newValue,
|
||||||
|
});
|
||||||
|
|
||||||
|
return newValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
static setGainNodeVolume(value: number) {
|
||||||
|
STATES.currentStream.audioGainNode && (STATES.currentStream.audioGainNode.gain.value = value / 100);
|
||||||
|
}
|
||||||
|
|
||||||
|
static muteUnmute() {
|
||||||
|
if (getPref(PrefKey.AUDIO_ENABLE_VOLUME_CONTROL) && STATES.currentStream.audioGainNode) {
|
||||||
|
const gainValue = STATES.currentStream.audioGainNode.gain.value;
|
||||||
|
const settingValue = getPref(PrefKey.AUDIO_VOLUME);
|
||||||
|
|
||||||
|
let targetValue: number;
|
||||||
|
if (settingValue === 0) { // settingValue is 0 => set to 100
|
||||||
|
targetValue = 100;
|
||||||
|
setPref(PrefKey.AUDIO_VOLUME, targetValue);
|
||||||
|
BxEvent.dispatch(window, BxEvent.GAINNODE_VOLUME_CHANGED, {
|
||||||
|
volume: targetValue,
|
||||||
|
});
|
||||||
|
} else if (gainValue === 0) { // is being muted => set to settingValue
|
||||||
|
targetValue = settingValue;
|
||||||
|
} else { // not being muted => mute
|
||||||
|
targetValue = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
let status: string;
|
||||||
|
if (targetValue === 0) {
|
||||||
|
status = t('muted');
|
||||||
|
} else {
|
||||||
|
status = targetValue + '%';
|
||||||
|
}
|
||||||
|
|
||||||
|
SoundShortcut.setGainNodeVolume(targetValue);
|
||||||
|
Toast.show(`${t('stream')} ❯ ${t('volume')}`, status, {instant: true});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let $media: HTMLMediaElement;
|
||||||
|
|
||||||
|
$media = document.querySelector('div[data-testid=media-container] audio') as HTMLAudioElement;
|
||||||
|
if (!$media) {
|
||||||
|
$media = document.querySelector('div[data-testid=media-container] video') as HTMLAudioElement;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($media) {
|
||||||
|
$media.muted = !$media.muted;
|
||||||
|
|
||||||
|
const status = $media.muted ? t('muted') : t('unmuted');
|
||||||
|
Toast.show(`${t('stream')} ❯ ${t('volume')}`, status, {instant: true});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
6
src/modules/shortcuts/shortcut-stream-ui.ts
Normal file
6
src/modules/shortcuts/shortcut-stream-ui.ts
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
export class StreamUiShortcut {
|
||||||
|
static showHideStreamMenu() {
|
||||||
|
// Show menu
|
||||||
|
window.BX_EXPOSED.showStreamMenu && window.BX_EXPOSED.showStreamMenu();
|
||||||
|
}
|
||||||
|
}
|
@ -66,9 +66,9 @@ export function injectStreamMenuButtons() {
|
|||||||
|
|
||||||
($screen as any).xObserving = true;
|
($screen as any).xObserving = true;
|
||||||
|
|
||||||
const $quickBar = document.querySelector('.bx-quick-settings-bar')!;
|
const $settingsDialog = document.querySelector('.bx-stream-settings-dialog')!;
|
||||||
const $parent = $screen.parentElement;
|
const $parent = $screen.parentElement;
|
||||||
const hideQuickBarFunc = (e?: MouseEvent | TouchEvent) => {
|
const hideSettingsFunc = (e?: MouseEvent | TouchEvent) => {
|
||||||
if (e) {
|
if (e) {
|
||||||
const $target = e.target as HTMLElement;
|
const $target = e.target as HTMLElement;
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
@ -76,15 +76,15 @@ export function injectStreamMenuButtons() {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if ($target.id === 'MultiTouchSurface') {
|
if ($target.id === 'MultiTouchSurface') {
|
||||||
$target.removeEventListener('touchstart', hideQuickBarFunc);
|
$target.removeEventListener('touchstart', hideSettingsFunc);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hide Quick settings bar
|
// Hide Stream settings dialog
|
||||||
$quickBar.classList.add('bx-gone');
|
$settingsDialog.classList.add('bx-gone');
|
||||||
|
|
||||||
$parent?.removeEventListener('click', hideQuickBarFunc);
|
$parent?.removeEventListener('click', hideSettingsFunc);
|
||||||
// $parent.removeEventListener('touchstart', hideQuickBarFunc);
|
// $parent.removeEventListener('touchstart', hideSettingsFunc);
|
||||||
}
|
}
|
||||||
|
|
||||||
let $btnStreamSettings: HTMLElement;
|
let $btnStreamSettings: HTMLElement;
|
||||||
@ -105,12 +105,6 @@ export function injectStreamMenuButtons() {
|
|||||||
if (!($node as HTMLElement).className || !($node as HTMLElement).className.startsWith) {
|
if (!($node as HTMLElement).className || !($node as HTMLElement).className.startsWith) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (($node as HTMLElement).className.startsWith('StreamMenu')) {
|
|
||||||
if (!document.querySelector('div[class^=PureInStreamConfirmationModal]')) {
|
|
||||||
BxEvent.dispatch(window, BxEvent.STREAM_MENU_HIDDEN);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
item.addedNodes.forEach(async $node => {
|
item.addedNodes.forEach(async $node => {
|
||||||
@ -139,16 +133,14 @@ export function injectStreamMenuButtons() {
|
|||||||
|
|
||||||
// Render badges
|
// Render badges
|
||||||
if ($elm.className?.startsWith('StreamMenu-module__container')) {
|
if ($elm.className?.startsWith('StreamMenu-module__container')) {
|
||||||
BxEvent.dispatch(window, BxEvent.STREAM_MENU_SHOWN);
|
|
||||||
|
|
||||||
const $btnCloseHud = document.querySelector('button[class*=StreamMenu-module__backButton]');
|
const $btnCloseHud = document.querySelector('button[class*=StreamMenu-module__backButton]');
|
||||||
if (!$btnCloseHud) {
|
if (!$btnCloseHud) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Hide Quick bar when closing HUD
|
// Hide Stream Settings dialog when closing HUD
|
||||||
$btnCloseHud && $btnCloseHud.addEventListener('click', e => {
|
$btnCloseHud && $btnCloseHud.addEventListener('click', e => {
|
||||||
$quickBar.classList.add('bx-gone');
|
$settingsDialog.classList.add('bx-gone');
|
||||||
});
|
});
|
||||||
|
|
||||||
// Create Refresh button from the Close button
|
// Create Refresh button from the Close button
|
||||||
@ -176,7 +168,7 @@ export function injectStreamMenuButtons() {
|
|||||||
const $menu = document.querySelector('div[class*=StreamMenu-module__menuContainer] > div[class*=Menu-module]');
|
const $menu = document.querySelector('div[class*=StreamMenu-module__menuContainer] > div[class*=Menu-module]');
|
||||||
$menu?.appendChild(await StreamBadges.render());
|
$menu?.appendChild(await StreamBadges.render());
|
||||||
|
|
||||||
hideQuickBarFunc();
|
hideSettingsFunc();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -210,25 +202,25 @@ export function injectStreamMenuButtons() {
|
|||||||
|
|
||||||
// Create Stream Settings button
|
// Create Stream Settings button
|
||||||
if (!$btnStreamSettings) {
|
if (!$btnStreamSettings) {
|
||||||
$btnStreamSettings = cloneStreamHudButton($orgButton, t('menu-stream-settings'), BxIcon.STREAM_SETTINGS);
|
$btnStreamSettings = cloneStreamHudButton($orgButton, t('stream-settings'), BxIcon.STREAM_SETTINGS);
|
||||||
$btnStreamSettings.addEventListener('click', e => {
|
$btnStreamSettings.addEventListener('click', e => {
|
||||||
hideGripHandle();
|
hideGripHandle();
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
|
||||||
// Show Quick settings bar
|
// Show Stream Settings dialog
|
||||||
$quickBar.classList.remove('bx-gone');
|
$settingsDialog.classList.remove('bx-gone');
|
||||||
|
|
||||||
$parent?.addEventListener('click', hideQuickBarFunc);
|
$parent?.addEventListener('click', hideSettingsFunc);
|
||||||
//$parent.addEventListener('touchstart', hideQuickBarFunc);
|
//$parent.addEventListener('touchstart', hideSettingsFunc);
|
||||||
|
|
||||||
const $touchSurface = document.getElementById('MultiTouchSurface');
|
const $touchSurface = document.getElementById('MultiTouchSurface');
|
||||||
$touchSurface && $touchSurface.style.display != 'none' && $touchSurface.addEventListener('touchstart', hideQuickBarFunc);
|
$touchSurface && $touchSurface.style.display != 'none' && $touchSurface.addEventListener('touchstart', hideSettingsFunc);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create Stream Stats button
|
// Create Stream Stats button
|
||||||
if (!$btnStreamStats) {
|
if (!$btnStreamStats) {
|
||||||
$btnStreamStats = cloneStreamHudButton($orgButton, t('menu-stream-stats'), BxIcon.STREAM_STATS);
|
$btnStreamStats = cloneStreamHudButton($orgButton, t('stream-stats'), BxIcon.STREAM_STATS);
|
||||||
$btnStreamStats.addEventListener('click', e => {
|
$btnStreamStats.addEventListener('click', e => {
|
||||||
hideGripHandle();
|
hideGripHandle();
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
@ -263,14 +255,14 @@ export function injectStreamMenuButtons() {
|
|||||||
|
|
||||||
|
|
||||||
export function showStreamSettings(tabId: string) {
|
export function showStreamSettings(tabId: string) {
|
||||||
const $wrapper = document.querySelector('.bx-quick-settings-bar');
|
const $wrapper = document.querySelector('.bx-stream-settings-dialog');
|
||||||
if (!$wrapper) {
|
if (!$wrapper) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Select tab
|
// Select tab
|
||||||
if (tabId) {
|
if (tabId) {
|
||||||
const $tab = $wrapper.querySelector(`.bx-quick-settings-tabs svg[data-group=${tabId}]`);
|
const $tab = $wrapper.querySelector(`.bx-stream-settings-tabs svg[data-group=${tabId}]`);
|
||||||
$tab && $tab.dispatchEvent(new Event('click'));
|
$tab && $tab.dispatchEvent(new Event('click'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2,10 +2,9 @@ import { STATES } from "@utils/global";
|
|||||||
import { escapeHtml } from "@utils/html";
|
import { escapeHtml } from "@utils/html";
|
||||||
import { Toast } from "@utils/toast";
|
import { Toast } from "@utils/toast";
|
||||||
import { BxEvent } from "@utils/bx-event";
|
import { BxEvent } from "@utils/bx-event";
|
||||||
import { BX_FLAGS } from "@utils/bx-flags";
|
import { BX_FLAGS, NATIVE_FETCH } from "@utils/bx-flags";
|
||||||
import { getPref, PrefKey } from "@utils/preferences";
|
import { getPref, PrefKey } from "@utils/preferences";
|
||||||
import { t } from "@utils/translation";
|
import { t } from "@utils/translation";
|
||||||
import { NATIVE_FETCH } from "@utils/network";
|
|
||||||
import { BxLogger } from "@utils/bx-logger";
|
import { BxLogger } from "@utils/bx-logger";
|
||||||
|
|
||||||
const LOG_TAG = 'TouchController';
|
const LOG_TAG = 'TouchController';
|
||||||
|
@ -4,7 +4,7 @@ import { BxIcon } from "@utils/bx-icon";
|
|||||||
import { getPreferredServerRegion } from "@utils/region";
|
import { getPreferredServerRegion } from "@utils/region";
|
||||||
import { UserAgent, UserAgentProfile } from "@utils/user-agent";
|
import { UserAgent, UserAgentProfile } from "@utils/user-agent";
|
||||||
import { getPref, Preferences, PrefKey, setPref, toPrefElement } from "@utils/preferences";
|
import { getPref, Preferences, PrefKey, setPref, toPrefElement } from "@utils/preferences";
|
||||||
import { t, refreshCurrentLocale } from "@utils/translation";
|
import { t, Translations } from "@utils/translation";
|
||||||
import { PatcherCache } from "../patcher";
|
import { PatcherCache } from "../patcher";
|
||||||
|
|
||||||
const SETTINGS_UI = {
|
const SETTINGS_UI = {
|
||||||
@ -55,6 +55,7 @@ const SETTINGS_UI = {
|
|||||||
|
|
||||||
[t('mouse-and-keyboard')]: {
|
[t('mouse-and-keyboard')]: {
|
||||||
items: [
|
items: [
|
||||||
|
PrefKey.NATIVE_MKB_DISABLED,
|
||||||
PrefKey.MKB_ENABLED,
|
PrefKey.MKB_ENABLED,
|
||||||
PrefKey.MKB_HIDE_IDLE_CURSOR,
|
PrefKey.MKB_HIDE_IDLE_CURSOR,
|
||||||
],
|
],
|
||||||
@ -83,6 +84,7 @@ const SETTINGS_UI = {
|
|||||||
[t('ui')]: {
|
[t('ui')]: {
|
||||||
items: [
|
items: [
|
||||||
PrefKey.UI_LAYOUT,
|
PrefKey.UI_LAYOUT,
|
||||||
|
PrefKey.UI_HOME_CONTEXT_MENU_DISABLED,
|
||||||
PrefKey.STREAM_SIMPLIFY_MENU,
|
PrefKey.STREAM_SIMPLIFY_MENU,
|
||||||
PrefKey.SKIP_SPLASH_VIDEO,
|
PrefKey.SKIP_SPLASH_VIDEO,
|
||||||
!AppInterface && PrefKey.UI_SCROLLBAR_HIDE,
|
!AppInterface && PrefKey.UI_SCROLLBAR_HIDE,
|
||||||
@ -115,7 +117,7 @@ export function setupSettingsUi() {
|
|||||||
const PREF_PREFERRED_REGION = getPreferredServerRegion();
|
const PREF_PREFERRED_REGION = getPreferredServerRegion();
|
||||||
const PREF_LATEST_VERSION = getPref(PrefKey.LATEST_VERSION);
|
const PREF_LATEST_VERSION = getPref(PrefKey.LATEST_VERSION);
|
||||||
|
|
||||||
let $reloadBtnWrapper: HTMLButtonElement;
|
let $btnReload: HTMLButtonElement;
|
||||||
|
|
||||||
// Setup Settings UI
|
// Setup Settings UI
|
||||||
const $container = CE<HTMLElement>('div', {
|
const $container = CE<HTMLElement>('div', {
|
||||||
@ -131,7 +133,12 @@ export function setupSettingsUi() {
|
|||||||
'href': SCRIPT_HOME,
|
'href': SCRIPT_HOME,
|
||||||
'target': '_blank',
|
'target': '_blank',
|
||||||
}, 'Better xCloud ' + SCRIPT_VERSION),
|
}, 'Better xCloud ' + SCRIPT_VERSION),
|
||||||
createButton({icon: BxIcon.QUESTION, label: t('help'), url: 'https://better-xcloud.github.io/features/'}),
|
createButton({
|
||||||
|
icon: BxIcon.QUESTION,
|
||||||
|
style: ButtonStyle.FOCUSABLE,
|
||||||
|
label: t('help'),
|
||||||
|
url: 'https://better-xcloud.github.io/features/',
|
||||||
|
}),
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
$updateAvailable = CE('a', {
|
$updateAvailable = CE('a', {
|
||||||
@ -148,8 +155,20 @@ export function setupSettingsUi() {
|
|||||||
$updateAvailable.classList.remove('bx-gone');
|
$updateAvailable.classList.remove('bx-gone');
|
||||||
}
|
}
|
||||||
|
|
||||||
// Show link to Android app
|
if (AppInterface) {
|
||||||
if (!AppInterface) {
|
// Show Android app settings button
|
||||||
|
const $btn = createButton({
|
||||||
|
label: t('android-app-settings'),
|
||||||
|
icon: BxIcon.STREAM_SETTINGS,
|
||||||
|
style: ButtonStyle.FULL_WIDTH | ButtonStyle.FOCUSABLE,
|
||||||
|
onClick: e => {
|
||||||
|
AppInterface.openAppSettings && AppInterface.openAppSettings();
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
$wrapper.appendChild($btn);
|
||||||
|
} else {
|
||||||
|
// Show link to Android app
|
||||||
const userAgent = UserAgent.getDefault().toLowerCase();
|
const userAgent = UserAgent.getDefault().toLowerCase();
|
||||||
if (userAgent.includes('android')) {
|
if (userAgent.includes('android')) {
|
||||||
const $btn = createButton({
|
const $btn = createButton({
|
||||||
@ -162,23 +181,23 @@ export function setupSettingsUi() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const onChange = (e: Event) => {
|
const onChange = async (e: Event) => {
|
||||||
if (!$reloadBtnWrapper) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$reloadBtnWrapper.classList.remove('bx-gone');
|
|
||||||
|
|
||||||
// Clear PatcherCache;
|
// Clear PatcherCache;
|
||||||
PatcherCache.clear();
|
PatcherCache.clear();
|
||||||
|
|
||||||
|
$btnReload.classList.add('bx-danger');
|
||||||
|
|
||||||
|
// Highlight the Settings button in the Header to remind user to reload the page
|
||||||
|
const $btnHeaderSettings = document.querySelector('.bx-header-settings-button');
|
||||||
|
$btnHeaderSettings && $btnHeaderSettings.classList.add('bx-danger');
|
||||||
|
|
||||||
if ((e.target as HTMLElement).id === 'bx_setting_' + PrefKey.BETTER_XCLOUD_LOCALE) {
|
if ((e.target as HTMLElement).id === 'bx_setting_' + PrefKey.BETTER_XCLOUD_LOCALE) {
|
||||||
// Update locale
|
// Update locale
|
||||||
refreshCurrentLocale();
|
Translations.refreshCurrentLocale();
|
||||||
|
await Translations.updateTranslations();
|
||||||
|
|
||||||
const $btn = $reloadBtnWrapper.firstElementChild! as HTMLButtonElement;
|
$btnReload.textContent = t('settings-reloading');
|
||||||
$btn.textContent = t('settings-reloading');
|
$btnReload.click();
|
||||||
$btn.click();
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -226,13 +245,16 @@ export function setupSettingsUi() {
|
|||||||
|
|
||||||
let $control: any;
|
let $control: any;
|
||||||
let $inpCustomUserAgent: HTMLInputElement;
|
let $inpCustomUserAgent: HTMLInputElement;
|
||||||
let labelAttrs = {};
|
let labelAttrs: any = {
|
||||||
|
tabindex: '-1',
|
||||||
|
};
|
||||||
|
|
||||||
if (settingId === PrefKey.USER_AGENT_PROFILE) {
|
if (settingId === PrefKey.USER_AGENT_PROFILE) {
|
||||||
let defaultUserAgent = (window.navigator as any).orgUserAgent || window.navigator.userAgent;
|
let defaultUserAgent = (window.navigator as any).orgUserAgent || window.navigator.userAgent;
|
||||||
$inpCustomUserAgent = CE('input', {
|
$inpCustomUserAgent = CE('input', {
|
||||||
'type': 'text',
|
id: `bx_setting_inp_${settingId}`,
|
||||||
'placeholder': defaultUserAgent,
|
type: 'text',
|
||||||
|
placeholder: defaultUserAgent,
|
||||||
'class': 'bx-settings-custom-user-agent',
|
'class': 'bx-settings-custom-user-agent',
|
||||||
});
|
});
|
||||||
$inpCustomUserAgent.addEventListener('change', e => {
|
$inpCustomUserAgent.addEventListener('change', e => {
|
||||||
@ -254,12 +276,16 @@ export function setupSettingsUi() {
|
|||||||
$inpCustomUserAgent.readOnly = !isCustom;
|
$inpCustomUserAgent.readOnly = !isCustom;
|
||||||
$inpCustomUserAgent.disabled = !isCustom;
|
$inpCustomUserAgent.disabled = !isCustom;
|
||||||
|
|
||||||
onChange(e);
|
!(e.target as HTMLInputElement).disabled && onChange(e);
|
||||||
});
|
});
|
||||||
} else if (settingId === PrefKey.SERVER_REGION) {
|
} else if (settingId === PrefKey.SERVER_REGION) {
|
||||||
let selectedValue;
|
let selectedValue;
|
||||||
|
|
||||||
$control = CE<HTMLSelectElement>('select', {id: `bx_setting_${settingId}`});
|
$control = CE<HTMLSelectElement>('select', {
|
||||||
|
id: `bx_setting_${settingId}`,
|
||||||
|
title: settingLabel,
|
||||||
|
tabindex: 0,
|
||||||
|
});
|
||||||
$control.name = $control.id;
|
$control.name = $control.id;
|
||||||
|
|
||||||
$control.addEventListener('change', (e: Event) => {
|
$control.addEventListener('change', (e: Event) => {
|
||||||
@ -305,7 +331,12 @@ export function setupSettingsUi() {
|
|||||||
} else {
|
} else {
|
||||||
$control = toPrefElement(settingId, onChange);
|
$control = toPrefElement(settingId, onChange);
|
||||||
}
|
}
|
||||||
labelAttrs = {'for': $control.id, 'tabindex': 0};
|
}
|
||||||
|
|
||||||
|
if (!!$control.id) {
|
||||||
|
labelAttrs['for'] = $control.id;
|
||||||
|
} else {
|
||||||
|
labelAttrs['for'] = `bx_setting_${settingId}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Disable unsupported settings
|
// Disable unsupported settings
|
||||||
@ -313,14 +344,19 @@ export function setupSettingsUi() {
|
|||||||
($control as HTMLInputElement).disabled = true;
|
($control as HTMLInputElement).disabled = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Make disabled control elements un-focusable
|
||||||
|
if ($control.disabled && !!$control.getAttribute('tabindex')) {
|
||||||
|
$control.setAttribute('tabindex', -1);
|
||||||
|
}
|
||||||
|
|
||||||
const $label = CE('label', labelAttrs, settingLabel);
|
const $label = CE('label', labelAttrs, settingLabel);
|
||||||
if (settingNote) {
|
if (settingNote) {
|
||||||
$label.appendChild(CE('b', {}, settingNote));
|
$label.appendChild(CE('b', {}, settingNote));
|
||||||
}
|
}
|
||||||
const $elm = CE<HTMLElement>('div', {'class': 'bx-settings-row'},
|
const $elm = CE<HTMLElement>('div', {'class': 'bx-settings-row'},
|
||||||
$label,
|
$label,
|
||||||
$control
|
$control,
|
||||||
);
|
);
|
||||||
|
|
||||||
$wrapper.appendChild($elm);
|
$wrapper.appendChild($elm);
|
||||||
|
|
||||||
@ -328,28 +364,35 @@ export function setupSettingsUi() {
|
|||||||
if (settingId === PrefKey.USER_AGENT_PROFILE) {
|
if (settingId === PrefKey.USER_AGENT_PROFILE) {
|
||||||
$wrapper.appendChild($inpCustomUserAgent!);
|
$wrapper.appendChild($inpCustomUserAgent!);
|
||||||
// Trigger 'change' event
|
// Trigger 'change' event
|
||||||
|
$control.disabled = true;
|
||||||
$control.dispatchEvent(new Event('change'));
|
$control.dispatchEvent(new Event('change'));
|
||||||
|
$control.disabled = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Setup Reload button
|
// Setup Reload button
|
||||||
const $reloadBtn = createButton({
|
$btnReload = createButton({
|
||||||
label: t('settings-reload'),
|
label: t('settings-reload'),
|
||||||
style: ButtonStyle.DANGER | ButtonStyle.FOCUSABLE | ButtonStyle.FULL_WIDTH,
|
classes: ['bx-settings-reload-button'],
|
||||||
|
style: ButtonStyle.FOCUSABLE | ButtonStyle.FULL_WIDTH,
|
||||||
onClick: e => {
|
onClick: e => {
|
||||||
window.location.reload();
|
window.location.reload();
|
||||||
$reloadBtn.disabled = true;
|
$btnReload.disabled = true;
|
||||||
$reloadBtn.textContent = t('settings-reloading');
|
$btnReload.textContent = t('settings-reloading');
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
$reloadBtn.setAttribute('tabindex', '0');
|
$btnReload.setAttribute('tabindex', '0');
|
||||||
|
|
||||||
$reloadBtnWrapper = CE<HTMLButtonElement>('div', {'class': 'bx-settings-reload-button-wrapper bx-gone'}, $reloadBtn);
|
$wrapper.appendChild($btnReload);
|
||||||
$wrapper.appendChild($reloadBtnWrapper);
|
|
||||||
|
|
||||||
// Donation link
|
// Donation link
|
||||||
const $donationLink = CE('a', {'class': 'bx-donation-link', href: 'https://ko-fi.com/redphx', target: '_blank'}, `❤️ ${t('support-better-xcloud')}`);
|
const $donationLink = CE('a', {
|
||||||
|
'class': 'bx-donation-link',
|
||||||
|
href: 'https://ko-fi.com/redphx',
|
||||||
|
target: '_blank',
|
||||||
|
tabindex: 0,
|
||||||
|
}, `❤️ ${t('support-better-xcloud')}`);
|
||||||
$wrapper.appendChild($donationLink);
|
$wrapper.appendChild($donationLink);
|
||||||
|
|
||||||
// Show Game Pass app version
|
// Show Game Pass app version
|
||||||
|
@ -4,12 +4,14 @@ import { BxIcon } from "@utils/bx-icon";
|
|||||||
import { UserAgent } from "@utils/user-agent";
|
import { UserAgent } from "@utils/user-agent";
|
||||||
import { BxEvent } from "@utils/bx-event";
|
import { BxEvent } from "@utils/bx-event";
|
||||||
import { MkbRemapper } from "@modules/mkb/mkb-remapper";
|
import { MkbRemapper } from "@modules/mkb/mkb-remapper";
|
||||||
import { getPref, PrefKey, toPrefElement } from "@utils/preferences";
|
import { getPref, Preferences, PrefKey, toPrefElement } from "@utils/preferences";
|
||||||
import { StreamStats } from "@modules/stream/stream-stats";
|
import { StreamStats } from "@modules/stream/stream-stats";
|
||||||
import { TouchController } from "@modules/touch-controller";
|
import { TouchController } from "@modules/touch-controller";
|
||||||
import { t } from "@utils/translation";
|
import { t } from "@utils/translation";
|
||||||
import { VibrationManager } from "@modules/vibration-manager";
|
import { VibrationManager } from "@modules/vibration-manager";
|
||||||
import { Screenshot } from "@/utils/screenshot";
|
import { Screenshot } from "@/utils/screenshot";
|
||||||
|
import { ControllerShortcut } from "../controller-shortcut";
|
||||||
|
import { SoundShortcut } from "../shortcuts/shortcut-sound";
|
||||||
|
|
||||||
|
|
||||||
export function localRedirect(path: string) {
|
export function localRedirect(path: string) {
|
||||||
@ -66,7 +68,7 @@ function getVideoPlayerFilterStyle() {
|
|||||||
return filters.join(' ');
|
return filters.join(' ');
|
||||||
}
|
}
|
||||||
|
|
||||||
function setupQuickSettingsBar() {
|
function setupStreamSettingsDialog() {
|
||||||
const isSafari = UserAgent.isSafari();
|
const isSafari = UserAgent.isSafari();
|
||||||
|
|
||||||
const SETTINGS_UI = [
|
const SETTINGS_UI = [
|
||||||
@ -94,13 +96,21 @@ function setupQuickSettingsBar() {
|
|||||||
items: [
|
items: [
|
||||||
{
|
{
|
||||||
pref: PrefKey.AUDIO_VOLUME,
|
pref: PrefKey.AUDIO_VOLUME,
|
||||||
label: t('volume'),
|
|
||||||
onChange: (e: any, value: number) => {
|
onChange: (e: any, value: number) => {
|
||||||
STATES.currentStream.audioGainNode && (STATES.currentStream.audioGainNode.gain.value = value / 100);
|
SoundShortcut.setGainNodeVolume(value);
|
||||||
},
|
},
|
||||||
params: {
|
params: {
|
||||||
disabled: !getPref(PrefKey.AUDIO_ENABLE_VOLUME_CONTROL),
|
disabled: !getPref(PrefKey.AUDIO_ENABLE_VOLUME_CONTROL),
|
||||||
},
|
},
|
||||||
|
onMounted: ($elm: HTMLElement) => {
|
||||||
|
const $range = $elm.querySelector('input[type=range') as HTMLInputElement;
|
||||||
|
window.addEventListener(BxEvent.GAINNODE_VOLUME_CHANGED, e => {
|
||||||
|
$range.value = (e as any).volume;
|
||||||
|
BxEvent.dispatch($range, 'input', {
|
||||||
|
ignoreOnChange: true,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
@ -112,32 +122,27 @@ function setupQuickSettingsBar() {
|
|||||||
items: [
|
items: [
|
||||||
{
|
{
|
||||||
pref: PrefKey.VIDEO_RATIO,
|
pref: PrefKey.VIDEO_RATIO,
|
||||||
label: t('ratio'),
|
|
||||||
onChange: updateVideoPlayerCss,
|
onChange: updateVideoPlayerCss,
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
pref: PrefKey.VIDEO_CLARITY,
|
pref: PrefKey.VIDEO_CLARITY,
|
||||||
label: t('clarity'),
|
|
||||||
onChange: updateVideoPlayerCss,
|
onChange: updateVideoPlayerCss,
|
||||||
unsupported: isSafari,
|
unsupported: isSafari,
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
pref: PrefKey.VIDEO_SATURATION,
|
pref: PrefKey.VIDEO_SATURATION,
|
||||||
label: t('saturation'),
|
|
||||||
onChange: updateVideoPlayerCss,
|
onChange: updateVideoPlayerCss,
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
pref: PrefKey.VIDEO_CONTRAST,
|
pref: PrefKey.VIDEO_CONTRAST,
|
||||||
label: t('contrast'),
|
|
||||||
onChange: updateVideoPlayerCss,
|
onChange: updateVideoPlayerCss,
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
pref: PrefKey.VIDEO_BRIGHTNESS,
|
pref: PrefKey.VIDEO_BRIGHTNESS,
|
||||||
label: t('brightness'),
|
|
||||||
onChange: updateVideoPlayerCss,
|
onChange: updateVideoPlayerCss,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
@ -156,21 +161,18 @@ function setupQuickSettingsBar() {
|
|||||||
items: [
|
items: [
|
||||||
{
|
{
|
||||||
pref: PrefKey.CONTROLLER_ENABLE_VIBRATION,
|
pref: PrefKey.CONTROLLER_ENABLE_VIBRATION,
|
||||||
label: t('controller-vibration'),
|
|
||||||
unsupported: !VibrationManager.supportControllerVibration(),
|
unsupported: !VibrationManager.supportControllerVibration(),
|
||||||
onChange: VibrationManager.updateGlobalVars,
|
onChange: VibrationManager.updateGlobalVars,
|
||||||
},
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
pref: PrefKey.CONTROLLER_DEVICE_VIBRATION,
|
pref: PrefKey.CONTROLLER_DEVICE_VIBRATION,
|
||||||
label: t('device-vibration'),
|
|
||||||
unsupported: !VibrationManager.supportDeviceVibration(),
|
unsupported: !VibrationManager.supportDeviceVibration(),
|
||||||
onChange: VibrationManager.updateGlobalVars,
|
onChange: VibrationManager.updateGlobalVars,
|
||||||
},
|
},
|
||||||
|
|
||||||
(VibrationManager.supportControllerVibration() || VibrationManager.supportDeviceVibration()) && {
|
(VibrationManager.supportControllerVibration() || VibrationManager.supportDeviceVibration()) && {
|
||||||
pref: PrefKey.CONTROLLER_VIBRATION_INTENSITY,
|
pref: PrefKey.CONTROLLER_VIBRATION_INTENSITY,
|
||||||
label: t('vibration-intensity'),
|
|
||||||
unsupported: !VibrationManager.supportDeviceVibration(),
|
unsupported: !VibrationManager.supportDeviceVibration(),
|
||||||
onChange: VibrationManager.updateGlobalVars,
|
onChange: VibrationManager.updateGlobalVars,
|
||||||
},
|
},
|
||||||
@ -239,54 +241,58 @@ function setupQuickSettingsBar() {
|
|||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
icon: BxIcon.COMMAND,
|
||||||
|
group: 'shortcuts',
|
||||||
|
items: [
|
||||||
|
{
|
||||||
|
group: 'shortcuts_controller',
|
||||||
|
label: t('controller-shortcuts'),
|
||||||
|
content: ControllerShortcut.renderSettings(),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
|
||||||
{
|
{
|
||||||
icon: BxIcon.STREAM_STATS,
|
icon: BxIcon.STREAM_STATS,
|
||||||
group: 'stats',
|
group: 'stats',
|
||||||
items: [
|
items: [
|
||||||
{
|
{
|
||||||
group: 'stats',
|
group: 'stats',
|
||||||
label: t('menu-stream-stats'),
|
label: t('stream-stats'),
|
||||||
help_url: 'https://better-xcloud.github.io/stream-stats/',
|
help_url: 'https://better-xcloud.github.io/stream-stats/',
|
||||||
items: [
|
items: [
|
||||||
{
|
{
|
||||||
pref: PrefKey.STATS_SHOW_WHEN_PLAYING,
|
pref: PrefKey.STATS_SHOW_WHEN_PLAYING,
|
||||||
label: t('show-stats-on-startup'),
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
pref: PrefKey.STATS_QUICK_GLANCE,
|
pref: PrefKey.STATS_QUICK_GLANCE,
|
||||||
label: '👀 ' + t('enable-quick-glance-mode'),
|
|
||||||
onChange: (e: InputEvent) => {
|
onChange: (e: InputEvent) => {
|
||||||
(e.target! as HTMLInputElement).checked ? StreamStats.quickGlanceSetup() : StreamStats.quickGlanceStop();
|
(e.target! as HTMLInputElement).checked ? StreamStats.quickGlanceSetup() : StreamStats.quickGlanceStop();
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
pref: PrefKey.STATS_ITEMS,
|
pref: PrefKey.STATS_ITEMS,
|
||||||
label: t('stats'),
|
|
||||||
onChange: StreamStats.refreshStyles,
|
onChange: StreamStats.refreshStyles,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
pref: PrefKey.STATS_POSITION,
|
pref: PrefKey.STATS_POSITION,
|
||||||
label: t('position'),
|
|
||||||
onChange: StreamStats.refreshStyles,
|
onChange: StreamStats.refreshStyles,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
pref: PrefKey.STATS_TEXT_SIZE,
|
pref: PrefKey.STATS_TEXT_SIZE,
|
||||||
label: t('text-size'),
|
|
||||||
onChange: StreamStats.refreshStyles,
|
onChange: StreamStats.refreshStyles,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
pref: PrefKey.STATS_OPACITY,
|
pref: PrefKey.STATS_OPACITY,
|
||||||
label: t('opacity'),
|
|
||||||
onChange: StreamStats.refreshStyles,
|
onChange: StreamStats.refreshStyles,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
pref: PrefKey.STATS_TRANSPARENT,
|
pref: PrefKey.STATS_TRANSPARENT,
|
||||||
label: t('transparent-background'),
|
|
||||||
onChange: StreamStats.refreshStyles,
|
onChange: StreamStats.refreshStyles,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
pref: PrefKey.STATS_CONDITIONAL_FORMATTING,
|
pref: PrefKey.STATS_CONDITIONAL_FORMATTING,
|
||||||
label: t('conditional-formatting'),
|
|
||||||
onChange: StreamStats.refreshStyles,
|
onChange: StreamStats.refreshStyles,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
@ -298,9 +304,9 @@ function setupQuickSettingsBar() {
|
|||||||
let $tabs: HTMLElement;
|
let $tabs: HTMLElement;
|
||||||
let $settings: HTMLElement;
|
let $settings: HTMLElement;
|
||||||
|
|
||||||
const $wrapper = CE<HTMLElement>('div', {'class': 'bx-quick-settings-bar bx-gone'},
|
const $wrapper = CE<HTMLElement>('div', {'class': 'bx-stream-settings-dialog bx-gone'},
|
||||||
$tabs = CE<HTMLElement>('div', {'class': 'bx-quick-settings-tabs'}),
|
$tabs = CE<HTMLElement>('div', {'class': 'bx-stream-settings-tabs'}),
|
||||||
$settings = CE<HTMLElement>('div', {'class': 'bx-quick-settings-tab-contents'}),
|
$settings = CE<HTMLElement>('div', {'class': 'bx-stream-settings-tab-contents'}),
|
||||||
);
|
);
|
||||||
|
|
||||||
for (const settingTab of SETTINGS_UI) {
|
for (const settingTab of SETTINGS_UI) {
|
||||||
@ -375,13 +381,17 @@ function setupQuickSettingsBar() {
|
|||||||
$control = toPrefElement(pref, setting.onChange, setting.params);
|
$control = toPrefElement(pref, setting.onChange, setting.params);
|
||||||
}
|
}
|
||||||
|
|
||||||
const $content = CE<HTMLElement>('div', {'class': 'bx-quick-settings-row', 'data-type': settingGroup.group},
|
const label = Preferences.SETTINGS[pref as PrefKey]?.label || setting.label;
|
||||||
CE('label', {for: `bx_setting_${pref}`},
|
const note = Preferences.SETTINGS[pref as PrefKey]?.note || setting.note;
|
||||||
setting.label,
|
|
||||||
setting.unsupported && CE<HTMLElement>('div', {'class': 'bx-quick-settings-bar-note'}, t('browser-unsupported-feature')),
|
const $content = CE('div', {'class': 'bx-stream-settings-row', 'data-type': settingGroup.group},
|
||||||
),
|
CE('label', {for: `bx_setting_${pref}`},
|
||||||
!setting.unsupported && $control,
|
label,
|
||||||
);
|
note && CE('div', {'class': 'bx-stream-settings-dialog-note'}, note),
|
||||||
|
setting.unsupported && CE('div', {'class': 'bx-stream-settings-dialog-note'}, t('browser-unsupported-feature')),
|
||||||
|
),
|
||||||
|
!setting.unsupported && $control,
|
||||||
|
);
|
||||||
|
|
||||||
$group.appendChild($content);
|
$group.appendChild($content);
|
||||||
|
|
||||||
@ -431,48 +441,85 @@ export function updateVideoPlayerCss() {
|
|||||||
Screenshot.updateCanvasFilters(filters);
|
Screenshot.updateCanvasFilters(filters);
|
||||||
}
|
}
|
||||||
|
|
||||||
const PREF_RATIO = getPref(PrefKey.VIDEO_RATIO);
|
|
||||||
if (PREF_RATIO && PREF_RATIO !== '16:9') {
|
|
||||||
if (PREF_RATIO.includes(':')) {
|
|
||||||
videoCss += `aspect-ratio: ${PREF_RATIO.replace(':', '/')}; object-fit: unset !important;`;
|
|
||||||
|
|
||||||
const tmp = PREF_RATIO.split(':');
|
|
||||||
const ratio = parseFloat(tmp[0]) / parseFloat(tmp[1]);
|
|
||||||
const maxRatio = window.innerWidth / window.innerHeight;
|
|
||||||
if (ratio < maxRatio) {
|
|
||||||
videoCss += 'width: fit-content !important;'
|
|
||||||
} else {
|
|
||||||
videoCss += 'height: fit-content !important;'
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
videoCss += `object-fit: ${PREF_RATIO} !important;`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let css = '';
|
let css = '';
|
||||||
if (videoCss) {
|
if (videoCss) {
|
||||||
css = `
|
css = `
|
||||||
div[data-testid="media-container"] {
|
|
||||||
display: flex;
|
|
||||||
}
|
|
||||||
|
|
||||||
#game-stream video {
|
#game-stream video {
|
||||||
margin: 0 auto;
|
|
||||||
align-self: center;
|
|
||||||
background: #000;
|
|
||||||
${videoCss}
|
${videoCss}
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
$elm.textContent = css;
|
$elm.textContent = css;
|
||||||
|
|
||||||
|
resizeVideoPlayer();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function resizeVideoPlayer() {
|
||||||
|
const $video = STATES.currentStream.$video;
|
||||||
|
if (!$video || !$video.parentElement) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const PREF_RATIO = getPref(PrefKey.VIDEO_RATIO);
|
||||||
|
if (PREF_RATIO.includes(':')) {
|
||||||
|
const tmp = PREF_RATIO.split(':');
|
||||||
|
|
||||||
|
// Get preferred ratio
|
||||||
|
const videoRatio = parseFloat(tmp[0]) / parseFloat(tmp[1]);
|
||||||
|
|
||||||
|
let width = 0;
|
||||||
|
let height = 0;
|
||||||
|
|
||||||
|
// Get parent's ratio
|
||||||
|
const parentRect = $video.parentElement.getBoundingClientRect();
|
||||||
|
const parentRatio = parentRect.width / parentRect.height;
|
||||||
|
|
||||||
|
// Get target width & height
|
||||||
|
if (parentRatio > videoRatio) {
|
||||||
|
height = parentRect.height;
|
||||||
|
width = height * videoRatio;
|
||||||
|
} else {
|
||||||
|
width = parentRect.width;
|
||||||
|
height = width / videoRatio;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prevent floating points
|
||||||
|
width = Math.min(parentRect.width, Math.ceil(width));
|
||||||
|
height = Math.min(parentRect.height, Math.ceil(height));
|
||||||
|
|
||||||
|
// Update size
|
||||||
|
$video.style.width = `${width}px`;
|
||||||
|
$video.style.height = `${height}px`;
|
||||||
|
$video.style.objectFit = 'fill';
|
||||||
|
} else {
|
||||||
|
$video.style.width = '100%';
|
||||||
|
$video.style.height = '100%';
|
||||||
|
$video.style.objectFit = PREF_RATIO;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
function preloadFonts() {
|
||||||
|
const $link = CE<HTMLLinkElement>('link', {
|
||||||
|
rel: 'preload',
|
||||||
|
href: 'https://redphx.github.io/better-xcloud/fonts/promptfont.otf',
|
||||||
|
as: 'font',
|
||||||
|
type: 'font/otf',
|
||||||
|
crossorigin: '',
|
||||||
|
});
|
||||||
|
|
||||||
|
document.querySelector('head')?.appendChild($link);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
export function setupStreamUi() {
|
export function setupStreamUi() {
|
||||||
// Prevent initializing multiple times
|
// Prevent initializing multiple times
|
||||||
if (!document.querySelector('.bx-quick-settings-bar')) {
|
if (!document.querySelector('.bx-stream-settings-dialog')) {
|
||||||
|
preloadFonts();
|
||||||
|
|
||||||
window.addEventListener('resize', updateVideoPlayerCss);
|
window.addEventListener('resize', updateVideoPlayerCss);
|
||||||
setupQuickSettingsBar();
|
setupStreamSettingsDialog();
|
||||||
StreamStats.render();
|
StreamStats.render();
|
||||||
|
|
||||||
Screenshot.setup();
|
Screenshot.setup();
|
||||||
|
28
src/types/index.d.ts
vendored
28
src/types/index.d.ts
vendored
@ -38,8 +38,6 @@ type BxStates = {
|
|||||||
titleInfo: XcloudTitleInfo;
|
titleInfo: XcloudTitleInfo;
|
||||||
|
|
||||||
$video: HTMLVideoElement | null;
|
$video: HTMLVideoElement | null;
|
||||||
$screenshotCanvas: HTMLCanvasElement | null;
|
|
||||||
screenshotCanvasContext: CanvasRenderingContext2D | null;
|
|
||||||
|
|
||||||
peerConnection: RTCPeerConnection;
|
peerConnection: RTCPeerConnection;
|
||||||
audioContext: AudioContext | null;
|
audioContext: AudioContext | null;
|
||||||
@ -61,6 +59,7 @@ type XcloudTitleInfo = {
|
|||||||
details: {
|
details: {
|
||||||
productId: string;
|
productId: string;
|
||||||
supportedInputTypes: InputType[];
|
supportedInputTypes: InputType[];
|
||||||
|
supportedTabs: any[];
|
||||||
hasTouchSupport: boolean;
|
hasTouchSupport: boolean;
|
||||||
hasFakeTouchSupport: boolean;
|
hasFakeTouchSupport: boolean;
|
||||||
hasMkbSupport: boolean;
|
hasMkbSupport: boolean;
|
||||||
@ -73,5 +72,26 @@ type XcloudTitleInfo = {
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
declare module "*.svg";
|
declare module '*.js';
|
||||||
declare module "*.styl";
|
declare module '*.svg';
|
||||||
|
declare module '*.styl';
|
||||||
|
|
||||||
|
type MkbMouseMove = {
|
||||||
|
movementX: number;
|
||||||
|
movementY: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
type MkbMouseClick = {
|
||||||
|
key: {
|
||||||
|
code: string;
|
||||||
|
name: string;
|
||||||
|
} | null;
|
||||||
|
pressed: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
type MkbMouseWheel = {
|
||||||
|
key: {
|
||||||
|
code: string;
|
||||||
|
name: string;
|
||||||
|
} | null;
|
||||||
|
}
|
||||||
|
2
src/types/preferences.d.ts
vendored
2
src/types/preferences.d.ts
vendored
@ -6,7 +6,7 @@ export type PreferenceSetting = {
|
|||||||
note?: string | HTMLElement;
|
note?: string | HTMLElement;
|
||||||
type?: SettingElementType;
|
type?: SettingElementType;
|
||||||
ready?: (setting: PreferenceSetting) => void;
|
ready?: (setting: PreferenceSetting) => void;
|
||||||
migrate?: (savedPrefs: any, value: any) => {};
|
migrate?: (this: Preferences, savedPrefs: any, value: any) => void;
|
||||||
min?: number;
|
min?: number;
|
||||||
max?: number;
|
max?: number;
|
||||||
steps?: number;
|
steps?: number;
|
||||||
|
@ -13,9 +13,6 @@ export enum BxEvent {
|
|||||||
STREAM_STOPPED = 'bx-stream-stopped',
|
STREAM_STOPPED = 'bx-stream-stopped',
|
||||||
STREAM_ERROR_PAGE = 'bx-stream-error-page',
|
STREAM_ERROR_PAGE = 'bx-stream-error-page',
|
||||||
|
|
||||||
STREAM_MENU_SHOWN = 'bx-stream-menu-shown',
|
|
||||||
STREAM_MENU_HIDDEN = 'bx-stream-menu-hidden',
|
|
||||||
|
|
||||||
STREAM_WEBRTC_CONNECTED = 'bx-stream-webrtc-connected',
|
STREAM_WEBRTC_CONNECTED = 'bx-stream-webrtc-connected',
|
||||||
STREAM_WEBRTC_DISCONNECTED = 'bx-stream-webrtc-disconnected',
|
STREAM_WEBRTC_DISCONNECTED = 'bx-stream-webrtc-disconnected',
|
||||||
|
|
||||||
@ -34,6 +31,15 @@ export enum BxEvent {
|
|||||||
|
|
||||||
GAME_BAR_ACTION_ACTIVATED = 'bx-game-bar-action-activated',
|
GAME_BAR_ACTION_ACTIVATED = 'bx-game-bar-action-activated',
|
||||||
MICROPHONE_STATE_CHANGED = 'bx-microphone-state-changed',
|
MICROPHONE_STATE_CHANGED = 'bx-microphone-state-changed',
|
||||||
|
|
||||||
|
CAPTURE_SCREENSHOT = 'bx-capture-screenshot',
|
||||||
|
GAINNODE_VOLUME_CHANGED = 'bx-gainnode-volume-changed',
|
||||||
|
|
||||||
|
// xCloud Dialog events
|
||||||
|
XCLOUD_DIALOG_SHOWN = 'bx-xcloud-dialog-shown',
|
||||||
|
XCLOUD_DIALOG_DISMISSED = 'bx-xcloud-dialog-dismissed',
|
||||||
|
|
||||||
|
XCLOUD_POLLING_MODE_CHANGED = 'bx-xcloud-polling-mode-changed',
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum XcloudEvent {
|
export enum XcloudEvent {
|
||||||
@ -59,3 +65,5 @@ export namespace BxEvent {
|
|||||||
target.dispatchEvent(event);
|
target.dispatchEvent(event);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
(window as any).BxEvent = BxEvent;
|
||||||
|
@ -1,10 +1,11 @@
|
|||||||
import { GameBar } from "@modules/game-bar/game-bar";
|
import { ControllerShortcut } from "@/modules/controller-shortcut";
|
||||||
import { BxEvent } from "@utils/bx-event";
|
import { BxEvent } from "@utils/bx-event";
|
||||||
import { STATES } from "@utils/global";
|
import { STATES } from "@utils/global";
|
||||||
import { getPref, PrefKey } from "@utils/preferences";
|
import { getPref, PrefKey } from "@utils/preferences";
|
||||||
import { UserAgent } from "@utils/user-agent";
|
import { UserAgent } from "@utils/user-agent";
|
||||||
|
import { BxLogger } from "./bx-logger";
|
||||||
|
|
||||||
enum InputType {
|
export enum InputType {
|
||||||
CONTROLLER = 'Controller',
|
CONTROLLER = 'Controller',
|
||||||
MKB = 'MKB',
|
MKB = 'MKB',
|
||||||
CUSTOM_TOUCH_OVERLAY = 'CustomTouchOverlay',
|
CUSTOM_TOUCH_OVERLAY = 'CustomTouchOverlay',
|
||||||
@ -14,23 +15,6 @@ enum InputType {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const BxExposed = {
|
export const BxExposed = {
|
||||||
// Enable/disable Game Bar when playing/pausing
|
|
||||||
onPollingModeChanged: (mode: 'All' | 'None') => {
|
|
||||||
if (getPref(PrefKey.GAME_BAR_POSITION) === 'off') {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const gameBar = GameBar.getInstance();
|
|
||||||
|
|
||||||
if (!STATES.isPlaying) {
|
|
||||||
gameBar.disable();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Toggle Game bar
|
|
||||||
mode !== 'None' ? gameBar.disable() : gameBar.enable();
|
|
||||||
},
|
|
||||||
|
|
||||||
getTitleInfo: () => STATES.currentStream.titleInfo,
|
getTitleInfo: () => STATES.currentStream.titleInfo,
|
||||||
|
|
||||||
modifyTitleInfo: (titleInfo: XcloudTitleInfo): XcloudTitleInfo => {
|
modifyTitleInfo: (titleInfo: XcloudTitleInfo): XcloudTitleInfo => {
|
||||||
@ -38,6 +22,12 @@ export const BxExposed = {
|
|||||||
titleInfo = structuredClone(titleInfo);
|
titleInfo = structuredClone(titleInfo);
|
||||||
|
|
||||||
let supportedInputTypes = titleInfo.details.supportedInputTypes;
|
let supportedInputTypes = titleInfo.details.supportedInputTypes;
|
||||||
|
|
||||||
|
// Remove native MKB support on mobile browsers or by user's choice
|
||||||
|
if (getPref(PrefKey.NATIVE_MKB_DISABLED) || UserAgent.isMobile()) {
|
||||||
|
supportedInputTypes = supportedInputTypes.filter(i => i !== InputType.MKB);
|
||||||
|
}
|
||||||
|
|
||||||
titleInfo.details.hasMkbSupport = supportedInputTypes.includes(InputType.MKB);
|
titleInfo.details.hasMkbSupport = supportedInputTypes.includes(InputType.MKB);
|
||||||
|
|
||||||
if (STATES.hasTouchSupport) {
|
if (STATES.hasTouchSupport) {
|
||||||
@ -58,14 +48,11 @@ export const BxExposed = {
|
|||||||
gamepadFound && (touchControllerAvailability = 'off');
|
gamepadFound && (touchControllerAvailability = 'off');
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove MKB support on mobile browsers
|
|
||||||
if (UserAgent.isMobile()) {
|
|
||||||
supportedInputTypes = supportedInputTypes.filter(i => i !== InputType.MKB);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (touchControllerAvailability === 'off') {
|
if (touchControllerAvailability === 'off') {
|
||||||
// Disable touch on all games (not native touch)
|
// Disable touch on all games (not native touch)
|
||||||
supportedInputTypes = supportedInputTypes.filter(i => i !== InputType.CUSTOM_TOUCH_OVERLAY && i !== InputType.GENERIC_TOUCH);
|
supportedInputTypes = supportedInputTypes.filter(i => i !== InputType.CUSTOM_TOUCH_OVERLAY && i !== InputType.GENERIC_TOUCH);
|
||||||
|
// Empty TABs
|
||||||
|
titleInfo.details.supportedTabs = [];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Pre-check supported input types
|
// Pre-check supported input types
|
||||||
@ -78,10 +65,10 @@ export const BxExposed = {
|
|||||||
titleInfo.details.hasFakeTouchSupport = true;
|
titleInfo.details.hasFakeTouchSupport = true;
|
||||||
supportedInputTypes.push(InputType.GENERIC_TOUCH);
|
supportedInputTypes.push(InputType.GENERIC_TOUCH);
|
||||||
}
|
}
|
||||||
|
|
||||||
titleInfo.details.supportedInputTypes = supportedInputTypes;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
titleInfo.details.supportedInputTypes = supportedInputTypes;
|
||||||
|
|
||||||
// Save this info in STATES
|
// Save this info in STATES
|
||||||
STATES.currentStream.titleInfo = titleInfo;
|
STATES.currentStream.titleInfo = titleInfo;
|
||||||
BxEvent.dispatch(window, BxEvent.TITLE_INFO_READY);
|
BxEvent.dispatch(window, BxEvent.TITLE_INFO_READY);
|
||||||
@ -103,10 +90,18 @@ export const BxExposed = {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const audioCtx = STATES.currentStream.audioContext!;
|
try {
|
||||||
const source = audioCtx.createMediaStreamSource(audioStream);
|
const audioCtx = STATES.currentStream.audioContext!;
|
||||||
|
const source = audioCtx.createMediaStreamSource(audioStream);
|
||||||
|
|
||||||
const gainNode = audioCtx.createGain(); // call monkey-patched createGain() in BxAudioContext
|
const gainNode = audioCtx.createGain(); // call monkey-patched createGain() in BxAudioContext
|
||||||
source.connect(gainNode).connect(audioCtx.destination);
|
source.connect(gainNode).connect(audioCtx.destination);
|
||||||
}
|
} catch (e) {
|
||||||
|
BxLogger.error('setupGainNode', e);
|
||||||
|
STATES.currentStream.audioGainNode = null;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
handleControllerShortcut: ControllerShortcut.handle,
|
||||||
|
resetControllerShortcut: ControllerShortcut.reset,
|
||||||
};
|
};
|
||||||
|
@ -23,3 +23,5 @@ export const BX_FLAGS = Object.assign(DEFAULT_FLAGS, window.BX_FLAGS || {});
|
|||||||
try {
|
try {
|
||||||
delete window.BX_FLAGS;
|
delete window.BX_FLAGS;
|
||||||
} catch (e) {}
|
} catch (e) {}
|
||||||
|
|
||||||
|
export const NATIVE_FETCH = window.fetch;
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import iconCommand from "@assets/svg/command.svg" with { type: "text" };
|
||||||
import iconController from "@assets/svg/controller.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 iconCopy from "@assets/svg/copy.svg" with { type: "text" };
|
||||||
import iconCursorText from "@assets/svg/cursor-text.svg" with { type: "text" };
|
import iconCursorText from "@assets/svg/cursor-text.svg" with { type: "text" };
|
||||||
@ -24,6 +25,7 @@ import iconMicrophoneMuted from "@assets/svg/microphone-slash.svg" with { type:
|
|||||||
export const BxIcon = {
|
export const BxIcon = {
|
||||||
STREAM_SETTINGS: iconStreamSettings,
|
STREAM_SETTINGS: iconStreamSettings,
|
||||||
STREAM_STATS: iconStreamStats,
|
STREAM_STATS: iconStreamStats,
|
||||||
|
COMMAND: iconCommand,
|
||||||
CONTROLLER: iconController,
|
CONTROLLER: iconController,
|
||||||
DISPLAY: iconDisplay,
|
DISPLAY: iconDisplay,
|
||||||
MOUSE: iconMouse,
|
MOUSE: iconMouse,
|
||||||
|
@ -77,7 +77,7 @@ export const createButton = <T=HTMLButtonElement>(options: BxButton): T => {
|
|||||||
$btn.href = options.url;
|
$btn.href = options.url;
|
||||||
$btn.target = '_blank';
|
$btn.target = '_blank';
|
||||||
} else {
|
} else {
|
||||||
$btn = CE('button', {'class': 'bx-button'}) as HTMLButtonElement;
|
$btn = CE('button', {'class': 'bx-button', type: 'button'}) as HTMLButtonElement;
|
||||||
}
|
}
|
||||||
|
|
||||||
const style = (options.style || 0) as number;
|
const style = (options.style || 0) as number;
|
||||||
|
@ -103,7 +103,7 @@ export function patchRtcPeerConnection() {
|
|||||||
try {
|
try {
|
||||||
const maxVideoBitrate = getPref(PrefKey.BITRATE_VIDEO_MAX);
|
const maxVideoBitrate = getPref(PrefKey.BITRATE_VIDEO_MAX);
|
||||||
if (maxVideoBitrate > 0) {
|
if (maxVideoBitrate > 0) {
|
||||||
arguments[0].sdp = patchSdpBitrate(arguments[0].sdp, maxVideoBitrate * 1000);
|
arguments[0].sdp = patchSdpBitrate(arguments[0].sdp, Math.round(maxVideoBitrate / 1000));
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
BxLogger.error('setLocalDescription', e);
|
BxLogger.error('setLocalDescription', e);
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { BxEvent } from "@utils/bx-event";
|
import { BxEvent } from "@utils/bx-event";
|
||||||
import { BX_FLAGS } from "@utils/bx-flags";
|
import { BX_FLAGS, NATIVE_FETCH } from "@utils/bx-flags";
|
||||||
import { LoadingScreen } from "@modules/loading-screen";
|
import { LoadingScreen } from "@modules/loading-screen";
|
||||||
import { PrefKey, getPref } from "@utils/preferences";
|
import { PrefKey, getPref } from "@utils/preferences";
|
||||||
import { RemotePlay } from "@modules/remote-play";
|
import { RemotePlay } from "@modules/remote-play";
|
||||||
@ -8,8 +8,8 @@ import { TouchController } from "@modules/touch-controller";
|
|||||||
import { STATES } from "@utils/global";
|
import { STATES } from "@utils/global";
|
||||||
import { getPreferredServerRegion } from "@utils/region";
|
import { getPreferredServerRegion } from "@utils/region";
|
||||||
import { GamePassCloudGallery } from "./gamepass-gallery";
|
import { GamePassCloudGallery } from "./gamepass-gallery";
|
||||||
|
import { InputType } from "./bx-exposed";
|
||||||
export const NATIVE_FETCH = window.fetch;
|
import { UserAgent } from "./user-agent";
|
||||||
|
|
||||||
enum RequestType {
|
enum RequestType {
|
||||||
XCLOUD = 'xcloud',
|
XCLOUD = 'xcloud',
|
||||||
@ -188,7 +188,7 @@ class XhomeInterceptor {
|
|||||||
let hasTouchSupport = inputConfigs.supportedTabs.length > 0;
|
let hasTouchSupport = inputConfigs.supportedTabs.length > 0;
|
||||||
if (!hasTouchSupport) {
|
if (!hasTouchSupport) {
|
||||||
const supportedInputTypes = inputConfigs.supportedInputTypes;
|
const supportedInputTypes = inputConfigs.supportedInputTypes;
|
||||||
hasTouchSupport = supportedInputTypes.includes('NativeTouch');
|
hasTouchSupport = supportedInputTypes.includes(InputType.NATIVE_TOUCH) || supportedInputTypes.includes(InputType.CUSTOM_TOUCH_OVERLAY);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (hasTouchSupport) {
|
if (hasTouchSupport) {
|
||||||
@ -438,6 +438,14 @@ class XcloudInterceptor {
|
|||||||
overrides.inputConfiguration = overrides.inputConfiguration || {};
|
overrides.inputConfiguration = overrides.inputConfiguration || {};
|
||||||
overrides.inputConfiguration.enableVibration = true;
|
overrides.inputConfiguration.enableVibration = true;
|
||||||
|
|
||||||
|
if (getPref(PrefKey.NATIVE_MKB_DISABLED) || UserAgent.isMobile()) {
|
||||||
|
overrides.inputConfiguration = Object.assign(overrides.inputConfiguration, {
|
||||||
|
enableMouseInput: false,
|
||||||
|
enableAbsoluteMouse: false,
|
||||||
|
enableKeyboardInput: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
overrides.videoConfiguration = overrides.videoConfiguration || {};
|
overrides.videoConfiguration = overrides.videoConfiguration || {};
|
||||||
overrides.videoConfiguration.setCodecPreferences = true;
|
overrides.videoConfiguration.setCodecPreferences = true;
|
||||||
|
|
||||||
@ -555,6 +563,29 @@ export function interceptHttpRequests() {
|
|||||||
BxEvent.dispatch(window, BxEvent.STREAM_STARTING);
|
BxEvent.dispatch(window, BxEvent.STREAM_STARTING);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Override experimentals
|
||||||
|
if (url.startsWith('https://emerald.xboxservices.com/xboxcomfd/experimentation')) {
|
||||||
|
try {
|
||||||
|
const response = await NATIVE_FETCH(request, init);
|
||||||
|
const json = await response.json();
|
||||||
|
|
||||||
|
const overrideTreatments: {[key: string]: boolean} = {};
|
||||||
|
|
||||||
|
if (getPref(PrefKey.UI_HOME_CONTEXT_MENU_DISABLED)) {
|
||||||
|
overrideTreatments['EnableHomeContextMenu'] = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const key in overrideTreatments) {
|
||||||
|
json.exp.treatments[key] = overrideTreatments[key]
|
||||||
|
}
|
||||||
|
|
||||||
|
response.json = () => Promise.resolve(json);
|
||||||
|
return response;
|
||||||
|
} catch (e) {
|
||||||
|
console.log(e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Add list of games with custom layouts to the official list
|
// Add list of games with custom layouts to the official list
|
||||||
if (STATES.hasTouchSupport && url.includes('catalog.gamepass.com/sigls/')) {
|
if (STATES.hasTouchSupport && url.includes('catalog.gamepass.com/sigls/')) {
|
||||||
const response = await NATIVE_FETCH(request, init);
|
const response = await NATIVE_FETCH(request, init);
|
||||||
|
@ -4,7 +4,7 @@ import { SettingElement, SettingElementType } from "@utils/settings";
|
|||||||
import { UserAgentProfile } from "@utils/user-agent";
|
import { UserAgentProfile } from "@utils/user-agent";
|
||||||
import { StreamStat } from "@modules/stream/stream-stats";
|
import { StreamStat } from "@modules/stream/stream-stats";
|
||||||
import type { PreferenceSetting, PreferenceSettings } from "@/types/preferences";
|
import type { PreferenceSetting, PreferenceSettings } from "@/types/preferences";
|
||||||
import { STATES } from "@utils/global";
|
import { AppInterface, STATES } from "@utils/global";
|
||||||
|
|
||||||
export enum PrefKey {
|
export enum PrefKey {
|
||||||
LAST_UPDATE_CHECK = 'version_last_check',
|
LAST_UPDATE_CHECK = 'version_last_check',
|
||||||
@ -44,6 +44,7 @@ export enum PrefKey {
|
|||||||
CONTROLLER_DEVICE_VIBRATION = 'controller_device_vibration',
|
CONTROLLER_DEVICE_VIBRATION = 'controller_device_vibration',
|
||||||
CONTROLLER_VIBRATION_INTENSITY = 'controller_vibration_intensity',
|
CONTROLLER_VIBRATION_INTENSITY = 'controller_vibration_intensity',
|
||||||
|
|
||||||
|
NATIVE_MKB_DISABLED = 'native_mkb_disabled',
|
||||||
MKB_ENABLED = 'mkb_enabled',
|
MKB_ENABLED = 'mkb_enabled',
|
||||||
MKB_HIDE_IDLE_CURSOR = 'mkb_hide_idle_cursor',
|
MKB_HIDE_IDLE_CURSOR = 'mkb_hide_idle_cursor',
|
||||||
MKB_ABSOLUTE_MOUSE = 'mkb_absolute_mouse',
|
MKB_ABSOLUTE_MOUSE = 'mkb_absolute_mouse',
|
||||||
@ -64,6 +65,8 @@ export enum PrefKey {
|
|||||||
UI_LAYOUT = 'ui_layout',
|
UI_LAYOUT = 'ui_layout',
|
||||||
UI_SCROLLBAR_HIDE = 'ui_scrollbar_hide',
|
UI_SCROLLBAR_HIDE = 'ui_scrollbar_hide',
|
||||||
|
|
||||||
|
UI_HOME_CONTEXT_MENU_DISABLED = 'ui_home_context_menu_disabled',
|
||||||
|
|
||||||
VIDEO_CLARITY = 'video_clarity',
|
VIDEO_CLARITY = 'video_clarity',
|
||||||
VIDEO_RATIO = 'video_ratio',
|
VIDEO_RATIO = 'video_ratio',
|
||||||
VIDEO_BRIGHTNESS = 'video_brightness',
|
VIDEO_BRIGHTNESS = 'video_brightness',
|
||||||
@ -318,25 +321,36 @@ export class Preferences {
|
|||||||
|
|
||||||
[PrefKey.BITRATE_VIDEO_MAX]: {
|
[PrefKey.BITRATE_VIDEO_MAX]: {
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
label: 'Maximum video bitrate',
|
label: t('bitrate-video-maximum'),
|
||||||
note: '⚠️ ' + t('unexpected-behavior'),
|
note: '⚠️ ' + t('unexpected-behavior'),
|
||||||
default: 0,
|
default: 0,
|
||||||
min: 0,
|
min: 0,
|
||||||
max: 14,
|
max: 14 * 1024 * 1000,
|
||||||
steps: 1,
|
steps: 100 * 1024,
|
||||||
params: {
|
params: {
|
||||||
suffix: ' Mb/s',
|
exactTicks: 5 * 1024 * 1000,
|
||||||
exactTicks: 5,
|
|
||||||
customTextValue: (value: any) => {
|
customTextValue: (value: any) => {
|
||||||
value = parseInt(value);
|
value = parseInt(value);
|
||||||
|
|
||||||
if (value === 0) {
|
if (value === 0) {
|
||||||
return t('unlimited');
|
return t('unlimited');
|
||||||
|
} else {
|
||||||
|
return (value / (1024 * 1000)).toFixed(1) + ' Mb/s';
|
||||||
}
|
}
|
||||||
|
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
migrate: function(savedPrefs: any, value: any) {
|
||||||
|
try {
|
||||||
|
value = parseInt(value);
|
||||||
|
if (value !== 0 && value < 100) {
|
||||||
|
value *= 1024 * 1000;
|
||||||
|
}
|
||||||
|
this.set(PrefKey.BITRATE_VIDEO_MAX, value, true);
|
||||||
|
savedPrefs[PrefKey.BITRATE_VIDEO_MAX] = value;
|
||||||
|
} catch (e) {}
|
||||||
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.GAME_BAR_POSITION]: {
|
[PrefKey.GAME_BAR_POSITION]: {
|
||||||
@ -370,10 +384,12 @@ export class Preferences {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.CONTROLLER_ENABLE_VIBRATION]: {
|
[PrefKey.CONTROLLER_ENABLE_VIBRATION]: {
|
||||||
|
label: t('controller-vibration'),
|
||||||
default: true,
|
default: true,
|
||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.CONTROLLER_DEVICE_VIBRATION]: {
|
[PrefKey.CONTROLLER_DEVICE_VIBRATION]: {
|
||||||
|
label: t('device-vibration'),
|
||||||
default: 'off',
|
default: 'off',
|
||||||
options: {
|
options: {
|
||||||
on: t('on'),
|
on: t('on'),
|
||||||
@ -383,6 +399,7 @@ export class Preferences {
|
|||||||
},
|
},
|
||||||
|
|
||||||
[PrefKey.CONTROLLER_VIBRATION_INTENSITY]: {
|
[PrefKey.CONTROLLER_VIBRATION_INTENSITY]: {
|
||||||
|
label: t('vibration-intensity'),
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 100,
|
default: 100,
|
||||||
min: 0,
|
min: 0,
|
||||||
@ -399,7 +416,7 @@ export class Preferences {
|
|||||||
default: false,
|
default: false,
|
||||||
unsupported: ((): string | boolean => {
|
unsupported: ((): string | boolean => {
|
||||||
const userAgent = ((window.navigator as any).orgUserAgent || window.navigator.userAgent || '').toLowerCase();
|
const userAgent = ((window.navigator as any).orgUserAgent || window.navigator.userAgent || '').toLowerCase();
|
||||||
return userAgent.match(/(android|iphone|ipad)/) ? t('browser-unsupported-feature') : false;
|
return !AppInterface && userAgent.match(/(android|iphone|ipad)/) ? t('browser-unsupported-feature') : false;
|
||||||
})(),
|
})(),
|
||||||
ready: (setting: PreferenceSetting) => {
|
ready: (setting: PreferenceSetting) => {
|
||||||
let note;
|
let note;
|
||||||
@ -419,6 +436,11 @@ export class Preferences {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
[PrefKey.NATIVE_MKB_DISABLED]: {
|
||||||
|
label: t('disable-native-mkb'),
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
|
||||||
[PrefKey.MKB_DEFAULT_PRESET_ID]: {
|
[PrefKey.MKB_DEFAULT_PRESET_ID]: {
|
||||||
default: 0,
|
default: 0,
|
||||||
},
|
},
|
||||||
@ -464,6 +486,11 @@ export class Preferences {
|
|||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
|
|
||||||
|
[PrefKey.UI_HOME_CONTEXT_MENU_DISABLED]: {
|
||||||
|
label: t('disable-home-context-menu'),
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
|
||||||
[PrefKey.BLOCK_SOCIAL_FEATURES]: {
|
[PrefKey.BLOCK_SOCIAL_FEATURES]: {
|
||||||
label: t('disable-social-features'),
|
label: t('disable-social-features'),
|
||||||
default: false,
|
default: false,
|
||||||
@ -488,6 +515,7 @@ export class Preferences {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
[PrefKey.VIDEO_CLARITY]: {
|
[PrefKey.VIDEO_CLARITY]: {
|
||||||
|
label: t('clarity'),
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 0,
|
default: 0,
|
||||||
min: 0,
|
min: 0,
|
||||||
@ -497,6 +525,8 @@ export class Preferences {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
[PrefKey.VIDEO_RATIO]: {
|
[PrefKey.VIDEO_RATIO]: {
|
||||||
|
label: t('ratio'),
|
||||||
|
note: t('stretch-note'),
|
||||||
default: '16:9',
|
default: '16:9',
|
||||||
options: {
|
options: {
|
||||||
'16:9': '16:9',
|
'16:9': '16:9',
|
||||||
@ -510,6 +540,7 @@ export class Preferences {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
[PrefKey.VIDEO_SATURATION]: {
|
[PrefKey.VIDEO_SATURATION]: {
|
||||||
|
label: t('saturation'),
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 100,
|
default: 100,
|
||||||
min: 50,
|
min: 50,
|
||||||
@ -520,6 +551,7 @@ export class Preferences {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
[PrefKey.VIDEO_CONTRAST]: {
|
[PrefKey.VIDEO_CONTRAST]: {
|
||||||
|
label: t('contrast'),
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 100,
|
default: 100,
|
||||||
min: 50,
|
min: 50,
|
||||||
@ -530,6 +562,7 @@ export class Preferences {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
[PrefKey.VIDEO_BRIGHTNESS]: {
|
[PrefKey.VIDEO_BRIGHTNESS]: {
|
||||||
|
label: t('brightness'),
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 100,
|
default: 100,
|
||||||
min: 50,
|
min: 50,
|
||||||
@ -549,6 +582,7 @@ export class Preferences {
|
|||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
[PrefKey.AUDIO_VOLUME]: {
|
[PrefKey.AUDIO_VOLUME]: {
|
||||||
|
label: t('volume'),
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 100,
|
default: 100,
|
||||||
min: 0,
|
min: 0,
|
||||||
@ -561,6 +595,7 @@ export class Preferences {
|
|||||||
|
|
||||||
|
|
||||||
[PrefKey.STATS_ITEMS]: {
|
[PrefKey.STATS_ITEMS]: {
|
||||||
|
label: t('stats'),
|
||||||
default: [StreamStat.PING, StreamStat.FPS, StreamStat.BITRATE, StreamStat.DECODE_TIME, StreamStat.PACKETS_LOST, StreamStat.FRAMES_LOST],
|
default: [StreamStat.PING, StreamStat.FPS, StreamStat.BITRATE, StreamStat.DECODE_TIME, StreamStat.PACKETS_LOST, StreamStat.FRAMES_LOST],
|
||||||
multipleOptions: {
|
multipleOptions: {
|
||||||
[StreamStat.PING]: `${StreamStat.PING.toUpperCase()}: ${t('stat-ping')}`,
|
[StreamStat.PING]: `${StreamStat.PING.toUpperCase()}: ${t('stat-ping')}`,
|
||||||
@ -575,12 +610,15 @@ export class Preferences {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
[PrefKey.STATS_SHOW_WHEN_PLAYING]: {
|
[PrefKey.STATS_SHOW_WHEN_PLAYING]: {
|
||||||
|
label: t('show-stats-on-startup'),
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
[PrefKey.STATS_QUICK_GLANCE]: {
|
[PrefKey.STATS_QUICK_GLANCE]: {
|
||||||
|
label: '👀 ' + t('enable-quick-glance-mode'),
|
||||||
default: true,
|
default: true,
|
||||||
},
|
},
|
||||||
[PrefKey.STATS_POSITION]: {
|
[PrefKey.STATS_POSITION]: {
|
||||||
|
label: t('position'),
|
||||||
default: 'top-right',
|
default: 'top-right',
|
||||||
options: {
|
options: {
|
||||||
'top-left': t('top-left'),
|
'top-left': t('top-left'),
|
||||||
@ -589,6 +627,7 @@ export class Preferences {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
[PrefKey.STATS_TEXT_SIZE]: {
|
[PrefKey.STATS_TEXT_SIZE]: {
|
||||||
|
label: t('text-size'),
|
||||||
default: '0.9rem',
|
default: '0.9rem',
|
||||||
options: {
|
options: {
|
||||||
'0.9rem': t('small'),
|
'0.9rem': t('small'),
|
||||||
@ -597,9 +636,11 @@ export class Preferences {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
[PrefKey.STATS_TRANSPARENT]: {
|
[PrefKey.STATS_TRANSPARENT]: {
|
||||||
|
label: t('transparent-background'),
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
[PrefKey.STATS_OPACITY]: {
|
[PrefKey.STATS_OPACITY]: {
|
||||||
|
label: t('opacity'),
|
||||||
type: SettingElementType.NUMBER_STEPPER,
|
type: SettingElementType.NUMBER_STEPPER,
|
||||||
default: 80,
|
default: 80,
|
||||||
min: 50,
|
min: 50,
|
||||||
@ -610,6 +651,7 @@ export class Preferences {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
[PrefKey.STATS_CONDITIONAL_FORMATTING]: {
|
[PrefKey.STATS_CONDITIONAL_FORMATTING]: {
|
||||||
|
label: t('conditional-formatting'),
|
||||||
default: false,
|
default: false,
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -658,11 +700,12 @@ export class Preferences {
|
|||||||
|
|
||||||
for (let settingId in Preferences.SETTINGS) {
|
for (let settingId in Preferences.SETTINGS) {
|
||||||
const setting = Preferences.SETTINGS[settingId];
|
const setting = Preferences.SETTINGS[settingId];
|
||||||
setting.ready && setting.ready.call(this, setting);
|
|
||||||
|
|
||||||
if (setting.migrate && settingId in savedPrefs) {
|
if (setting.migrate && settingId in savedPrefs) {
|
||||||
setting.migrate.call(this, savedPrefs, savedPrefs[settingId]);
|
setting.migrate.call(this, savedPrefs, savedPrefs[settingId]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
setting.ready && setting.ready.call(this, setting);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (let settingId in Preferences.SETTINGS) {
|
for (let settingId in Preferences.SETTINGS) {
|
||||||
@ -673,7 +716,7 @@ export class Preferences {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ignore deprecated settings
|
// Ignore deprecated/migrated settings
|
||||||
if (setting.migrate) {
|
if (setting.migrate) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -740,11 +783,13 @@ export class Preferences {
|
|||||||
return this.#prefs[key];
|
return this.#prefs[key];
|
||||||
}
|
}
|
||||||
|
|
||||||
set(key: PrefKey, value: any) {
|
set(key: PrefKey, value: any, skipSave?: boolean): any {
|
||||||
value = this.#validateValue(key, value);
|
value = this.#validateValue(key, value);
|
||||||
|
|
||||||
this.#prefs[key] = value;
|
this.#prefs[key] = value;
|
||||||
this.#updateStorage();
|
!skipSave && this.#updateStorage();
|
||||||
|
|
||||||
|
return value;
|
||||||
}
|
}
|
||||||
|
|
||||||
#updateStorage() {
|
#updateStorage() {
|
||||||
|
@ -2,6 +2,7 @@ import { STATES } from "@utils/global";
|
|||||||
import { BxLogger } from "./bx-logger";
|
import { BxLogger } from "./bx-logger";
|
||||||
import { TouchController } from "@modules/touch-controller";
|
import { TouchController } from "@modules/touch-controller";
|
||||||
import { GamePassCloudGallery } from "./gamepass-gallery";
|
import { GamePassCloudGallery } from "./gamepass-gallery";
|
||||||
|
import { getPref, PrefKey } from "./preferences";
|
||||||
|
|
||||||
const LOG_TAG = 'PreloadState';
|
const LOG_TAG = 'PreloadState';
|
||||||
|
|
||||||
@ -41,6 +42,14 @@ export function overridePreloadState() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (getPref(PrefKey.UI_HOME_CONTEXT_MENU_DISABLED)) {
|
||||||
|
try {
|
||||||
|
state.experiments.experimentationInfo.data.treatments.EnableHomeContextMenu = false;
|
||||||
|
} catch (e) {
|
||||||
|
BxLogger.error(LOG_TAG, e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
_state = state;
|
_state = state;
|
||||||
STATES.appContext = structuredClone(state.appContext);
|
STATES.appContext = structuredClone(state.appContext);
|
||||||
|
32
src/utils/prompt-font.ts
Normal file
32
src/utils/prompt-font.ts
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
export enum PrompFont {
|
||||||
|
A = '⇓',
|
||||||
|
B = '⇒',
|
||||||
|
X = '⇐',
|
||||||
|
Y = '⇑',
|
||||||
|
|
||||||
|
LB = '↘',
|
||||||
|
RB = '↙',
|
||||||
|
LT = '↖',
|
||||||
|
RT = '↗',
|
||||||
|
|
||||||
|
SELECT = '⇺',
|
||||||
|
START = '⇻',
|
||||||
|
HOME = '',
|
||||||
|
|
||||||
|
UP = '≻',
|
||||||
|
DOWN = '≽',
|
||||||
|
LEFT = '≺',
|
||||||
|
RIGHT = '≼',
|
||||||
|
|
||||||
|
L3 = '↺',
|
||||||
|
LS_UP = '↾',
|
||||||
|
LS_DOWN = '⇂',
|
||||||
|
LS_LEFT = '↼',
|
||||||
|
LS_RIGHT = '⇀',
|
||||||
|
|
||||||
|
R3 = '↻',
|
||||||
|
RS_UP = '↿',
|
||||||
|
RS_DOWN = '⇃',
|
||||||
|
RS_LEFT = '↽',
|
||||||
|
RS_RIGHT = '⇁',
|
||||||
|
}
|
@ -3,21 +3,24 @@ import { CE } from "./html";
|
|||||||
|
|
||||||
|
|
||||||
export class Screenshot {
|
export class Screenshot {
|
||||||
static setup() {
|
static #$canvas: HTMLCanvasElement;
|
||||||
const currentStream = STATES.currentStream;
|
static #canvasContext: CanvasRenderingContext2D;
|
||||||
if (!currentStream.$screenshotCanvas) {
|
|
||||||
currentStream.$screenshotCanvas = CE<HTMLCanvasElement>('canvas', {'class': 'bx-gone'});
|
|
||||||
|
|
||||||
currentStream.screenshotCanvasContext = currentStream.$screenshotCanvas.getContext('2d', {
|
static setup() {
|
||||||
alpha: false,
|
if (Screenshot.#$canvas) {
|
||||||
willReadFrequently: false,
|
return;
|
||||||
});
|
|
||||||
}
|
}
|
||||||
// document.documentElement.appendChild(currentStream.$screenshotCanvas!);
|
|
||||||
|
Screenshot.#$canvas = CE<HTMLCanvasElement>('canvas', {'class': 'bx-gone'});
|
||||||
|
|
||||||
|
Screenshot.#canvasContext = Screenshot.#$canvas.getContext('2d', {
|
||||||
|
alpha: false,
|
||||||
|
willReadFrequently: false,
|
||||||
|
})!;
|
||||||
}
|
}
|
||||||
|
|
||||||
static updateCanvasSize(width: number, height: number) {
|
static updateCanvasSize(width: number, height: number) {
|
||||||
const $canvas = STATES.currentStream.$screenshotCanvas;
|
const $canvas = Screenshot.#$canvas;
|
||||||
if ($canvas) {
|
if ($canvas) {
|
||||||
$canvas.width = width;
|
$canvas.width = width;
|
||||||
$canvas.height = height;
|
$canvas.height = height;
|
||||||
@ -25,7 +28,7 @@ export class Screenshot {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static updateCanvasFilters(filters: string) {
|
static updateCanvasFilters(filters: string) {
|
||||||
STATES.currentStream.screenshotCanvasContext && (STATES.currentStream.screenshotCanvasContext.filter = filters);
|
Screenshot.#canvasContext.filter = filters;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static onAnimationEnd(e: Event) {
|
private static onAnimationEnd(e: Event) {
|
||||||
@ -35,7 +38,7 @@ export class Screenshot {
|
|||||||
static takeScreenshot(callback?: any) {
|
static takeScreenshot(callback?: any) {
|
||||||
const currentStream = STATES.currentStream;
|
const currentStream = STATES.currentStream;
|
||||||
const $video = currentStream.$video;
|
const $video = currentStream.$video;
|
||||||
const $canvas = currentStream.$screenshotCanvas;
|
const $canvas = Screenshot.#$canvas;
|
||||||
if (!$video || !$canvas) {
|
if (!$video || !$canvas) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -43,7 +46,7 @@ export class Screenshot {
|
|||||||
$video.parentElement?.addEventListener('animationend', this.onAnimationEnd);
|
$video.parentElement?.addEventListener('animationend', this.onAnimationEnd);
|
||||||
$video.parentElement?.classList.add('bx-taking-screenshot');
|
$video.parentElement?.classList.add('bx-taking-screenshot');
|
||||||
|
|
||||||
const canvasContext = currentStream.screenshotCanvasContext!;
|
const canvasContext = Screenshot.#canvasContext;
|
||||||
canvasContext.drawImage($video, 0, 0, $canvas.width, $canvas.height);
|
canvasContext.drawImage($video, 0, 0, $canvas.width, $canvas.height);
|
||||||
|
|
||||||
// Get data URL and pass to parent app
|
// Get data URL and pass to parent app
|
||||||
|
@ -26,7 +26,10 @@ export enum SettingElementType {
|
|||||||
|
|
||||||
export class SettingElement {
|
export class SettingElement {
|
||||||
static #renderOptions(key: string, setting: PreferenceSetting, currentValue: any, onChange: any) {
|
static #renderOptions(key: string, setting: PreferenceSetting, currentValue: any, onChange: any) {
|
||||||
const $control = CE<HTMLSelectElement>('select') as HTMLSelectElement;
|
const $control = CE<HTMLSelectElement>('select', {
|
||||||
|
title: setting.label,
|
||||||
|
tabindex: 0,
|
||||||
|
}) as HTMLSelectElement;
|
||||||
for (let value in setting.options) {
|
for (let value in setting.options) {
|
||||||
const label = setting.options[value];
|
const label = setting.options[value];
|
||||||
|
|
||||||
@ -50,7 +53,11 @@ export class SettingElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static #renderMultipleOptions(key: string, setting: PreferenceSetting, currentValue: any, onChange: any, params: MultipleOptionsParams={}) {
|
static #renderMultipleOptions(key: string, setting: PreferenceSetting, currentValue: any, onChange: any, params: MultipleOptionsParams={}) {
|
||||||
const $control = CE<HTMLSelectElement>('select', {'multiple': true});
|
const $control = CE<HTMLSelectElement>('select', {
|
||||||
|
title: setting.label,
|
||||||
|
multiple: true,
|
||||||
|
tabindex: 0,
|
||||||
|
});
|
||||||
if (params && params.size) {
|
if (params && params.size) {
|
||||||
$control.setAttribute('size', params.size.toString());
|
$control.setAttribute('size', params.size.toString());
|
||||||
}
|
}
|
||||||
@ -93,7 +100,7 @@ export class SettingElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static #renderNumber(key: string, setting: PreferenceSetting, currentValue: any, onChange: any) {
|
static #renderNumber(key: string, setting: PreferenceSetting, currentValue: any, onChange: any) {
|
||||||
const $control = CE('input', {'type': 'number', 'min': setting.min, 'max': setting.max}) as HTMLInputElement;
|
const $control = CE('input', {'tabindex': 0, 'type': 'number', 'min': setting.min, 'max': setting.max}) as HTMLInputElement;
|
||||||
$control.value = currentValue;
|
$control.value = currentValue;
|
||||||
onChange && $control.addEventListener('change', (e: Event) => {
|
onChange && $control.addEventListener('change', (e: Event) => {
|
||||||
const target = e.target as HTMLInputElement;
|
const target = e.target as HTMLInputElement;
|
||||||
@ -108,7 +115,7 @@ export class SettingElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static #renderCheckbox(key: string, setting: PreferenceSetting, currentValue: any, onChange: any) {
|
static #renderCheckbox(key: string, setting: PreferenceSetting, currentValue: any, onChange: any) {
|
||||||
const $control = CE('input', {'type': 'checkbox'}) as HTMLInputElement;
|
const $control = CE('input', {'type': 'checkbox', 'tabindex': 0}) as HTMLInputElement;
|
||||||
$control.checked = currentValue;
|
$control.checked = currentValue;
|
||||||
|
|
||||||
onChange && $control.addEventListener('change', e => {
|
onChange && $control.addEventListener('change', e => {
|
||||||
@ -149,17 +156,34 @@ export class SettingElement {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const $wrapper = CE('div', {'class': 'bx-number-stepper'},
|
const $wrapper = CE('div', {'class': 'bx-number-stepper'},
|
||||||
$decBtn = CE('button', {'data-type': 'dec'}, '-') as HTMLButtonElement,
|
$decBtn = CE('button', {
|
||||||
$text = CE('span', {}, renderTextValue(value)) as HTMLSpanElement,
|
'data-type': 'dec',
|
||||||
$incBtn = CE('button', {'data-type': 'inc'}, '+') as HTMLButtonElement,
|
type: 'button',
|
||||||
);
|
tabindex: -1,
|
||||||
|
}, '-') as HTMLButtonElement,
|
||||||
|
$text = CE('span', {}, renderTextValue(value)) as HTMLSpanElement,
|
||||||
|
$incBtn = CE('button', {
|
||||||
|
'data-type': 'inc',
|
||||||
|
type: 'button',
|
||||||
|
tabindex: -1,
|
||||||
|
}, '+') as HTMLButtonElement,
|
||||||
|
);
|
||||||
|
|
||||||
if (!options.disabled && !options.hideSlider) {
|
if (!options.disabled && !options.hideSlider) {
|
||||||
$range = CE('input', {'type': 'range', 'min': MIN, 'max': MAX, 'value': value, 'step': STEPS}) as HTMLInputElement;
|
$range = CE('input', {
|
||||||
|
id: `bx_setting_${key}`,
|
||||||
|
type: 'range',
|
||||||
|
min: MIN,
|
||||||
|
max: MAX,
|
||||||
|
value: value,
|
||||||
|
step: STEPS,
|
||||||
|
tabindex: 0,
|
||||||
|
}) as HTMLInputElement;
|
||||||
|
|
||||||
$range.addEventListener('input', e => {
|
$range.addEventListener('input', e => {
|
||||||
value = parseInt((e.target as HTMLInputElement).value);
|
value = parseInt((e.target as HTMLInputElement).value);
|
||||||
$text.textContent = renderTextValue(value);
|
$text.textContent = renderTextValue(value);
|
||||||
onChange && onChange(e, value);
|
!(e as any).ignoreOnChange && onChange && onChange(e, value);
|
||||||
});
|
});
|
||||||
$wrapper.appendChild($range);
|
$wrapper.appendChild($range);
|
||||||
|
|
||||||
@ -282,7 +306,10 @@ export class SettingElement {
|
|||||||
const method = SettingElement.#METHOD_MAP[type];
|
const method = SettingElement.#METHOD_MAP[type];
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
const $control = method(...Array.from(arguments).slice(1)) as HTMLElement;
|
const $control = method(...Array.from(arguments).slice(1)) as HTMLElement;
|
||||||
$control.id = `bx_setting_${key}`;
|
|
||||||
|
if (type !== SettingElementType.NUMBER_STEPPER) {
|
||||||
|
$control.id = `bx_setting_${key}`;
|
||||||
|
}
|
||||||
|
|
||||||
// Add "name" property to "select" elements
|
// Add "name" property to "select" elements
|
||||||
if (type === SettingElementType.OPTIONS || type === SettingElementType.MULTIPLE_OPTIONS) {
|
if (type === SettingElementType.OPTIONS || type === SettingElementType.MULTIPLE_OPTIONS) {
|
||||||
|
@ -15,7 +15,7 @@ export class Toast {
|
|||||||
static #timeout?: number | null;
|
static #timeout?: number | null;
|
||||||
static #DURATION = 3000;
|
static #DURATION = 3000;
|
||||||
|
|
||||||
static show(msg: string, status?: string, options: Partial<ToastOptions>={}) {
|
static show(msg: string, status?: string, options: Partial<ToastOptions> = {}) {
|
||||||
options = options || {};
|
options = options || {};
|
||||||
|
|
||||||
const args = Array.from(arguments) as [string, string, ToastOptions];
|
const args = Array.from(arguments) as [string, string, ToastOptions];
|
||||||
@ -43,7 +43,7 @@ export class Toast {
|
|||||||
// Get values from item
|
// Get values from item
|
||||||
const [msg, status, options] = Toast.#stack.shift()!;
|
const [msg, status, options] = Toast.#stack.shift()!;
|
||||||
|
|
||||||
if (options.html) {
|
if (options && options.html) {
|
||||||
Toast.#$msg.innerHTML = msg;
|
Toast.#$msg.innerHTML = msg;
|
||||||
} else {
|
} else {
|
||||||
Toast.#$msg.textContent = msg;
|
Toast.#$msg.textContent = msg;
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -1,6 +1,7 @@
|
|||||||
import { PrefKey, getPref, setPref } from "@utils/preferences";
|
import { PrefKey, getPref, setPref } from "@utils/preferences";
|
||||||
import { SCRIPT_VERSION } from "@utils/global";
|
import { AppInterface, SCRIPT_VERSION } from "@utils/global";
|
||||||
import { UserAgent } from "@utils/user-agent";
|
import { UserAgent } from "@utils/user-agent";
|
||||||
|
import { Translations } from "./translation";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check for update
|
* Check for update
|
||||||
@ -25,6 +26,9 @@ export function checkForUpdate() {
|
|||||||
setPref(PrefKey.LATEST_VERSION, json.tag_name.substring(1));
|
setPref(PrefKey.LATEST_VERSION, json.tag_name.substring(1));
|
||||||
setPref(PrefKey.CURRENT_VERSION, SCRIPT_VERSION);
|
setPref(PrefKey.CURRENT_VERSION, SCRIPT_VERSION);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Update translations
|
||||||
|
Translations.updateTranslations(currentVersion === SCRIPT_VERSION);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -38,7 +42,7 @@ export function disablePwa() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Check if it's Safari on mobile
|
// Check if it's Safari on mobile
|
||||||
if (UserAgent.isSafari(true)) {
|
if (!!AppInterface || UserAgent.isSafari(true)) {
|
||||||
// Disable the PWA prompt
|
// Disable the PWA prompt
|
||||||
Object.defineProperty(window.navigator, 'standalone', {
|
Object.defineProperty(window.navigator, 'standalone', {
|
||||||
value: true,
|
value: true,
|
||||||
@ -61,3 +65,28 @@ export function hashCode(str: string): number {
|
|||||||
|
|
||||||
return hash;
|
return hash;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export function renderString(str: string, obj: any){
|
||||||
|
return str.replace(/\$\{.+?\}/g, match => {
|
||||||
|
const key = match.substring(2, match.length - 1);
|
||||||
|
if (key in obj) {
|
||||||
|
return obj[key];
|
||||||
|
}
|
||||||
|
|
||||||
|
return match;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export function ceilToNearest(value: number, interval: number): number {
|
||||||
|
return Math.ceil(value / interval) * interval;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function floorToNearest(value: number, interval: number): number {
|
||||||
|
return Math.floor(value / interval) * interval;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function roundToNearest(value: number, interval: number): number {
|
||||||
|
return Math.round(value / interval) * interval;
|
||||||
|
}
|
||||||
|
Reference in New Issue
Block a user