Merge pull request #10146 from missionfloyd/gamepad-option

Fix gamepad navigation
This commit is contained in:
AUTOMATIC1111 2023-05-08 08:09:12 +03:00 committed by GitHub
commit 66428667c5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 55 additions and 32 deletions

View File

@ -1,27 +1,36 @@
let delay = 350//ms
window.addEventListener('gamepadconnected', (e) => { window.addEventListener('gamepadconnected', (e) => {
console.log("Gamepad connected!") const index = e.gamepad.index;
const gamepad = e.gamepad; let isWaiting = false;
setInterval(() => { setInterval(async () => {
const xValue = gamepad.axes[0].toFixed(2); if (!opts.js_modal_lightbox_gamepad || isWaiting) return;
if (xValue < -0.3) { const gamepad = navigator.getGamepads()[index];
const xValue = gamepad.axes[0];
if (xValue <= -0.3) {
modalPrevImage(e); modalPrevImage(e);
} else if (xValue > 0.3) { isWaiting = true;
} else if (xValue >= 0.3) {
modalNextImage(e); modalNextImage(e);
isWaiting = true;
} }
if (isWaiting) {
}, delay); await sleepUntil(() => {
const xValue = navigator.getGamepads()[index].axes[0]
if (xValue < 0.3 && xValue > -0.3) {
return true;
}
}, opts.js_modal_lightbox_gamepad_repeat);
isWaiting = false;
}
}, 10);
}); });
/* /*
Primarily for vr controller type pointer devices. Primarily for vr controller type pointer devices.
I use the wheel event because there's currently no way to do it properly with web xr. I use the wheel event because there's currently no way to do it properly with web xr.
*/ */
let isScrolling = false; let isScrolling = false;
window.addEventListener('wheel', (e) => { window.addEventListener('wheel', (e) => {
if (isScrolling) return; if (!opts.js_modal_lightbox_gamepad || isScrolling) return;
isScrolling = true; isScrolling = true;
if (e.deltaX <= -0.6) { if (e.deltaX <= -0.6) {
@ -32,5 +41,17 @@
setTimeout(() => { setTimeout(() => {
isScrolling = false; isScrolling = false;
}, delay); }, opts.js_modal_lightbox_gamepad_repeat);
}); });
function sleepUntil(f, timeout) {
return new Promise((resolve) => {
const timeStart = new Date();
const wait = setInterval(function() {
if (f() || new Date() - timeStart > timeout) {
clearInterval(wait);
resolve();
}
}, 20);
});
}

View File

@ -400,6 +400,8 @@ options_templates.update(options_section(('ui', "User interface"), {
"font": OptionInfo("", "Font for image grids that have text"), "font": OptionInfo("", "Font for image grids that have text"),
"js_modal_lightbox": OptionInfo(True, "Enable full page image viewer"), "js_modal_lightbox": OptionInfo(True, "Enable full page image viewer"),
"js_modal_lightbox_initially_zoomed": OptionInfo(True, "Show images zoomed in by default in full page image viewer"), "js_modal_lightbox_initially_zoomed": OptionInfo(True, "Show images zoomed in by default in full page image viewer"),
"js_modal_lightbox_gamepad": OptionInfo(True, "Navigate image viewer with gamepad"),
"js_modal_lightbox_gamepad_repeat": OptionInfo(250, "Gamepad repeat period, in milliseconds"),
"show_progress_in_title": OptionInfo(True, "Show generation progress in window title."), "show_progress_in_title": OptionInfo(True, "Show generation progress in window title."),
"samplers_in_dropdown": OptionInfo(True, "Use dropdown for sampler selection instead of radio group"), "samplers_in_dropdown": OptionInfo(True, "Use dropdown for sampler selection instead of radio group"),
"dimensions_and_batch_together": OptionInfo(True, "Show Width/Height and Batch sliders in same row"), "dimensions_and_batch_together": OptionInfo(True, "Show Width/Height and Batch sliders in same row"),