คุณต้องเข้าสู่ระบบหรือลงทะเบียนก่อนดำเนินการต่อ
Implement joystick control in Pixiv using the HTML5 Gamepad API.
// ==UserScript== // @name Pixiv-JoystickControl // @name:zh-CN Pixiv-摇杆控制 // @name:ja Pixiv-コントローラー制御 // @namespace https://github.com/Mehver // @version 1.0 // @description Implement joystick control in Pixiv using the HTML5 Gamepad API. // @description:zh-CN 使用HTML5 Gamepad API在Pixiv上实现操纵杆控制。 // @description:ja HTML5 Gamepad APIを使用してPixivでジョイスティックコントロールを実装します。 // @author https://github.com/Mehver // @icon  // @match http*://pixiv.net // @match http*://pixiv.net/* // @match http*://www.pixiv.net // @match http*://www.pixiv.net/* // @license MPL-2.0 // @license Mozilla Public License 2.0 // @homepageURL https://github.com/SynRGB/Pixiv-JoystickControl // @contributionURL https://github.com/SynRGB/Pixiv-JoystickControl // @copyright Copyright © 2022-PRESENT, Mehver (https://github.com/Mehver) // @charset UTF-8 // @grant GM_registerMenuCommand // @grant GM_setValue // @grant GM_getValue // @grant unsafeWindow // @run-at document-end // ==/UserScript== (function () { 'use strict'; let buttonA, buttonB, buttonX, buttonY, buttonRB, buttonLB, buttonRT, buttonLT; let buttonESC, buttonLike, buttonPageBack, buttonPageForward; // let buttonTabindexExit, buttonShiftTab, buttonTab, buttonEnter; let buttonPageBackPrevious = false; let buttonPageForwardPrevious = false; let buttonTabindexExitPrevious = false; let buttonShiftTabPrevious = false; let buttonTabPrevious = false; let buttonEnterPrevious = false; let preset = GM_getValue("preset", "Xbox"); function setupPreset() { buttonA = preset === "Xbox" ? 0 : 2; buttonB = preset === "Xbox" ? 1 : 1; buttonX = preset === "Xbox" ? 2 : 0; buttonY = preset === "Xbox" ? 3 : 3; buttonRB = preset === "Xbox" ? 5 : 5; buttonLB = preset === "Xbox" ? 4 : 4; buttonRT = preset === "Xbox" ? 7 : 7; buttonLT = preset === "Xbox" ? 6 : 6; // buttonESC = buttonA; // buttonLike = buttonB; // buttonPageBack = buttonX; // buttonPageForward = buttonY; } function switchToXbox() { preset = "Xbox"; GM_setValue("preset", "Xbox"); setupPreset(); } function switchToPlayStation() { preset = "PlayStation"; GM_setValue("preset", "PlayStation"); setupPreset(); } GM_registerMenuCommand("Switch to Xbox preset", switchToXbox); GM_registerMenuCommand("Switch to PlayStation preset", switchToPlayStation); setupPreset(); window.addEventListener("gamepadconnected", (event) => { console.log("A gamepad was connected:", event.gamepad); }); let lastAxis0 = null; let lastAxis1 = null; function simulateKeyPress(key, extraParams = {}) { const eventInitDict = {bubbles: true, cancelable: true, composed: true, ...extraParams}; document.dispatchEvent(new KeyboardEvent('keydown', {...eventInitDict, key})); document.dispatchEvent(new KeyboardEvent('keyup', {...eventInitDict, key})); } function focusNextElement(reverse = false) { const focusableElements = 'button, [href], input, select, textarea, [tabindex]:not([tabindex="-1"])'; const focusables = Array.from(document.querySelectorAll(focusableElements)); const currentIndex = focusables.findIndex(el => el === document.activeElement); let nextIndex = reverse ? currentIndex - 1 : currentIndex + 1; if (nextIndex < 0) nextIndex = focusables.length - 1; if (nextIndex >= focusables.length) nextIndex = 0; focusables[nextIndex]?.focus(); } function pollGamepad() { const gamepads = navigator.getGamepads(); if (gamepads[0]) { const gp = gamepads[0]; if (lastAxis0 === null) lastAxis0 = gp.axes[0]; if (lastAxis1 === null) lastAxis1 = gp.axes[1]; if (Math.abs(gp.axes[0] - lastAxis0) > 0.7) { if (gp.axes[0] > 0.7) { simulateKeyPress("ArrowRight"); } else if (gp.axes[0] < -0.7) { simulateKeyPress("ArrowLeft"); } lastAxis0 = gp.axes[0]; } if (Math.abs(gp.axes[1] - lastAxis1) > 0.7) { if (gp.axes[1] > 0.7) { simulateKeyPress("ArrowDown"); } else if (gp.axes[1] < -0.7) { simulateKeyPress("ArrowUp"); } lastAxis1 = gp.axes[1]; } if (gp.buttons[buttonESC].pressed) { simulateKeyPress("Escape"); } if (gp.buttons[buttonLike].pressed) { const url = window.location.href; const artworksRegex1 = /\/artworks\/\d+/; if (artworksRegex1.test(url)) { const xpath1 = '//*[@id="root"]/div[2]/div/div[3]/div/div/div[1]/main/section/div[1]/div/div[4]/div/div[2]/section/div[3]/button'; const xpath2 = '//*[@id="root"]/div[2]/div/div[3]/div/div/div[1]/main/section/div[1]/div/div[4]/div/div[2]/section/div[3]/a'; const xpath3 = '//*[@id="root"]/div[2]/div/div[3]/div/div/div[1]/main/section/div[1]/div/div[5]/div/div[2]/section/div[3]/a'; const element1 = document.evaluate(xpath1, document, null, XPathR###lt.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; const element2 = document.evaluate(xpath2, document, null, XPathR###lt.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; const element3 = document.evaluate(xpath3, document, null, XPathR###lt.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; if (element1) { element1.click(); } else if (element2) { element2.click(); } else if (element3) { element3.click(); } } const artworksRegex2 = /\/bookmark_add\.php\?type=illust&illust_id=\d+/; if (artworksRegex2.test(url)) { const xpath = '//*[@id="wrapper"]/div[1]/section/form[2]/input[7]'; const element = document.evaluate(xpath, document, null, XPathR###lt.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue; if (element) { element.click(); } } } if (gp.buttons[buttonPageBack].pressed && !buttonPageBackPrevious) { window.history.back(); } buttonPageBackPrevious = gp.buttons[buttonPageBack].pressed; if (gp.buttons[buttonPageForward].pressed && !buttonPageForwardPrevious) { window.history.forward(); } buttonPageForwardPrevious = gp.buttons[buttonPageForward].pressed; // if (gp.buttons[buttonTabindexExit].pressed) { // document.elementFromPoint(10, 10)?.click(); // } // buttonTabindexExitPrevious = gp.buttons[buttonTabindexExit].pressed; // // if (gp.buttons[buttonEnter].pressed && !buttonEnterPrevious) { // simulateKeyPress("Enter"); // } // buttonEnterPrevious = gp.buttons[buttonEnter].pressed; // // if (gp.buttons[buttonShiftTab].pressed && !buttonShiftTabPrevious) { // focusNextElement(true); // } // buttonShiftTabPrevious = gp.buttons[buttonShiftTab].pressed; // // if (gp.buttons[buttonTab].pressed && !buttonTabPrevious) { // focusNextElement(); // } // buttonTabPrevious = gp.buttons[buttonTab].pressed; } requestAnimationFrame(pollGamepad); } pollGamepad(); console.log("JS script Pixiv-JoystickControl (Pixiv-摇杆控制) loaded. See more details at https://github.com/SynRGB/Pixiv-JoystickControl"); })();