1911 lines
65 KiB
JavaScript
1911 lines
65 KiB
JavaScript
(window["webpackJsonpGUI"] = window["webpackJsonpGUI"] || []).push([["addon-entry-gamepad"],{
|
|
|
|
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/gamepad/gamepadlib.css":
|
|
/*!****************************************************************************!*\
|
|
!*** ./node_modules/css-loader!./src/addons/addons/gamepad/gamepadlib.css ***!
|
|
\****************************************************************************/
|
|
/*! no static exports found */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
var escape = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/url/escape.js */ "./node_modules/css-loader/lib/url/escape.js");
|
|
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
|
|
// imports
|
|
|
|
|
|
// module
|
|
exports.push([module.i, ".gamepadlib-selector {\n width: 100%;\n margin-bottom: 3px;\n}\n\n.gamepadlib-content {\n display: flex;\n flex-direction: row;\n flex-wrap: wrap;\n}\n.gamepadlib-content-buttons {\n padding-right: 10px;\n}\n\n.gamepadlib-mapping {\n display: flex;\n align-items: center;\n margin-bottom: 3px;\n}\n.gamepadlib-mapping-label {\n width: 100px;\n text-align: center;\n}\n.gamepadlib-keyinput {\n text-align: center;\n width: 75px;\n height: 25px;\n border-radius: 0;\n border: 1px solid black;\n background: white;\n color: black;\n box-sizing: border-box;\n padding: 0;\n margin: 0;\n}\n[theme=\"dark\"] .gamepadlib-keyinput {\n background-color: var(--ui-tertiary);\n border-color: var(--ui-black-transparent);\n color: var(--text-primary);\n}\n.gamepadlib-mapping[data-value=\"1\"] .gamepadlib-keyinput {\n background: yellow;\n}\n[theme=\"dark\"] .gamepadlib-mapping[data-value=\"1\"] .gamepadlib-keyinput {\n background: hsl(60, 100%, 20%)\n}\n.gamepadlib-keyinput[data-accepting-input=\"true\"] {\n background: #d6fff9;\n}\n[theme=\"dark\"] .gamepadlib-keyinput[data-accepting-input=\"true\"] {\n background: hsl(171, 100%, 20%);\n}\n.gamepadlib-keyinput[data-empty=\"true\"]:not([data-accepting-input=\"true\"]) {\n color: #aaa;\n font-style: italic;\n}\n\n.gamepadlib-axis {\n margin-bottom: 8px;\n text-align: center;\n}\n.gamepadlib-axis-circle {\n position: relative;\n width: 150px;\n height: 150px;\n border: 1px solid black;\n overflow: hidden;\n}\n[theme=\"dark\"] .gamepadlib-axis-circle {\n border-color: var(--ui-black-transparent);\n}\n.gamepadlib-axis-dot {\n position: absolute;\n top: 50%;\n left: 50%;\n background-image: url(" + escape(__webpack_require__(/*! ./dot.svg */ "./src/addons/addons/gamepad/dot.svg")) + ");\n width: 8px;\n height: 8px;\n transform: translate(-50%, -50%);\n pointer-events: none;\n}\n.gamepadlib-axis-mapping {\n width: 100%;\n}\n\n.gamepadlib-axis-circle-overlay {\n position: absolute;\n top: 0;\n left: 0;\n width: 100%;\n height: 100%;\n}\n.gamepadlib-axis-circle-overlay > .gamepadlib-axis-mapper {\n position: absolute;\n}\n.gamepadlib-axis-circle-overlay > .gamepadlib-axis-mapper:nth-of-type(1) {\n left: 50%;\n top: 0;\n transform: translateX(-50%);\n}\n.gamepadlib-axis-circle-overlay > .gamepadlib-axis-mapper:nth-of-type(2) {\n left: 0;\n top: 50%;\n transform: translateY(-50%);\n}\n.gamepadlib-axis-circle-overlay > .gamepadlib-axis-mapper:nth-of-type(3) {\n right: 0;\n top: 50%;\n transform: translateY(-50%);\n}\n.gamepadlib-axis-circle-overlay > .gamepadlib-axis-mapper:nth-of-type(4) {\n left: 50%;\n bottom: 0;\n transform: translateX(-50%);\n}\n", ""]);
|
|
|
|
// exports
|
|
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./node_modules/css-loader/index.js!./src/addons/addons/gamepad/style.css":
|
|
/*!***********************************************************************!*\
|
|
!*** ./node_modules/css-loader!./src/addons/addons/gamepad/style.css ***!
|
|
\***********************************************************************/
|
|
/*! no static exports found */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
exports = module.exports = __webpack_require__(/*! ../../../../node_modules/css-loader/lib/css-base.js */ "./node_modules/css-loader/lib/css-base.js")(false);
|
|
// imports
|
|
|
|
|
|
// module
|
|
exports.push([module.i, "[dir=\"ltr\"] .sa-gamepad-container {\n margin-right: 0.2rem;\n}\n[dir=\"rtl\"] .sa-gamepad-container {\n margin-left: 0.2rem;\n}\n\n.sa-gamepad-popup-outer {\n /* above fullscreen */\n z-index: 99999;\n}\n.sa-gamepad-popup {\n box-sizing: border-box;\n width: 700px;\n max-height: min(800px, 85vh);\n height: 100%;\n max-width: 85%;\n margin: 50px auto;\n display: flex;\n flex-direction: column;\n}\n.sa-gamepad-popup-content {\n padding: 1.5rem 2.25rem;\n height: 100%;\n overflow-y: auto;\n}\n\n.sa-gamepad-popup [class*=\"modal_header-item-title\"] {\n margin: 0 -20rem 0 0;\n}\n\n.sa-gamepad-cursor {\n position: absolute;\n top: 0;\n left: 0;\n z-index: 9999;\n user-select: none;\n pointer-events: none;\n will-change: transform;\n image-rendering: optimizeSpeed;\n image-rendering: crisp-edges;\n image-rendering: pixelated;\n}\n.sa-gamepad-cursor-down {\n filter: invert(100%);\n}\n\n.sa-gamepad-small .sa-gamepad-container[data-editor-mode=\"editor\"] {\n display: none !important;\n}\n\n.sa-gamepad-hide-cursor [class^=\"stage_stage_\"] {\n cursor: none;\n}\n\n.sa-gamepad-browser-support-warning {\n font-weight: bold;\n margin-bottom: 10px;\n}\n\n.sa-gamepad-extra-options {\n display: none;\n}\n.sa-gamepad-has-controller .sa-gamepad-extra-options {\n display: block;\n}\n\n.sa-gamepad-store-settings {\n display: block;\n}\n.sa-gamepad-store-settings > input {\n margin-right: 4px;\n}\n\n.sa-gamepad-reset-button {\n margin: 8px 8px 8px 0;\n}\n", ""]);
|
|
|
|
// exports
|
|
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/active.png":
|
|
/*!************************************************************************************!*\
|
|
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/active.png ***!
|
|
\************************************************************************************/
|
|
/*! exports provided: default */
|
|
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
|
|
|
"use strict";
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony default export */ __webpack_exports__["default"] = ("");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/close.svg":
|
|
/*!***********************************************************************************!*\
|
|
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/close.svg ***!
|
|
\***********************************************************************************/
|
|
/*! exports provided: default */
|
|
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
|
|
|
"use strict";
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony default export */ __webpack_exports__["default"] = ("");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/cursor.png":
|
|
/*!************************************************************************************!*\
|
|
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/cursor.png ***!
|
|
\************************************************************************************/
|
|
/*! exports provided: default */
|
|
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
|
|
|
"use strict";
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony default export */ __webpack_exports__["default"] = ("");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/dot.svg":
|
|
/*!*********************************************************************************!*\
|
|
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/dot.svg ***!
|
|
\*********************************************************************************/
|
|
/*! exports provided: default */
|
|
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
|
|
|
"use strict";
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony default export */ __webpack_exports__["default"] = ("");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/gamepad.svg":
|
|
/*!*************************************************************************************!*\
|
|
!*** ./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/gamepad.svg ***!
|
|
\*************************************************************************************/
|
|
/*! exports provided: default */
|
|
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
|
|
|
"use strict";
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony default export */ __webpack_exports__["default"] = ("");
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./src/addons/addons/gamepad/_runtime_entry.js":
|
|
/*!*****************************************************!*\
|
|
!*** ./src/addons/addons/gamepad/_runtime_entry.js ***!
|
|
\*****************************************************/
|
|
/*! exports provided: resources */
|
|
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
|
|
|
"use strict";
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony export (binding) */ __webpack_require__.d(__webpack_exports__, "resources", function() { return resources; });
|
|
/* harmony import */ var _userscript_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./userscript.js */ "./src/addons/addons/gamepad/userscript.js");
|
|
/* harmony import */ var _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! css-loader!./style.css */ "./node_modules/css-loader/index.js!./src/addons/addons/gamepad/style.css");
|
|
/* harmony import */ var _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_css_loader_style_css__WEBPACK_IMPORTED_MODULE_1__);
|
|
/* harmony import */ var _css_loader_gamepadlib_css__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! css-loader!./gamepadlib.css */ "./node_modules/css-loader/index.js!./src/addons/addons/gamepad/gamepadlib.css");
|
|
/* harmony import */ var _css_loader_gamepadlib_css__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(_css_loader_gamepadlib_css__WEBPACK_IMPORTED_MODULE_2__);
|
|
/* harmony import */ var _url_loader_active_png__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! url-loader!./active.png */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/active.png");
|
|
/* harmony import */ var _url_loader_close_svg__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! url-loader!./close.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/close.svg");
|
|
/* harmony import */ var _url_loader_cursor_png__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! url-loader!./cursor.png */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/cursor.png");
|
|
/* harmony import */ var _url_loader_dot_svg__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! url-loader!./dot.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/dot.svg");
|
|
/* harmony import */ var _url_loader_gamepad_svg__WEBPACK_IMPORTED_MODULE_7__ = __webpack_require__(/*! url-loader!./gamepad.svg */ "./node_modules/url-loader/dist/cjs.js!./src/addons/addons/gamepad/gamepad.svg");
|
|
/* generated by pull.js */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const resources = {
|
|
"userscript.js": _userscript_js__WEBPACK_IMPORTED_MODULE_0__["default"],
|
|
"style.css": _css_loader_style_css__WEBPACK_IMPORTED_MODULE_1___default.a,
|
|
"gamepadlib.css": _css_loader_gamepadlib_css__WEBPACK_IMPORTED_MODULE_2___default.a,
|
|
"active.png": _url_loader_active_png__WEBPACK_IMPORTED_MODULE_3__["default"],
|
|
"close.svg": _url_loader_close_svg__WEBPACK_IMPORTED_MODULE_4__["default"],
|
|
"cursor.png": _url_loader_cursor_png__WEBPACK_IMPORTED_MODULE_5__["default"],
|
|
"dot.svg": _url_loader_dot_svg__WEBPACK_IMPORTED_MODULE_6__["default"],
|
|
"gamepad.svg": _url_loader_gamepad_svg__WEBPACK_IMPORTED_MODULE_7__["default"]
|
|
};
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./src/addons/addons/gamepad/dot.svg":
|
|
/*!*******************************************!*\
|
|
!*** ./src/addons/addons/gamepad/dot.svg ***!
|
|
\*******************************************/
|
|
/*! no static exports found */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
module.exports = __webpack_require__.p + "static/assets/88a77444f0bb453209bf1c62becbd37c.svg";
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./src/addons/addons/gamepad/gamepadlib.js":
|
|
/*!*************************************************!*\
|
|
!*** ./src/addons/addons/gamepad/gamepadlib.js ***!
|
|
\*************************************************/
|
|
/*! exports provided: default */
|
|
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
|
|
|
"use strict";
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony import */ var _event_target_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../event-target.js */ "./src/addons/event-target.js");
|
|
|
|
/* inserted by pull.js */
|
|
|
|
let console = window.console;
|
|
/*
|
|
Mapping types:
|
|
|
|
type: "key" maps a button to a keyboard key
|
|
All key names will be interpreted as a KeyboardEvent.key value (https://developer.mozilla.org/en-US/docs/Web/API/KeyboardEvent/key/Key_Values)
|
|
high: "KeyName" is the name of the key to dispatch when a button reads a HIGH value
|
|
low: "KeyName" is the name of the key to dispatch when a button reads a LOW value
|
|
deadZone: 0.5 controls the minimum value necessary to be read in either + or - to trigger either high or low
|
|
The high/low distinction is necessary for axes. Buttons will only use high
|
|
|
|
type: "mousedown" maps a button to control whether the mouse is down or not
|
|
deadZone: 0.5 controls the minimum value to trigger a mousedown
|
|
button: 0, 1, 2, etc. controls which button to press
|
|
|
|
type: "virtual_cursor" maps a button to control the "virtual cursor"
|
|
deadZone: 0.5 again controls the minimum value to trigger a movement
|
|
sensitivity: 10 controls the speed
|
|
high: "+y"/"-y"/"+x"/"-x" defines what happens when an axis reads high
|
|
low: "+y"/"-y"/"+x"/"-x" defines what happens when an axis reads low
|
|
+y increases y, -y decreases y, +x increases x, -x decreases x.
|
|
*/
|
|
|
|
const defaultAxesMappings = {
|
|
arrows: [{
|
|
type: "key",
|
|
high: "ArrowRight",
|
|
low: "ArrowLeft",
|
|
deadZone: 0.5
|
|
}, {
|
|
type: "key",
|
|
high: "ArrowDown",
|
|
low: "ArrowUp",
|
|
deadZone: 0.5
|
|
}],
|
|
wasd: [{
|
|
type: "key",
|
|
high: "d",
|
|
low: "a",
|
|
deadZone: 0.5
|
|
}, {
|
|
type: "key",
|
|
high: "s",
|
|
low: "w",
|
|
deadZone: 0.5
|
|
}],
|
|
cursor: [{
|
|
type: "virtual_cursor",
|
|
high: "+x",
|
|
low: "-x",
|
|
sensitivity: 0.6,
|
|
deadZone: 0.2
|
|
}, {
|
|
type: "virtual_cursor",
|
|
high: "-y",
|
|
low: "+y",
|
|
sensitivity: 0.6,
|
|
deadZone: 0.2
|
|
}]
|
|
};
|
|
|
|
const emptyMapping = () => ({
|
|
type: "key",
|
|
high: null,
|
|
low: null
|
|
});
|
|
|
|
const transformAndCopyMapping = mapping => {
|
|
if (typeof mapping !== "object" || !mapping) {
|
|
console.warn("invalid mapping", mapping);
|
|
return emptyMapping();
|
|
}
|
|
|
|
const copy = Object.assign({}, mapping);
|
|
|
|
if (copy.type === "key") {
|
|
if (typeof copy.deadZone === "undefined") {
|
|
copy.deadZone = 0.5;
|
|
}
|
|
|
|
if (typeof copy.high === "undefined") {
|
|
copy.high = "";
|
|
}
|
|
|
|
if (typeof copy.low === "undefined") {
|
|
copy.low = "";
|
|
}
|
|
} else if (copy.type === "mousedown") {
|
|
if (typeof copy.deadZone === "undefined") {
|
|
copy.deadZone = 0.5;
|
|
}
|
|
|
|
if (typeof copy.button === "undefined") {
|
|
copy.button = 0;
|
|
}
|
|
} else if (copy.type === "virtual_cursor") {
|
|
if (typeof copy.high === "undefined") {
|
|
copy.high = "";
|
|
}
|
|
|
|
if (typeof copy.low === "undefined") {
|
|
copy.low = "";
|
|
}
|
|
|
|
if (typeof copy.sensitivity === "undefined") {
|
|
copy.sensitivity = 10;
|
|
}
|
|
|
|
if (typeof copy.deadZone === "undefined") {
|
|
copy.deadZone = 0.5;
|
|
}
|
|
} else {
|
|
console.warn("unknown mapping type", copy.type);
|
|
return emptyMapping();
|
|
}
|
|
|
|
return copy;
|
|
};
|
|
|
|
const prepareMappingForExport = mapping => Object.assign({}, mapping);
|
|
|
|
const prepareAxisMappingForExport = prepareMappingForExport;
|
|
|
|
const prepareButtonMappingForExport = mapping => {
|
|
const copy = prepareMappingForExport(mapping);
|
|
delete copy.deadZone;
|
|
delete copy.low;
|
|
return copy;
|
|
};
|
|
|
|
const padWithEmptyMappings = (array, length) => {
|
|
// Keep adding empty mappings until the list is full
|
|
while (array.length < length) {
|
|
array.push(emptyMapping());
|
|
} // In case the input array is longer than the desired length
|
|
|
|
|
|
array.length = length;
|
|
return array;
|
|
};
|
|
|
|
const createEmptyMappingList = length => padWithEmptyMappings([], length);
|
|
|
|
const getMovementConfiguration = usedKeys => ({
|
|
usesArrows: usedKeys.has("ArrowUp") || usedKeys.has("ArrowDown") || usedKeys.has("ArrowRight") || usedKeys.has("ArrowLeft"),
|
|
usesWASD: usedKeys.has("w") && usedKeys.has("s") || usedKeys.has("a") && usedKeys.has("d")
|
|
});
|
|
|
|
const getGamepadId = gamepad => "".concat(gamepad.id, " (").concat(gamepad.index, ")");
|
|
|
|
class GamepadData {
|
|
/**
|
|
* @param {Gamepad} gamepad Source Gamepad
|
|
* @param {GamepadLib} gamepadLib Parent GamepadLib
|
|
*/
|
|
constructor(gamepad, gamepadLib) {
|
|
this.gamepad = gamepad;
|
|
this.gamepadLib = gamepadLib;
|
|
this.resetMappings();
|
|
}
|
|
|
|
resetMappings() {
|
|
this.hints = this.gamepadLib.getHints();
|
|
this.buttonMappings = this.getDefaultButtonMappings().map(transformAndCopyMapping);
|
|
this.axesMappings = this.getDefaultAxisMappings().map(transformAndCopyMapping);
|
|
}
|
|
|
|
clearMappings() {
|
|
this.buttonMappings = createEmptyMappingList(this.gamepad.buttons.length);
|
|
this.axesMappings = createEmptyMappingList(this.gamepad.axes.length);
|
|
}
|
|
|
|
getDefaultButtonMappings() {
|
|
let buttons;
|
|
|
|
if (this.hints.importedSettings) {
|
|
buttons = this.hints.importedSettings.buttons;
|
|
} else {
|
|
const usedKeys = this.hints.usedKeys;
|
|
const alreadyUsedKeys = new Set();
|
|
const {
|
|
usesArrows,
|
|
usesWASD
|
|
} = getMovementConfiguration(usedKeys);
|
|
|
|
if (usesWASD) {
|
|
alreadyUsedKeys.add("w");
|
|
alreadyUsedKeys.add("a");
|
|
alreadyUsedKeys.add("s");
|
|
alreadyUsedKeys.add("d");
|
|
}
|
|
|
|
const possiblePauseKeys = [// Restart keys, pause keys, other potentially dangerous keys
|
|
"p", "q", "r"];
|
|
const possibleActionKeys = [" ", "Enter", "e", "f", "z", "x", "c", ...Array.from(usedKeys).filter(i => i.length === 1 && !possiblePauseKeys.includes(i))];
|
|
|
|
const findKey = keys => {
|
|
for (const key of keys) {
|
|
if (usedKeys.has(key) && !alreadyUsedKeys.has(key)) {
|
|
alreadyUsedKeys.add(key);
|
|
return key;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
const getPrimaryAction = () => {
|
|
if (usesArrows && usedKeys.has("ArrowUp")) {
|
|
return "ArrowUp";
|
|
}
|
|
|
|
if (usesWASD && usedKeys.has("w")) {
|
|
return "w";
|
|
}
|
|
|
|
return findKey(possibleActionKeys);
|
|
};
|
|
|
|
const getSecondaryAction = () => findKey(possibleActionKeys);
|
|
|
|
const getPauseKey = () => findKey(possiblePauseKeys);
|
|
|
|
const getUp = () => {
|
|
if (usesArrows || !usesWASD) return "ArrowUp";
|
|
return "w";
|
|
};
|
|
|
|
const getDown = () => {
|
|
if (usesArrows || !usesWASD) return "ArrowDown";
|
|
return "s";
|
|
};
|
|
|
|
const getRight = () => {
|
|
if (usesArrows || !usesWASD) return "ArrowRight";
|
|
return "d";
|
|
};
|
|
|
|
const getLeft = () => {
|
|
if (usesArrows || !usesWASD) return "ArrowLeft";
|
|
return "a";
|
|
};
|
|
|
|
const action1 = getPrimaryAction();
|
|
let action2 = getSecondaryAction();
|
|
let action3 = getSecondaryAction();
|
|
let action4 = getSecondaryAction(); // When only 1 or 2 action keys are detected, bind the other buttons to the same things.
|
|
|
|
if (action1 && !action2 && !action3 && !action4) {
|
|
action2 = action1;
|
|
action3 = action1;
|
|
action4 = action1;
|
|
}
|
|
|
|
if (action1 && action2 && !action3 && !action4) {
|
|
action3 = action1;
|
|
action4 = action2;
|
|
} // Set indices "manually" because we don't evaluate them in order.
|
|
|
|
|
|
buttons = [];
|
|
buttons[0] = {
|
|
/*
|
|
Xbox: A
|
|
SNES-like: B
|
|
*/
|
|
type: "key",
|
|
high: action1
|
|
};
|
|
buttons[1] = {
|
|
/*
|
|
Xbox: B
|
|
SNES-like: A
|
|
*/
|
|
type: "key",
|
|
high: action2
|
|
};
|
|
buttons[2] = {
|
|
/*
|
|
Xbox: X
|
|
SNES-like: Y
|
|
*/
|
|
type: "key",
|
|
high: action3
|
|
};
|
|
buttons[3] = {
|
|
/*
|
|
Xbox: Y
|
|
SNES-like: X
|
|
*/
|
|
type: "key",
|
|
high: action4
|
|
};
|
|
buttons[4] = {
|
|
/*
|
|
Xbox: LB
|
|
SNES-like: Left trigger
|
|
*/
|
|
type: "mousedown"
|
|
};
|
|
buttons[5] = {
|
|
/*
|
|
Xbox: RB
|
|
*/
|
|
type: "mousedown"
|
|
};
|
|
buttons[6] = {
|
|
/*
|
|
Xbox: LT
|
|
*/
|
|
type: "mousedown"
|
|
};
|
|
buttons[7] = {
|
|
/*
|
|
Xbox: RT
|
|
SNES-like: Right trigger
|
|
*/
|
|
type: "mousedown"
|
|
};
|
|
buttons[9] = {
|
|
/*
|
|
Xbox: Menu
|
|
SNES-like: Start
|
|
*/
|
|
type: "key",
|
|
high: getPauseKey()
|
|
};
|
|
buttons[8] = {
|
|
/*
|
|
Xbox: Change view
|
|
SNES-like: Select
|
|
*/
|
|
type: "key",
|
|
high: getPauseKey()
|
|
}; // Xbox: Left analog press
|
|
|
|
buttons[10] = emptyMapping(); // Xbox: Right analog press
|
|
|
|
buttons[11] = emptyMapping();
|
|
buttons[12] = {
|
|
/*
|
|
Xbox: D-pad up
|
|
*/
|
|
type: "key",
|
|
high: getUp()
|
|
};
|
|
buttons[13] = {
|
|
/*
|
|
Xbox: D-pad down
|
|
*/
|
|
type: "key",
|
|
high: getDown()
|
|
};
|
|
buttons[14] = {
|
|
/*
|
|
Xbox: D-pad left
|
|
*/
|
|
type: "key",
|
|
high: getLeft()
|
|
};
|
|
buttons[15] = {
|
|
/*
|
|
Xbox: D-pad right
|
|
*/
|
|
type: "key",
|
|
high: getRight()
|
|
};
|
|
}
|
|
|
|
return padWithEmptyMappings(buttons, this.gamepad.buttons.length);
|
|
}
|
|
|
|
getDefaultAxisMappings() {
|
|
let axes = [];
|
|
|
|
if (this.hints.importedSettings) {
|
|
axes = this.hints.importedSettings.axes;
|
|
} else {
|
|
// Only return default axis mappings when there are 4 axes, like an xbox controller
|
|
// If there isn't exactly 4, we can't really predict what the axes mean
|
|
// Some controllers map the dpad to *both* buttons and axes at the same time, which would cause conflicts.
|
|
if (this.gamepad.axes.length === 4) {
|
|
const usedKeys = this.hints.usedKeys;
|
|
const {
|
|
usesArrows,
|
|
usesWASD
|
|
} = getMovementConfiguration(usedKeys);
|
|
|
|
if (usesWASD) {
|
|
axes.push(defaultAxesMappings.wasd[0]);
|
|
axes.push(defaultAxesMappings.wasd[1]);
|
|
} else if (usesArrows) {
|
|
axes.push(defaultAxesMappings.arrows[0]);
|
|
axes.push(defaultAxesMappings.arrows[1]);
|
|
} else {
|
|
axes.push(defaultAxesMappings.cursor[0]);
|
|
axes.push(defaultAxesMappings.cursor[1]);
|
|
}
|
|
|
|
axes.push(defaultAxesMappings.cursor[0]);
|
|
axes.push(defaultAxesMappings.cursor[1]);
|
|
}
|
|
}
|
|
|
|
return padWithEmptyMappings(axes, this.gamepad.axes.length);
|
|
}
|
|
|
|
}
|
|
|
|
const defaultHints = () => ({
|
|
usedKeys: new Set(),
|
|
importedSettings: null,
|
|
generated: false
|
|
});
|
|
|
|
class GamepadLib extends _event_target_js__WEBPACK_IMPORTED_MODULE_0__["default"] {
|
|
constructor() {
|
|
super();
|
|
/** @type {Map<string, GamepadData>} */
|
|
|
|
this.gamepads = new Map();
|
|
this.handleConnect = this.handleConnect.bind(this);
|
|
this.handleDisconnect = this.handleDisconnect.bind(this);
|
|
this.update = this.update.bind(this);
|
|
this.animationFrame = null;
|
|
this.currentTime = null;
|
|
this.deltaTime = 0;
|
|
this.virtualCursor = {
|
|
x: 0,
|
|
y: 0,
|
|
maxX: Infinity,
|
|
minX: -Infinity,
|
|
maxY: Infinity,
|
|
minY: -Infinity,
|
|
modified: false
|
|
};
|
|
this._editor = null;
|
|
this.connectCallbacks = [];
|
|
this.keysPressedThisFrame = new Set();
|
|
this.oldKeysPressed = new Set();
|
|
this.mouseButtonsPressedThisFrame = new Set();
|
|
this.oldMouseDown = new Set();
|
|
this.addEventHandlers();
|
|
}
|
|
|
|
addEventHandlers() {
|
|
window.addEventListener("gamepadconnected", this.handleConnect);
|
|
window.addEventListener("gamepaddisconnected", this.handleDisconnect);
|
|
}
|
|
|
|
removeEventHandlers() {
|
|
window.removeEventListener("gamepadconnected", this.handleConnect);
|
|
window.removeEventListener("gamepaddisconnected", this.handleDisconnect);
|
|
}
|
|
|
|
gamepadConnected() {
|
|
if (this.gamepads.size > 0) {
|
|
return Promise.resolve();
|
|
}
|
|
|
|
return new Promise(resolve => {
|
|
this.connectCallbacks.push(resolve);
|
|
});
|
|
}
|
|
|
|
getHints() {
|
|
return Object.assign(defaultHints(), this.getUserHints());
|
|
}
|
|
|
|
getUserHints() {
|
|
// to be overridden by users
|
|
return {};
|
|
}
|
|
|
|
resetControls() {
|
|
for (const gamepad of this.gamepads.values()) {
|
|
gamepad.resetMappings();
|
|
}
|
|
}
|
|
|
|
clearControls() {
|
|
for (const gamepad of this.gamepads.values()) {
|
|
gamepad.clearMappings();
|
|
}
|
|
}
|
|
|
|
handleConnect(e) {
|
|
for (const callback of this.connectCallbacks) {
|
|
callback();
|
|
}
|
|
|
|
this.connectCallbacks = [];
|
|
const gamepad = e.gamepad;
|
|
const id = getGamepadId(gamepad);
|
|
console.log("connected", gamepad);
|
|
const gamepadData = new GamepadData(gamepad, this);
|
|
this.gamepads.set(id, gamepadData);
|
|
|
|
if (this.animationFrame === null) {
|
|
this.animationFrame = requestAnimationFrame(this.update);
|
|
}
|
|
|
|
this.dispatchEvent(new CustomEvent("gamepadconnected", {
|
|
detail: gamepadData
|
|
}));
|
|
}
|
|
|
|
handleDisconnect(e) {
|
|
const gamepad = e.gamepad;
|
|
const id = getGamepadId(gamepad);
|
|
console.log("disconnected", gamepad);
|
|
const gamepadData = this.gamepads.get(id);
|
|
this.gamepads.delete(id);
|
|
this.dispatchEvent(new CustomEvent("gamepaddisconnected", {
|
|
detail: gamepadData
|
|
}));
|
|
|
|
if (this.gamepads.size === 0) {
|
|
cancelAnimationFrame(this.animationFrame);
|
|
this.animationFrame = null;
|
|
this.currentTime = null;
|
|
}
|
|
}
|
|
|
|
dispatchKey(key, pressed) {
|
|
if (pressed) {
|
|
this.dispatchEvent(new CustomEvent("keydown", {
|
|
detail: key
|
|
}));
|
|
} else {
|
|
this.dispatchEvent(new CustomEvent("keyup", {
|
|
detail: key
|
|
}));
|
|
}
|
|
}
|
|
|
|
dispatchMouse(button, down) {
|
|
if (down) {
|
|
this.dispatchEvent(new CustomEvent("mousedown", {
|
|
detail: button
|
|
}));
|
|
} else {
|
|
this.dispatchEvent(new CustomEvent("mouseup", {
|
|
detail: button
|
|
}));
|
|
}
|
|
}
|
|
|
|
dispatchMouseMove(x, y) {
|
|
this.dispatchEvent(new CustomEvent("mousemove", {
|
|
detail: {
|
|
x,
|
|
y
|
|
}
|
|
}));
|
|
}
|
|
|
|
updateButton(value, mapping) {
|
|
if (mapping.type === "key") {
|
|
if (value >= mapping.deadZone) {
|
|
if (mapping.high) {
|
|
this.keysPressedThisFrame.add(mapping.high);
|
|
}
|
|
} else if (value <= -mapping.deadZone) {
|
|
if (mapping.low) {
|
|
this.keysPressedThisFrame.add(mapping.low);
|
|
}
|
|
}
|
|
} else if (mapping.type === "mousedown") {
|
|
const isDown = Math.abs(value) >= mapping.deadZone;
|
|
|
|
if (isDown) {
|
|
this.mouseButtonsPressedThisFrame.add(mapping.button);
|
|
}
|
|
} else if (mapping.type === "virtual_cursor") {
|
|
const deadZone = mapping.deadZone;
|
|
let action;
|
|
if (value >= deadZone) action = mapping.high;
|
|
if (value <= -deadZone) action = mapping.low;
|
|
|
|
if (action) {
|
|
// an axis value just beyond the deadzone should have a multiplier near 0, a high value should have a multiplier of 1
|
|
const multiplier = (Math.abs(value) - deadZone) / (1 - deadZone);
|
|
const speed = multiplier * multiplier * mapping.sensitivity * this.deltaTime;
|
|
|
|
if (action === "+x") {
|
|
this.virtualCursor.x += speed;
|
|
} else if (action === "-x") {
|
|
this.virtualCursor.x -= speed;
|
|
} else if (action === "+y") {
|
|
this.virtualCursor.y += speed;
|
|
} else if (action === "-y") {
|
|
this.virtualCursor.y -= speed;
|
|
}
|
|
|
|
this.virtualCursor.modified = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
update(time) {
|
|
this.oldKeysPressed = this.keysPressedThisFrame;
|
|
this.oldMouseButtonsPressed = this.mouseButtonsPressedThisFrame;
|
|
this.keysPressedThisFrame = new Set();
|
|
this.mouseButtonsPressedThisFrame = new Set();
|
|
|
|
if (this.currentTime === null) {
|
|
this.deltaTime = 0; // doesn't matter what this is, it's just the first frame
|
|
} else {
|
|
this.deltaTime = time - this.currentTime;
|
|
}
|
|
|
|
this.deltaTime = Math.max(Math.min(this.deltaTime, 1000), 0);
|
|
this.currentTime = time;
|
|
this.animationFrame = requestAnimationFrame(this.update);
|
|
const gamepads = navigator.getGamepads();
|
|
|
|
for (const gamepad of gamepads) {
|
|
if (gamepad === null) {
|
|
continue;
|
|
}
|
|
|
|
const id = getGamepadId(gamepad);
|
|
const data = this.gamepads.get(id);
|
|
|
|
for (let i = 0; i < gamepad.buttons.length; i++) {
|
|
const button = gamepad.buttons[i];
|
|
const value = button.value;
|
|
const mapping = data.buttonMappings[i];
|
|
this.updateButton(value, mapping);
|
|
}
|
|
|
|
for (let i = 0; i < gamepad.axes.length; i++) {
|
|
const axis = gamepad.axes[i];
|
|
const mapping = data.axesMappings[i];
|
|
this.updateButton(axis, mapping);
|
|
}
|
|
}
|
|
|
|
if (this._editor) {
|
|
this._editor.update(gamepads);
|
|
}
|
|
|
|
for (const key of this.keysPressedThisFrame) {
|
|
if (!this.oldKeysPressed.has(key)) {
|
|
this.dispatchKey(key, true);
|
|
}
|
|
}
|
|
|
|
for (const key of this.oldKeysPressed) {
|
|
if (!this.keysPressedThisFrame.has(key)) {
|
|
this.dispatchKey(key, false);
|
|
}
|
|
}
|
|
|
|
for (const button of this.mouseButtonsPressedThisFrame) {
|
|
if (!this.oldMouseButtonsPressed.has(button)) {
|
|
this.dispatchMouse(button, true);
|
|
}
|
|
}
|
|
|
|
for (const button of this.oldMouseButtonsPressed) {
|
|
if (!this.mouseButtonsPressedThisFrame.has(button)) {
|
|
this.dispatchMouse(button, false);
|
|
}
|
|
}
|
|
|
|
if (this.virtualCursor.modified) {
|
|
this.virtualCursor.modified = false;
|
|
|
|
if (this.virtualCursor.x > this.virtualCursor.maxX) {
|
|
this.virtualCursor.x = this.virtualCursor.maxX;
|
|
}
|
|
|
|
if (this.virtualCursor.x < this.virtualCursor.minX) {
|
|
this.virtualCursor.x = this.virtualCursor.minX;
|
|
}
|
|
|
|
if (this.virtualCursor.y > this.virtualCursor.maxY) {
|
|
this.virtualCursor.y = this.virtualCursor.maxY;
|
|
}
|
|
|
|
if (this.virtualCursor.y < this.virtualCursor.minY) {
|
|
this.virtualCursor.y = this.virtualCursor.minY;
|
|
}
|
|
|
|
this.dispatchMouseMove(this.virtualCursor.x, this.virtualCursor.y);
|
|
}
|
|
}
|
|
|
|
editor() {
|
|
if (!this._editor) {
|
|
this._editor = new GamepadEditor(this);
|
|
}
|
|
|
|
return this._editor;
|
|
}
|
|
|
|
}
|
|
|
|
GamepadLib.browserHasBrokenGamepadAPI = () => {
|
|
// Check that the gamepad API is supported at all
|
|
if (!navigator.getGamepads) {
|
|
return true;
|
|
} // Firefox on Linux has a broken gamepad API implementation that results in strange and sometimes unusable mappings
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=1643358
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=1643835
|
|
|
|
|
|
if (navigator.userAgent.includes("Firefox") && navigator.userAgent.includes("Linux")) {
|
|
return true;
|
|
} // Firefox on macOS has other bugs that result in strange and unusable mappings
|
|
// eg. https://bugzilla.mozilla.org/show_bug.cgi?id=1434408
|
|
|
|
|
|
if (navigator.userAgent.includes("Firefox") && navigator.userAgent.includes("Mac OS")) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
GamepadLib.setConsole = n => console = n;
|
|
|
|
const removeAllChildren = el => {
|
|
while (el.firstChild) {
|
|
el.removeChild(el.firstChild);
|
|
}
|
|
};
|
|
|
|
const buttonHtmlId = index => "gamepadlib-button-".concat(index);
|
|
|
|
const axisHtmlId = n => "gamepadlib-axis-".concat(n);
|
|
|
|
class GamepadEditor extends _event_target_js__WEBPACK_IMPORTED_MODULE_0__["default"] {
|
|
constructor(gamepadLib) {
|
|
super();
|
|
/** @type {GamepadLib} */
|
|
|
|
this.gamepadLib = gamepadLib;
|
|
this.root = Object.assign(document.createElement("div"), {
|
|
className: "gamepadlib-root"
|
|
});
|
|
this.selector = Object.assign(document.createElement("select"), {
|
|
className: "gamepadlib-selector"
|
|
});
|
|
this.content = Object.assign(document.createElement("div"), {
|
|
className: "gamepadlib-content"
|
|
});
|
|
this.root.appendChild(this.selector);
|
|
this.root.appendChild(this.content);
|
|
this.onSelectorChange = this.onSelectorChange.bind(this);
|
|
this.onGamepadsChange = this.onGamepadsChange.bind(this);
|
|
this.selector.addEventListener("change", this.onSelectorChange);
|
|
this.gamepadLib.addEventListener("gamepadconnected", this.onGamepadsChange);
|
|
this.gamepadLib.addEventListener("gamepaddisconnected", this.onGamepadsChange);
|
|
this.buttonIdToElement = new Map();
|
|
this.axisIdToElement = new Map();
|
|
this.hidden = false; // should be overridden later
|
|
|
|
this.msg = (id, opts) => id;
|
|
}
|
|
|
|
onSelectorChange() {
|
|
this.updateContent();
|
|
this.dispatchEvent(new CustomEvent("gamepad-changed"));
|
|
}
|
|
|
|
onGamepadsChange() {
|
|
this.updateAllContent();
|
|
this.dispatchEvent(new CustomEvent("gamepad-changed"));
|
|
}
|
|
|
|
updateAllContent() {
|
|
this.updateDropdown();
|
|
this.updateContent();
|
|
this.focus();
|
|
}
|
|
|
|
updateDropdown() {
|
|
removeAllChildren(this.selector);
|
|
const gamepads = Array.from(this.gamepadLib.gamepads.entries());
|
|
|
|
if (gamepads.length === 0) {
|
|
this.selector.hidden = true;
|
|
return;
|
|
}
|
|
|
|
this.selector.hidden = false;
|
|
|
|
for (const [id, _] of gamepads) {
|
|
const option = document.createElement("option");
|
|
option.textContent = id;
|
|
option.value = id;
|
|
this.selector.appendChild(option);
|
|
}
|
|
}
|
|
|
|
keyToString(key) {
|
|
if (key === " ") return this.msg("key-space");
|
|
if (key === "ArrowUp") return this.msg("key-up");
|
|
if (key === "ArrowDown") return this.msg("key-down");
|
|
if (key === "ArrowLeft") return this.msg("key-left");
|
|
if (key === "ArrowRight") return this.msg("key-right");
|
|
if (key === "Enter") return this.msg("key-enter");
|
|
|
|
if (key.length === 1) {
|
|
return key.toUpperCase();
|
|
} // Convert eg. "PageUp" -> "Page Up"
|
|
|
|
|
|
return key.replace(/[a-z]([A-Z])/, n => "".concat(n[0], " ").concat(n[1]));
|
|
}
|
|
|
|
createButtonMapping(mappingList, index, {
|
|
property = "high",
|
|
allowClick = true
|
|
} = {}) {
|
|
const input = document.createElement("input");
|
|
input.readOnly = true;
|
|
input.className = "gamepadlib-keyinput";
|
|
input.title = this.msg("keyinput-title");
|
|
input.dataset.index = index;
|
|
|
|
const update = () => {
|
|
const mapping = mappingList[index];
|
|
input.dataset.empty = false;
|
|
|
|
if (mapping.type === "key") {
|
|
if (mapping[property] === null) {
|
|
input.value = this.msg("key-none");
|
|
input.dataset.empty = true;
|
|
} else {
|
|
input.value = this.keyToString(mapping[property]);
|
|
}
|
|
} else if (mapping.type === "mousedown") {
|
|
let value = this.msg("key-click");
|
|
|
|
if (mapping.button !== 0) {
|
|
value += " (".concat(mapping.button, ")");
|
|
}
|
|
|
|
input.value = value;
|
|
} else {
|
|
// should never happen
|
|
input.value = "??? ".concat(mapping.type);
|
|
}
|
|
};
|
|
|
|
const changedMapping = () => {
|
|
mappingList[index] = transformAndCopyMapping(mappingList[index]);
|
|
isAcceptingInput = false;
|
|
input.blur();
|
|
update();
|
|
input.dispatchEvent(new CustomEvent("mapping-changed"));
|
|
this.changed();
|
|
};
|
|
|
|
let isAcceptingInput = false;
|
|
|
|
const handleClick = e => {
|
|
e.preventDefault();
|
|
|
|
if (isAcceptingInput) {
|
|
if (allowClick) {
|
|
const mapping = mappingList[index];
|
|
mapping.type = "mousedown";
|
|
mapping.button = e.button;
|
|
changedMapping();
|
|
} else {
|
|
handleBlur();
|
|
}
|
|
} else {
|
|
input.value = "...";
|
|
input.dataset.acceptingInput = true;
|
|
isAcceptingInput = true;
|
|
}
|
|
};
|
|
|
|
const handleKeyEvent = e => {
|
|
if (isAcceptingInput) {
|
|
e.preventDefault();
|
|
const key = e.key; // TW: We allow binding to control and shift
|
|
|
|
if (["Alt"].includes(key)) {
|
|
return;
|
|
}
|
|
|
|
const mapping = mappingList[index];
|
|
const KEYS = ["ArrowUp", "ArrowDown", "ArrowRight", "ArrowLeft", "Enter", // TW: We support more keys
|
|
// "Backspace",
|
|
// "Delete",
|
|
"Shift", "CapsLock", "ScrollLock", "Control", // "Escape",
|
|
"Insert", "Home", "End", "PageUp", "PageDown"];
|
|
|
|
if (key.length === 1 || KEYS.includes(key)) {
|
|
mapping.type = "key";
|
|
mapping[property] = key;
|
|
} else if (key !== "Escape") {
|
|
mapping.type = "key";
|
|
mapping[property] = null;
|
|
}
|
|
|
|
changedMapping();
|
|
} else if (e.key === "Enter") {
|
|
e.preventDefault();
|
|
e.target.click();
|
|
}
|
|
};
|
|
|
|
const MODIFIER_KEYS = ["Shift", "Control"];
|
|
|
|
const handleKeyDown = e => {
|
|
if (!MODIFIER_KEYS.includes(e.key)) handleKeyEvent(e);
|
|
};
|
|
|
|
const handleKeyUp = e => {
|
|
if (MODIFIER_KEYS.includes(e.key)) handleKeyEvent(e);
|
|
};
|
|
|
|
const handleBlur = () => {
|
|
input.dataset.acceptingInput = false;
|
|
|
|
if (isAcceptingInput) {
|
|
isAcceptingInput = false;
|
|
update();
|
|
}
|
|
};
|
|
|
|
input.addEventListener("contextmenu", e => {
|
|
e.preventDefault();
|
|
});
|
|
input.addEventListener("mouseup", handleClick);
|
|
input.addEventListener("keydown", handleKeyDown);
|
|
input.addEventListener("keyup", handleKeyUp);
|
|
input.addEventListener("blur", handleBlur);
|
|
update();
|
|
return input;
|
|
}
|
|
|
|
createAxisMapping(mappingList, index) {
|
|
const selector = document.createElement("select");
|
|
selector.className = "gamepadlib-axis-mapping";
|
|
selector.id = axisHtmlId(index);
|
|
selector.appendChild(Object.assign(document.createElement("option"), {
|
|
textContent: this.msg("axis-none"),
|
|
value: "none"
|
|
}));
|
|
selector.appendChild(Object.assign(document.createElement("option"), {
|
|
textContent: this.msg("axis-cursor"),
|
|
value: "cursor"
|
|
}));
|
|
selector.appendChild(Object.assign(document.createElement("option"), {
|
|
// doesn't really make sense to translate
|
|
textContent: "WASD",
|
|
value: "wasd"
|
|
}));
|
|
selector.appendChild(Object.assign(document.createElement("option"), {
|
|
textContent: this.msg("axis-arrows"),
|
|
value: "arrows"
|
|
}));
|
|
selector.appendChild(Object.assign(document.createElement("option"), {
|
|
textContent: this.msg("axis-custom"),
|
|
value: "custom"
|
|
}));
|
|
|
|
const updateDropdownValue = () => {
|
|
if (mappingList[index].type === "key" || mappingList[index].type === "mousedown") {
|
|
if (mappingList[index].high === null && mappingList[index].low === null && mappingList[index + 1].high === null && mappingList[index + 1].low === null) {
|
|
selector.value = "none";
|
|
} else if (mappingList[index].high === defaultAxesMappings.wasd[0].high && mappingList[index].low === defaultAxesMappings.wasd[0].low && mappingList[index + 1].high === defaultAxesMappings.wasd[1].high && mappingList[index + 1].low === defaultAxesMappings.wasd[1].low) {
|
|
selector.value = "wasd";
|
|
} else if (mappingList[index].high === defaultAxesMappings.arrows[0].high && mappingList[index].low === defaultAxesMappings.arrows[0].low && mappingList[index + 1].high === defaultAxesMappings.arrows[1].high && mappingList[index + 1].low === defaultAxesMappings.arrows[1].low) {
|
|
selector.value = "arrows";
|
|
} else {
|
|
selector.value = "custom";
|
|
}
|
|
} else if (mappingList[index].type === "virtual_cursor") {
|
|
selector.value = "cursor";
|
|
} else {
|
|
// should never happen
|
|
selector.value = "none";
|
|
}
|
|
};
|
|
|
|
updateDropdownValue();
|
|
const circleOverlay = document.createElement("div");
|
|
circleOverlay.className = "gamepadlib-axis-circle-overlay";
|
|
|
|
const updateOverlay = () => {
|
|
removeAllChildren(circleOverlay);
|
|
|
|
if (mappingList[index].type === "key") {
|
|
const buttons = [this.createButtonMapping(mappingList, index + 1, {
|
|
property: "low",
|
|
allowClick: false
|
|
}), this.createButtonMapping(mappingList, index, {
|
|
property: "low",
|
|
allowClick: false
|
|
}), this.createButtonMapping(mappingList, index, {
|
|
property: "high",
|
|
allowClick: false
|
|
}), this.createButtonMapping(mappingList, index + 1, {
|
|
property: "high",
|
|
allowClick: false
|
|
})];
|
|
|
|
for (const button of buttons) {
|
|
button.classList.add("gamepadlib-axis-mapper");
|
|
button.addEventListener("mapping-changed", updateDropdownValue);
|
|
circleOverlay.appendChild(button);
|
|
}
|
|
}
|
|
};
|
|
|
|
updateOverlay();
|
|
selector.addEventListener("change", () => {
|
|
if (selector.value === "custom") {
|
|
// If key mappings already exist, leave them as-is
|
|
if (mappingList[index].type !== "key") {
|
|
mappingList[index] = transformAndCopyMapping(defaultAxesMappings.arrows[0]);
|
|
mappingList[index + 1] = transformAndCopyMapping(defaultAxesMappings.arrows[1]);
|
|
}
|
|
} else if (selector.value === "arrows") {
|
|
mappingList[index] = transformAndCopyMapping(defaultAxesMappings.arrows[0]);
|
|
mappingList[index + 1] = transformAndCopyMapping(defaultAxesMappings.arrows[1]);
|
|
} else if (selector.value === "wasd") {
|
|
mappingList[index] = transformAndCopyMapping(defaultAxesMappings.wasd[0]);
|
|
mappingList[index + 1] = transformAndCopyMapping(defaultAxesMappings.wasd[1]);
|
|
} else if (selector.value === "cursor") {
|
|
mappingList[index] = transformAndCopyMapping(defaultAxesMappings.cursor[0]);
|
|
mappingList[index + 1] = transformAndCopyMapping(defaultAxesMappings.cursor[1]);
|
|
} else {
|
|
mappingList[index] = transformAndCopyMapping(emptyMapping());
|
|
mappingList[index + 1] = transformAndCopyMapping(emptyMapping());
|
|
}
|
|
|
|
updateOverlay();
|
|
this.changed();
|
|
});
|
|
return {
|
|
circleOverlay,
|
|
selector
|
|
};
|
|
}
|
|
|
|
hasControllerSelected() {
|
|
return !!this.selector.value;
|
|
}
|
|
|
|
updateContent() {
|
|
removeAllChildren(this.content);
|
|
|
|
if (this.hidden) {
|
|
return;
|
|
}
|
|
|
|
const selectedId = this.selector.value;
|
|
|
|
if (!selectedId) {
|
|
const message = document.createElement("div");
|
|
message.textContent = this.msg("no-controllers");
|
|
this.content.appendChild(message);
|
|
return;
|
|
}
|
|
|
|
const gamepadData = this.gamepadLib.gamepads.get(selectedId);
|
|
|
|
if (!gamepadData) {
|
|
// Users should never be able to see this
|
|
const message = document.createElement("div");
|
|
message.textContent = "Cannot find controller: ".concat(selectedId);
|
|
this.content.appendChild(message);
|
|
return;
|
|
}
|
|
|
|
this.buttonIdToElement.clear();
|
|
this.axisIdToElement.clear();
|
|
const mappingsContainer = document.createElement("div");
|
|
mappingsContainer.className = "gamepadlib-content-buttons";
|
|
const buttonMappings = gamepadData.buttonMappings;
|
|
|
|
for (let i = 0; i < buttonMappings.length; i++) {
|
|
const container = document.createElement("div");
|
|
container.className = "gamepadlib-mapping";
|
|
container.dataset.id = i;
|
|
const label = document.createElement("label");
|
|
label.className = "gamepadlib-mapping-label";
|
|
label.textContent = this.msg("button-n", {
|
|
n: i
|
|
});
|
|
const id = buttonHtmlId(i);
|
|
label.htmlFor = id;
|
|
const options = document.createElement("div");
|
|
options.className = "gamepadlib-mapping-options";
|
|
const mappingInput = this.createButtonMapping(buttonMappings, i);
|
|
mappingInput.id = id;
|
|
options.appendChild(mappingInput);
|
|
container.appendChild(label);
|
|
container.appendChild(options);
|
|
mappingsContainer.appendChild(container);
|
|
this.buttonIdToElement.set(i, container);
|
|
}
|
|
|
|
const axesContainer = document.createElement("div");
|
|
axesContainer.className = "gamepadlib-content-axes";
|
|
const axesMappings = gamepadData.axesMappings;
|
|
|
|
for (let i = 0; i < axesMappings.length; i += 2) {
|
|
const container = document.createElement("div");
|
|
container.className = "gamepadlib-axis";
|
|
const label = document.createElement("label");
|
|
label.textContent = this.msg("axes-a-b", {
|
|
a: i,
|
|
b: i + 1
|
|
});
|
|
label.htmlFor = axisHtmlId(i);
|
|
const circle = document.createElement("div");
|
|
circle.className = "gamepadlib-axis-circle";
|
|
const {
|
|
circleOverlay,
|
|
selector
|
|
} = this.createAxisMapping(axesMappings, i);
|
|
circle.appendChild(circleOverlay);
|
|
const dot = document.createElement("div");
|
|
dot.className = "gamepadlib-axis-dot";
|
|
circle.appendChild(dot);
|
|
container.appendChild(label);
|
|
container.appendChild(circle);
|
|
container.appendChild(selector);
|
|
axesContainer.appendChild(container);
|
|
this.axisIdToElement.set(i, dot);
|
|
}
|
|
|
|
this.content.appendChild(mappingsContainer);
|
|
this.content.appendChild(axesContainer);
|
|
}
|
|
|
|
update(gamepads) {
|
|
if (this.hidden) {
|
|
return;
|
|
}
|
|
|
|
const selectedId = this.selector.value;
|
|
|
|
if (!selectedId) {
|
|
return;
|
|
}
|
|
|
|
const gamepad = Array.from(gamepads).find(i => i && getGamepadId(i) === this.selector.value);
|
|
|
|
if (!gamepad) {
|
|
return;
|
|
}
|
|
|
|
for (let i = 0; i < gamepad.buttons.length; i++) {
|
|
const element = this.buttonIdToElement.get(i);
|
|
|
|
if (element) {
|
|
const button = gamepad.buttons[i];
|
|
const value = button.value.toString();
|
|
|
|
if (value !== element.dataset.value) {
|
|
element.dataset.value = value;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (let i = 0; i < gamepad.axes.length; i += 2) {
|
|
const element = this.axisIdToElement.get(i);
|
|
|
|
if (element) {
|
|
const x = gamepad.axes[i];
|
|
const y = gamepad.axes[i + 1] || 0;
|
|
const size = 150 / 2;
|
|
element.style.transform = "translate(-50%, -50%) translate(".concat(x * size, "px, ").concat(y * size, "px)");
|
|
}
|
|
}
|
|
}
|
|
|
|
export() {
|
|
const selectedId = this.selector.value;
|
|
|
|
if (!selectedId) {
|
|
return null;
|
|
}
|
|
|
|
const gamepadData = this.gamepadLib.gamepads.get(selectedId);
|
|
|
|
if (!gamepadData) {
|
|
return null;
|
|
}
|
|
|
|
return {
|
|
axes: gamepadData.axesMappings.map(prepareAxisMappingForExport),
|
|
buttons: gamepadData.buttonMappings.map(prepareButtonMappingForExport)
|
|
};
|
|
}
|
|
|
|
changed() {
|
|
this.dispatchEvent(new CustomEvent("mapping-changed"));
|
|
}
|
|
|
|
hide() {
|
|
this.hidden = true;
|
|
this.updateContent();
|
|
}
|
|
|
|
focus() {
|
|
if (this.selector.value) {
|
|
this.selector.focus();
|
|
}
|
|
}
|
|
|
|
generateEditor() {
|
|
this.hidden = false;
|
|
this.updateAllContent();
|
|
return this.root;
|
|
}
|
|
|
|
}
|
|
|
|
/* harmony default export */ __webpack_exports__["default"] = (GamepadLib);
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./src/addons/addons/gamepad/userscript.js":
|
|
/*!*************************************************!*\
|
|
!*** ./src/addons/addons/gamepad/userscript.js ***!
|
|
\*************************************************/
|
|
/*! exports provided: default */
|
|
/***/ (function(module, __webpack_exports__, __webpack_require__) {
|
|
|
|
"use strict";
|
|
__webpack_require__.r(__webpack_exports__);
|
|
/* harmony import */ var _gamepadlib_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./gamepadlib.js */ "./src/addons/addons/gamepad/gamepadlib.js");
|
|
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) { symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); } keys.push.apply(keys, symbols); } return keys; }
|
|
|
|
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
|
|
|
|
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
|
|
|
|
|
|
/* harmony default export */ __webpack_exports__["default"] = (async function ({
|
|
addon,
|
|
console,
|
|
msg
|
|
}) {
|
|
const vm = addon.tab.traps.vm; // Wait for the project to finish loading. Renderer and scripts will not be fully available until this happens.
|
|
|
|
await new Promise(resolve => {
|
|
if (vm.editingTarget) return resolve();
|
|
vm.runtime.once("PROJECT_LOADED", resolve);
|
|
});
|
|
|
|
const vmStarted = () => vm.runtime._steppingInterval !== null;
|
|
|
|
const scratchKeyToKey = key => {
|
|
switch (key) {
|
|
case "right arrow":
|
|
return "ArrowRight";
|
|
|
|
case "up arrow":
|
|
return "ArrowUp";
|
|
|
|
case "left arrow":
|
|
return "ArrowLeft";
|
|
|
|
case "down arrow":
|
|
return "ArrowDown";
|
|
|
|
case "enter":
|
|
return "Enter";
|
|
|
|
case "space":
|
|
return " ";
|
|
}
|
|
|
|
return key.toLowerCase().charAt(0);
|
|
};
|
|
|
|
const getKeysUsedByProject = () => {
|
|
const allBlocks = [vm.runtime.getTargetForStage(), ...vm.runtime.targets].filter(i => i.isOriginal).map(i => i.blocks);
|
|
const result = new Set();
|
|
|
|
for (const blocks of allBlocks) {
|
|
for (const block of Object.values(blocks._blocks)) {
|
|
if (block.opcode === "event_whenkeypressed" || block.opcode === "sensing_keyoptions") {
|
|
// For blocks like "key (my variable) pressed?", the sensing_keyoptions still exists but has a null parent.
|
|
if (block.opcode === "sensing_keyoptions" && !block.parent) {
|
|
continue;
|
|
}
|
|
|
|
const key = block.fields.KEY_OPTION.value;
|
|
result.add(scratchKeyToKey(key));
|
|
}
|
|
}
|
|
}
|
|
|
|
return result;
|
|
};
|
|
|
|
const GAMEPAD_CONFIG_MAGIC = " // _gamepad_";
|
|
|
|
const findOptionsComment = () => {
|
|
const target = vm.runtime.getTargetForStage();
|
|
const comments = target.comments;
|
|
|
|
for (const comment of Object.values(comments)) {
|
|
if (comment.text.includes(GAMEPAD_CONFIG_MAGIC)) {
|
|
return comment;
|
|
}
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
const parseOptionsComment = () => {
|
|
const comment = findOptionsComment();
|
|
|
|
if (!comment) {
|
|
return null;
|
|
}
|
|
|
|
const lineWithMagic = comment.text.split("\n").find(i => i.endsWith(GAMEPAD_CONFIG_MAGIC));
|
|
|
|
if (!lineWithMagic) {
|
|
console.warn("Gamepad comment does not contain valid line");
|
|
return null;
|
|
}
|
|
|
|
const jsonText = lineWithMagic.substr(0, lineWithMagic.length - GAMEPAD_CONFIG_MAGIC.length);
|
|
let parsed;
|
|
|
|
try {
|
|
parsed = JSON.parse(jsonText);
|
|
|
|
if (!parsed || typeof parsed !== "object" || !Array.isArray(parsed.buttons) || !Array.isArray(parsed.axes)) {
|
|
throw new Error("Invalid data");
|
|
}
|
|
} catch (e) {
|
|
console.warn("Gamepad comment has invalid JSON", e);
|
|
return null;
|
|
}
|
|
|
|
return parsed;
|
|
};
|
|
|
|
_gamepadlib_js__WEBPACK_IMPORTED_MODULE_0__["default"].setConsole(console);
|
|
const gamepad = new _gamepadlib_js__WEBPACK_IMPORTED_MODULE_0__["default"]();
|
|
|
|
gamepad.getUserHints = () => {
|
|
const parsedOptions = parseOptionsComment();
|
|
|
|
if (parsedOptions) {
|
|
return {
|
|
importedSettings: parsedOptions
|
|
};
|
|
}
|
|
|
|
return {
|
|
usedKeys: getKeysUsedByProject()
|
|
};
|
|
};
|
|
|
|
vm.runtime.on("PROJECT_LOADED", () => {
|
|
gamepad.resetControls();
|
|
});
|
|
|
|
if (addon.settings.get("hide")) {
|
|
await new Promise(resolve => {
|
|
const end = () => {
|
|
addon.settings.removeEventListener("change", listener);
|
|
resolve();
|
|
};
|
|
|
|
const listener = () => {
|
|
if (!addon.settings.get("hide")) {
|
|
end();
|
|
}
|
|
};
|
|
|
|
gamepad.gamepadConnected().then(end);
|
|
addon.settings.addEventListener("change", listener);
|
|
});
|
|
}
|
|
|
|
const renderer = vm.runtime.renderer;
|
|
const width = renderer._xRight - renderer._xLeft;
|
|
const height = renderer._yTop - renderer._yBottom;
|
|
const canvas = renderer.canvas;
|
|
const container = document.createElement("div");
|
|
container.className = "sa-gamepad-container";
|
|
addon.tab.displayNoneWhileDisabled(container, {
|
|
display: "flex"
|
|
});
|
|
const buttonContainer = document.createElement("span");
|
|
buttonContainer.className = addon.tab.scratchClass("button_outlined-button", "stage-header_stage-button");
|
|
const buttonContent = document.createElement("div");
|
|
buttonContent.className = addon.tab.scratchClass("button_content");
|
|
const buttonImage = document.createElement("img");
|
|
buttonImage.className = addon.tab.scratchClass("stage-header_stage-button-icon");
|
|
buttonImage.draggable = false;
|
|
buttonImage.src = addon.self.getResource("/gamepad.svg")
|
|
/* rewritten by pull.js */
|
|
;
|
|
buttonContent.appendChild(buttonImage);
|
|
buttonContainer.appendChild(buttonContent);
|
|
container.appendChild(buttonContainer);
|
|
let editor;
|
|
let shouldStoreSettingsInProject = false;
|
|
|
|
const didChangeProject = () => {
|
|
vm.runtime.emitProjectChanged();
|
|
|
|
if (vm.editingTarget === vm.runtime.getTargetForStage()) {
|
|
vm.emitWorkspaceUpdate();
|
|
}
|
|
};
|
|
|
|
const storeMappings = () => {
|
|
const exported = editor.export();
|
|
|
|
if (!exported) {
|
|
console.warn("Could not export gamepad settings");
|
|
return;
|
|
}
|
|
|
|
const text = "".concat(msg("config-header"), "\n").concat(JSON.stringify(exported)).concat(GAMEPAD_CONFIG_MAGIC);
|
|
const existingComment = findOptionsComment();
|
|
|
|
if (existingComment) {
|
|
existingComment.text = text;
|
|
} else {
|
|
const target = vm.runtime.getTargetForStage();
|
|
target.createComment( // comment ID, just has to be a random string
|
|
Math.random() + "", // block ID
|
|
null, // text
|
|
text, // x, y, width, height
|
|
50, 50, 350, 150, // minimized
|
|
false);
|
|
}
|
|
|
|
didChangeProject();
|
|
};
|
|
|
|
const removeStoredMappings = () => {
|
|
const comment = findOptionsComment();
|
|
|
|
if (comment) {
|
|
const target = vm.runtime.getTargetForStage();
|
|
delete target.comments[comment.id];
|
|
didChangeProject();
|
|
}
|
|
};
|
|
|
|
const handleGamepadMappingChanged = () => {
|
|
if (shouldStoreSettingsInProject) {
|
|
storeMappings();
|
|
}
|
|
};
|
|
|
|
const handleStoreSettingsCheckboxChanged = e => {
|
|
shouldStoreSettingsInProject = !!e.target.checked;
|
|
|
|
if (shouldStoreSettingsInProject) {
|
|
storeMappings();
|
|
} else {
|
|
removeStoredMappings();
|
|
}
|
|
};
|
|
|
|
const handleEditorControllerChanged = () => {
|
|
document.body.classList.toggle("sa-gamepad-has-controller", editor.hasControllerSelected());
|
|
handleGamepadMappingChanged();
|
|
};
|
|
|
|
buttonContainer.addEventListener("click", () => {
|
|
if (!editor) {
|
|
editor = gamepad.editor();
|
|
editor.msg = msg;
|
|
editor.addEventListener("mapping-changed", handleGamepadMappingChanged);
|
|
editor.addEventListener("gamepad-changed", handleEditorControllerChanged);
|
|
}
|
|
|
|
const editorEl = editor.generateEditor();
|
|
handleEditorControllerChanged();
|
|
const {
|
|
backdrop,
|
|
container,
|
|
content,
|
|
closeButton,
|
|
remove
|
|
} = addon.tab.createModal(msg("settings"), {
|
|
isOpen: true,
|
|
useEditorClasses: true
|
|
});
|
|
|
|
const handleKeyDown = e => {
|
|
if (e.key === "Escape" && !e.target.closest("[data-accepting-input]")) {
|
|
remove();
|
|
}
|
|
};
|
|
|
|
backdrop.addEventListener("click", remove);
|
|
window.addEventListener("keydown", handleKeyDown);
|
|
addon.self.addEventListener("disabled", remove);
|
|
backdrop.classList.add("sa-gamepad-popup-outer");
|
|
container.classList.add("sa-gamepad-popup");
|
|
closeButton.tabIndex = "0";
|
|
closeButton.setAttribute("role", "button");
|
|
closeButton.addEventListener("click", remove);
|
|
closeButton.addEventListener("keydown", e => {
|
|
if (e.key === "Enter" || e.key === " ") {
|
|
remove();
|
|
}
|
|
});
|
|
content.classList.add("sa-gamepad-popup-content");
|
|
|
|
if (_gamepadlib_js__WEBPACK_IMPORTED_MODULE_0__["default"].browserHasBrokenGamepadAPI()) {
|
|
const warning = document.createElement("div");
|
|
warning.textContent = msg("browser-support");
|
|
warning.className = "sa-gamepad-browser-support-warning";
|
|
content.appendChild(warning);
|
|
}
|
|
|
|
content.appendChild(editorEl);
|
|
const extraOptionsContainer = document.createElement("div");
|
|
extraOptionsContainer.className = "sa-gamepad-extra-options";
|
|
content.appendChild(extraOptionsContainer);
|
|
|
|
const mappingsWereResetOrCleared = () => {
|
|
editor.updateAllContent();
|
|
storeSettingsCheckbox.checked = false;
|
|
shouldStoreSettingsInProject = false;
|
|
};
|
|
|
|
const resetButton = document.createElement("button");
|
|
resetButton.className = "sa-gamepad-reset-button";
|
|
resetButton.textContent = msg("reset");
|
|
resetButton.addEventListener("click", () => {
|
|
gamepad.resetControls();
|
|
mappingsWereResetOrCleared();
|
|
});
|
|
extraOptionsContainer.appendChild(resetButton);
|
|
const clearButton = document.createElement("button");
|
|
clearButton.className = "sa-gamepad-reset-button";
|
|
clearButton.textContent = msg("clear");
|
|
clearButton.addEventListener("click", () => {
|
|
gamepad.clearControls();
|
|
mappingsWereResetOrCleared();
|
|
});
|
|
extraOptionsContainer.appendChild(clearButton);
|
|
const storeSettingsLabel = document.createElement("label");
|
|
storeSettingsLabel.className = "sa-gamepad-store-settings";
|
|
storeSettingsLabel.textContent = msg("store-in-project");
|
|
const storeSettingsCheckbox = document.createElement("input");
|
|
storeSettingsCheckbox.type = "checkbox";
|
|
storeSettingsCheckbox.checked = shouldStoreSettingsInProject;
|
|
storeSettingsCheckbox.addEventListener("change", handleStoreSettingsCheckboxChanged);
|
|
storeSettingsLabel.prepend(storeSettingsCheckbox);
|
|
extraOptionsContainer.appendChild(storeSettingsLabel);
|
|
editor.focus();
|
|
});
|
|
|
|
if (addon.tab.redux.state && addon.tab.redux.state.scratchGui.stageSize.stageSize === "small") {
|
|
document.body.classList.add("sa-gamepad-small");
|
|
}
|
|
|
|
document.addEventListener("click", e => {
|
|
if (e.target.closest("[class*='stage-header_stage-button-first']:not(.sa-hide-stage-button)")) {
|
|
document.body.classList.add("sa-gamepad-small");
|
|
} else if (e.target.closest("[class*='stage-header_stage-button-last']") || e.target.closest(".sa-hide-stage-button")) {
|
|
document.body.classList.remove("sa-gamepad-small");
|
|
}
|
|
}, {
|
|
capture: true
|
|
});
|
|
const virtualCursorElement = document.createElement("img");
|
|
virtualCursorElement.hidden = true;
|
|
virtualCursorElement.className = "sa-gamepad-cursor";
|
|
virtualCursorElement.src = addon.self.getResource("/cursor.png")
|
|
/* rewritten by pull.js */
|
|
;
|
|
addon.self.addEventListener("disabled", () => {
|
|
virtualCursorElement.hidden = true;
|
|
});
|
|
let hideCursorTimeout;
|
|
|
|
const hideRealCursor = () => {
|
|
document.body.classList.add("sa-gamepad-hide-cursor");
|
|
};
|
|
|
|
const showRealCursor = () => {
|
|
document.body.classList.remove("sa-gamepad-hide-cursor");
|
|
};
|
|
|
|
const virtualCursorSetVisible = visible => {
|
|
virtualCursorElement.hidden = !visible;
|
|
clearTimeout(hideCursorTimeout);
|
|
|
|
if (visible) {
|
|
hideRealCursor();
|
|
hideCursorTimeout = setTimeout(virtualCursorHide, 8000);
|
|
}
|
|
};
|
|
|
|
const virtualCursorHide = () => {
|
|
virtualCursorSetVisible(false);
|
|
};
|
|
|
|
const virtualCursorSetDown = down => {
|
|
virtualCursorSetVisible(true);
|
|
virtualCursorElement.classList.toggle("sa-gamepad-cursor-down", down);
|
|
};
|
|
|
|
const virtualCursorSetPosition = (x, y) => {
|
|
virtualCursorSetVisible(true);
|
|
const CURSOR_SIZE = 6;
|
|
const stageX = width / 2 + x - CURSOR_SIZE / 2;
|
|
const stageY = height / 2 - y - CURSOR_SIZE / 2;
|
|
virtualCursorElement.style.transform = "translate(".concat(stageX, "px, ").concat(stageY, "px)");
|
|
};
|
|
|
|
document.addEventListener("mousemove", () => {
|
|
virtualCursorSetVisible(false);
|
|
showRealCursor();
|
|
});
|
|
let getCanvasSize; // Support modern ResizeObserver and slow getBoundingClientRect version for improved browser support (matters for TurboWarp)
|
|
|
|
if (window.ResizeObserver) {
|
|
let canvasWidth = width;
|
|
let canvasHeight = height;
|
|
const resizeObserver = new ResizeObserver(entries => {
|
|
for (const entry of entries) {
|
|
canvasWidth = entry.contentRect.width;
|
|
canvasHeight = entry.contentRect.height;
|
|
}
|
|
});
|
|
resizeObserver.observe(canvas);
|
|
|
|
getCanvasSize = () => [canvasWidth, canvasHeight];
|
|
} else {
|
|
getCanvasSize = () => {
|
|
const rect = canvas.getBoundingClientRect();
|
|
return [rect.width, rect.height];
|
|
};
|
|
} // Both in Scratch space
|
|
|
|
|
|
let virtualX = 0;
|
|
let virtualY = 0;
|
|
|
|
const postMouseData = data => {
|
|
if (addon.self.disabled || !vmStarted()) return;
|
|
const [rectWidth, rectHeight] = getCanvasSize();
|
|
vm.postIOData("mouse", _objectSpread(_objectSpread({}, data), {}, {
|
|
canvasWidth: rectWidth,
|
|
canvasHeight: rectHeight,
|
|
x: (virtualX + width / 2) * (rectWidth / width),
|
|
y: (height / 2 - virtualY) * (rectHeight / height)
|
|
}));
|
|
};
|
|
|
|
const postKeyboardData = (key, isDown) => {
|
|
if (addon.self.disabled || !vmStarted()) return;
|
|
vm.postIOData("keyboard", {
|
|
key,
|
|
isDown
|
|
});
|
|
};
|
|
|
|
const handleGamepadButtonDown = e => postKeyboardData(e.detail, true);
|
|
|
|
const handleGamepadButtonUp = e => postKeyboardData(e.detail, false);
|
|
|
|
const handleGamepadMouseDown = e => {
|
|
virtualCursorSetDown(true);
|
|
postMouseData({
|
|
isDown: true,
|
|
button: e.detail
|
|
});
|
|
};
|
|
|
|
const handleGamepadMouseUp = e => {
|
|
virtualCursorSetDown(false);
|
|
postMouseData({
|
|
isDown: false,
|
|
button: e.detail
|
|
});
|
|
};
|
|
|
|
const handleGamepadMouseMove = e => {
|
|
virtualX = e.detail.x;
|
|
virtualY = e.detail.y;
|
|
virtualCursorSetPosition(virtualX, virtualY);
|
|
postMouseData({});
|
|
};
|
|
|
|
gamepad.virtualCursor.maxX = renderer._xRight;
|
|
gamepad.virtualCursor.minX = renderer._xLeft;
|
|
gamepad.virtualCursor.maxY = renderer._yTop;
|
|
gamepad.virtualCursor.minY = renderer._yBottom;
|
|
gamepad.addEventListener("keydown", handleGamepadButtonDown);
|
|
gamepad.addEventListener("keyup", handleGamepadButtonUp);
|
|
gamepad.addEventListener("mousedown", handleGamepadMouseDown);
|
|
gamepad.addEventListener("mouseup", handleGamepadMouseUp);
|
|
gamepad.addEventListener("mousemove", handleGamepadMouseMove);
|
|
|
|
while (true) {
|
|
const target = await addon.tab.waitForElement('[class^="stage-header_embed-buttons_"], [class^="stage-header_stage-size-row"], [class^="stage-header_stage-menu-wrapper"] > [class^="button_outlined-button"]', {
|
|
markAsSeen: true,
|
|
reduxEvents: ["scratch-gui/mode/SET_PLAYER", "scratch-gui/mode/SET_FULL_SCREEN", "fontsLoaded/SET_FONTS_LOADED", "scratch-gui/locales/SELECT_LOCALE"]
|
|
});
|
|
container.dataset.editorMode = addon.tab.editorMode;
|
|
|
|
if (target.className.includes("stage-size-row")) {
|
|
addon.tab.appendToSharedSpace({
|
|
space: "stageHeader",
|
|
element: container,
|
|
order: 1
|
|
});
|
|
} else {
|
|
addon.tab.appendToSharedSpace({
|
|
space: "fullscreenStageHeader",
|
|
element: container,
|
|
order: 0
|
|
});
|
|
}
|
|
|
|
const monitorListScaler = document.querySelector("[class^='monitor-list_monitor-list-scaler']");
|
|
monitorListScaler.appendChild(virtualCursorElement);
|
|
}
|
|
});
|
|
|
|
/***/ })
|
|
|
|
}]);
|
|
//# sourceMappingURL=addon-entry-gamepad.js.map
|