Greasy Fork is available in English.

Pixiv-JoystickControl

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, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
                    const element2 = document.evaluate(xpath2, document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null).singleNodeValue;
                    const element3 = document.evaluate(xpath3, document, null, XPathResult.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, XPathResult.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");
})();