diff --git a/extensions/SharkPool/Better-Input.js b/extensions/SharkPool/Better-Input.js
new file mode 100644
index 0000000000..63134100e9
--- /dev/null
+++ b/extensions/SharkPool/Better-Input.js
@@ -0,0 +1,1324 @@
+// Name: Better Input
+// ID: BetterInputSP
+// Description: Expansion of the "ask and wait" Blocks.
+// By: SharkPool
+// Licence: MLP-2.0
+
+// Version V.4.2.11
+
+(function (Scratch) {
+ "use strict";
+ if (!Scratch.extensions.unsandboxed) throw new Error("Better Input must run unsandboxed");
+
+ const menuIconURI =
+"";
+ const blockIconURI =
+"";
+
+ const formatIcon =
+"";
+ const colorIcon =
+"";
+ const effectIcon =
+"";
+
+ let laidImgContain = "";
+ const vm = Scratch.vm;
+ const fontMenu = [
+ "Sans Serif", "Serif", "Handwriting",
+ "Marker", "Curly", "Pixel", "Scratch"
+ ];
+
+ const xmlEscape = function (unsafe) {
+ return Scratch.Cast.toString(unsafe).replace(/[<>&'"]/g, c => {
+ switch (c) {
+ case "<": return "<";
+ case ">": return ">";
+ case "&": return "&";
+ case "'": return "'";
+ case "\"": return """;
+ }
+ });
+ };
+
+ class BetterInputSP {
+ constructor() {
+ this.activeOverlays = []; this.askBoxPromises = [];
+ this.isDropdownOpen = false;
+ this.userInput = " "; this.defaultValue = "";
+ this.textBoxX = 0; this.textBoxY = 0;
+ this.askBoxInfo = [0, 1]; this.appendTarget = ["window", false];
+ this.forceInput = "Disabled";
+ this.overlayInput = null;
+ this.uiOrder = ["question", "input", "buttons"];
+
+ this.optionList = ["Option 1", "Option 2", "Option 3"];
+ this.sliderInfo = [0, 100, 50];
+ this.Timeout = 0;
+
+ this.inputType = "Enabled";
+ this.fontSize = "14px"; this.fontFamily = "Sans Serif"; this.textAlign = "left";
+ // overlay + Image, input, dropdown button
+ this.mainUIinfo = {
+ // Border Radius
+ dimensions: ["auto", "auto"],
+ overlayRad: 5,
+ inputRad: 4,
+ dropBtnRad: 5,
+ // Border Information
+ overlayBord: "1px none #000000",
+ inputBord: "1px solid #000000",
+ dropBtnBord: "1px none #000000",
+ // Text Padding
+ overlayPad: "15px",
+ inputPad: "5px",
+ dropBtnPad: "5px 10px",
+ // Text Shadow
+ overlayTxtShad: "none",
+ inputTxtShad: "none",
+ dropBtnTxtShad: "none",
+ // Outline: Color + Thickness
+ overlayOutline: ["", 0],
+ inputOutline: ["", 0],
+ dropBtnOutline: ["", 0]
+ };
+ this.DropdownText = "Dropdown";
+ this.lastPressBtn = "";
+ this.buttonJSON = {
+ "Submit": {
+ color: "#0074D9", textColor: "#ffffff",
+ name: "Submit", image: "", imgScale: 100,
+ borderRadius: 5, border: "1px none #000000",
+ padding: "5px 10px", dropShadow: "none", outline: ["", 0]
+ },
+ "Cancel": {
+ color: "#d9534f", textColor: "#ffffff",
+ name: "Cancel", image: "", imgScale: 100,
+ borderRadius: 5, border: "1px none #000000",
+ padding: "5px 10px", dropShadow: "none", outline: ["", 0]
+ },
+ };
+
+ this.questionColor = "#000000"; this.inputColor = "#000000";
+ this.textBoxColor = ["#ffffff"]; this.inputFieldColor = "#a5aec3";
+ this.dropdwnBtnColor = ["#5f5f5f", "#ffffff"];
+ this.overlayImage = [" ", " ", " "];
+
+ this.Blur = 0; this.Brightness = 0; this.Opacity = 100;
+ this.Invert = 0; this.Saturation = 100; this.Hue = 0;
+ this.Sepia = 0; this.Contrast = 100; this.Scale = 100;
+ this.SkewX = 0; this.SkewY = 0; this.Rotation = 90;
+ this.imgScale = [100, 100, 100];
+ this.shadowEnabled = true;
+ this.shadowS = [0, 0, 5, "#000000"];
+ }
+
+ getInfo() {
+ return {
+ id: "BetterInputSP",
+ name: "Better Input",
+ color1: "#9400ff",
+ color2: "#7800cd",
+ color3: "#6900b3",
+ menuIconURI,
+ blockIconURI,
+ blocks: [
+ {
+ opcode: "askAndWait",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "ask [question] and wait",
+ arguments: {
+ question: { type: Scratch.ArgumentType.STRING, defaultValue: "What is your name?" }
+ },
+ },
+ {
+ opcode: "askAndWaitForInput",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "ask [question] and wait",
+ arguments: {
+ question: { type: Scratch.ArgumentType.STRING, defaultValue: "What is your name?" }
+ },
+ },
+ {
+ opcode: "getUserInput", blockType: Scratch.BlockType.REPORTER,
+ text: "user input"
+ },
+ {
+ opcode: "setDefaultV",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set default value to [defaultV]",
+ arguments: {
+ defaultV: { type: Scratch.ArgumentType.STRING, defaultValue: "My Name Is..." }
+ },
+ },
+ {
+ opcode: "removeAskBoxes", blockType: Scratch.BlockType.COMMAND,
+ text: "remove all ask boxes"
+ },
+ {
+ opcode: "resetInput", blockType: Scratch.BlockType.COMMAND,
+ text: "reset user input"
+ },
+ { blockType: Scratch.BlockType.LABEL, text: "Formatting" },
+ {
+ opcode: "setFontSize",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set font size to [SIZE]",
+ blockIconURI: formatIcon,
+ arguments: {
+ SIZE: { type: Scratch.ArgumentType.NUMBER, defaultValue: 14 }
+ },
+ },
+ {
+ opcode: "setTextAlignment",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set alignment to [ALIGNMENT]",
+ blockIconURI: formatIcon,
+ arguments: {
+ ALIGNMENT: { type: Scratch.ArgumentType.STRING, menu: "alignmentMenu" }
+ },
+ },
+ {
+ opcode: "setFontFamily",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set font to [FONT]",
+ blockIconURI: formatIcon,
+ arguments: {
+ FONT: { type: Scratch.ArgumentType.STRING, menu: "fontMenu" }
+ },
+ },
+ "---",
+ {
+ opcode: "setInputType",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set input type to [ACTION]",
+ blockIconURI: formatIcon,
+ arguments: {
+ ACTION: { type: Scratch.ArgumentType.STRING, menu: "inputActionMenu" }
+ },
+ },
+ {
+ opcode: "setDropdown",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set dropdown options to array: [DROPDOWN]",
+ blockIconURI: formatIcon,
+ arguments: {
+ DROPDOWN: { type: Scratch.ArgumentType.STRING, defaultValue: "[\"Option 1\", \"Option 2\", \"Option 3\"]" }
+ },
+ },
+ {
+ opcode: "setSlider",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set slider to min: [MIN] max: [MAX] default: [DEFAULT]",
+ blockIconURI: formatIcon,
+ arguments: {
+ MIN: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 },
+ MAX: { type: Scratch.ArgumentType.NUMBER, defaultValue: 100 },
+ DEFAULT: { type: Scratch.ArgumentType.NUMBER, defaultValue: 50 }
+ },
+ },
+ { blockType: Scratch.BlockType.LABEL, text: "Buttons" },
+ {
+ opcode: "setButton",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "[BUTTON] button named [NAME]",
+ blockIconURI: formatIcon,
+ arguments: {
+ BUTTON: { type: Scratch.ArgumentType.STRING, menu: "buttonType" },
+ NAME: { type: Scratch.ArgumentType.STRING, defaultValue: "Submit" }
+ },
+ },
+ {
+ opcode: "deleteAllButtons", blockType: Scratch.BlockType.COMMAND,
+ text: "remove all buttons", blockIconURI: formatIcon
+ },
+ {
+ opcode: "setButtonText",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set [BUTTON_MENU] button name to [TEXT]",
+ blockIconURI: formatIcon,
+ arguments: {
+ BUTTON_MENU: { type: Scratch.ArgumentType.STRING, menu: "buttonMenu" },
+ TEXT: { type: Scratch.ArgumentType.STRING, defaultValue: "my dropdown" }
+ },
+ },
+ {
+ opcode: "lastButton", blockType: Scratch.BlockType.REPORTER,
+ text: "last pressed button", blockIconURI: formatIcon
+ },
+ { blockType: Scratch.BlockType.LABEL, text: "Positioning" },
+ {
+ opcode: "setPrePosition",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "preset textbox position to x: [X] y: [Y]",
+ blockIconURI: formatIcon,
+ arguments: {
+ X: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 },
+ Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }
+ },
+ },
+ {
+ opcode: "setPosition",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set textbox position to x: [X] y: [Y]",
+ blockIconURI: formatIcon,
+ arguments: {
+ X: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 },
+ Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }
+ },
+ },
+ {
+ opcode: "changePosition",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "change textbox position by x: [X] y: [Y]",
+ blockIconURI: formatIcon,
+ arguments: {
+ X: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 },
+ Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }
+ },
+ },
+ {
+ opcode: "getXpos", blockType: Scratch.BlockType.REPORTER,
+ blockIconURI: formatIcon, text: "x position"
+ },
+ {
+ opcode: "getYpos", blockType: Scratch.BlockType.REPORTER,
+ blockIconURI: formatIcon, text: "y position"
+ },
+ {
+ opcode: "setDirection",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set direction to [ROTATE]",
+ blockIconURI: formatIcon,
+ arguments: {
+ ROTATE: { type: Scratch.ArgumentType.ANGLE, defaultValue: 90 }
+ },
+ },
+ {
+ opcode: "changeDirection",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "change direction by [ROTATE]",
+ blockIconURI: formatIcon,
+ arguments: {
+ ROTATE: { type: Scratch.ArgumentType.ANGLE, defaultValue: 15 }
+ },
+ },
+ {
+ opcode: "reportDirection", blockType: Scratch.BlockType.REPORTER,
+ text: "direction", blockIconURI: formatIcon
+ },
+ { blockType: Scratch.BlockType.LABEL, text: "Visual Settings" },
+ {
+ opcode: "setColorSettings",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set [COLOR_TYPE] color to [COLOR]",
+ blockIconURI: colorIcon,
+ arguments: {
+ COLOR_TYPE: { type: Scratch.ArgumentType.STRING, menu: "colorSettingsMenu" },
+ COLOR: { type: Scratch.ArgumentType.COLOR }
+ },
+ },
+ "---",
+ {
+ opcode: "setImage",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set [ELEMENT] image to [IMAGE]",
+ blockIconURI: colorIcon,
+ arguments: {
+ ELEMENT: { type: Scratch.ArgumentType.STRING, menu: "elementMenu" },
+ IMAGE: { type: Scratch.ArgumentType.STRING, defaultValue: "input-url-here" }
+ },
+ },
+ {
+ opcode: "scaleImage",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "scale [ELEMENT] image to [SCALE]%",
+ blockIconURI: colorIcon,
+ arguments: {
+ ELEMENT: { type: Scratch.ArgumentType.STRING, menu: "elementMenu" },
+ SCALE: { type: Scratch.ArgumentType.NUMBER, defaultValue: 100 }
+ },
+ },
+ "---",
+ {
+ opcode: "enableShadow",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set box shadow to [ACTION]",
+ blockIconURI: colorIcon,
+ arguments: {
+ ACTION: { type: Scratch.ArgumentType.STRING, menu: "buttonActionMenu" }
+ },
+ },
+ {
+ opcode: "setShadow",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set box shadow [SHADOW] to [AMT]",
+ blockIconURI: colorIcon,
+ arguments: {
+ SHADOW: { type: Scratch.ArgumentType.STRING, menu: "shadowStuff" },
+ AMT: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 }
+ },
+ },
+ {
+ opcode: "setDropShadow",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set [ELEMENT] shadow to x [x] y [y] z [z] color [COLOR]",
+ blockIconURI: colorIcon,
+ arguments: {
+ ELEMENT: { type: Scratch.ArgumentType.STRING, menu: "textsMenu" },
+ x: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 },
+ y: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 },
+ z: { type: Scratch.ArgumentType.NUMBER, defaultValue: 2 },
+ COLOR: { type: Scratch.ArgumentType.COLOR }
+ },
+ },
+ {
+ opcode: "setOutline",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set [ELEMENT] outline to [COLOR] thickness [THICK]",
+ blockIconURI: colorIcon,
+ arguments: {
+ ELEMENT: { type: Scratch.ArgumentType.STRING, menu: "textsMenu" },
+ COLOR: { type: Scratch.ArgumentType.COLOR },
+ THICK: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 }
+ },
+ },
+ "---",
+ {
+ opcode: "setBorder",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set [ELEMENT] border to [TYPE] color [COLOR] width [WIDTH]",
+ blockIconURI: colorIcon,
+ arguments: {
+ ELEMENT: { type: Scratch.ArgumentType.STRING, menu: "elementMenu" },
+ TYPE: { type: Scratch.ArgumentType.STRING, menu: "borderTypes" },
+ COLOR: { type: Scratch.ArgumentType.COLOR },
+ WIDTH: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 },
+ },
+ },
+ {
+ opcode: "setBorderRadius",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set [ELEMENT] border radius to [VALUE]",
+ blockIconURI: colorIcon,
+ arguments: {
+ ELEMENT: { type: Scratch.ArgumentType.STRING, menu: "elementMenu" },
+ VALUE: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 },
+ },
+ },
+ "---",
+ {
+ opcode: "setPadding",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set [ELEMENT] padding to T: [N1] B: [N3] L: [N4] R: [N2]",
+ blockIconURI: colorIcon,
+ arguments: {
+ ELEMENT: { type: Scratch.ArgumentType.STRING, menu: "elementMenu" },
+ N1: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 },
+ N2: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 },
+ N3: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 },
+ N4: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 }
+ },
+ },
+ {
+ opcode: "setDimension",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set Textbox width [W] height [H]",
+ blockIconURI: colorIcon,
+ arguments: {
+ W: { type: Scratch.ArgumentType.NUMBER, defaultValue: 100 },
+ H: { type: Scratch.ArgumentType.NUMBER, defaultValue: 100 }
+ },
+ },
+ { blockType: Scratch.BlockType.LABEL, text: "Effects" },
+ {
+ opcode: "resetEffect",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "reset effects",
+ blockIconURI: effectIcon
+ },
+ {
+ opcode: "setEffect",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set effect [EFFECT] to [AMT]",
+ blockIconURI: effectIcon,
+ arguments: {
+ EFFECT: { type: Scratch.ArgumentType.STRING, menu: "effectMenu" },
+ AMT: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 },
+ },
+ },
+ {
+ opcode: "changeEffect",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "change effect [EFFECT] by [AMT]",
+ blockIconURI: effectIcon,
+ arguments: {
+ EFFECT: { type: Scratch.ArgumentType.STRING, menu: "effectMenu" },
+ AMT: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 }
+ },
+ },
+ {
+ opcode: "showEffect",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "effect [EFFECT]",
+ blockIconURI: effectIcon,
+ arguments: {
+ EFFECT: { type: Scratch.ArgumentType.STRING, menu: "effectMenu" }
+ },
+ },
+ "---",
+ {
+ opcode: "setTimeout",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "when submitted delete textbox after [TIME] secs",
+ blockIconURI: effectIcon,
+ arguments: {
+ TIME: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 }
+ },
+ },
+ {
+ opcode: "reportTimeout",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "current textbox timeout",
+ blockIconURI: effectIcon
+ },
+ { blockType: Scratch.BlockType.LABEL, text: "Operations" },
+ {
+ opcode: "setUI",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set UI order to [ARRAY]",
+ arguments: {
+ ARRAY: { type: Scratch.ArgumentType.STRING, defaultValue: "[\"question\", \"input\", \"buttons\"]" }
+ },
+ },
+ {
+ opcode: "getUIOrder",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "UI order"
+ },
+ "---",
+ {
+ opcode: "setAppend",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "append next textbox to [TARGET]",
+ arguments: {
+ TARGET: { type: Scratch.ArgumentType.STRING, menu: "appendMenu" }
+ },
+ },
+ {
+ opcode: "setFocus",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "toggle focus mode to [TYPE]",
+ arguments: {
+ TYPE: { type: Scratch.ArgumentType.STRING, menu: "buttonActionMenu" }
+ },
+ },
+ "---",
+ {
+ opcode: "isWaitingInput",
+ blockType: Scratch.BlockType.BOOLEAN,
+ text: "is waiting?"
+ },
+ {
+ opcode: "isDropdown",
+ blockType: Scratch.BlockType.BOOLEAN,
+ text: "is dropdown open?"
+ },
+ {
+ opcode: "setSubmitEvent",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set force input to [ENTER]",
+ arguments: {
+ ENTER: { type: Scratch.ArgumentType.STRING, menu: "enterMenu" }
+ },
+ },
+ {
+ opcode: "setMaxBoxCount",
+ blockType: Scratch.BlockType.COMMAND,
+ text: "set max box count to: [MAX]",
+ arguments: {
+ MAX: { type: Scratch.ArgumentType.NUMBER, defaultValue: 1 }
+ },
+ },
+ {
+ opcode: "getBoxInfo",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "textbox [INFO]",
+ arguments: {
+ INFO: { type: Scratch.ArgumentType.STRING, menu: "boxInfo" }
+ },
+ }
+ ],
+ menus: {
+ fontMenu: { acceptReporters: true, items: "allFonts" },
+ buttonMenu: {
+ acceptReporters: true,
+ items: this.allButtons(["Dropdown"], false),
+ },
+ elementMenu: {
+ acceptReporters: true,
+ items: this.allButtons(["Textbox", "Input Box", "Dropdown Button"], false),
+ },
+ colorSettingsMenu: {
+ acceptReporters: true,
+ items: this.allButtons([
+ "Textbox", "Question Text", "Textbox Shadow",
+ "Input Text", "Input Box",
+ "Dropdown Button", "Dropdown Text"
+ ], true),
+ },
+ textsMenu: {
+ acceptReporters: true,
+ items: this.allButtons(["Question Text", "Input Text", "Dropdown Text"], true, true),
+ },
+ appendMenu: ["window", "canvas"],
+ buttonType: { acceptReporters: true, items: ["add", "remove"] },
+ buttonActionMenu: { acceptReporters: true, items: ["Enabled", "Disabled"] },
+ alignmentMenu: { acceptReporters: true, items: ["left", "right", "center"] },
+ shadowStuff: { acceptReporters: true, items: ["Size", "X", "Y"] },
+ boxInfo: {
+ acceptReporters: true,
+ items: ["count", "limit", "button count", "button names"],
+ },
+ inputActionMenu: {
+ acceptReporters: true,
+ items: [
+ "None", "Text", "Text Area", "Password", "Number", "Color",
+ "Dropdown", "Single Dropdown", "Multi-Select Dropdown",
+ "Horizontal Slider", "Vertical Slider"
+ ],
+ },
+ effectMenu: {
+ acceptReporters: true,
+ items: [
+ "Blur", "Brightness", "Opacity", "Invert",
+ "Saturation", "Hue", "Sepia", "Contrast",
+ "Scale", "SkewX", "SkewY",
+ ],
+ },
+ enterMenu: {
+ acceptReporters: true,
+ items: ["Disabled", "Enter Key", "Shift + Enter Key"],
+ },
+ borderTypes: {
+ acceptReporters: true,
+ items: [
+ "none", "solid", "dotted", "dashed",
+ "double", "groove", "ridge", "inset", "outset"
+ ],
+ }
+ },
+ };
+ }
+
+ allFonts() {
+ const custFonts = vm.runtime.fontManager ? vm.runtime.fontManager.getFonts().map((i) => ({ text: i.name, value: i.family })) : [];
+ return [ ...fontMenu, ...custFonts ];
+ }
+
+ allButtons(array, enableTxt, justTxt) {
+ let customBtn = Object.keys(this.buttonJSON);
+ if (justTxt) customBtn = customBtn.map(btn => btn + " Text");
+ else if (enableTxt) customBtn.forEach((btn) => { customBtn.push(btn + " Text") });
+ return [ ...array, ...customBtn ];
+ }
+
+ updateOverlayPos(overlay) {
+ if (this.Rotation > 359) this.Rotation = 0;
+ else if (this.Rotation < 1) this.Rotation = 360;
+ if (this.textBoxX !== null && this.textBoxY !== null) {
+ if (this.appendTarget[0] === "window") {
+ overlay.style.left = `${50 + this.textBoxX}%`;
+ overlay.style.top = `${50 + this.textBoxY}%`;
+ }
+ overlay.style.transform = `
+ translate${this.appendTarget[0] === "window" ? "(-50%, -50%)" : `(${-50 + this.textBoxX}%, ${-50 + this.textBoxY}%)` }
+ SkewX(${this.SkewX}deg) SkewY(${this.SkewY}deg)
+ rotate(${this.Rotation - 90}deg) scale(${this.Scale / 100})
+ `;
+ } else {
+ overlay.style.left = "50%";
+ overlay.style.top = "50%";
+ }
+ }
+ updateOverlay(overlay) {
+ const newOpacity = this.Opacity / 100;
+ const newBrightness = this.Brightness + 100;
+ overlay.style.backgroundImage = "";
+ overlay.style[this.textBoxColor[0].includes("gradient") ? "backgroundImage" : "backgroundColor"] = this.textBoxColor[0];
+ overlay.style.boxShadow = this.shadowEnabled ? `${this.shadowS[0]}px ${this.shadowS[1]}px ${this.shadowS[2]}px ${this.shadowS[3]}` : "none";
+ overlay.style.transform = `
+ translate${this.appendTarget[0] === "window" ? "(-50%, -50%)" : `(${-50 + this.textBoxX}%, ${-50 + this.textBoxY}%)` }
+ SkewX(${this.SkewX}deg) SkewY(${this.SkewY}deg)
+ rotate(${this.Rotation - 90}deg) scale(${this.Scale / 100})
+ `;
+ overlay.style.filter = `
+ blur(${this.Blur}px) brightness(${newBrightness}%)
+ invert(${this.Invert}%) saturate(${this.Saturation}%)
+ hue-rotate(${this.Hue}deg) sepia(${this.Sepia}%)
+ contrast(${this.Contrast}%)
+ `;
+ overlay.style.opacity = newOpacity;
+ overlay.style.border = this.mainUIinfo.overlayBord;
+ overlay.style.padding = this.mainUIinfo.overlayPad;
+ overlay.style.fontFamily = this.fontFamily;
+ overlay.style.textAlign = this.textAlign;
+ overlay.style.borderRadius = `${this.mainUIinfo.overlayRad}px`;
+ overlay.style.width = this.mainUIinfo.dimensions[0];
+ overlay.style.height = this.mainUIinfo.dimensions[1];
+ laidImgContain.style.borderRadius = `${this.mainUIinfo.overlayRad}px`;
+ laidImgContain.style.background = "";
+ this.setImageStyles(laidImgContain, this.overlayImage[0], this.imgScale[0]);
+ this.updateButtonImages(overlay);
+ }
+ updateButtonImages(overlay) {
+ let text = overlay.querySelector(".question");
+ if (text) {
+ text.style.color = this.questionColor;
+ text.style.textShadow = this.mainUIinfo.overlayTxtShad;
+ this.tryOutline(text, this.mainUIinfo.overlayOutline[0], this.mainUIinfo.overlayOutline[1]);
+ }
+ const inputField = overlay.querySelector(this.inputType.includes("Single") ? "select" : this.inputType === "Text Area" ? "textarea" : "input");
+ if (inputField) {
+ const inpWidth = parseInt(this.mainUIinfo.dimensions[0]);
+ inputField.style.width = this.inputType === "Color" || this.inputType.includes("Single") ? "100%" :
+ this.inputType === "Horizontal Slider" ? "95%" : isNaN(inpWidth) || this.inputType.includes("Dropdown") ? "auto" : `${inpWidth - 10}px`;
+ inputField.style.background = "";
+ inputField.style.fontFamily = this.fontFamily;
+ inputField.style[this.inputFieldColor.includes("gradient") ? "backgroundImage" : "backgroundColor"] = this.inputFieldColor;
+ inputField.style.color = this.inputColor; inputField.style.accentColor = this.inputFieldColor;
+ inputField.style.textShadow = this.mainUIinfo.inputTxtShad;
+ this.tryOutline(inputField, this.mainUIinfo.inputOutline[0], this.mainUIinfo.inputOutline[1]);
+ inputField.style.border = this.mainUIinfo.inputBord;
+ inputField.style.borderRadius = `${this.mainUIinfo.inputRad}px`;
+ inputField.style.padding = this.mainUIinfo.inputPad;
+ this.setImageStyles(inputField, this.overlayImage[1], this.imgScale[1]);
+ }
+
+ const dropBtn = overlay.querySelector("button.dropbtn");
+ if (dropBtn) {
+ dropBtn.style.backgroundImage = "";
+ dropBtn.style.fontFamily = this.fontFamily;
+ dropBtn.style.color = this.dropdwnBtnColor[1];
+ dropBtn.style.borderRadius = `${this.mainUIinfo.dropBtnRad}px`;
+ dropBtn.style.border = this.mainUIinfo.dropBtnBord;
+ dropBtn.style.padding = this.mainUIinfo.dropBtnPad;
+ dropBtn.style.textShadow = this.mainUIinfo.dropBtnTxtShad;
+ this.tryOutline(dropBtn, this.mainUIinfo.dropBtnOutline[0], this.mainUIinfo.dropBtnOutline[1]);
+ dropBtn.style[this.dropdwnBtnColor[0].includes("gradient") ? "backgroundImage" : "backgroundColor"] = this.dropdwnBtnColor[0];
+ this.setImageStyles(dropBtn, this.overlayImage[2], this.imgScale[2]);
+ }
+ const btnContain = overlay.querySelector(".button-container");
+ if (btnContain) {
+ const buttons = btnContain.querySelectorAll("button");
+ buttons.forEach((button, index) => {
+ const buttonName = Object.keys(this.buttonJSON)[index];
+ const buttonInfo = this.buttonJSON[buttonName];
+ if (buttonInfo) {
+ button.style.color = buttonInfo.textColor;
+ button.style.fontFamily = this.fontFamily;
+ button.style.fontSize = this.fontSize;
+ button.style.borderRadius = `${buttonInfo.borderRadius}px`;
+ button.style.border = buttonInfo.border;
+ button.style.padding = buttonInfo.padding;
+ button.style.textShadow = buttonInfo.dropShadow;
+ this.tryOutline(button, buttonInfo.outline[0], buttonInfo.outline[1]);
+ button.style.background = "";
+ button.style[buttonInfo.color.includes("gradient") ? "backgroundImage" : "background"] = buttonInfo.color;
+ this.setImageStyles(button, buttonInfo.image, buttonInfo.imgScale);
+ }
+ });
+ }
+ }
+ tryOutline(element, color, thick) {
+ element.style.webkitTextStrokeColor = color;
+ element.style.webkitTextStrokeWidth = `${thick}px`;
+ //multi-platform support cuz we cant have nice things
+ element.style.textStrokeColor = color;
+ element.style.textStrokeWidth = `${thick}px`;
+ element.style.mozTextStrokeColor = color;
+ element.style.mozTextStrokeWidth = `${thick}px`;
+ }
+
+ setImageStyles(element, url, scale) {
+ if (Scratch.Cast.toString(url).length > 5) {
+ Scratch.canFetch(encodeURI(url)).then((canFetch) => {
+ if (canFetch) {
+ element.style.background = `url(${encodeURI(url)})`;
+ element.style.backgroundSize = `${scale}%`;
+ } else { console.warn("Cannot fetch content from the URL") }
+ });
+ }
+ }
+
+ showEffect(args) { return this[args.EFFECT] }
+
+ setEffect(args) {
+ this[args.EFFECT] = args.AMT;
+ this.activeOverlays.forEach((overlay) => { this.updateOverlay(overlay) });
+ }
+
+ changeEffect(args) {
+ const effect = args.EFFECT;
+ this[effect] = this[effect] + args.AMT;
+ this.activeOverlays.forEach((overlay) => { this.updateOverlay(overlay) });
+ }
+
+ resetEffect() {
+ this.Blur = 0; this.Brightness = 0; this.Opacity = 100; this.Invert = 0;
+ this.Saturation = 100; this.Hue = 0; this.Sepia = 0; this.Contrast = 100;
+ this.Scale = 100; this.SkewX = 0; this.SkewY = 0;
+ this.activeOverlays.forEach((overlay) => { this.updateOverlay(overlay) });
+ }
+
+ setColorSettings(args) {
+ const colorType = args.COLOR_TYPE;
+ const colorValue = args.COLOR;
+ const colorTypeMap = {
+ "Question Text": () => this.questionColor = colorValue,
+ "Input Text": () => this.inputColor = colorValue,
+ "Textbox": () => { this.textBoxColor[0] = colorValue; this.overlayImage[0] = " "; },
+ "Textbox Shadow": () => { this.shadowS[3] = colorValue },
+ "Input Box": () => { this.inputFieldColor = colorValue; this.overlayImage[1] = " "; },
+ "Dropdown Button": () => { this.dropdwnBtnColor[0] = colorValue; this.overlayImage[2] = " "; },
+ "Dropdown Text": () => this.dropdwnBtnColor[1] = colorValue,
+ };
+ const buttonInfo = this.buttonJSON[colorType] || this.buttonJSON[colorType.replace(" Text", "")];
+ if (buttonInfo) {
+ if (colorType.includes(" Text")) buttonInfo.textColor = colorValue;
+ else {
+ buttonInfo.color = colorValue;
+ buttonInfo.image = " ";
+ }
+ }
+ const applyColor = colorTypeMap[colorType];
+ if (applyColor) applyColor();
+ this.activeOverlays.forEach(overlay => this.updateOverlay(overlay));
+ }
+
+ findGradientType(menu) {
+ const colorTypeMap = {
+ Textbox: { newColorType: "textBoxColor", ind: 0 },
+ "Dropdown Button": { newColorType: "dropdwnBtnColor", ind: 2 }
+ };
+ if (colorTypeMap[menu]) {
+ const { newColorType, ind } = colorTypeMap[menu];
+ this.overlayImage[ind] = " ";
+ return newColorType;
+ } else if (this.buttonJSON[menu]) { return ["button", menu] }
+ return menu;
+ }
+
+ callStyling(element, value, type, elements) {
+ const elementID = elements[element];
+ if (elementID !== undefined) this.mainUIinfo[elementID] = value;
+ else if (this.buttonJSON[element]) this.buttonJSON[element][type] = value;
+ this.activeOverlays.forEach(overlay => this.updateOverlay(overlay));
+ }
+
+ setBorder(args) {
+ const width = Scratch.Cast.toNumber(args.WIDTH);
+ const string = `${width}px ${args.TYPE} ${args.COLOR}`;
+ this.callStyling(
+ args.ELEMENT, string, "border",
+ { Textbox: "overlayBord", "Input Box": "inputBord", "Dropdown Button": "dropBtnBord" }
+ );
+ }
+
+ setBorderRadius(args) {
+ this.callStyling(
+ args.ELEMENT, Math.max(args.VALUE, 0), "borderRadius",
+ { Textbox: "overlayRad", "Input Box": "inputRad", "Dropdown Button": "dropBtnRad" }
+ );
+ }
+
+ setPadding(args) {
+ const casted = [
+ Scratch.Cast.toNumber(args.N1), Scratch.Cast.toNumber(args.N2),
+ Scratch.Cast.toNumber(args.N3), Scratch.Cast.toNumber(args.N4)
+ ];
+ let pad = `${casted[0]}px ${casted[1]}px ${casted[2]}px ${casted[3]}px`;
+ this.callStyling(
+ args.ELEMENT, pad, "padding",
+ { Textbox: "overlayPad", "Input Box": "inputPad", "Dropdown Button": "dropBtnPad" }
+ );
+ }
+
+ setDropShadow(args) {
+ const casted = [
+ Scratch.Cast.toNumber(args.x), Scratch.Cast.toNumber(args.y), Scratch.Cast.toNumber(args.z)
+ ];
+ let shadow = args.z === 0 ? "none" : `${casted[0]}px ${casted[1] * -1}px ${casted[2]}px ${args.COLOR}`;
+ this.callStyling(
+ args.ELEMENT.slice(0, -5), shadow, "dropShadow",
+ { "Question": "overlayTxtShad", "Input": "inputTxtShad", "Dropdown": "dropBtnTxtShad" }
+ );
+ }
+
+ setOutline(args) {
+ const thick = Scratch.Cast.toNumber(args.THICK);
+ this.callStyling(
+ args.ELEMENT.slice(0, -5), [args.COLOR, thick], "outline",
+ { "Question": "overlayOutline", "Input": "inputOutline", "Dropdown": "dropBtnOutline" }
+ );
+ }
+
+ setShadow(args) {
+ const shadowMap = { Size: 2, X: 0, Y: 1 };
+ const propertyIndex = shadowMap[args.SHADOW];
+ if (propertyIndex !== undefined) this.shadowS[propertyIndex] = args.AMT;
+ this.activeOverlays.forEach(overlay => this.updateOverlay(overlay));
+ }
+
+ setImage(args) {
+ const elementMap = { Textbox: 0, "Input Box": 1, "Dropdown Button": 2 };
+ const elementIndex = elementMap[args.ELEMENT];
+ if (elementIndex !== undefined) this.overlayImage[elementIndex] = args.IMAGE;
+ else if (this.buttonJSON[args.ELEMENT]) this.buttonJSON[args.ELEMENT].image = args.IMAGE;
+ this.activeOverlays.forEach(overlay => this.updateOverlay(overlay));
+ }
+
+ scaleImage(args) {
+ const elementMap = { Textbox: 0, "Input Box": 1, "Dropdown Button": 2 };
+ const elementIndex = elementMap[args.ELEMENT];
+ if (elementIndex !== undefined) this.imgScale[elementIndex] = args.SCALE;
+ else if (this.buttonJSON[args.ELEMENT]) this.buttonJSON[args.ELEMENT].imgScale = args.SCALE;
+ this.activeOverlays.forEach(overlay => this.updateOverlay(overlay));
+ }
+
+ setDimension(args) {
+ const w = `${Scratch.Cast.toNumber(args.W)}px`;
+ const h = `${Scratch.Cast.toNumber(args.H)}px`;
+ // Negative numbers result in auto-dimensions
+ this.mainUIinfo.dimensions = [w.includes("-") ? "auto" : w, h.includes("-") ? "auto" : h];
+ this.activeOverlays.forEach(overlay => this.updateOverlay(overlay));
+ }
+
+ setDirection(args) {
+ this.Rotation = Scratch.Cast.toNumber(args.ROTATE);
+ this.activeOverlays.forEach((overlay) => { this.updateOverlay(overlay) });
+ }
+
+ changeDirection(args) {
+ this.Rotation = this.Rotation + Scratch.Cast.toNumber(args.ROTATE);
+ this.activeOverlays.forEach((overlay) => { this.updateOverlay(overlay) });
+ }
+
+ reportDirection() { return this.Rotation }
+
+ setPrePosition(args) {
+ this.textBoxX = Scratch.Cast.toNumber(args.X) / (screen.width / 400);
+ this.textBoxY = Scratch.Cast.toNumber(args.Y) / (screen.height / -300);
+ }
+
+ setPosition(args) {
+ this.textBoxX = Scratch.Cast.toNumber(args.X) / (screen.width / 400);
+ this.textBoxY = Scratch.Cast.toNumber(args.Y) / (screen.height / -300);
+ this.activeOverlays.forEach((overlay) => { this.updateOverlayPos(overlay) });
+ }
+
+ changePosition(args) {
+ this.textBoxX = this.textBoxX + Scratch.Cast.toNumber(args.X) / (screen.width / 400);
+ this.textBoxY = this.textBoxY + Scratch.Cast.toNumber(args.Y) / (screen.height / -300);
+ this.activeOverlays.forEach((overlay) => { this.updateOverlayPos(overlay) });
+ }
+
+ getXpos() { return this.textBoxX * (screen.width / 400) }
+ getYpos() { return this.textBoxY * (screen.height / -300) }
+
+ setFontSize(args) { this.fontSize = args.SIZE + "px" }
+
+ setTextAlignment(args) {
+ this.textAlign = args.ALIGNMENT;
+ this.activeOverlays.forEach((overlay) => { this.updateOverlay(overlay) });
+ }
+
+ setFontFamily(args) {
+ this.fontFamily = args.FONT;
+ this.activeOverlays.forEach((overlay) => { this.updateOverlay(overlay) });
+ }
+
+ setSlider(args) { this.sliderInfo = [args.MIN, args.MAX, args.DEFAULT] }
+
+ setInputType(args) {
+ if (args.ACTION === "Text" || args.ACTION === "None") this.inputType = args.ACTION === "Text" ? "Enabled" : "Disabled";
+ else this.inputType = args.ACTION;
+ }
+
+ enableShadow(args) { this.shadowEnabled = args.ACTION === "Enabled" }
+
+ setButtonText(args) {
+ const buttonMenu = args.BUTTON_MENU;
+ const text = args.TEXT;
+ if (buttonMenu === "Dropdown") this.DropdownText = text;
+ else if (this.buttonJSON[buttonMenu]) {
+ this.buttonJSON[buttonMenu].name = text;
+ vm.extensionManager.refreshBlocks();
+ }
+ }
+
+ setDropdown(args) {
+ try {
+ this.optionList = JSON.parse(args.DROPDOWN);
+ } catch { this.optionList = ["Invalid Array"] }
+ }
+
+ removeAskBoxes() {
+ const overlaysToRemove = [];
+ this.activeOverlays.forEach((overlay) => {
+ if (overlay) {
+ if (this.appendTarget[0] === "window" && overlay.parentNode) overlay.parentNode.removeChild(overlay);
+ else if (overlay.parentNode.parentNode !== document.documentElement) overlay.parentNode.parentNode.removeChild(overlay.parentNode);
+ overlaysToRemove.push(overlay);
+ }
+ if (this.askBoxPromises) {
+ const index = this.activeOverlays.indexOf(overlay);
+ if (index !== -1) this.askBoxPromises[index].resolve("removed");
+ }
+ });
+ this.askBoxPromises = [];
+ this.activeOverlays = this.activeOverlays.filter((overlay) => !overlaysToRemove.includes(overlay));
+ this.askBoxInfo[0] = 0;
+ this.isDropdownOpen = false;
+ const bugged = document.querySelectorAll(`div[class^="SP-ask-box"]`);
+ bugged.forEach((box) => { box.parentNode.removeChild(box) });
+ }
+
+ resetInput() { this.userInput = this.askBoxInfo[1] > 1 ? [] : "" }
+
+ askAndWaitForInput(args) {
+ if (this.askBoxInfo[0] < this.askBoxInfo[1] ) {
+ return this.askAndWait(args).then(() => { return this.getUserInput() });
+ }
+ }
+
+ askAndWait(args) {
+ if (this.askBoxInfo[0] < this.askBoxInfo[1]) {
+ const question = args.question;
+ let hasDecreased = false; // for the box counter
+ const index = this.askBoxInfo[0];
+ this.lastPressBtn = "";
+ this.askBoxInfo[0]++;
+ let selectOpts = [];
+ return new Promise((resolve) => {
+ this.askBoxPromises.push({ resolve });
+ const overlay = document.createElement("div");
+ overlay.classList.add("SP-ask-box");
+ overlay.style.pointerEvents = "auto";
+ overlay.style.position = "fixed";
+ overlay.style.fontSize = this.fontSize;
+ overlay.style.left = this.appendTarget[0] === "window" ? `${50 + this.textBoxX}%` : "0%";
+ overlay.style.top = this.appendTarget[0] === "window" ? `${50 + this.textBoxY}%` : "0%";
+
+ const focusBG = document.createElement("div");
+ focusBG.style.cssText = "pointer-events: auto; position: fixed; width: 100%; height: 100%; background-color: rgba(0, 0, 0, 0.5); z-index: 9998;";
+ focusBG.className = "SP-ask-boxBG";
+ focusBG.id = this.appendTarget[0];
+ focusBG.style.left = this.appendTarget[0] === "window" ? "0%" : "-50%";
+ focusBG.style.top = this.appendTarget[0] === "window" ? "0%" : "-50%";
+
+ laidImgContain = document.createElement("div");
+ laidImgContain.style.width = "100%";
+ laidImgContain.style.height = "100%";
+ laidImgContain.style.position = "absolute";
+ laidImgContain.style.top = 0;
+ laidImgContain.style.left = 0;
+ laidImgContain.style.zIndex = "-1";
+ if (this.forceInput !== "Disabled") {
+ const overlayInput = this.forceInput === "Enter Key" ? "Enter" : this.forceInput === "Shift + Enter Key" ? "ShiftEnter" : this.forceInput;
+ const handleKeydown = (event) => {
+ if ((overlayInput === "ShiftEnter" && event.shiftKey && event.key === "Enter") || event.key === overlayInput) {
+ setInpValue(inputField.value);
+ this.closeOverlay(overlay, hasDecreased);
+ hasDecreased = true;
+ resolve();
+ }
+ };
+ const observer = new MutationObserver((mutationsList) => {
+ for (const mutation of mutationsList) {
+ if (mutation.type === "childList" && !document.contains(overlay)) {
+ document.removeEventListener("keydown", handleKeydown);
+ observer.disconnect();
+ }
+ }
+ });
+ observer.observe(document.body, { childList: true });
+ document.addEventListener("keydown", handleKeydown);
+ }
+
+ const questionText = document.createElement("div");
+ questionText.classList.add("question");
+ questionText.style.fontSize = this.fontSize;
+ if (this.uiOrder[0] !== "question") questionText.style.marginTop = "10px";
+ if (this.uiOrder[0] === "question") questionText.style.marginBottom = "10px";
+ questionText.innerHTML = xmlEscape(question).replace(/\n/g, "
");
+
+ const inputField = document.createElement(this.inputType === "Text Area" ? "textarea" : "input");
+ inputField.style.display = this.inputType ? "block" : "none";
+ inputField.style.fontSize = this.fontSize;
+ inputField.style.margin = "0 auto";
+ if (this.inputType !== "Text Area") inputField.type = this.inputType.toLowerCase();
+ const setInpValue = (val) => {
+ inputField.value = val;
+ if (this.askBoxInfo[1] == 1) this.userInput = inputField.value;
+ else {
+ const newInput = [...this.userInput];
+ newInput[index] = inputField.value;
+ this.userInput = newInput;
+ }
+ }
+ inputField.addEventListener("input", () => { setInpValue(inputField.value) });
+ const btnContain = document.createElement("div");
+ btnContain.classList.add("button-container");
+ for (const buttonName in this.buttonJSON) {
+ const btnInfo = this.buttonJSON[buttonName];
+ if (btnInfo.name.includes("")) btnContain.appendChild(document.createElement("br"));
+ else {
+ const btn = document.createElement("button");
+ if (this.uiOrder[0] !== "buttons") btn.style.marginTop = "10px";
+ if (this.uiOrder[2] !== "buttons") btn.style.marginBottom = "10px";
+ btn.style.marginRight = "5px";
+ btn.style.cursor = "pointer";
+ btn.innerHTML = xmlEscape(btnInfo.name).replace(/\n/g, "
");
+ btn.style.display = "inline-block";
+ btn.addEventListener("click", () => {
+ this.lastPressBtn = btnInfo.name;
+ setInpValue(this.inputType === "Disabled" ? btnInfo.name : this.userInput);
+ this.closeOverlay(overlay, hasDecreased);
+ hasDecreased = true;
+ resolve();
+ });
+ btnContain.appendChild(btn);
+ }
+ }
+ let dropdwnCont, dropdwnBtn, sliderContain, valTxt;
+ if (this.inputType.includes("Dropdown")) {
+ const dropdown = document.createElement("div");
+ dropdown.className = "dropdown";
+ if (this.inputType === "Single Dropdown") {
+ dropdwnBtn = document.createElement("select");
+ this.optionList.forEach((label) => {
+ let opt = document.createElement("option");
+ opt.value = label; opt.text = label;
+ dropdwnBtn.appendChild(opt);
+ });
+ dropdwnBtn.addEventListener("input", () => { setInpValue(dropdwnBtn.value) });
+ dropdwnBtn.value = this.defaultValue || dropdwnBtn.value;
+ setInpValue(dropdwnBtn.value);
+ } else {
+ const isMulti = this.inputType.includes("Multi-Select");
+ let defaultOpts = [];
+ if (isMulti) {
+ try {
+ defaultOpts = JSON.parse(this.defaultValue);
+ selectOpts = defaultOpts;
+ } catch {}
+ }
+ dropdwnBtn = document.createElement("button");
+ dropdwnBtn.className = "dropbtn";
+ dropdwnBtn.innerHTML = xmlEscape(this.DropdownText).replace(/\n/g, "
");
+ dropdwnCont = document.createElement("div");
+ dropdwnCont.id = "myDropdown";
+ dropdwnCont.className = "dropdown-content";
+ dropdwnCont.style.display = "none";
+ this.optionList.forEach((label, index) => {
+ const optTxt = document.createElement("label");
+ optTxt.style.color = this.questionColor;
+ optTxt.textContent = "";
+ const optRadio = document.createElement("input");
+ optRadio.type = this.inputType === "Dropdown" ? "radio" : "checkbox";
+ if (isMulti) optRadio.checked = defaultOpts.indexOf(label) > -1;
+ else optRadio.checked = label === this.defaultValue;
+ optRadio.name = "dropdownOptions";
+ optRadio.value = index;
+ optRadio.classList.add("dropdown-radio");
+ optRadio.addEventListener("click", () => {
+ if (isMulti) {
+ if (selectOpts.includes(label)) selectOpts = selectOpts.filter(item => item !== label);
+ else selectOpts.push(label);
+ inputField.value = selectOpts.length > 0 ? JSON.stringify(selectOpts) : "";
+ } else { inputField.value = label }
+ setInpValue(inputField.value)
+ });
+ optTxt.append(optRadio, document.createTextNode(" " + label), document.createElement("br"));
+ dropdwnCont.appendChild(optTxt);
+ });
+ dropdwnBtn.addEventListener("click", () => {
+ this.lastPressBtn = this.DropdownText;
+ dropdwnCont.style.display = this.isDropdownOpen ? "none" : "block";
+ this.isDropdownOpen = !this.isDropdownOpen;
+ });
+ setInpValue(this.defaultValue);
+ }
+ } else if (this.inputType.includes("Slider")) {
+ sliderContain = document.createElement("div");
+ sliderContain.classList.add("slider-container");
+ const slider = document.createElement("input");
+ slider.type = "range";
+ slider.min = this.sliderInfo[0]; slider.max = this.sliderInfo[1]; slider.value = this.sliderInfo[2];
+ if (this.inputType.includes("Vertical")) {
+ slider.style.writingMode = "vertical-lr";
+ slider.style.direction = "rtl";
+ }
+ sliderContain.appendChild(slider);
+ valTxt = document.createElement("span");
+ valTxt.classList.add("slider-value");
+ sliderContain.appendChild(valTxt);
+ valTxt.style.color = this.questionColor;
+ valTxt.textContent = slider.value;
+ slider.addEventListener("input", () => {
+ valTxt.textContent = slider.value;
+ setInpValue(valTxt.textContent);
+ });
+ setInpValue(valTxt.textContent);
+ }
+ for (const item of this.uiOrder) {
+ switch (item) {
+ case "question": { overlay.appendChild(questionText); break }
+ case "input":
+ if (this.inputType !== "Disabled") {
+ const createBr = () => { return document.createElement("br") };
+ if (this.inputType === "Single Dropdown") overlay.append(dropdwnBtn, createBr());
+ else if (this.inputType.includes("Dropdown")) overlay.append(dropdwnBtn, dropdwnCont, createBr());
+ else if (this.inputType.includes("Slider")) overlay.append(sliderContain, valTxt, createBr());
+ else {
+ setInpValue(this.defaultValue);
+ overlay.appendChild(inputField);
+ }
+ }
+ break;
+ case "buttons": { overlay.appendChild(btnContain); break }
+ }
+ }
+ overlay.appendChild(laidImgContain);
+ if (this.appendTarget[0] === "window") {
+ document.body.appendChild(overlay);
+ if (this.appendTarget[1]) document.body.appendChild(focusBG);
+ }
+ inputField.focus();
+ this.activeOverlays.push(overlay);
+ if (this.appendTarget[0] === "window") {
+ const resizeHandler = () => {
+ overlay.style.left = `${this.textBoxX !== null ? 50 + this.textBoxX : 50}%`;
+ overlay.style.top = `${this.textBoxY !== null ? 50 + this.textBoxY : 50}%`;
+ };
+ document.addEventListener("fullscreenchange", resizeHandler);
+ document.addEventListener("webkitfullscreenchange", resizeHandler);
+ document.addEventListener("mozfullscreenchange", resizeHandler);
+ document.addEventListener("MSFullscreenChange", resizeHandler);
+ const observer = new MutationObserver((mutationsList) => {
+ for (const mutation of mutationsList) {
+ if (mutation.type === "childList" && Array.from(mutation.removedNodes).includes(overlay)) {
+ document.removeEventListener("fullscreenchange", resizeHandler);
+ document.removeEventListener("webkitfullscreenchange", resizeHandler);
+ document.removeEventListener("mozfullscreenchange", resizeHandler);
+ document.removeEventListener("MSFullscreenChange", resizeHandler);
+ observer.disconnect();
+ }
+ }
+ });
+ observer.observe(overlay.parentNode, { childList: true });
+ document.body.appendChild(overlay);
+ } else {
+ if (this.appendTarget[1]) vm.renderer.addOverlay(focusBG, "scale-centered");
+ vm.renderer.addOverlay(overlay, "scale-centered");
+ }
+ inputField.focus();
+ if (this.appendTarget[0] === "window") overlay.style.zIndex = "9999";
+ else overlay.parentNode.style.zIndex = "9999";
+ this.updateOverlay(overlay);
+ });
+ }
+ }
+ closeOverlay(overlay, doneBefore) {
+ this.isDropdownOpen = false;
+ if (!doneBefore) {
+ this.askBoxInfo[0]--;
+ let usedBG = document.querySelectorAll(`div[class="SP-ask-boxBG"]`);
+ usedBG = usedBG[usedBG.length - 1];
+ // ^ Prioritizes Textboxes on Window
+ const index = this.activeOverlays.indexOf(overlay);
+ setTimeout(() => {
+ if (index !== -1) {
+ this.activeOverlays.splice(index, 1);
+ this.askBoxPromises.splice(index, 1);
+ }
+ if (this.appendTarget[0] === "window") document.body.removeChild(overlay);
+ else vm.renderer.removeOverlay(overlay);
+ if (usedBG) {
+ if (usedBG.id === "window") document.body.removeChild(usedBG);
+ else vm.renderer.removeOverlay(usedBG);
+ }
+ }, this.Timeout * 1000);
+ }
+ }
+
+ setButton(args) {
+ if (args.BUTTON === "add") {
+ this.buttonJSON[args.NAME] = {
+ borderRadius: 5, border: "1px none #000000",
+ color: "#0074D9", textColor: "#ffffff",
+ name: args.NAME, padding: "5px 10px",
+ image: "", imgScale: 100,
+ dropShadow: "none", outline: ["", 0]
+ };
+ } else { delete this.buttonJSON[args.NAME] }
+ vm.extensionManager.refreshBlocks();
+ }
+
+ deleteAllButtons() {
+ this.buttonJSON = {};
+ vm.extensionManager.refreshBlocks();
+ }
+
+ lastButton() { return this.lastPressBtn }
+
+ isWaitingInput() { return this.activeOverlays.length > 0 }
+
+ isDropdown() { return this.isDropdownOpen }
+
+ setMaxBoxCount(args) {
+ this.askBoxInfo[1] = Scratch.Cast.toNumber(args.MAX);
+ if (this.askBoxInfo[1] > 1 && !Array.isArray(this.userInput)) this.userInput = [this.userInput];
+ }
+
+ setTimeout(args) { this.Timeout = Scratch.Cast.toNumber(args.TIME) }
+
+ reportTimeout() { return this.Timeout }
+
+ getUserInput() {
+ if (this.askBoxInfo[1] > 1) return this.userInput === null ? "[]" : JSON.stringify(this.userInput);
+ else return this.userInput === null ? "" : this.userInput;
+ }
+
+ getBoxInfo(args) {
+ if (args.INFO.includes("button")) {
+ const buttons = Object.keys(this.buttonJSON);
+ return args.INFO.includes("names") ? JSON.stringify(buttons) : buttons.length;
+ } else { return this.askBoxInfo[args.INFO === "count" ? 0 : 1] }
+ }
+
+ setSubmitEvent(args) { this.forceInput = args.ENTER }
+
+ setDefaultV(args) { this.defaultValue = args.defaultV }
+
+ setAppend(args) { this.appendTarget[0] = args.TARGET }
+ setFocus(args) { this.appendTarget[1] = args.TYPE === "Enabled" }
+
+ setUI(args) {
+ let array;
+ try { array = JSON.parse(args.ARRAY.toLowerCase()) } catch { return }
+ if (!Array.isArray(array)) return;
+ const allowedUI = ["question", "input", "buttons"];
+ let filteredArray = [...new Set(array.filter(element => allowedUI.includes(element)))];
+ allowedUI.forEach(element => {
+ if (!filteredArray.includes(element)) filteredArray.push(element);
+ });
+ this.uiOrder = filteredArray;
+ }
+
+ getUIOrder() { return JSON.stringify(this.uiOrder) }
+ }
+
+ Scratch.extensions.register(new BetterInputSP());
+})(Scratch);
diff --git a/extensions/extensions.json b/extensions/extensions.json
index 58a0294d03..d872c652da 100644
--- a/extensions/extensions.json
+++ b/extensions/extensions.json
@@ -56,6 +56,7 @@
"Lily/Cast",
"-SIPC-/time",
"-SIPC-/consoles",
+ "SharkPool/Better-Input",
"ZXMushroom63/searchApi",
"TheShovel/ShovelUtils",
"Lily/Assets",
diff --git a/images/SharkPool/Better-Input.svg b/images/SharkPool/Better-Input.svg
new file mode 100644
index 0000000000..25557244fe
--- /dev/null
+++ b/images/SharkPool/Better-Input.svg
@@ -0,0 +1,65 @@
+
+