diff --git a/extensions/extensions.json b/extensions/extensions.json index d8ed71e33d..f57e0c3c11 100644 --- a/extensions/extensions.json +++ b/extensions/extensions.json @@ -7,6 +7,7 @@ "files", "pointerlock", "cursor", + "legobrainbiker/MoreMouseControls", "runtime-options", "fetch", "text", diff --git a/extensions/legobrainbiker/MoreMouseControls.js b/extensions/legobrainbiker/MoreMouseControls.js new file mode 100644 index 0000000000..fc37564b1a --- /dev/null +++ b/extensions/legobrainbiker/MoreMouseControls.js @@ -0,0 +1,185 @@ +// Name: More Mouse Controls +// ID: legobrainbikerMoreMouseControls +// Description: Adds more options to detect properties of the cursor, including its position, clicking, and scrolling. +// By: legobrainbiker +// License: MPL-2.0 +(function (Scratch) { + "use strict"; + const iconURI = + ""; + class moreMouseControls { + constructor() { + this.mouseX = 0; + this.mouseY = 0; + this.mouseWheelDelta = 0; + this.mouseWheelDeltaX = 0; + this.mouseZoomDelta = 0; + this.buttons = []; + this.contextMenuDissabled = false; + document.addEventListener("contextmenu", (e) => { + if (this.contextMenuDissabled) { + e.preventDefault(); + } + }); + this.scrollDissabled = false; + document.addEventListener( + "wheel", + (e) => { + if (this.scrollDissabled) { + e.preventDefault(); + } + if (e.ctrlKey) { + this.mouseZoomDelta = e.deltaY; + this.mouseWheelDelta = 0; + this.mouseWheelDeltaX = 0; + } else { + this.mouseZoomDelta = 0; + this.mouseWheelDelta = e.deltaY; + this.mouseWheelDeltaX = e.deltaX; + } + Scratch.vm.runtime.startHats( + "legobrainbikerMoreMouseControls_onScroll" + ); + }, + { passive: false } + ); + document.addEventListener("mouseup", (e) => { + this.buttons[e.button] = false; + }); + document.addEventListener("mousedown", (e) => { + this.buttons[e.button] = true; + }); + document.addEventListener("mousemove", (e) => { + this.mouseX = e.clientX; + this.mouseY = e.clientY; + }); + document.addEventListener("mouseleave", (e) => { + this.mouseX = NaN; + this.mouseY = NaN; + // this could be problomatic for some projects. it's not great to add another edge case + }); + } + getInfo() { + return { + id: "legobrainbikerMoreMouseControls", + name: Scratch.translate("More Mouse Controls"), + menuIconURI: iconURI, + blockIconURI: iconURI, + color1: "#99bbbb", + color2: "#88bbbb", + color3: "#88bbbb", + blocks: [ + { + opcode: "onMouseUp", + blockType: Scratch.BlockType.HAT, + text: Scratch.translate("when mouse button [button] is released"), + isEdgeActivated: true, + arguments: { + button: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + { + opcode: "onMouseDown", + blockType: Scratch.BlockType.HAT, + text: Scratch.translate("when mouse button [button] is pressed"), + isEdgeActivated: true, + arguments: { + button: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + { + opcode: "buttonPressed", + blockType: Scratch.BlockType.BOOLEAN, + text: Scratch.translate("mouse button [button] is pressed"), + arguments: { + button: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + }, + }, + { + opcode: "toggleContextMenu", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("toggle context menu"), + }, + { + opcode: "toggleScroll", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("toggle scrolling and zooming"), + }, + { + opcode: "onScroll", + blockType: Scratch.BlockType.EVENT, + text: Scratch.translate("when scrolling"), + isEdgeActivated: false, + }, + { + opcode: "scrollAmount", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("scroll amount"), + }, + { + opcode: "horozontalScrollAmount", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("horozontal scroll amount"), + }, + { + opcode: "zoomAmount", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("zoom amount"), + }, + { + opcode: "mouseGlobalX", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("mouse global x position"), + }, + { + opcode: "mouseGlobalY", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("mouse global y position"), + }, + ], + }; + } + onMouseUp(args) { + return !this.buttonPressed(args); + } + onMouseDown(args) { + return this.buttonPressed(args); + } + buttonPressed({ button }) { + return this.buttons[Scratch.Cast.toNumber(button)] || false; + } + toggleContextMenu() { + this.contextMenuDissabled = !this.contextMenuDissabled; + } + toggleScroll() { + this.scrollDissabled = !this.scrollDissabled; + } + scrollAmount() { + return -this.mouseWheelDelta; + } + horozontalScrollAmount() { + return this.mouseWheelDeltaX; + } + zoomAmount() { + return -this.mouseZoomDelta; + } + mouseGlobalX() { + const canvas = Scratch.renderer.canvas.getBoundingClientRect(); + return this.mouseX - canvas.x - canvas.width / 2; + } + mouseGlobalY() { + const canvas = Scratch.renderer.canvas.getBoundingClientRect(); + return -(this.mouseY - canvas.y - canvas.height / 2); + } + } + Scratch.extensions.register(new moreMouseControls()); +})(Scratch); diff --git a/images/legobrainbiker/MoreMouseControls.svg b/images/legobrainbiker/MoreMouseControls.svg new file mode 100644 index 0000000000..0d46c87595 --- /dev/null +++ b/images/legobrainbiker/MoreMouseControls.svg @@ -0,0 +1 @@ + \ No newline at end of file