From 20fbb438a204d1d799d794807c161935b97c36e1 Mon Sep 17 00:00:00 2001 From: HanHanDeYaYa Date: Mon, 10 Jun 2024 21:59:39 +0800 Subject: [PATCH 1/8] =?UTF-8?q?=E5=88=9B=E5=BB=BA=20Variable=20beautificat?= =?UTF-8?q?ion.js?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Variable beautification.js | 783 ++++++++++++++++++ 1 file changed, 783 insertions(+) create mode 100644 Variable beautification/Variable beautification.js diff --git a/Variable beautification/Variable beautification.js b/Variable beautification/Variable beautification.js new file mode 100644 index 00000000..59cbfe15 --- /dev/null +++ b/Variable beautification/Variable beautification.js @@ -0,0 +1,783 @@ +// Name: Custom Styles +// ID: shovelcss +// Description: Customize the appearance of variable monitors and prompts in your project. +// By: TheShovel +// License: MIT + +// Thanks LilyMakesThings for the awesome banner! +(function (Scratch) { + "use strict"; + + // Styles + let monitorText = ""; + let monitorBorder = ""; + let monitorBackgroundColor = ""; + let variableValueBackground = ""; + let variableValueTextColor = ""; + let listFooterBackground = ""; + let listHeaderBackground = ""; + let listValueText = ""; + let listValueBackground = ""; + let variableValueRoundness = -1; + let listValueRoundness = -1; + let monitorBackgroundRoundness = -1; + let monitorBackgroundBorderWidth = -1; + let allowScrolling = ""; + let askBackground = ""; + let askBackgroundRoundness = -1; + let askBackgroundBorderWidth = -1; + let askButtonBackground = ""; + let askButtonRoundness = -1; + let askInputBackground = ""; + let askInputRoundness = -1; + let askInputBorderWidth = -1; + let askBoxIcon = ""; + let askInputText = ""; + let askQuestionText = ""; + let askButtonImage = ""; + let askInputBorder = ""; + + // CSS selectors + let monitorRoot; + let monitorValue; + let monitorListHeader; + let monitorListFooter; + let monitorRowValueOuter; + let monitorRowsInner; + let monitorRowsScroller; + let monitorRowIndex; + let monitorValueLarge; + let askBoxBG; + let askBoxButton; + let askBoxInner; + let askBoxText; + let askBoxBorderMain; + let askBoxBorderOuter; + if (typeof scaffolding !== "undefined") { + monitorRoot = ".sc-monitor-root"; + monitorValue = ".sc-monitor-value"; + monitorListHeader = ".sc-monitor-list-label"; + monitorListFooter = ".sc-monitor-list-footer"; + monitorRowValueOuter = ".sc-monitor-row-value-outer"; + monitorRowsInner = ".sc-monitor-rows-inner"; + monitorRowsScroller = monitorRowsInner; + monitorRowIndex = ".sc-monitor-row-index"; + monitorValueLarge = ".sc-monitor-large-value"; + askBoxBG = ".sc-question-inner"; + askBoxButton = ".sc-question-submit-button"; + askBoxInner = ".sc-question-input"; + askBoxText = ".sc-question-text"; + askBoxBorderMain = ".sc-question-input:hover"; + askBoxBorderOuter = ".sc-question-input:focus"; + } else { + monitorRoot = 'div[class^="monitor_monitor-container_"]'; + monitorValue = 'div[class^="monitor_value_"]'; + monitorListHeader = 'div[class^="monitor_list-header_"]'; + monitorListFooter = 'div[class^="monitor_list-footer_"]'; + monitorRowValueOuter = 'div[class^="monitor_list-value_"]'; + monitorRowsInner = 'div[class^="monitor_list-body_"]'; + monitorRowsScroller = + 'div[class^="monitor_list-body_"] > .ReactVirtualized__List'; + monitorRowIndex = 'div[class^="monitor_list-index_"]'; + monitorValueLarge = 'div[class^="monitor_large-value_"]'; + askBoxBG = 'div[class^="question_question-container_"]'; + askBoxButton = 'button[class^="question_question-submit-button_"]'; + askBoxInner = + '[class^="question_question-container_"] input[class^="input_input-form_"]'; + askBoxText = + '[class^="question_question-container_"] div[class^="question_question-label_"]'; + askBoxIcon = 'img[class^="question_question-submit-button-icon_"]'; + askBoxBorderMain = + '[class^="question_question-input_"] input:focus, [class^="question_question-input_"] input:hover'; + askBoxBorderOuter = '[class^="question_question-input_"] > input:focus'; + } + + const ColorIcon = + ""; + const BorderIcon = + ""; + const extensionIcon = + ""; + const miscIcon = + ""; + const TransparentIcon = + ""; + const GradientIcon = + ""; + const PictureIcon = + ""; + const ResetIcon = + ""; + + const stylesheet = document.createElement("style"); + stylesheet.className = "shovelcss-style"; + // end of for higher precedence than other sheets + document.body.appendChild(stylesheet); + + const applyCSS = () => { + let css = ""; + + // We assume all values are sanitized when they are set, so then we can just use them as-is here. + + if (monitorText) { + css += `${monitorRoot}, ${monitorListFooter}, ${monitorListHeader}, ${monitorRowIndex} { color: ${monitorText}; }`; + } + if (monitorBackgroundColor) { + css += `${monitorRoot}, ${monitorRowsInner} { background: ${monitorBackgroundColor}; }`; + } + if (monitorBorder) { + css += `${monitorRoot} { border-color: ${monitorBorder}; }`; + } + if (monitorBackgroundRoundness >= 0) { + css += `${monitorRoot} { border-radius: ${monitorBackgroundRoundness}px; }`; + } + if (monitorBackgroundBorderWidth >= 0) { + css += `${monitorRoot} { border-width: ${monitorBackgroundBorderWidth}px; }`; + } + if (variableValueBackground) { + css += `${monitorValue}, ${monitorValueLarge} { background: ${variableValueBackground} !important; }`; + } + if (variableValueTextColor) { + css += `${monitorValue}, ${monitorValueLarge} { color: ${variableValueTextColor}; }`; + } + if (variableValueRoundness >= 0) { + css += `${monitorValue} { border-radius: ${variableValueRoundness}px; }`; + } + if (listHeaderBackground) { + css += `${monitorListHeader} { background: ${listHeaderBackground}; }`; + } + if (listFooterBackground) { + css += `${monitorListFooter} { background: ${listHeaderBackground}; }`; + } + if (listValueBackground) { + css += `${monitorRowValueOuter} { background: ${listValueBackground} !important; }`; + } + if (listValueText) { + css += `${monitorRowValueOuter} { color: ${listValueText}; }`; + } + if (listValueRoundness >= 0) { + css += `${monitorRowValueOuter} { border-radius: ${listValueRoundness}px; }`; + } + if (allowScrolling) { + css += `${monitorRowsScroller} { overflow: ${allowScrolling} !important; }`; + } + if (askBackground) { + css += `${askBoxBG} { background: ${askBackground} !important; border: none !important; }`; + } + if (askBackgroundRoundness >= 0) { + css += `${askBoxBG} { border-radius: ${askBackgroundRoundness}px !important; }`; + } + if (askBackgroundBorderWidth >= 0) { + css += `${askBoxBG} { border-width: ${askBackgroundBorderWidth}px !important; }`; + } + if (askButtonBackground) { + css += `${askBoxButton} { background-color: ${askButtonBackground}; }`; + } + if (askButtonRoundness >= 0) { + css += `${askBoxButton} { border-radius: ${askButtonRoundness}px !important; }`; + } + if (askInputBackground) { + css += `${askBoxInner} { background: ${askInputBackground} !important; }`; + css += `${askBoxInner} { border: none !important; }`; + } + if (askInputText) { + css += `${askBoxInner} { color: ${askInputText} !important; }`; + } + if (askQuestionText) { + css += `${askBoxText} { color: ${askQuestionText} !important; }`; + } + if (askInputRoundness >= 0) { + css += `${askBoxInner} { border-radius: ${askInputRoundness}px !important; }`; + } + if (askInputBorderWidth >= 0) { + css += `${askBoxInner} { border-width: ${askInputBorderWidth}px !important; }`; + } + if (askButtonImage) { + css += `${askBoxButton} { background-image: url("${encodeURI( + askButtonImage + )}") !important; background-repeat: no-repeat; background-size: contain; }`; + css += `${askBoxIcon} { visibility: hidden; }`; + } + if (askInputBorder) { + css += `${askBoxBorderMain}, ${askBoxBorderOuter} { border-color: ${askInputBorder} !important; }`; + css += `${askBoxBorderOuter} { box-shadow: none !important; }`; + } + + stylesheet.textContent = css; + }; + + const resetStyles = () => { + monitorText = ""; + monitorBorder = ""; + monitorBackgroundColor = ""; + variableValueBackground = ""; + variableValueTextColor = ""; + listFooterBackground = ""; + listHeaderBackground = ""; + listValueText = ""; + listValueBackground = ""; + variableValueRoundness = -1; + listValueRoundness = -1; + monitorBackgroundRoundness = -1; + monitorBackgroundBorderWidth = -1; + allowScrolling = ""; + askBackground = ""; + askBackgroundRoundness = -1; + askBackgroundBorderWidth = -1; + askButtonBackground = ""; + askButtonRoundness = -1; + askInputBackground = ""; + askInputRoundness = -1; + askInputBorderWidth = -1; + askBoxIcon = ""; + askInputText = ""; + askQuestionText = ""; + askButtonImage = ""; + askInputBorder = ""; + + applyCSS(); + }; + + const getMonitorRoot = (id) => { + const allMonitors = document.querySelectorAll(monitorRoot); + for (const monitor of allMonitors) { + if (monitor.dataset.id === id) { + return monitor; + } + } + return null; + }; + + /** + * @param {string} id + * @param {number} x + * @param {number} y + */ + const setMonitorPosition = (id, x, y) => { + const root = getMonitorRoot(id); + if (root) { + root.style.transform = `translate(${x}px, ${y}px)`; + root.style.left = "0px"; + root.style.top = "0px"; + } + }; + + /** + * @param {VM.Target} target + * @param {string} name + * @param {VM.VariableType} type + * @param {number} x + * @param {number} y + */ + const setVariableMonitorPosition = (target, name, type, x, y) => { + // @ts-expect-error + const variable = target.lookupVariableByNameAndType(name, type); + if (variable) { + // @ts-expect-error + setMonitorPosition(variable.id, x, y); + } + }; + + const parseColor = (color, callback) => { + color = Scratch.Cast.toString(color); + + // These might have some exponential backtracking/ReDoS, but that's not really a concern here. + // If a project wanted to get stuck in an infinite loop, there are so many other ways to do that. + + // Simple color code or name + if (/^#?[a-z0-9]+$/.test(color)) { + callback(color); + return; + } + + // General gradient pattern + if (/^[a-z-]+-gradient\([a-z0-9,#%. ]+\)$/i.test(color)) { + callback(color); + return; + } + + // URL + // see list of non-escaped characters: + // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/encodeURI#description + const match = color.match(/^url\("([A-Za-z0-9\-_.!~*'();/?:@&=+$,#]+)"\)$/); + if (match) { + const url = match[1]; + return Scratch.canFetch(url).then((allowed) => { + if (allowed) { + callback(color); + } + }); + } + + console.error("Invalid color", color); + }; + + Scratch.vm.runtime.on("RUNTIME_DISPOSED", resetStyles); + + class MonitorStyles { + getInfo() { + return { + id: "shovelcss", + name: "Custom Styles", + menuIconURI: extensionIcon, + color1: "#0072d6", + color2: "#0064bc", + color3: "#01539b", + blocks: [ + { + blockIconURI: ColorIcon, + opcode: "changecss", + blockType: Scratch.BlockType.COMMAND, + text: "set [COLORABLE] to [COLOR]", + arguments: { + COLORABLE: { + type: Scratch.ArgumentType.STRING, + menu: "COLORABLE_MENU", + }, + COLOR: { + type: Scratch.ArgumentType.COLOR, + defaultValue: "#ff0000", + }, + }, + }, + { + blockIconURI: GradientIcon, + opcode: "gradientAngle", + blockType: Scratch.BlockType.REPORTER, + text: "make a gradient with [COLOR1] and [COLOR2] at angle [ANGLE]", + arguments: { + COLOR1: { + type: Scratch.ArgumentType.COLOR, + defaultValue: "#ff0000", + }, + COLOR2: { + type: Scratch.ArgumentType.COLOR, + defaultValue: "#6ed02d", + }, + ANGLE: { + type: Scratch.ArgumentType.ANGLE, + defaultValue: "90", + }, + }, + }, + { + blockIconURI: TransparentIcon, + disableMonitor: true, + opcode: "transparentinput", + blockType: Scratch.BlockType.REPORTER, + text: "transparent", + }, + { + blockIconURI: PictureIcon, + disableMonitor: true, + opcode: "pictureinput", + blockType: Scratch.BlockType.REPORTER, + text: "image [URL]", + arguments: { + URL: { + type: Scratch.ArgumentType.STRING, + defaultValue: "https://extensions.turbowarp.org/dango.png", + }, + }, + }, + "---", + { + blockIconURI: PictureIcon, + disableMonitor: true, + opcode: "setAskURI", + blockType: Scratch.BlockType.COMMAND, + text: "set ask prompt button image to [URL]", + arguments: { + URL: { + type: Scratch.ArgumentType.STRING, + defaultValue: "https://extensions.turbowarp.org/dango.png", + }, + }, + }, + "---", + { + blockIconURI: BorderIcon, + opcode: "setbordersize", + blockType: Scratch.BlockType.COMMAND, + text: "set border width of [BORDER] to [SIZE]", + arguments: { + BORDER: { + type: Scratch.ArgumentType.STRING, + menu: "BORDER_WIDTH_MENU", + }, + SIZE: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "2", + }, + }, + }, + { + blockIconURI: BorderIcon, + opcode: "setborderradius", + blockType: Scratch.BlockType.COMMAND, + text: "set roundness of [CORNER] to [SIZE]", + arguments: { + SIZE: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "4", + }, + CORNER: { + type: Scratch.ArgumentType.STRING, + menu: "BORDER_ROUNDNESS_MENU", + }, + }, + }, + "---", + { + blockIconURI: ResetIcon, + opcode: "clearCSS", + blockType: Scratch.BlockType.COMMAND, + text: "reset styles", + }, + "---", + { + blockIconURI: miscIcon, + opcode: "allowscrollrule", + blockType: Scratch.BlockType.COMMAND, + text: "set list scrolling to [SCROLLRULE]", + arguments: { + SCROLLRULE: { + type: Scratch.ArgumentType.STRING, + menu: "SCROLL_MENU", + }, + }, + }, + { + blockIconURI: miscIcon, + opcode: "getValue", + blockType: Scratch.BlockType.REPORTER, + text: "get [ITEM]", + arguments: { + ITEM: { + type: Scratch.ArgumentType.STRING, + menu: "VALUEGET_LIST", + }, + }, + }, + "---", + { + blockIconURI: miscIcon, + opcode: "setvarpos", + blockType: Scratch.BlockType.COMMAND, + text: "set position of variable [NAME] to x: [X] y: [Y]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my variable", + }, + }, + }, + { + blockIconURI: miscIcon, + opcode: "setlistpos", + blockType: Scratch.BlockType.COMMAND, + text: "set position of list [NAME] to x: [X] y: [Y]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my variable", + }, + }, + }, + ], + // Accepting reporters because there can't be errors in case the value is not correct + menus: { + COLORABLE_MENU: { + acceptReporters: true, + items: [ + "monitor text", + "monitor background", + "monitor border", + "variable value background", + "variable value text", + "list header background", + "list footer background", + "list value background", + "list value text", + "ask prompt background", + "ask prompt button background", + "ask prompt input background", + "ask prompt question text", + "ask prompt input text", + "ask prompt input border", + ], + }, + BORDER_WIDTH_MENU: { + acceptReporters: true, + items: [ + "monitor background", + "ask prompt background", + "ask prompt input", + ], + }, + BORDER_ROUNDNESS_MENU: { + acceptReporters: true, + items: [ + "monitor background", + "variable value", + "list value", + "ask prompt background", + "ask prompt button", + "ask prompt input", + ], + }, + SCROLL_MENU: { + acceptReporters: true, + items: ["enabled", "disabled"], + }, + VALUEGET_LIST: { + acceptReporters: true, + items: [ + "monitor text", + "monitor background", + "monitor border color", + "variable value background", + "variable value text", + "list header background", + "list footer background", + "list value background", + "list value text", + "ask prompt background", + "ask prompt button background", + "ask prompt input background", + "ask prompt question text", + "ask prompt input text", + "ask prompt input border", + "monitor background border width", + "ask prompt background border width", + "ask prompt input border width", + "monitor background roundness", + "variable value roundness", + "list value roundness", + "ask prompt background roundness", + "ask prompt button roundness", + "ask prompt input roundness", + "ask prompt button image", + "list scroll rule", + ], + }, + }, + }; + } + + changecss(args) { + return parseColor(args.COLOR, (color) => { + if (args.COLORABLE === "monitor text") { + monitorText = color; + } else if (args.COLORABLE === "monitor background") { + monitorBackgroundColor = color; + } else if (args.COLORABLE === "monitor border") { + monitorBorder = color; + } else if (args.COLORABLE === "variable value background") { + variableValueBackground = color; + } else if (args.COLORABLE === "variable value text") { + variableValueTextColor = color; + } else if (args.COLORABLE === "list header background") { + listHeaderBackground = color; + } else if (args.COLORABLE === "list footer background") { + listFooterBackground = color; + } else if (args.COLORABLE === "list value background") { + listValueBackground = color; + } else if (args.COLORABLE === "list value text") { + listValueText = color; + } else if (args.COLORABLE === "ask prompt background") { + askBackground = color; + } else if (args.COLORABLE === "ask prompt button background") { + askButtonBackground = color; + } else if (args.COLORABLE === "ask prompt input background") { + askInputBackground = color; + } else if (args.COLORABLE === "ask prompt question text") { + askQuestionText = color; + } else if (args.COLORABLE === "ask prompt input text") { + askInputText = color; + } else if (args.COLORABLE === "ask prompt input border") { + askInputBorder = color; + } + + applyCSS(); + }); + } + + gradientAngle(args) { + return ( + "linear-gradient(" + + args.ANGLE + + "deg," + + args.COLOR1 + + "," + + args.COLOR2 + + ")" + ); + } + + setbordersize(args) { + const size = Scratch.Cast.toNumber(args.SIZE); + if (args.BORDER === "monitor background") { + monitorBackgroundBorderWidth = size; + } else if (args.BORDER === "ask prompt background") { + askBackgroundBorderWidth = size; + } else if (args.BORDER === "ask prompt input") { + askInputBorderWidth = size; + } + applyCSS(); + } + + setborderradius(args) { + const size = Scratch.Cast.toNumber(args.SIZE); + if (args.CORNER === "monitor background") { + monitorBackgroundRoundness = size; + } else if (args.CORNER === "variable value") { + variableValueRoundness = size; + } else if (args.CORNER === "list value") { + listValueRoundness = size; + } else if (args.CORNER === "ask prompt background") { + askBackgroundRoundness = size; + } else if (args.CORNER === "ask prompt button") { + askButtonRoundness = size; + } else if (args.CORNER === "ask prompt input") { + askInputRoundness = size; + } + applyCSS(); + } + + allowscrollrule(args) { + if (args.SCROLLRULE === "enabled") { + allowScrolling = "auto"; + } else { + allowScrolling = "hidden"; + } + applyCSS(); + } + + setvarpos(args, util) { + setVariableMonitorPosition( + util.target, + args.NAME, + "", + Scratch.Cast.toNumber(args.X) + Scratch.vm.runtime.stageWidth / 2, + Scratch.vm.runtime.stageHeight / 2 - Scratch.Cast.toNumber(args.Y) + ); + } + + setlistpos(args, util) { + setVariableMonitorPosition( + util.target, + args.NAME, + "list", + Scratch.Cast.toNumber(args.X) + Scratch.vm.runtime.stageWidth / 2, + Scratch.vm.runtime.stageHeight / 2 - Scratch.Cast.toNumber(args.Y) + ); + } + + help() { + alert( + "\nThis is a short introduction to how to use the Monitor Styles extension!\n\n๐—Ÿ๐—ผ๐—ผ๐—ธ๐˜€ ๐—ฏ๐—น๐—ผ๐—ฐ๐—ธ๐˜€\nThese blocks change the appearance of the variable and list didsplays. You can use the drop-down menu to select what component you want to modify. ๐™๐™๐™š ๐™˜๐™ค๐™ก๐™ค๐™ง ๐™—๐™ก๐™ค๐™˜๐™  modifieas the color of a component. You can use the ๐™œ๐™ง๐™–๐™™๐™ž๐™š๐™ฃ๐™ฉ block inside the color input, to create gradients or the ๐™„๐™ข๐™–๐™œ๐™š block to use a image instead of solid colors. ๐™๐™๐™š๐™จ๐™š ๐™ฉ๐™ฌ๐™ค ๐™ค๐™ฃ๐™ก๐™ฎ ๐™ฌ๐™ค๐™ง๐™  ๐™ค๐™ฃ ๐™˜๐™š๐™ง๐™ฉ๐™–๐™ž๐™ฃ ๐™˜๐™ค๐™ข๐™ฅ๐™ค๐™ฃ๐™š๐™ฃ๐™ฉ๐™จ! You can also use the ๐™ฉ๐™ง๐™–๐™ฃ๐™จ๐™ฅ๐™–๐™ง๐™š๐™ฃ๐™ฉ ๐™—๐™ก๐™ค๐™˜๐™  as a color input, to make components invisible. The ๐™—๐™ค๐™ง๐™™๐™š๐™ง ๐™—๐™ก๐™ค๐™˜๐™ ๐™จ modify the borders of components.\n\n๐—ฆ๐—ฒ๐—ป๐˜€๐—ถ๐—ป๐—ด ๐—ฏ๐—น๐—ผ๐—ฐ๐—ธ๐˜€\nThese blocks can change the behaviour of certain components. The ๐™จ๐™˜๐™ง๐™ค๐™ก๐™ก ๐™ง๐™ช๐™ก๐™š block change the behaviour for lists. On 'auto' they will show the scroll bar, and allow you to school, but on 'hidden', they won't let you do that, and the scroll bar will be hidden.\n\n๐— ๐—ผ๐˜๐—ถ๐—ผ๐—ป ๐—ฏ๐—น๐—ผ๐—ฐ๐—ธ๐˜€\nThese blocks allow you to move variable and list displays around. You need to use their ๐™ก๐™–๐™—๐™š๐™ก ๐™ฃ๐™–๐™ข๐™š. The label name is the text that displays on the monitor. For example, a 'for this sprite only' variable will be like 'Sprite1: my variable'." + ); + } + + transparentinput() { + return "transparent"; + } + + pictureinput(args) { + return `url("${encodeURI(args.URL)}")`; + } + + clearCSS() { + resetStyles(); + } + + getValue(args) { + if (args.ITEM === "monitor text") { + return monitorText; + } else if (args.ITEM === "monitor background") { + return monitorBackgroundColor; + } else if (args.ITEM === "monitor border color") { + return monitorBorder; + } else if (args.ITEM === "variable value background") { + return variableValueBackground; + } else if (args.ITEM === "variable value text") { + return variableValueTextColor; + } else if (args.ITEM === "list header background") { + return listHeaderBackground; + } else if (args.ITEM === "list footer background") { + return listFooterBackground; + } else if (args.ITEM === "list value background") { + return listValueBackground; + } else if (args.ITEM === "list value text") { + return listValueText; + } else if (args.ITEM === "ask prompt background") { + return askBackground; + } else if (args.ITEM === "ask prompt button background") { + return askButtonBackground; + } else if (args.ITEM === "ask prompt input background") { + return askInputBackground; + } else if (args.ITEM === "ask prompt question text") { + return askQuestionText; + } else if (args.ITEM === "ask prompt input text") { + return askInputText; + } else if (args.ITEM === "ask prompt input border") { + return askInputBorder; + } else if (args.ITEM === "monitor background border width") { + return monitorBackgroundBorderWidth; + } else if (args.ITEM === "ask prompt background border width") { + return askBackgroundBorderWidth; + } else if (args.ITEM === "ask prompt input border width") { + return askInputBorderWidth; + } else if (args.ITEM === "monitor background roundness") { + return monitorBackgroundRoundness; + } else if (args.ITEM === "variable value roundness") { + return variableValueRoundness; + } else if (args.ITEM === "list value roundness") { + return listValueRoundness; + } else if (args.ITEM === "ask prompt background roundness") { + return askBackgroundRoundness; + } else if (args.ITEM === "ask prompt button roundness") { + return askButtonRoundness; + } else if (args.ITEM === "ask prompt input roundness") { + return askInputRoundness; + } else if (args.ITEM === "ask prompt button image") { + return askButtonImage; + } else if (args.ITEM === "list scrolling") { + if (allowScrolling === "auto") { + return "enabled"; + } else { + return "disabled"; + } + } + return ""; + } + + setAskURI(args) { + return Scratch.canFetch(args.URL).then((allowed) => { + if (allowed) { + askButtonImage = args.URL; + applyCSS(); + } + }); + } + } + + Scratch.extensions.register(new MonitorStyles()); +})(Scratch); \ No newline at end of file From 6a805b9a94cbd143f45ae271fb7b377ed0a188d3 Mon Sep 17 00:00:00 2001 From: HanHanDeYaYa Date: Mon, 10 Jun 2024 22:03:13 +0800 Subject: [PATCH 2/8] =?UTF-8?q?=E5=88=A0=E9=99=A4=20Variable=20beautificat?= =?UTF-8?q?ion.js?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Variable beautification.js | 783 ------------------ 1 file changed, 783 deletions(-) delete mode 100644 Variable beautification/Variable beautification.js diff --git a/Variable beautification/Variable beautification.js b/Variable beautification/Variable beautification.js deleted file mode 100644 index 59cbfe15..00000000 --- a/Variable beautification/Variable beautification.js +++ /dev/null @@ -1,783 +0,0 @@ -// Name: Custom Styles -// ID: shovelcss -// Description: Customize the appearance of variable monitors and prompts in your project. -// By: TheShovel -// License: MIT - -// Thanks LilyMakesThings for the awesome banner! -(function (Scratch) { - "use strict"; - - // Styles - let monitorText = ""; - let monitorBorder = ""; - let monitorBackgroundColor = ""; - let variableValueBackground = ""; - let variableValueTextColor = ""; - let listFooterBackground = ""; - let listHeaderBackground = ""; - let listValueText = ""; - let listValueBackground = ""; - let variableValueRoundness = -1; - let listValueRoundness = -1; - let monitorBackgroundRoundness = -1; - let monitorBackgroundBorderWidth = -1; - let allowScrolling = ""; - let askBackground = ""; - let askBackgroundRoundness = -1; - let askBackgroundBorderWidth = -1; - let askButtonBackground = ""; - let askButtonRoundness = -1; - let askInputBackground = ""; - let askInputRoundness = -1; - let askInputBorderWidth = -1; - let askBoxIcon = ""; - let askInputText = ""; - let askQuestionText = ""; - let askButtonImage = ""; - let askInputBorder = ""; - - // CSS selectors - let monitorRoot; - let monitorValue; - let monitorListHeader; - let monitorListFooter; - let monitorRowValueOuter; - let monitorRowsInner; - let monitorRowsScroller; - let monitorRowIndex; - let monitorValueLarge; - let askBoxBG; - let askBoxButton; - let askBoxInner; - let askBoxText; - let askBoxBorderMain; - let askBoxBorderOuter; - if (typeof scaffolding !== "undefined") { - monitorRoot = ".sc-monitor-root"; - monitorValue = ".sc-monitor-value"; - monitorListHeader = ".sc-monitor-list-label"; - monitorListFooter = ".sc-monitor-list-footer"; - monitorRowValueOuter = ".sc-monitor-row-value-outer"; - monitorRowsInner = ".sc-monitor-rows-inner"; - monitorRowsScroller = monitorRowsInner; - monitorRowIndex = ".sc-monitor-row-index"; - monitorValueLarge = ".sc-monitor-large-value"; - askBoxBG = ".sc-question-inner"; - askBoxButton = ".sc-question-submit-button"; - askBoxInner = ".sc-question-input"; - askBoxText = ".sc-question-text"; - askBoxBorderMain = ".sc-question-input:hover"; - askBoxBorderOuter = ".sc-question-input:focus"; - } else { - monitorRoot = 'div[class^="monitor_monitor-container_"]'; - monitorValue = 'div[class^="monitor_value_"]'; - monitorListHeader = 'div[class^="monitor_list-header_"]'; - monitorListFooter = 'div[class^="monitor_list-footer_"]'; - monitorRowValueOuter = 'div[class^="monitor_list-value_"]'; - monitorRowsInner = 'div[class^="monitor_list-body_"]'; - monitorRowsScroller = - 'div[class^="monitor_list-body_"] > .ReactVirtualized__List'; - monitorRowIndex = 'div[class^="monitor_list-index_"]'; - monitorValueLarge = 'div[class^="monitor_large-value_"]'; - askBoxBG = 'div[class^="question_question-container_"]'; - askBoxButton = 'button[class^="question_question-submit-button_"]'; - askBoxInner = - '[class^="question_question-container_"] input[class^="input_input-form_"]'; - askBoxText = - '[class^="question_question-container_"] div[class^="question_question-label_"]'; - askBoxIcon = 'img[class^="question_question-submit-button-icon_"]'; - askBoxBorderMain = - '[class^="question_question-input_"] input:focus, [class^="question_question-input_"] input:hover'; - askBoxBorderOuter = '[class^="question_question-input_"] > input:focus'; - } - - const ColorIcon = - ""; - const BorderIcon = - ""; - const extensionIcon = - ""; - const miscIcon = - ""; - const TransparentIcon = - ""; - const GradientIcon = - ""; - const PictureIcon = - ""; - const ResetIcon = - ""; - - const stylesheet = document.createElement("style"); - stylesheet.className = "shovelcss-style"; - // end of for higher precedence than other sheets - document.body.appendChild(stylesheet); - - const applyCSS = () => { - let css = ""; - - // We assume all values are sanitized when they are set, so then we can just use them as-is here. - - if (monitorText) { - css += `${monitorRoot}, ${monitorListFooter}, ${monitorListHeader}, ${monitorRowIndex} { color: ${monitorText}; }`; - } - if (monitorBackgroundColor) { - css += `${monitorRoot}, ${monitorRowsInner} { background: ${monitorBackgroundColor}; }`; - } - if (monitorBorder) { - css += `${monitorRoot} { border-color: ${monitorBorder}; }`; - } - if (monitorBackgroundRoundness >= 0) { - css += `${monitorRoot} { border-radius: ${monitorBackgroundRoundness}px; }`; - } - if (monitorBackgroundBorderWidth >= 0) { - css += `${monitorRoot} { border-width: ${monitorBackgroundBorderWidth}px; }`; - } - if (variableValueBackground) { - css += `${monitorValue}, ${monitorValueLarge} { background: ${variableValueBackground} !important; }`; - } - if (variableValueTextColor) { - css += `${monitorValue}, ${monitorValueLarge} { color: ${variableValueTextColor}; }`; - } - if (variableValueRoundness >= 0) { - css += `${monitorValue} { border-radius: ${variableValueRoundness}px; }`; - } - if (listHeaderBackground) { - css += `${monitorListHeader} { background: ${listHeaderBackground}; }`; - } - if (listFooterBackground) { - css += `${monitorListFooter} { background: ${listHeaderBackground}; }`; - } - if (listValueBackground) { - css += `${monitorRowValueOuter} { background: ${listValueBackground} !important; }`; - } - if (listValueText) { - css += `${monitorRowValueOuter} { color: ${listValueText}; }`; - } - if (listValueRoundness >= 0) { - css += `${monitorRowValueOuter} { border-radius: ${listValueRoundness}px; }`; - } - if (allowScrolling) { - css += `${monitorRowsScroller} { overflow: ${allowScrolling} !important; }`; - } - if (askBackground) { - css += `${askBoxBG} { background: ${askBackground} !important; border: none !important; }`; - } - if (askBackgroundRoundness >= 0) { - css += `${askBoxBG} { border-radius: ${askBackgroundRoundness}px !important; }`; - } - if (askBackgroundBorderWidth >= 0) { - css += `${askBoxBG} { border-width: ${askBackgroundBorderWidth}px !important; }`; - } - if (askButtonBackground) { - css += `${askBoxButton} { background-color: ${askButtonBackground}; }`; - } - if (askButtonRoundness >= 0) { - css += `${askBoxButton} { border-radius: ${askButtonRoundness}px !important; }`; - } - if (askInputBackground) { - css += `${askBoxInner} { background: ${askInputBackground} !important; }`; - css += `${askBoxInner} { border: none !important; }`; - } - if (askInputText) { - css += `${askBoxInner} { color: ${askInputText} !important; }`; - } - if (askQuestionText) { - css += `${askBoxText} { color: ${askQuestionText} !important; }`; - } - if (askInputRoundness >= 0) { - css += `${askBoxInner} { border-radius: ${askInputRoundness}px !important; }`; - } - if (askInputBorderWidth >= 0) { - css += `${askBoxInner} { border-width: ${askInputBorderWidth}px !important; }`; - } - if (askButtonImage) { - css += `${askBoxButton} { background-image: url("${encodeURI( - askButtonImage - )}") !important; background-repeat: no-repeat; background-size: contain; }`; - css += `${askBoxIcon} { visibility: hidden; }`; - } - if (askInputBorder) { - css += `${askBoxBorderMain}, ${askBoxBorderOuter} { border-color: ${askInputBorder} !important; }`; - css += `${askBoxBorderOuter} { box-shadow: none !important; }`; - } - - stylesheet.textContent = css; - }; - - const resetStyles = () => { - monitorText = ""; - monitorBorder = ""; - monitorBackgroundColor = ""; - variableValueBackground = ""; - variableValueTextColor = ""; - listFooterBackground = ""; - listHeaderBackground = ""; - listValueText = ""; - listValueBackground = ""; - variableValueRoundness = -1; - listValueRoundness = -1; - monitorBackgroundRoundness = -1; - monitorBackgroundBorderWidth = -1; - allowScrolling = ""; - askBackground = ""; - askBackgroundRoundness = -1; - askBackgroundBorderWidth = -1; - askButtonBackground = ""; - askButtonRoundness = -1; - askInputBackground = ""; - askInputRoundness = -1; - askInputBorderWidth = -1; - askBoxIcon = ""; - askInputText = ""; - askQuestionText = ""; - askButtonImage = ""; - askInputBorder = ""; - - applyCSS(); - }; - - const getMonitorRoot = (id) => { - const allMonitors = document.querySelectorAll(monitorRoot); - for (const monitor of allMonitors) { - if (monitor.dataset.id === id) { - return monitor; - } - } - return null; - }; - - /** - * @param {string} id - * @param {number} x - * @param {number} y - */ - const setMonitorPosition = (id, x, y) => { - const root = getMonitorRoot(id); - if (root) { - root.style.transform = `translate(${x}px, ${y}px)`; - root.style.left = "0px"; - root.style.top = "0px"; - } - }; - - /** - * @param {VM.Target} target - * @param {string} name - * @param {VM.VariableType} type - * @param {number} x - * @param {number} y - */ - const setVariableMonitorPosition = (target, name, type, x, y) => { - // @ts-expect-error - const variable = target.lookupVariableByNameAndType(name, type); - if (variable) { - // @ts-expect-error - setMonitorPosition(variable.id, x, y); - } - }; - - const parseColor = (color, callback) => { - color = Scratch.Cast.toString(color); - - // These might have some exponential backtracking/ReDoS, but that's not really a concern here. - // If a project wanted to get stuck in an infinite loop, there are so many other ways to do that. - - // Simple color code or name - if (/^#?[a-z0-9]+$/.test(color)) { - callback(color); - return; - } - - // General gradient pattern - if (/^[a-z-]+-gradient\([a-z0-9,#%. ]+\)$/i.test(color)) { - callback(color); - return; - } - - // URL - // see list of non-escaped characters: - // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/encodeURI#description - const match = color.match(/^url\("([A-Za-z0-9\-_.!~*'();/?:@&=+$,#]+)"\)$/); - if (match) { - const url = match[1]; - return Scratch.canFetch(url).then((allowed) => { - if (allowed) { - callback(color); - } - }); - } - - console.error("Invalid color", color); - }; - - Scratch.vm.runtime.on("RUNTIME_DISPOSED", resetStyles); - - class MonitorStyles { - getInfo() { - return { - id: "shovelcss", - name: "Custom Styles", - menuIconURI: extensionIcon, - color1: "#0072d6", - color2: "#0064bc", - color3: "#01539b", - blocks: [ - { - blockIconURI: ColorIcon, - opcode: "changecss", - blockType: Scratch.BlockType.COMMAND, - text: "set [COLORABLE] to [COLOR]", - arguments: { - COLORABLE: { - type: Scratch.ArgumentType.STRING, - menu: "COLORABLE_MENU", - }, - COLOR: { - type: Scratch.ArgumentType.COLOR, - defaultValue: "#ff0000", - }, - }, - }, - { - blockIconURI: GradientIcon, - opcode: "gradientAngle", - blockType: Scratch.BlockType.REPORTER, - text: "make a gradient with [COLOR1] and [COLOR2] at angle [ANGLE]", - arguments: { - COLOR1: { - type: Scratch.ArgumentType.COLOR, - defaultValue: "#ff0000", - }, - COLOR2: { - type: Scratch.ArgumentType.COLOR, - defaultValue: "#6ed02d", - }, - ANGLE: { - type: Scratch.ArgumentType.ANGLE, - defaultValue: "90", - }, - }, - }, - { - blockIconURI: TransparentIcon, - disableMonitor: true, - opcode: "transparentinput", - blockType: Scratch.BlockType.REPORTER, - text: "transparent", - }, - { - blockIconURI: PictureIcon, - disableMonitor: true, - opcode: "pictureinput", - blockType: Scratch.BlockType.REPORTER, - text: "image [URL]", - arguments: { - URL: { - type: Scratch.ArgumentType.STRING, - defaultValue: "https://extensions.turbowarp.org/dango.png", - }, - }, - }, - "---", - { - blockIconURI: PictureIcon, - disableMonitor: true, - opcode: "setAskURI", - blockType: Scratch.BlockType.COMMAND, - text: "set ask prompt button image to [URL]", - arguments: { - URL: { - type: Scratch.ArgumentType.STRING, - defaultValue: "https://extensions.turbowarp.org/dango.png", - }, - }, - }, - "---", - { - blockIconURI: BorderIcon, - opcode: "setbordersize", - blockType: Scratch.BlockType.COMMAND, - text: "set border width of [BORDER] to [SIZE]", - arguments: { - BORDER: { - type: Scratch.ArgumentType.STRING, - menu: "BORDER_WIDTH_MENU", - }, - SIZE: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: "2", - }, - }, - }, - { - blockIconURI: BorderIcon, - opcode: "setborderradius", - blockType: Scratch.BlockType.COMMAND, - text: "set roundness of [CORNER] to [SIZE]", - arguments: { - SIZE: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: "4", - }, - CORNER: { - type: Scratch.ArgumentType.STRING, - menu: "BORDER_ROUNDNESS_MENU", - }, - }, - }, - "---", - { - blockIconURI: ResetIcon, - opcode: "clearCSS", - blockType: Scratch.BlockType.COMMAND, - text: "reset styles", - }, - "---", - { - blockIconURI: miscIcon, - opcode: "allowscrollrule", - blockType: Scratch.BlockType.COMMAND, - text: "set list scrolling to [SCROLLRULE]", - arguments: { - SCROLLRULE: { - type: Scratch.ArgumentType.STRING, - menu: "SCROLL_MENU", - }, - }, - }, - { - blockIconURI: miscIcon, - opcode: "getValue", - blockType: Scratch.BlockType.REPORTER, - text: "get [ITEM]", - arguments: { - ITEM: { - type: Scratch.ArgumentType.STRING, - menu: "VALUEGET_LIST", - }, - }, - }, - "---", - { - blockIconURI: miscIcon, - opcode: "setvarpos", - blockType: Scratch.BlockType.COMMAND, - text: "set position of variable [NAME] to x: [X] y: [Y]", - arguments: { - X: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: "0", - }, - Y: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: "0", - }, - NAME: { - type: Scratch.ArgumentType.STRING, - defaultValue: "my variable", - }, - }, - }, - { - blockIconURI: miscIcon, - opcode: "setlistpos", - blockType: Scratch.BlockType.COMMAND, - text: "set position of list [NAME] to x: [X] y: [Y]", - arguments: { - X: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: "0", - }, - Y: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: "0", - }, - NAME: { - type: Scratch.ArgumentType.STRING, - defaultValue: "my variable", - }, - }, - }, - ], - // Accepting reporters because there can't be errors in case the value is not correct - menus: { - COLORABLE_MENU: { - acceptReporters: true, - items: [ - "monitor text", - "monitor background", - "monitor border", - "variable value background", - "variable value text", - "list header background", - "list footer background", - "list value background", - "list value text", - "ask prompt background", - "ask prompt button background", - "ask prompt input background", - "ask prompt question text", - "ask prompt input text", - "ask prompt input border", - ], - }, - BORDER_WIDTH_MENU: { - acceptReporters: true, - items: [ - "monitor background", - "ask prompt background", - "ask prompt input", - ], - }, - BORDER_ROUNDNESS_MENU: { - acceptReporters: true, - items: [ - "monitor background", - "variable value", - "list value", - "ask prompt background", - "ask prompt button", - "ask prompt input", - ], - }, - SCROLL_MENU: { - acceptReporters: true, - items: ["enabled", "disabled"], - }, - VALUEGET_LIST: { - acceptReporters: true, - items: [ - "monitor text", - "monitor background", - "monitor border color", - "variable value background", - "variable value text", - "list header background", - "list footer background", - "list value background", - "list value text", - "ask prompt background", - "ask prompt button background", - "ask prompt input background", - "ask prompt question text", - "ask prompt input text", - "ask prompt input border", - "monitor background border width", - "ask prompt background border width", - "ask prompt input border width", - "monitor background roundness", - "variable value roundness", - "list value roundness", - "ask prompt background roundness", - "ask prompt button roundness", - "ask prompt input roundness", - "ask prompt button image", - "list scroll rule", - ], - }, - }, - }; - } - - changecss(args) { - return parseColor(args.COLOR, (color) => { - if (args.COLORABLE === "monitor text") { - monitorText = color; - } else if (args.COLORABLE === "monitor background") { - monitorBackgroundColor = color; - } else if (args.COLORABLE === "monitor border") { - monitorBorder = color; - } else if (args.COLORABLE === "variable value background") { - variableValueBackground = color; - } else if (args.COLORABLE === "variable value text") { - variableValueTextColor = color; - } else if (args.COLORABLE === "list header background") { - listHeaderBackground = color; - } else if (args.COLORABLE === "list footer background") { - listFooterBackground = color; - } else if (args.COLORABLE === "list value background") { - listValueBackground = color; - } else if (args.COLORABLE === "list value text") { - listValueText = color; - } else if (args.COLORABLE === "ask prompt background") { - askBackground = color; - } else if (args.COLORABLE === "ask prompt button background") { - askButtonBackground = color; - } else if (args.COLORABLE === "ask prompt input background") { - askInputBackground = color; - } else if (args.COLORABLE === "ask prompt question text") { - askQuestionText = color; - } else if (args.COLORABLE === "ask prompt input text") { - askInputText = color; - } else if (args.COLORABLE === "ask prompt input border") { - askInputBorder = color; - } - - applyCSS(); - }); - } - - gradientAngle(args) { - return ( - "linear-gradient(" + - args.ANGLE + - "deg," + - args.COLOR1 + - "," + - args.COLOR2 + - ")" - ); - } - - setbordersize(args) { - const size = Scratch.Cast.toNumber(args.SIZE); - if (args.BORDER === "monitor background") { - monitorBackgroundBorderWidth = size; - } else if (args.BORDER === "ask prompt background") { - askBackgroundBorderWidth = size; - } else if (args.BORDER === "ask prompt input") { - askInputBorderWidth = size; - } - applyCSS(); - } - - setborderradius(args) { - const size = Scratch.Cast.toNumber(args.SIZE); - if (args.CORNER === "monitor background") { - monitorBackgroundRoundness = size; - } else if (args.CORNER === "variable value") { - variableValueRoundness = size; - } else if (args.CORNER === "list value") { - listValueRoundness = size; - } else if (args.CORNER === "ask prompt background") { - askBackgroundRoundness = size; - } else if (args.CORNER === "ask prompt button") { - askButtonRoundness = size; - } else if (args.CORNER === "ask prompt input") { - askInputRoundness = size; - } - applyCSS(); - } - - allowscrollrule(args) { - if (args.SCROLLRULE === "enabled") { - allowScrolling = "auto"; - } else { - allowScrolling = "hidden"; - } - applyCSS(); - } - - setvarpos(args, util) { - setVariableMonitorPosition( - util.target, - args.NAME, - "", - Scratch.Cast.toNumber(args.X) + Scratch.vm.runtime.stageWidth / 2, - Scratch.vm.runtime.stageHeight / 2 - Scratch.Cast.toNumber(args.Y) - ); - } - - setlistpos(args, util) { - setVariableMonitorPosition( - util.target, - args.NAME, - "list", - Scratch.Cast.toNumber(args.X) + Scratch.vm.runtime.stageWidth / 2, - Scratch.vm.runtime.stageHeight / 2 - Scratch.Cast.toNumber(args.Y) - ); - } - - help() { - alert( - "\nThis is a short introduction to how to use the Monitor Styles extension!\n\n๐—Ÿ๐—ผ๐—ผ๐—ธ๐˜€ ๐—ฏ๐—น๐—ผ๐—ฐ๐—ธ๐˜€\nThese blocks change the appearance of the variable and list didsplays. You can use the drop-down menu to select what component you want to modify. ๐™๐™๐™š ๐™˜๐™ค๐™ก๐™ค๐™ง ๐™—๐™ก๐™ค๐™˜๐™  modifieas the color of a component. You can use the ๐™œ๐™ง๐™–๐™™๐™ž๐™š๐™ฃ๐™ฉ block inside the color input, to create gradients or the ๐™„๐™ข๐™–๐™œ๐™š block to use a image instead of solid colors. ๐™๐™๐™š๐™จ๐™š ๐™ฉ๐™ฌ๐™ค ๐™ค๐™ฃ๐™ก๐™ฎ ๐™ฌ๐™ค๐™ง๐™  ๐™ค๐™ฃ ๐™˜๐™š๐™ง๐™ฉ๐™–๐™ž๐™ฃ ๐™˜๐™ค๐™ข๐™ฅ๐™ค๐™ฃ๐™š๐™ฃ๐™ฉ๐™จ! You can also use the ๐™ฉ๐™ง๐™–๐™ฃ๐™จ๐™ฅ๐™–๐™ง๐™š๐™ฃ๐™ฉ ๐™—๐™ก๐™ค๐™˜๐™  as a color input, to make components invisible. The ๐™—๐™ค๐™ง๐™™๐™š๐™ง ๐™—๐™ก๐™ค๐™˜๐™ ๐™จ modify the borders of components.\n\n๐—ฆ๐—ฒ๐—ป๐˜€๐—ถ๐—ป๐—ด ๐—ฏ๐—น๐—ผ๐—ฐ๐—ธ๐˜€\nThese blocks can change the behaviour of certain components. The ๐™จ๐™˜๐™ง๐™ค๐™ก๐™ก ๐™ง๐™ช๐™ก๐™š block change the behaviour for lists. On 'auto' they will show the scroll bar, and allow you to school, but on 'hidden', they won't let you do that, and the scroll bar will be hidden.\n\n๐— ๐—ผ๐˜๐—ถ๐—ผ๐—ป ๐—ฏ๐—น๐—ผ๐—ฐ๐—ธ๐˜€\nThese blocks allow you to move variable and list displays around. You need to use their ๐™ก๐™–๐™—๐™š๐™ก ๐™ฃ๐™–๐™ข๐™š. The label name is the text that displays on the monitor. For example, a 'for this sprite only' variable will be like 'Sprite1: my variable'." - ); - } - - transparentinput() { - return "transparent"; - } - - pictureinput(args) { - return `url("${encodeURI(args.URL)}")`; - } - - clearCSS() { - resetStyles(); - } - - getValue(args) { - if (args.ITEM === "monitor text") { - return monitorText; - } else if (args.ITEM === "monitor background") { - return monitorBackgroundColor; - } else if (args.ITEM === "monitor border color") { - return monitorBorder; - } else if (args.ITEM === "variable value background") { - return variableValueBackground; - } else if (args.ITEM === "variable value text") { - return variableValueTextColor; - } else if (args.ITEM === "list header background") { - return listHeaderBackground; - } else if (args.ITEM === "list footer background") { - return listFooterBackground; - } else if (args.ITEM === "list value background") { - return listValueBackground; - } else if (args.ITEM === "list value text") { - return listValueText; - } else if (args.ITEM === "ask prompt background") { - return askBackground; - } else if (args.ITEM === "ask prompt button background") { - return askButtonBackground; - } else if (args.ITEM === "ask prompt input background") { - return askInputBackground; - } else if (args.ITEM === "ask prompt question text") { - return askQuestionText; - } else if (args.ITEM === "ask prompt input text") { - return askInputText; - } else if (args.ITEM === "ask prompt input border") { - return askInputBorder; - } else if (args.ITEM === "monitor background border width") { - return monitorBackgroundBorderWidth; - } else if (args.ITEM === "ask prompt background border width") { - return askBackgroundBorderWidth; - } else if (args.ITEM === "ask prompt input border width") { - return askInputBorderWidth; - } else if (args.ITEM === "monitor background roundness") { - return monitorBackgroundRoundness; - } else if (args.ITEM === "variable value roundness") { - return variableValueRoundness; - } else if (args.ITEM === "list value roundness") { - return listValueRoundness; - } else if (args.ITEM === "ask prompt background roundness") { - return askBackgroundRoundness; - } else if (args.ITEM === "ask prompt button roundness") { - return askButtonRoundness; - } else if (args.ITEM === "ask prompt input roundness") { - return askInputRoundness; - } else if (args.ITEM === "ask prompt button image") { - return askButtonImage; - } else if (args.ITEM === "list scrolling") { - if (allowScrolling === "auto") { - return "enabled"; - } else { - return "disabled"; - } - } - return ""; - } - - setAskURI(args) { - return Scratch.canFetch(args.URL).then((allowed) => { - if (allowed) { - askButtonImage = args.URL; - applyCSS(); - } - }); - } - } - - Scratch.extensions.register(new MonitorStyles()); -})(Scratch); \ No newline at end of file From 910f361d576488ddd5877f6d4d85cbd40cf403e2 Mon Sep 17 00:00:00 2001 From: HanHanDeYaYa Date: Mon, 10 Jun 2024 22:05:17 +0800 Subject: [PATCH 3/8] =?UTF-8?q?=E5=88=9B=E5=BB=BA=20Variable=20beautificat?= =?UTF-8?q?ion.js?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Variable beautification.js | 783 ++++++++++++++++++ 1 file changed, 783 insertions(+) create mode 100644 Variable beautification/Variable beautification.js diff --git a/Variable beautification/Variable beautification.js b/Variable beautification/Variable beautification.js new file mode 100644 index 00000000..59cbfe15 --- /dev/null +++ b/Variable beautification/Variable beautification.js @@ -0,0 +1,783 @@ +// Name: Custom Styles +// ID: shovelcss +// Description: Customize the appearance of variable monitors and prompts in your project. +// By: TheShovel +// License: MIT + +// Thanks LilyMakesThings for the awesome banner! +(function (Scratch) { + "use strict"; + + // Styles + let monitorText = ""; + let monitorBorder = ""; + let monitorBackgroundColor = ""; + let variableValueBackground = ""; + let variableValueTextColor = ""; + let listFooterBackground = ""; + let listHeaderBackground = ""; + let listValueText = ""; + let listValueBackground = ""; + let variableValueRoundness = -1; + let listValueRoundness = -1; + let monitorBackgroundRoundness = -1; + let monitorBackgroundBorderWidth = -1; + let allowScrolling = ""; + let askBackground = ""; + let askBackgroundRoundness = -1; + let askBackgroundBorderWidth = -1; + let askButtonBackground = ""; + let askButtonRoundness = -1; + let askInputBackground = ""; + let askInputRoundness = -1; + let askInputBorderWidth = -1; + let askBoxIcon = ""; + let askInputText = ""; + let askQuestionText = ""; + let askButtonImage = ""; + let askInputBorder = ""; + + // CSS selectors + let monitorRoot; + let monitorValue; + let monitorListHeader; + let monitorListFooter; + let monitorRowValueOuter; + let monitorRowsInner; + let monitorRowsScroller; + let monitorRowIndex; + let monitorValueLarge; + let askBoxBG; + let askBoxButton; + let askBoxInner; + let askBoxText; + let askBoxBorderMain; + let askBoxBorderOuter; + if (typeof scaffolding !== "undefined") { + monitorRoot = ".sc-monitor-root"; + monitorValue = ".sc-monitor-value"; + monitorListHeader = ".sc-monitor-list-label"; + monitorListFooter = ".sc-monitor-list-footer"; + monitorRowValueOuter = ".sc-monitor-row-value-outer"; + monitorRowsInner = ".sc-monitor-rows-inner"; + monitorRowsScroller = monitorRowsInner; + monitorRowIndex = ".sc-monitor-row-index"; + monitorValueLarge = ".sc-monitor-large-value"; + askBoxBG = ".sc-question-inner"; + askBoxButton = ".sc-question-submit-button"; + askBoxInner = ".sc-question-input"; + askBoxText = ".sc-question-text"; + askBoxBorderMain = ".sc-question-input:hover"; + askBoxBorderOuter = ".sc-question-input:focus"; + } else { + monitorRoot = 'div[class^="monitor_monitor-container_"]'; + monitorValue = 'div[class^="monitor_value_"]'; + monitorListHeader = 'div[class^="monitor_list-header_"]'; + monitorListFooter = 'div[class^="monitor_list-footer_"]'; + monitorRowValueOuter = 'div[class^="monitor_list-value_"]'; + monitorRowsInner = 'div[class^="monitor_list-body_"]'; + monitorRowsScroller = + 'div[class^="monitor_list-body_"] > .ReactVirtualized__List'; + monitorRowIndex = 'div[class^="monitor_list-index_"]'; + monitorValueLarge = 'div[class^="monitor_large-value_"]'; + askBoxBG = 'div[class^="question_question-container_"]'; + askBoxButton = 'button[class^="question_question-submit-button_"]'; + askBoxInner = + '[class^="question_question-container_"] input[class^="input_input-form_"]'; + askBoxText = + '[class^="question_question-container_"] div[class^="question_question-label_"]'; + askBoxIcon = 'img[class^="question_question-submit-button-icon_"]'; + askBoxBorderMain = + '[class^="question_question-input_"] input:focus, [class^="question_question-input_"] input:hover'; + askBoxBorderOuter = '[class^="question_question-input_"] > input:focus'; + } + + const ColorIcon = + ""; + const BorderIcon = + ""; + const extensionIcon = + ""; + const miscIcon = + ""; + const TransparentIcon = + ""; + const GradientIcon = + ""; + const PictureIcon = + ""; + const ResetIcon = + ""; + + const stylesheet = document.createElement("style"); + stylesheet.className = "shovelcss-style"; + // end of for higher precedence than other sheets + document.body.appendChild(stylesheet); + + const applyCSS = () => { + let css = ""; + + // We assume all values are sanitized when they are set, so then we can just use them as-is here. + + if (monitorText) { + css += `${monitorRoot}, ${monitorListFooter}, ${monitorListHeader}, ${monitorRowIndex} { color: ${monitorText}; }`; + } + if (monitorBackgroundColor) { + css += `${monitorRoot}, ${monitorRowsInner} { background: ${monitorBackgroundColor}; }`; + } + if (monitorBorder) { + css += `${monitorRoot} { border-color: ${monitorBorder}; }`; + } + if (monitorBackgroundRoundness >= 0) { + css += `${monitorRoot} { border-radius: ${monitorBackgroundRoundness}px; }`; + } + if (monitorBackgroundBorderWidth >= 0) { + css += `${monitorRoot} { border-width: ${monitorBackgroundBorderWidth}px; }`; + } + if (variableValueBackground) { + css += `${monitorValue}, ${monitorValueLarge} { background: ${variableValueBackground} !important; }`; + } + if (variableValueTextColor) { + css += `${monitorValue}, ${monitorValueLarge} { color: ${variableValueTextColor}; }`; + } + if (variableValueRoundness >= 0) { + css += `${monitorValue} { border-radius: ${variableValueRoundness}px; }`; + } + if (listHeaderBackground) { + css += `${monitorListHeader} { background: ${listHeaderBackground}; }`; + } + if (listFooterBackground) { + css += `${monitorListFooter} { background: ${listHeaderBackground}; }`; + } + if (listValueBackground) { + css += `${monitorRowValueOuter} { background: ${listValueBackground} !important; }`; + } + if (listValueText) { + css += `${monitorRowValueOuter} { color: ${listValueText}; }`; + } + if (listValueRoundness >= 0) { + css += `${monitorRowValueOuter} { border-radius: ${listValueRoundness}px; }`; + } + if (allowScrolling) { + css += `${monitorRowsScroller} { overflow: ${allowScrolling} !important; }`; + } + if (askBackground) { + css += `${askBoxBG} { background: ${askBackground} !important; border: none !important; }`; + } + if (askBackgroundRoundness >= 0) { + css += `${askBoxBG} { border-radius: ${askBackgroundRoundness}px !important; }`; + } + if (askBackgroundBorderWidth >= 0) { + css += `${askBoxBG} { border-width: ${askBackgroundBorderWidth}px !important; }`; + } + if (askButtonBackground) { + css += `${askBoxButton} { background-color: ${askButtonBackground}; }`; + } + if (askButtonRoundness >= 0) { + css += `${askBoxButton} { border-radius: ${askButtonRoundness}px !important; }`; + } + if (askInputBackground) { + css += `${askBoxInner} { background: ${askInputBackground} !important; }`; + css += `${askBoxInner} { border: none !important; }`; + } + if (askInputText) { + css += `${askBoxInner} { color: ${askInputText} !important; }`; + } + if (askQuestionText) { + css += `${askBoxText} { color: ${askQuestionText} !important; }`; + } + if (askInputRoundness >= 0) { + css += `${askBoxInner} { border-radius: ${askInputRoundness}px !important; }`; + } + if (askInputBorderWidth >= 0) { + css += `${askBoxInner} { border-width: ${askInputBorderWidth}px !important; }`; + } + if (askButtonImage) { + css += `${askBoxButton} { background-image: url("${encodeURI( + askButtonImage + )}") !important; background-repeat: no-repeat; background-size: contain; }`; + css += `${askBoxIcon} { visibility: hidden; }`; + } + if (askInputBorder) { + css += `${askBoxBorderMain}, ${askBoxBorderOuter} { border-color: ${askInputBorder} !important; }`; + css += `${askBoxBorderOuter} { box-shadow: none !important; }`; + } + + stylesheet.textContent = css; + }; + + const resetStyles = () => { + monitorText = ""; + monitorBorder = ""; + monitorBackgroundColor = ""; + variableValueBackground = ""; + variableValueTextColor = ""; + listFooterBackground = ""; + listHeaderBackground = ""; + listValueText = ""; + listValueBackground = ""; + variableValueRoundness = -1; + listValueRoundness = -1; + monitorBackgroundRoundness = -1; + monitorBackgroundBorderWidth = -1; + allowScrolling = ""; + askBackground = ""; + askBackgroundRoundness = -1; + askBackgroundBorderWidth = -1; + askButtonBackground = ""; + askButtonRoundness = -1; + askInputBackground = ""; + askInputRoundness = -1; + askInputBorderWidth = -1; + askBoxIcon = ""; + askInputText = ""; + askQuestionText = ""; + askButtonImage = ""; + askInputBorder = ""; + + applyCSS(); + }; + + const getMonitorRoot = (id) => { + const allMonitors = document.querySelectorAll(monitorRoot); + for (const monitor of allMonitors) { + if (monitor.dataset.id === id) { + return monitor; + } + } + return null; + }; + + /** + * @param {string} id + * @param {number} x + * @param {number} y + */ + const setMonitorPosition = (id, x, y) => { + const root = getMonitorRoot(id); + if (root) { + root.style.transform = `translate(${x}px, ${y}px)`; + root.style.left = "0px"; + root.style.top = "0px"; + } + }; + + /** + * @param {VM.Target} target + * @param {string} name + * @param {VM.VariableType} type + * @param {number} x + * @param {number} y + */ + const setVariableMonitorPosition = (target, name, type, x, y) => { + // @ts-expect-error + const variable = target.lookupVariableByNameAndType(name, type); + if (variable) { + // @ts-expect-error + setMonitorPosition(variable.id, x, y); + } + }; + + const parseColor = (color, callback) => { + color = Scratch.Cast.toString(color); + + // These might have some exponential backtracking/ReDoS, but that's not really a concern here. + // If a project wanted to get stuck in an infinite loop, there are so many other ways to do that. + + // Simple color code or name + if (/^#?[a-z0-9]+$/.test(color)) { + callback(color); + return; + } + + // General gradient pattern + if (/^[a-z-]+-gradient\([a-z0-9,#%. ]+\)$/i.test(color)) { + callback(color); + return; + } + + // URL + // see list of non-escaped characters: + // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/encodeURI#description + const match = color.match(/^url\("([A-Za-z0-9\-_.!~*'();/?:@&=+$,#]+)"\)$/); + if (match) { + const url = match[1]; + return Scratch.canFetch(url).then((allowed) => { + if (allowed) { + callback(color); + } + }); + } + + console.error("Invalid color", color); + }; + + Scratch.vm.runtime.on("RUNTIME_DISPOSED", resetStyles); + + class MonitorStyles { + getInfo() { + return { + id: "shovelcss", + name: "Custom Styles", + menuIconURI: extensionIcon, + color1: "#0072d6", + color2: "#0064bc", + color3: "#01539b", + blocks: [ + { + blockIconURI: ColorIcon, + opcode: "changecss", + blockType: Scratch.BlockType.COMMAND, + text: "set [COLORABLE] to [COLOR]", + arguments: { + COLORABLE: { + type: Scratch.ArgumentType.STRING, + menu: "COLORABLE_MENU", + }, + COLOR: { + type: Scratch.ArgumentType.COLOR, + defaultValue: "#ff0000", + }, + }, + }, + { + blockIconURI: GradientIcon, + opcode: "gradientAngle", + blockType: Scratch.BlockType.REPORTER, + text: "make a gradient with [COLOR1] and [COLOR2] at angle [ANGLE]", + arguments: { + COLOR1: { + type: Scratch.ArgumentType.COLOR, + defaultValue: "#ff0000", + }, + COLOR2: { + type: Scratch.ArgumentType.COLOR, + defaultValue: "#6ed02d", + }, + ANGLE: { + type: Scratch.ArgumentType.ANGLE, + defaultValue: "90", + }, + }, + }, + { + blockIconURI: TransparentIcon, + disableMonitor: true, + opcode: "transparentinput", + blockType: Scratch.BlockType.REPORTER, + text: "transparent", + }, + { + blockIconURI: PictureIcon, + disableMonitor: true, + opcode: "pictureinput", + blockType: Scratch.BlockType.REPORTER, + text: "image [URL]", + arguments: { + URL: { + type: Scratch.ArgumentType.STRING, + defaultValue: "https://extensions.turbowarp.org/dango.png", + }, + }, + }, + "---", + { + blockIconURI: PictureIcon, + disableMonitor: true, + opcode: "setAskURI", + blockType: Scratch.BlockType.COMMAND, + text: "set ask prompt button image to [URL]", + arguments: { + URL: { + type: Scratch.ArgumentType.STRING, + defaultValue: "https://extensions.turbowarp.org/dango.png", + }, + }, + }, + "---", + { + blockIconURI: BorderIcon, + opcode: "setbordersize", + blockType: Scratch.BlockType.COMMAND, + text: "set border width of [BORDER] to [SIZE]", + arguments: { + BORDER: { + type: Scratch.ArgumentType.STRING, + menu: "BORDER_WIDTH_MENU", + }, + SIZE: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "2", + }, + }, + }, + { + blockIconURI: BorderIcon, + opcode: "setborderradius", + blockType: Scratch.BlockType.COMMAND, + text: "set roundness of [CORNER] to [SIZE]", + arguments: { + SIZE: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "4", + }, + CORNER: { + type: Scratch.ArgumentType.STRING, + menu: "BORDER_ROUNDNESS_MENU", + }, + }, + }, + "---", + { + blockIconURI: ResetIcon, + opcode: "clearCSS", + blockType: Scratch.BlockType.COMMAND, + text: "reset styles", + }, + "---", + { + blockIconURI: miscIcon, + opcode: "allowscrollrule", + blockType: Scratch.BlockType.COMMAND, + text: "set list scrolling to [SCROLLRULE]", + arguments: { + SCROLLRULE: { + type: Scratch.ArgumentType.STRING, + menu: "SCROLL_MENU", + }, + }, + }, + { + blockIconURI: miscIcon, + opcode: "getValue", + blockType: Scratch.BlockType.REPORTER, + text: "get [ITEM]", + arguments: { + ITEM: { + type: Scratch.ArgumentType.STRING, + menu: "VALUEGET_LIST", + }, + }, + }, + "---", + { + blockIconURI: miscIcon, + opcode: "setvarpos", + blockType: Scratch.BlockType.COMMAND, + text: "set position of variable [NAME] to x: [X] y: [Y]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my variable", + }, + }, + }, + { + blockIconURI: miscIcon, + opcode: "setlistpos", + blockType: Scratch.BlockType.COMMAND, + text: "set position of list [NAME] to x: [X] y: [Y]", + arguments: { + X: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + Y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "0", + }, + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "my variable", + }, + }, + }, + ], + // Accepting reporters because there can't be errors in case the value is not correct + menus: { + COLORABLE_MENU: { + acceptReporters: true, + items: [ + "monitor text", + "monitor background", + "monitor border", + "variable value background", + "variable value text", + "list header background", + "list footer background", + "list value background", + "list value text", + "ask prompt background", + "ask prompt button background", + "ask prompt input background", + "ask prompt question text", + "ask prompt input text", + "ask prompt input border", + ], + }, + BORDER_WIDTH_MENU: { + acceptReporters: true, + items: [ + "monitor background", + "ask prompt background", + "ask prompt input", + ], + }, + BORDER_ROUNDNESS_MENU: { + acceptReporters: true, + items: [ + "monitor background", + "variable value", + "list value", + "ask prompt background", + "ask prompt button", + "ask prompt input", + ], + }, + SCROLL_MENU: { + acceptReporters: true, + items: ["enabled", "disabled"], + }, + VALUEGET_LIST: { + acceptReporters: true, + items: [ + "monitor text", + "monitor background", + "monitor border color", + "variable value background", + "variable value text", + "list header background", + "list footer background", + "list value background", + "list value text", + "ask prompt background", + "ask prompt button background", + "ask prompt input background", + "ask prompt question text", + "ask prompt input text", + "ask prompt input border", + "monitor background border width", + "ask prompt background border width", + "ask prompt input border width", + "monitor background roundness", + "variable value roundness", + "list value roundness", + "ask prompt background roundness", + "ask prompt button roundness", + "ask prompt input roundness", + "ask prompt button image", + "list scroll rule", + ], + }, + }, + }; + } + + changecss(args) { + return parseColor(args.COLOR, (color) => { + if (args.COLORABLE === "monitor text") { + monitorText = color; + } else if (args.COLORABLE === "monitor background") { + monitorBackgroundColor = color; + } else if (args.COLORABLE === "monitor border") { + monitorBorder = color; + } else if (args.COLORABLE === "variable value background") { + variableValueBackground = color; + } else if (args.COLORABLE === "variable value text") { + variableValueTextColor = color; + } else if (args.COLORABLE === "list header background") { + listHeaderBackground = color; + } else if (args.COLORABLE === "list footer background") { + listFooterBackground = color; + } else if (args.COLORABLE === "list value background") { + listValueBackground = color; + } else if (args.COLORABLE === "list value text") { + listValueText = color; + } else if (args.COLORABLE === "ask prompt background") { + askBackground = color; + } else if (args.COLORABLE === "ask prompt button background") { + askButtonBackground = color; + } else if (args.COLORABLE === "ask prompt input background") { + askInputBackground = color; + } else if (args.COLORABLE === "ask prompt question text") { + askQuestionText = color; + } else if (args.COLORABLE === "ask prompt input text") { + askInputText = color; + } else if (args.COLORABLE === "ask prompt input border") { + askInputBorder = color; + } + + applyCSS(); + }); + } + + gradientAngle(args) { + return ( + "linear-gradient(" + + args.ANGLE + + "deg," + + args.COLOR1 + + "," + + args.COLOR2 + + ")" + ); + } + + setbordersize(args) { + const size = Scratch.Cast.toNumber(args.SIZE); + if (args.BORDER === "monitor background") { + monitorBackgroundBorderWidth = size; + } else if (args.BORDER === "ask prompt background") { + askBackgroundBorderWidth = size; + } else if (args.BORDER === "ask prompt input") { + askInputBorderWidth = size; + } + applyCSS(); + } + + setborderradius(args) { + const size = Scratch.Cast.toNumber(args.SIZE); + if (args.CORNER === "monitor background") { + monitorBackgroundRoundness = size; + } else if (args.CORNER === "variable value") { + variableValueRoundness = size; + } else if (args.CORNER === "list value") { + listValueRoundness = size; + } else if (args.CORNER === "ask prompt background") { + askBackgroundRoundness = size; + } else if (args.CORNER === "ask prompt button") { + askButtonRoundness = size; + } else if (args.CORNER === "ask prompt input") { + askInputRoundness = size; + } + applyCSS(); + } + + allowscrollrule(args) { + if (args.SCROLLRULE === "enabled") { + allowScrolling = "auto"; + } else { + allowScrolling = "hidden"; + } + applyCSS(); + } + + setvarpos(args, util) { + setVariableMonitorPosition( + util.target, + args.NAME, + "", + Scratch.Cast.toNumber(args.X) + Scratch.vm.runtime.stageWidth / 2, + Scratch.vm.runtime.stageHeight / 2 - Scratch.Cast.toNumber(args.Y) + ); + } + + setlistpos(args, util) { + setVariableMonitorPosition( + util.target, + args.NAME, + "list", + Scratch.Cast.toNumber(args.X) + Scratch.vm.runtime.stageWidth / 2, + Scratch.vm.runtime.stageHeight / 2 - Scratch.Cast.toNumber(args.Y) + ); + } + + help() { + alert( + "\nThis is a short introduction to how to use the Monitor Styles extension!\n\n๐—Ÿ๐—ผ๐—ผ๐—ธ๐˜€ ๐—ฏ๐—น๐—ผ๐—ฐ๐—ธ๐˜€\nThese blocks change the appearance of the variable and list didsplays. You can use the drop-down menu to select what component you want to modify. ๐™๐™๐™š ๐™˜๐™ค๐™ก๐™ค๐™ง ๐™—๐™ก๐™ค๐™˜๐™  modifieas the color of a component. You can use the ๐™œ๐™ง๐™–๐™™๐™ž๐™š๐™ฃ๐™ฉ block inside the color input, to create gradients or the ๐™„๐™ข๐™–๐™œ๐™š block to use a image instead of solid colors. ๐™๐™๐™š๐™จ๐™š ๐™ฉ๐™ฌ๐™ค ๐™ค๐™ฃ๐™ก๐™ฎ ๐™ฌ๐™ค๐™ง๐™  ๐™ค๐™ฃ ๐™˜๐™š๐™ง๐™ฉ๐™–๐™ž๐™ฃ ๐™˜๐™ค๐™ข๐™ฅ๐™ค๐™ฃ๐™š๐™ฃ๐™ฉ๐™จ! You can also use the ๐™ฉ๐™ง๐™–๐™ฃ๐™จ๐™ฅ๐™–๐™ง๐™š๐™ฃ๐™ฉ ๐™—๐™ก๐™ค๐™˜๐™  as a color input, to make components invisible. The ๐™—๐™ค๐™ง๐™™๐™š๐™ง ๐™—๐™ก๐™ค๐™˜๐™ ๐™จ modify the borders of components.\n\n๐—ฆ๐—ฒ๐—ป๐˜€๐—ถ๐—ป๐—ด ๐—ฏ๐—น๐—ผ๐—ฐ๐—ธ๐˜€\nThese blocks can change the behaviour of certain components. The ๐™จ๐™˜๐™ง๐™ค๐™ก๐™ก ๐™ง๐™ช๐™ก๐™š block change the behaviour for lists. On 'auto' they will show the scroll bar, and allow you to school, but on 'hidden', they won't let you do that, and the scroll bar will be hidden.\n\n๐— ๐—ผ๐˜๐—ถ๐—ผ๐—ป ๐—ฏ๐—น๐—ผ๐—ฐ๐—ธ๐˜€\nThese blocks allow you to move variable and list displays around. You need to use their ๐™ก๐™–๐™—๐™š๐™ก ๐™ฃ๐™–๐™ข๐™š. The label name is the text that displays on the monitor. For example, a 'for this sprite only' variable will be like 'Sprite1: my variable'." + ); + } + + transparentinput() { + return "transparent"; + } + + pictureinput(args) { + return `url("${encodeURI(args.URL)}")`; + } + + clearCSS() { + resetStyles(); + } + + getValue(args) { + if (args.ITEM === "monitor text") { + return monitorText; + } else if (args.ITEM === "monitor background") { + return monitorBackgroundColor; + } else if (args.ITEM === "monitor border color") { + return monitorBorder; + } else if (args.ITEM === "variable value background") { + return variableValueBackground; + } else if (args.ITEM === "variable value text") { + return variableValueTextColor; + } else if (args.ITEM === "list header background") { + return listHeaderBackground; + } else if (args.ITEM === "list footer background") { + return listFooterBackground; + } else if (args.ITEM === "list value background") { + return listValueBackground; + } else if (args.ITEM === "list value text") { + return listValueText; + } else if (args.ITEM === "ask prompt background") { + return askBackground; + } else if (args.ITEM === "ask prompt button background") { + return askButtonBackground; + } else if (args.ITEM === "ask prompt input background") { + return askInputBackground; + } else if (args.ITEM === "ask prompt question text") { + return askQuestionText; + } else if (args.ITEM === "ask prompt input text") { + return askInputText; + } else if (args.ITEM === "ask prompt input border") { + return askInputBorder; + } else if (args.ITEM === "monitor background border width") { + return monitorBackgroundBorderWidth; + } else if (args.ITEM === "ask prompt background border width") { + return askBackgroundBorderWidth; + } else if (args.ITEM === "ask prompt input border width") { + return askInputBorderWidth; + } else if (args.ITEM === "monitor background roundness") { + return monitorBackgroundRoundness; + } else if (args.ITEM === "variable value roundness") { + return variableValueRoundness; + } else if (args.ITEM === "list value roundness") { + return listValueRoundness; + } else if (args.ITEM === "ask prompt background roundness") { + return askBackgroundRoundness; + } else if (args.ITEM === "ask prompt button roundness") { + return askButtonRoundness; + } else if (args.ITEM === "ask prompt input roundness") { + return askInputRoundness; + } else if (args.ITEM === "ask prompt button image") { + return askButtonImage; + } else if (args.ITEM === "list scrolling") { + if (allowScrolling === "auto") { + return "enabled"; + } else { + return "disabled"; + } + } + return ""; + } + + setAskURI(args) { + return Scratch.canFetch(args.URL).then((allowed) => { + if (allowed) { + askButtonImage = args.URL; + applyCSS(); + } + }); + } + } + + Scratch.extensions.register(new MonitorStyles()); +})(Scratch); \ No newline at end of file From 19c9cf94070424a932f2d7873aa19d8bacbf7b01 Mon Sep 17 00:00:00 2001 From: HanHanDeYaYa Date: Tue, 11 Jun 2024 12:51:21 +0800 Subject: [PATCH 4/8] Create an extension Better timer You're going to create an extension --- Better timer/Better timer.js | 290 +++++++++++++++++++++++++++++++++++ 1 file changed, 290 insertions(+) create mode 100644 Better timer/Better timer.js diff --git a/Better timer/Better timer.js b/Better timer/Better timer.js new file mode 100644 index 00000000..f3f8c5de --- /dev/null +++ b/Better timer/Better timer.js @@ -0,0 +1,290 @@ +// Name: More Timers +// ID: lmsTimers +// Description: Control several timers at once. +// By: LilyMakesThings +// License: MIT AND LGPL-3.0 +// It is just to provide port, and if the original author has comments, tell the administrator of this main project, you can immediately deactivate this extension +// Due to the extended architecture, which does not provide bilingual conversion, contributors are being called... + +(function (Scratch) { + "use strict"; + + const vm = Scratch.vm; + + /** + * @typedef Timer + * @property {number} startTime + * @property {number} pauseTime + * @property {boolean} paused + */ + + /** @type {Record} */ + let timers = Object.create(null); + + /** + * @param {Timer} timer + * @return {number} + */ + const timerValue = (timer) => { + return ( + ((timer.paused ? 0 : Date.now() - timer.startTime) + timer.pauseTime) / + 1000 + ); + }; + + class Timers { + constructor() { + vm.runtime.on("PROJECT_START", () => { + timers = Object.create(null); + }); + } + + getInfo() { + return { + id: "lmsTimers", + name: "ๆ›ดๅฅฝ็š„่ฎกๆ—ถๅ™จ", + color1: "#5cb1d6", + color2: "#428faf", + color3: "#3281a3", + blocks: [ + { + opcode: "whenTimerOp", + blockType: Scratch.BlockType.HAT, + extensions: ["colours_sensing"], + text: "ๅฝ“่ฎกๆ—ถๅ™จ[TIMER][OP][NUM]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + OP: { + type: Scratch.ArgumentType.STRING, + menu: "operation", + }, + NUM: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "5", + }, + }, + }, + + "---", + + { + opcode: "startResetTimer", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "ๅฏๅŠจ/ๅคไฝ ่ฎกๆ—ถๅ™จ[TIMER]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + }, + }, + { + opcode: "valueOfTimer", + blockType: Scratch.BlockType.REPORTER, + extensions: ["colours_sensing"], + text: "่ฎกๆ—ถๅ™จ[TIMER]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + }, + }, + + "---", + + { + opcode: "pauseTimer", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "ๆš‚ๅœ่ฎกๆ—ถๅ™จ[TIMER]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + }, + }, + { + opcode: "resumeTimer", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "ๆขๅค่ฎกๆ—ถๅ™จ[TIMER]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + }, + }, + + "---", + + { + opcode: "setTimer", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "่ฎพ็ฝฎ่ฎกๆ—ถๅ™จ[TIMER]ไธบ[NUM]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + NUM: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + }, + }, + { + opcode: "changeTimer", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "ๆ›ดๆ”น่ฎกๆ—ถๅ™จ[TIMER]็”ฑ[NUM]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + NUM: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: "10", + }, + }, + }, + + "---", + + { + opcode: "removeTimer", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "ๅˆ ้™ค่ฎกๆ—ถๅ™จ[TIMER]", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + }, + }, + { + opcode: "removeTimers", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_sensing"], + text: "ๅˆ ้™คๅ…จ้ƒจ่ฎกๆ—ถๅ™จ", + }, + { + opcode: "timerExists", + blockType: Scratch.BlockType.BOOLEAN, + extensions: ["colours_sensing"], + text: "่ฎกๆ—ถๅ™จ[TIMER]ๅญ˜ๅœจๅ—๏ผŸ", + arguments: { + TIMER: { + type: Scratch.ArgumentType.STRING, + defaultValue: "timer", + }, + }, + }, + { + opcode: "listExistingTimers", + blockType: Scratch.BlockType.REPORTER, + extensions: ["colours_sensing"], + text: "ๅˆ—ๅ‡บๆ‰€ๆœ‰่ฎกๆ—ถๅ™จ", + disableMonitor: false, + }, + ], + menus: { + operation: { + // false for Scratch parity + acceptReporters: false, + items: [">", "<"], + }, + }, + }; + } + + whenTimerOp(args) { + if (!timers[args.TIMER]) return false; + const value = timerValue(timers[args.TIMER]); + if (args.OP === ">") return value > args.NUM; + if (args.OP === "<") return value < args.NUM; + return false; + } + + startResetTimer(args) { + timers[args.TIMER] = { + startTime: Date.now(), + pauseTime: 0, + paused: false, + }; + } + + pauseTimer(args) { + const timer = timers[args.TIMER]; + if (!timer) return; + timer.pauseTime = timerValue(timer) * 1000; + timer.paused = true; + } + + resumeTimer(args) { + const timer = timers[args.TIMER]; + if (!timer) return; + if (timer.paused === false) return; + timer.paused = false; + timer.startTime = Date.now(); + } + + valueOfTimer(args) { + if (!timers[args.TIMER]) return ""; + return timerValue(timers[args.TIMER]); + } + + setTimer(args) { + timers[args.TIMER] = { + paused: false, + startTime: Date.now(), + pauseTime: Scratch.Cast.toNumber(args.NUM) * 1000, + }; + } + + changeTimer(args) { + if (!timers[args.TIMER]) this.startResetTimer(args); + timers[args.TIMER].pauseTime += Scratch.Cast.toNumber(args.NUM) * 1000; + } + + removeTimers(args) { + timers = Object.create(null); + } + + removeTimer(args) { + Reflect.deleteProperty(timers, args.TIMER); + } + + timerExists(args) { + return !!timers[args.TIMER]; + } + + listExistingTimers(args) { + return Object.keys(timers).join(","); + } + } + + // "Extension" option reimplementation by Xeltalliv + // https://github.com/Xeltalliv/extensions/blob/examples/examples/extension-colors.js + + // const cbfsb = Scratch.vm.runtime._convertBlockForScratchBlocks.bind(Scratch.vm.runtime); + // Scratch.vm.runtime._convertBlockForScratchBlocks = function(blockInfo, categoryInfo) { + // const res = cbfsb(blockInfo, categoryInfo); + // if (blockInfo.extensions) { + // if (!res.json.extensions) res.json.extensions = []; + // res.json.extensions.push(...blockInfo.extensions); + // } + // return res; + // }; + + Scratch.extensions.register(new Timers()); +})(Scratch); \ No newline at end of file From 6127c7f127c8a71b99244f9e61b31bbf0424b7c9 Mon Sep 17 00:00:00 2001 From: HanHanDeYaYa Date: Tue, 11 Jun 2024 12:53:33 +0800 Subject: [PATCH 5/8] Delete Variable beautification.js --- .../Variable beautification.js | 783 ------------------ 1 file changed, 783 deletions(-) delete mode 100644 Variable beautification/Variable beautification.js diff --git a/Variable beautification/Variable beautification.js b/Variable beautification/Variable beautification.js deleted file mode 100644 index 59cbfe15..00000000 --- a/Variable beautification/Variable beautification.js +++ /dev/null @@ -1,783 +0,0 @@ -// Name: Custom Styles -// ID: shovelcss -// Description: Customize the appearance of variable monitors and prompts in your project. -// By: TheShovel -// License: MIT - -// Thanks LilyMakesThings for the awesome banner! -(function (Scratch) { - "use strict"; - - // Styles - let monitorText = ""; - let monitorBorder = ""; - let monitorBackgroundColor = ""; - let variableValueBackground = ""; - let variableValueTextColor = ""; - let listFooterBackground = ""; - let listHeaderBackground = ""; - let listValueText = ""; - let listValueBackground = ""; - let variableValueRoundness = -1; - let listValueRoundness = -1; - let monitorBackgroundRoundness = -1; - let monitorBackgroundBorderWidth = -1; - let allowScrolling = ""; - let askBackground = ""; - let askBackgroundRoundness = -1; - let askBackgroundBorderWidth = -1; - let askButtonBackground = ""; - let askButtonRoundness = -1; - let askInputBackground = ""; - let askInputRoundness = -1; - let askInputBorderWidth = -1; - let askBoxIcon = ""; - let askInputText = ""; - let askQuestionText = ""; - let askButtonImage = ""; - let askInputBorder = ""; - - // CSS selectors - let monitorRoot; - let monitorValue; - let monitorListHeader; - let monitorListFooter; - let monitorRowValueOuter; - let monitorRowsInner; - let monitorRowsScroller; - let monitorRowIndex; - let monitorValueLarge; - let askBoxBG; - let askBoxButton; - let askBoxInner; - let askBoxText; - let askBoxBorderMain; - let askBoxBorderOuter; - if (typeof scaffolding !== "undefined") { - monitorRoot = ".sc-monitor-root"; - monitorValue = ".sc-monitor-value"; - monitorListHeader = ".sc-monitor-list-label"; - monitorListFooter = ".sc-monitor-list-footer"; - monitorRowValueOuter = ".sc-monitor-row-value-outer"; - monitorRowsInner = ".sc-monitor-rows-inner"; - monitorRowsScroller = monitorRowsInner; - monitorRowIndex = ".sc-monitor-row-index"; - monitorValueLarge = ".sc-monitor-large-value"; - askBoxBG = ".sc-question-inner"; - askBoxButton = ".sc-question-submit-button"; - askBoxInner = ".sc-question-input"; - askBoxText = ".sc-question-text"; - askBoxBorderMain = ".sc-question-input:hover"; - askBoxBorderOuter = ".sc-question-input:focus"; - } else { - monitorRoot = 'div[class^="monitor_monitor-container_"]'; - monitorValue = 'div[class^="monitor_value_"]'; - monitorListHeader = 'div[class^="monitor_list-header_"]'; - monitorListFooter = 'div[class^="monitor_list-footer_"]'; - monitorRowValueOuter = 'div[class^="monitor_list-value_"]'; - monitorRowsInner = 'div[class^="monitor_list-body_"]'; - monitorRowsScroller = - 'div[class^="monitor_list-body_"] > .ReactVirtualized__List'; - monitorRowIndex = 'div[class^="monitor_list-index_"]'; - monitorValueLarge = 'div[class^="monitor_large-value_"]'; - askBoxBG = 'div[class^="question_question-container_"]'; - askBoxButton = 'button[class^="question_question-submit-button_"]'; - askBoxInner = - '[class^="question_question-container_"] input[class^="input_input-form_"]'; - askBoxText = - '[class^="question_question-container_"] div[class^="question_question-label_"]'; - askBoxIcon = 'img[class^="question_question-submit-button-icon_"]'; - askBoxBorderMain = - '[class^="question_question-input_"] input:focus, [class^="question_question-input_"] input:hover'; - askBoxBorderOuter = '[class^="question_question-input_"] > input:focus'; - } - - const ColorIcon = - ""; - const BorderIcon = - ""; - const extensionIcon = - ""; - const miscIcon = - ""; - const TransparentIcon = - ""; - const GradientIcon = - ""; - const PictureIcon = - ""; - const ResetIcon = - ""; - - const stylesheet = document.createElement("style"); - stylesheet.className = "shovelcss-style"; - // end of for higher precedence than other sheets - document.body.appendChild(stylesheet); - - const applyCSS = () => { - let css = ""; - - // We assume all values are sanitized when they are set, so then we can just use them as-is here. - - if (monitorText) { - css += `${monitorRoot}, ${monitorListFooter}, ${monitorListHeader}, ${monitorRowIndex} { color: ${monitorText}; }`; - } - if (monitorBackgroundColor) { - css += `${monitorRoot}, ${monitorRowsInner} { background: ${monitorBackgroundColor}; }`; - } - if (monitorBorder) { - css += `${monitorRoot} { border-color: ${monitorBorder}; }`; - } - if (monitorBackgroundRoundness >= 0) { - css += `${monitorRoot} { border-radius: ${monitorBackgroundRoundness}px; }`; - } - if (monitorBackgroundBorderWidth >= 0) { - css += `${monitorRoot} { border-width: ${monitorBackgroundBorderWidth}px; }`; - } - if (variableValueBackground) { - css += `${monitorValue}, ${monitorValueLarge} { background: ${variableValueBackground} !important; }`; - } - if (variableValueTextColor) { - css += `${monitorValue}, ${monitorValueLarge} { color: ${variableValueTextColor}; }`; - } - if (variableValueRoundness >= 0) { - css += `${monitorValue} { border-radius: ${variableValueRoundness}px; }`; - } - if (listHeaderBackground) { - css += `${monitorListHeader} { background: ${listHeaderBackground}; }`; - } - if (listFooterBackground) { - css += `${monitorListFooter} { background: ${listHeaderBackground}; }`; - } - if (listValueBackground) { - css += `${monitorRowValueOuter} { background: ${listValueBackground} !important; }`; - } - if (listValueText) { - css += `${monitorRowValueOuter} { color: ${listValueText}; }`; - } - if (listValueRoundness >= 0) { - css += `${monitorRowValueOuter} { border-radius: ${listValueRoundness}px; }`; - } - if (allowScrolling) { - css += `${monitorRowsScroller} { overflow: ${allowScrolling} !important; }`; - } - if (askBackground) { - css += `${askBoxBG} { background: ${askBackground} !important; border: none !important; }`; - } - if (askBackgroundRoundness >= 0) { - css += `${askBoxBG} { border-radius: ${askBackgroundRoundness}px !important; }`; - } - if (askBackgroundBorderWidth >= 0) { - css += `${askBoxBG} { border-width: ${askBackgroundBorderWidth}px !important; }`; - } - if (askButtonBackground) { - css += `${askBoxButton} { background-color: ${askButtonBackground}; }`; - } - if (askButtonRoundness >= 0) { - css += `${askBoxButton} { border-radius: ${askButtonRoundness}px !important; }`; - } - if (askInputBackground) { - css += `${askBoxInner} { background: ${askInputBackground} !important; }`; - css += `${askBoxInner} { border: none !important; }`; - } - if (askInputText) { - css += `${askBoxInner} { color: ${askInputText} !important; }`; - } - if (askQuestionText) { - css += `${askBoxText} { color: ${askQuestionText} !important; }`; - } - if (askInputRoundness >= 0) { - css += `${askBoxInner} { border-radius: ${askInputRoundness}px !important; }`; - } - if (askInputBorderWidth >= 0) { - css += `${askBoxInner} { border-width: ${askInputBorderWidth}px !important; }`; - } - if (askButtonImage) { - css += `${askBoxButton} { background-image: url("${encodeURI( - askButtonImage - )}") !important; background-repeat: no-repeat; background-size: contain; }`; - css += `${askBoxIcon} { visibility: hidden; }`; - } - if (askInputBorder) { - css += `${askBoxBorderMain}, ${askBoxBorderOuter} { border-color: ${askInputBorder} !important; }`; - css += `${askBoxBorderOuter} { box-shadow: none !important; }`; - } - - stylesheet.textContent = css; - }; - - const resetStyles = () => { - monitorText = ""; - monitorBorder = ""; - monitorBackgroundColor = ""; - variableValueBackground = ""; - variableValueTextColor = ""; - listFooterBackground = ""; - listHeaderBackground = ""; - listValueText = ""; - listValueBackground = ""; - variableValueRoundness = -1; - listValueRoundness = -1; - monitorBackgroundRoundness = -1; - monitorBackgroundBorderWidth = -1; - allowScrolling = ""; - askBackground = ""; - askBackgroundRoundness = -1; - askBackgroundBorderWidth = -1; - askButtonBackground = ""; - askButtonRoundness = -1; - askInputBackground = ""; - askInputRoundness = -1; - askInputBorderWidth = -1; - askBoxIcon = ""; - askInputText = ""; - askQuestionText = ""; - askButtonImage = ""; - askInputBorder = ""; - - applyCSS(); - }; - - const getMonitorRoot = (id) => { - const allMonitors = document.querySelectorAll(monitorRoot); - for (const monitor of allMonitors) { - if (monitor.dataset.id === id) { - return monitor; - } - } - return null; - }; - - /** - * @param {string} id - * @param {number} x - * @param {number} y - */ - const setMonitorPosition = (id, x, y) => { - const root = getMonitorRoot(id); - if (root) { - root.style.transform = `translate(${x}px, ${y}px)`; - root.style.left = "0px"; - root.style.top = "0px"; - } - }; - - /** - * @param {VM.Target} target - * @param {string} name - * @param {VM.VariableType} type - * @param {number} x - * @param {number} y - */ - const setVariableMonitorPosition = (target, name, type, x, y) => { - // @ts-expect-error - const variable = target.lookupVariableByNameAndType(name, type); - if (variable) { - // @ts-expect-error - setMonitorPosition(variable.id, x, y); - } - }; - - const parseColor = (color, callback) => { - color = Scratch.Cast.toString(color); - - // These might have some exponential backtracking/ReDoS, but that's not really a concern here. - // If a project wanted to get stuck in an infinite loop, there are so many other ways to do that. - - // Simple color code or name - if (/^#?[a-z0-9]+$/.test(color)) { - callback(color); - return; - } - - // General gradient pattern - if (/^[a-z-]+-gradient\([a-z0-9,#%. ]+\)$/i.test(color)) { - callback(color); - return; - } - - // URL - // see list of non-escaped characters: - // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/encodeURI#description - const match = color.match(/^url\("([A-Za-z0-9\-_.!~*'();/?:@&=+$,#]+)"\)$/); - if (match) { - const url = match[1]; - return Scratch.canFetch(url).then((allowed) => { - if (allowed) { - callback(color); - } - }); - } - - console.error("Invalid color", color); - }; - - Scratch.vm.runtime.on("RUNTIME_DISPOSED", resetStyles); - - class MonitorStyles { - getInfo() { - return { - id: "shovelcss", - name: "Custom Styles", - menuIconURI: extensionIcon, - color1: "#0072d6", - color2: "#0064bc", - color3: "#01539b", - blocks: [ - { - blockIconURI: ColorIcon, - opcode: "changecss", - blockType: Scratch.BlockType.COMMAND, - text: "set [COLORABLE] to [COLOR]", - arguments: { - COLORABLE: { - type: Scratch.ArgumentType.STRING, - menu: "COLORABLE_MENU", - }, - COLOR: { - type: Scratch.ArgumentType.COLOR, - defaultValue: "#ff0000", - }, - }, - }, - { - blockIconURI: GradientIcon, - opcode: "gradientAngle", - blockType: Scratch.BlockType.REPORTER, - text: "make a gradient with [COLOR1] and [COLOR2] at angle [ANGLE]", - arguments: { - COLOR1: { - type: Scratch.ArgumentType.COLOR, - defaultValue: "#ff0000", - }, - COLOR2: { - type: Scratch.ArgumentType.COLOR, - defaultValue: "#6ed02d", - }, - ANGLE: { - type: Scratch.ArgumentType.ANGLE, - defaultValue: "90", - }, - }, - }, - { - blockIconURI: TransparentIcon, - disableMonitor: true, - opcode: "transparentinput", - blockType: Scratch.BlockType.REPORTER, - text: "transparent", - }, - { - blockIconURI: PictureIcon, - disableMonitor: true, - opcode: "pictureinput", - blockType: Scratch.BlockType.REPORTER, - text: "image [URL]", - arguments: { - URL: { - type: Scratch.ArgumentType.STRING, - defaultValue: "https://extensions.turbowarp.org/dango.png", - }, - }, - }, - "---", - { - blockIconURI: PictureIcon, - disableMonitor: true, - opcode: "setAskURI", - blockType: Scratch.BlockType.COMMAND, - text: "set ask prompt button image to [URL]", - arguments: { - URL: { - type: Scratch.ArgumentType.STRING, - defaultValue: "https://extensions.turbowarp.org/dango.png", - }, - }, - }, - "---", - { - blockIconURI: BorderIcon, - opcode: "setbordersize", - blockType: Scratch.BlockType.COMMAND, - text: "set border width of [BORDER] to [SIZE]", - arguments: { - BORDER: { - type: Scratch.ArgumentType.STRING, - menu: "BORDER_WIDTH_MENU", - }, - SIZE: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: "2", - }, - }, - }, - { - blockIconURI: BorderIcon, - opcode: "setborderradius", - blockType: Scratch.BlockType.COMMAND, - text: "set roundness of [CORNER] to [SIZE]", - arguments: { - SIZE: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: "4", - }, - CORNER: { - type: Scratch.ArgumentType.STRING, - menu: "BORDER_ROUNDNESS_MENU", - }, - }, - }, - "---", - { - blockIconURI: ResetIcon, - opcode: "clearCSS", - blockType: Scratch.BlockType.COMMAND, - text: "reset styles", - }, - "---", - { - blockIconURI: miscIcon, - opcode: "allowscrollrule", - blockType: Scratch.BlockType.COMMAND, - text: "set list scrolling to [SCROLLRULE]", - arguments: { - SCROLLRULE: { - type: Scratch.ArgumentType.STRING, - menu: "SCROLL_MENU", - }, - }, - }, - { - blockIconURI: miscIcon, - opcode: "getValue", - blockType: Scratch.BlockType.REPORTER, - text: "get [ITEM]", - arguments: { - ITEM: { - type: Scratch.ArgumentType.STRING, - menu: "VALUEGET_LIST", - }, - }, - }, - "---", - { - blockIconURI: miscIcon, - opcode: "setvarpos", - blockType: Scratch.BlockType.COMMAND, - text: "set position of variable [NAME] to x: [X] y: [Y]", - arguments: { - X: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: "0", - }, - Y: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: "0", - }, - NAME: { - type: Scratch.ArgumentType.STRING, - defaultValue: "my variable", - }, - }, - }, - { - blockIconURI: miscIcon, - opcode: "setlistpos", - blockType: Scratch.BlockType.COMMAND, - text: "set position of list [NAME] to x: [X] y: [Y]", - arguments: { - X: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: "0", - }, - Y: { - type: Scratch.ArgumentType.NUMBER, - defaultValue: "0", - }, - NAME: { - type: Scratch.ArgumentType.STRING, - defaultValue: "my variable", - }, - }, - }, - ], - // Accepting reporters because there can't be errors in case the value is not correct - menus: { - COLORABLE_MENU: { - acceptReporters: true, - items: [ - "monitor text", - "monitor background", - "monitor border", - "variable value background", - "variable value text", - "list header background", - "list footer background", - "list value background", - "list value text", - "ask prompt background", - "ask prompt button background", - "ask prompt input background", - "ask prompt question text", - "ask prompt input text", - "ask prompt input border", - ], - }, - BORDER_WIDTH_MENU: { - acceptReporters: true, - items: [ - "monitor background", - "ask prompt background", - "ask prompt input", - ], - }, - BORDER_ROUNDNESS_MENU: { - acceptReporters: true, - items: [ - "monitor background", - "variable value", - "list value", - "ask prompt background", - "ask prompt button", - "ask prompt input", - ], - }, - SCROLL_MENU: { - acceptReporters: true, - items: ["enabled", "disabled"], - }, - VALUEGET_LIST: { - acceptReporters: true, - items: [ - "monitor text", - "monitor background", - "monitor border color", - "variable value background", - "variable value text", - "list header background", - "list footer background", - "list value background", - "list value text", - "ask prompt background", - "ask prompt button background", - "ask prompt input background", - "ask prompt question text", - "ask prompt input text", - "ask prompt input border", - "monitor background border width", - "ask prompt background border width", - "ask prompt input border width", - "monitor background roundness", - "variable value roundness", - "list value roundness", - "ask prompt background roundness", - "ask prompt button roundness", - "ask prompt input roundness", - "ask prompt button image", - "list scroll rule", - ], - }, - }, - }; - } - - changecss(args) { - return parseColor(args.COLOR, (color) => { - if (args.COLORABLE === "monitor text") { - monitorText = color; - } else if (args.COLORABLE === "monitor background") { - monitorBackgroundColor = color; - } else if (args.COLORABLE === "monitor border") { - monitorBorder = color; - } else if (args.COLORABLE === "variable value background") { - variableValueBackground = color; - } else if (args.COLORABLE === "variable value text") { - variableValueTextColor = color; - } else if (args.COLORABLE === "list header background") { - listHeaderBackground = color; - } else if (args.COLORABLE === "list footer background") { - listFooterBackground = color; - } else if (args.COLORABLE === "list value background") { - listValueBackground = color; - } else if (args.COLORABLE === "list value text") { - listValueText = color; - } else if (args.COLORABLE === "ask prompt background") { - askBackground = color; - } else if (args.COLORABLE === "ask prompt button background") { - askButtonBackground = color; - } else if (args.COLORABLE === "ask prompt input background") { - askInputBackground = color; - } else if (args.COLORABLE === "ask prompt question text") { - askQuestionText = color; - } else if (args.COLORABLE === "ask prompt input text") { - askInputText = color; - } else if (args.COLORABLE === "ask prompt input border") { - askInputBorder = color; - } - - applyCSS(); - }); - } - - gradientAngle(args) { - return ( - "linear-gradient(" + - args.ANGLE + - "deg," + - args.COLOR1 + - "," + - args.COLOR2 + - ")" - ); - } - - setbordersize(args) { - const size = Scratch.Cast.toNumber(args.SIZE); - if (args.BORDER === "monitor background") { - monitorBackgroundBorderWidth = size; - } else if (args.BORDER === "ask prompt background") { - askBackgroundBorderWidth = size; - } else if (args.BORDER === "ask prompt input") { - askInputBorderWidth = size; - } - applyCSS(); - } - - setborderradius(args) { - const size = Scratch.Cast.toNumber(args.SIZE); - if (args.CORNER === "monitor background") { - monitorBackgroundRoundness = size; - } else if (args.CORNER === "variable value") { - variableValueRoundness = size; - } else if (args.CORNER === "list value") { - listValueRoundness = size; - } else if (args.CORNER === "ask prompt background") { - askBackgroundRoundness = size; - } else if (args.CORNER === "ask prompt button") { - askButtonRoundness = size; - } else if (args.CORNER === "ask prompt input") { - askInputRoundness = size; - } - applyCSS(); - } - - allowscrollrule(args) { - if (args.SCROLLRULE === "enabled") { - allowScrolling = "auto"; - } else { - allowScrolling = "hidden"; - } - applyCSS(); - } - - setvarpos(args, util) { - setVariableMonitorPosition( - util.target, - args.NAME, - "", - Scratch.Cast.toNumber(args.X) + Scratch.vm.runtime.stageWidth / 2, - Scratch.vm.runtime.stageHeight / 2 - Scratch.Cast.toNumber(args.Y) - ); - } - - setlistpos(args, util) { - setVariableMonitorPosition( - util.target, - args.NAME, - "list", - Scratch.Cast.toNumber(args.X) + Scratch.vm.runtime.stageWidth / 2, - Scratch.vm.runtime.stageHeight / 2 - Scratch.Cast.toNumber(args.Y) - ); - } - - help() { - alert( - "\nThis is a short introduction to how to use the Monitor Styles extension!\n\n๐—Ÿ๐—ผ๐—ผ๐—ธ๐˜€ ๐—ฏ๐—น๐—ผ๐—ฐ๐—ธ๐˜€\nThese blocks change the appearance of the variable and list didsplays. You can use the drop-down menu to select what component you want to modify. ๐™๐™๐™š ๐™˜๐™ค๐™ก๐™ค๐™ง ๐™—๐™ก๐™ค๐™˜๐™  modifieas the color of a component. You can use the ๐™œ๐™ง๐™–๐™™๐™ž๐™š๐™ฃ๐™ฉ block inside the color input, to create gradients or the ๐™„๐™ข๐™–๐™œ๐™š block to use a image instead of solid colors. ๐™๐™๐™š๐™จ๐™š ๐™ฉ๐™ฌ๐™ค ๐™ค๐™ฃ๐™ก๐™ฎ ๐™ฌ๐™ค๐™ง๐™  ๐™ค๐™ฃ ๐™˜๐™š๐™ง๐™ฉ๐™–๐™ž๐™ฃ ๐™˜๐™ค๐™ข๐™ฅ๐™ค๐™ฃ๐™š๐™ฃ๐™ฉ๐™จ! You can also use the ๐™ฉ๐™ง๐™–๐™ฃ๐™จ๐™ฅ๐™–๐™ง๐™š๐™ฃ๐™ฉ ๐™—๐™ก๐™ค๐™˜๐™  as a color input, to make components invisible. The ๐™—๐™ค๐™ง๐™™๐™š๐™ง ๐™—๐™ก๐™ค๐™˜๐™ ๐™จ modify the borders of components.\n\n๐—ฆ๐—ฒ๐—ป๐˜€๐—ถ๐—ป๐—ด ๐—ฏ๐—น๐—ผ๐—ฐ๐—ธ๐˜€\nThese blocks can change the behaviour of certain components. The ๐™จ๐™˜๐™ง๐™ค๐™ก๐™ก ๐™ง๐™ช๐™ก๐™š block change the behaviour for lists. On 'auto' they will show the scroll bar, and allow you to school, but on 'hidden', they won't let you do that, and the scroll bar will be hidden.\n\n๐— ๐—ผ๐˜๐—ถ๐—ผ๐—ป ๐—ฏ๐—น๐—ผ๐—ฐ๐—ธ๐˜€\nThese blocks allow you to move variable and list displays around. You need to use their ๐™ก๐™–๐™—๐™š๐™ก ๐™ฃ๐™–๐™ข๐™š. The label name is the text that displays on the monitor. For example, a 'for this sprite only' variable will be like 'Sprite1: my variable'." - ); - } - - transparentinput() { - return "transparent"; - } - - pictureinput(args) { - return `url("${encodeURI(args.URL)}")`; - } - - clearCSS() { - resetStyles(); - } - - getValue(args) { - if (args.ITEM === "monitor text") { - return monitorText; - } else if (args.ITEM === "monitor background") { - return monitorBackgroundColor; - } else if (args.ITEM === "monitor border color") { - return monitorBorder; - } else if (args.ITEM === "variable value background") { - return variableValueBackground; - } else if (args.ITEM === "variable value text") { - return variableValueTextColor; - } else if (args.ITEM === "list header background") { - return listHeaderBackground; - } else if (args.ITEM === "list footer background") { - return listFooterBackground; - } else if (args.ITEM === "list value background") { - return listValueBackground; - } else if (args.ITEM === "list value text") { - return listValueText; - } else if (args.ITEM === "ask prompt background") { - return askBackground; - } else if (args.ITEM === "ask prompt button background") { - return askButtonBackground; - } else if (args.ITEM === "ask prompt input background") { - return askInputBackground; - } else if (args.ITEM === "ask prompt question text") { - return askQuestionText; - } else if (args.ITEM === "ask prompt input text") { - return askInputText; - } else if (args.ITEM === "ask prompt input border") { - return askInputBorder; - } else if (args.ITEM === "monitor background border width") { - return monitorBackgroundBorderWidth; - } else if (args.ITEM === "ask prompt background border width") { - return askBackgroundBorderWidth; - } else if (args.ITEM === "ask prompt input border width") { - return askInputBorderWidth; - } else if (args.ITEM === "monitor background roundness") { - return monitorBackgroundRoundness; - } else if (args.ITEM === "variable value roundness") { - return variableValueRoundness; - } else if (args.ITEM === "list value roundness") { - return listValueRoundness; - } else if (args.ITEM === "ask prompt background roundness") { - return askBackgroundRoundness; - } else if (args.ITEM === "ask prompt button roundness") { - return askButtonRoundness; - } else if (args.ITEM === "ask prompt input roundness") { - return askInputRoundness; - } else if (args.ITEM === "ask prompt button image") { - return askButtonImage; - } else if (args.ITEM === "list scrolling") { - if (allowScrolling === "auto") { - return "enabled"; - } else { - return "disabled"; - } - } - return ""; - } - - setAskURI(args) { - return Scratch.canFetch(args.URL).then((allowed) => { - if (allowed) { - askButtonImage = args.URL; - applyCSS(); - } - }); - } - } - - Scratch.extensions.register(new MonitorStyles()); -})(Scratch); \ No newline at end of file From 46eb58c2a7edffca2371acaf4d0c445d9b10f439 Mon Sep 17 00:00:00 2001 From: HanHanDeYaYa Date: Tue, 11 Jun 2024 19:07:00 +0800 Subject: [PATCH 6/8] Test --- Test/Test.js | 619 +++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 619 insertions(+) create mode 100644 Test/Test.js diff --git a/Test/Test.js b/Test/Test.js new file mode 100644 index 00000000..2ff17a72 --- /dev/null +++ b/Test/Test.js @@ -0,0 +1,619 @@ +// Name: Text +// ID: strings +// Description: Manipulate characters and text. +// Original: CST1229 +// License: MIT AND MPL-2.0 + +/* generated l10n code */Scratch.translate.setup({"es":{"_Text":"Texto"},"fi":{"_Text":"Teksti"},"it":{"_Text":"Testo"},"ja":{"_Text":"ใƒ†ใ‚ญใ‚นใƒˆ"},"ko":{"_Text":"ํ…์ŠคํŠธ"},"nb":{"_Text":"Tekst"},"nl":{"_Exactly Title Case":"Alleen Beginhoofdletters","_MiXeD CaSe":"DoOr eLkAaR","_Text":"Tekst","_Title Case":"Alles Met Beginhoofdletter","_UPPERCASE":"HOOFDLETTERS","_[STRING] matches regex /[REGEX]/[FLAGS]?":"[STRING] komt overeen met regex /[REGEX]/[FLAGS]?","_convert [STRING] to [TEXTCASE]":"zet [STRING] om naar [TEXTCASE]","_count [SUBSTRING] in [STRING]":"aantal [SUBSTRING] in [STRING]","_count regex /[REGEX]/[FLAGS] in [STRING]":"aantal overeenkomsten van regex /[REGEX]/[FLAGS] met [STRING]","_index of [SUBSTRING] in [STRING]":"positie van [SUBSTRING] in [STRING]","_is [OPERAND1] identical to [OPERAND2]?":"[OPERAND1] identiek aan [OPERAND2]?","_item [ITEM] of [STRING] matched by regex /[REGEX]/[FLAGS]":"item [ITEM] van [STRING] waarmee regex /[REGEX]/[FLAGS] overeenkomt","_item [ITEM] of [STRING] split by [SPLIT]":"item [ITEM] van [STRING] gesplitst door [SPLIT]","_letters [LETTER1] to [LETTER2] of [STRING]":"letters [LETTER1] t/m [LETTER2] van [STRING]","_lowercase":"kleine letters","_repeat [STRING] [REPEAT] times":"herhaal [STRING] [REPEAT] keer","_replace [SUBSTRING] in [STRING] with [REPLACE]":"vervang [SUBSTRING] in [STRING] door [REPLACE]","_replace regex /[REGEX]/[FLAGS] in [STRING] with [REPLACE]":"vervang regex /[REGEX]/[FLAGS] in [STRING] door [REPLACE]","_unicode [NUM] as letter":"unicode [NUM] als letter","_unicode of [STRING]":"unicode van [STRING]"},"pt-br":{"_Text":"Texto"},"ru":{"_Exactly Title Case":"ะะฐัั‚ะพัั‰ะธะน ะขะฐะนั‚ะป ะšะตะนั","_MiXeD CaSe":"ะกะผะ•ัˆะะฝะั‹ะ™ ะšะตะ™ั","_Text":"ะขะตะบัั‚","_Title Case":"ะขะฐะนั‚ะป ะšะตะนั","_UPPERCASE":"ะ’ะ•ะ ะฅะะ˜ะ™ ะ ะ•ะ“ะ˜ะกะขะ ","_[STRING] matches regex /[REGEX]/[FLAGS]?":"[STRING] ัะพะฒะฟะฐะดะฐะตั‚ ั regex'ะพะผ /[REGEX]/[FLAGS]?","_convert [STRING] to [TEXTCASE]":"ัะบะพะฝะฒะตั€ั‚ะธั€ะพะฒะฐั‚ัŒ ัั‚ั€ะพะบัƒ [STRING] ะฒ [TEXTCASE]","_count [SUBSTRING] in [STRING]":"ะบะพะปะธั‡ะตัั‚ะฒะพ [SUBSTRING] ะฒ [STRING]","_count regex /[REGEX]/[FLAGS] in [STRING]":"ะฟะพัั‡ะธั‚ะฐั‚ัŒ regex /[REGEX]/[FLAGS] ะฒ [STRING]","_index of [SUBSTRING] in [STRING]":"ะธะฝะดะตะบั [SUBSTRING] ะฒ [STRING]","_is [OPERAND1] identical to [OPERAND2]?":"[OPERAND1] ะธะดะตะฝั‚ะธั‡ะฝั‹ะน ั [OPERAND2]?","_is [STRING] [TEXTCASE]?":"ัั‚ั€ะพะบะฐ [STRING] ัั‚ะพ [TEXTCASE]?","_item [ITEM] of [STRING] matched by regex /[REGEX]/[FLAGS]":"ะฟั€ะตะดะผะตั‚ [ITEM] ัั‚ั€ะพะบะธ [STRING] ัะพะฒะฟะฐะดะฐัŽั‰ะธะน regex'ะพะผ /[REGEX]/[FLAGS]","_item [ITEM] of [STRING] split by [SPLIT]":"ะฟั€ะตะดะผะตั‚ [ITEM] ัั‚ั€ะพะบะธ [STRING] ั€ะฐัะฟั€ะตะดะตะปั‘ะฝะฝั‹ะน [SPLIT]","_letters [LETTER1] to [LETTER2] of [STRING]":"ะฑัƒะบะฒั‹ ั [LETTER1] ะดะพ [LETTER2] ัั‚ั€ะพะบะธ [STRING]","_lowercase":"ะฝะธะถะฝะธะน ั€ะตะณะธัั‚ั€","_repeat [STRING] [REPEAT] times":"ะฟะพะฒั‚ะพั€ะธั‚ัŒ [STRING] [REPEAT] ั€ะฐะท","_replace [SUBSTRING] in [STRING] with [REPLACE]":"ะทะฐะผะตะฝะธั‚ัŒ [SUBSTRING] ะฒ [STRING] ะฝะฐ [REPLACE]","_replace regex /[REGEX]/[FLAGS] in [STRING] with [REPLACE]":"ะทะฐะผะตะฝะธั‚ัŒ regex /[REGEX]/[FLAGS] ะฒ [STRING] ะฝะฐ [REPLACE]","_unicode [NUM] as letter":"ัŽะฝะธะบะพะด [NUM] ะบะฐะบ ะฑัƒะบะฒัƒ","_unicode of [STRING]":"ัŽะฝะธะบะพะด [STRING]"},"tr":{"_Text":"Metin"},"uk":{"_Text":"ะขะตะบัั‚"},"zh-cn":{"_Exactly Title Case":"็ฒพ็กฎๆ ‡้ข˜","_MiXeD CaSe":"ๆททๅˆๅคงๅฐๅ†™","_Text":"ๆ–‡ๆœฌ","_Title Case":"ๆ ‡้ข˜","_UPPERCASE":"ๅคงๅ†™","_[STRING] matches regex /[REGEX]/[FLAGS]?":"[STRING] ๆปก่ถณๆญฃๅˆ™่กจ่พพๅผ /[REGEX]/[FLAGS]๏ผŸ","_convert [STRING] to [TEXTCASE]":"่ฝฌๆข[STRING]ไธบ[TEXTCASE]","_count [SUBSTRING] in [STRING]":"[STRING]ไธญ[SUBSTRING]็š„ๆ•ฐ้‡","_count regex /[REGEX]/[FLAGS] in [STRING]":"ไฝฟ็”จๆญฃๅˆ™่กจ่พพๅผ /[REGEX]/[FLAGS] ๅœจ [STRING] ๅŒน้…็š„ๆ•ฐ้‡","_index of [SUBSTRING] in [STRING]":"[STRING]ไธญ[SUBSTRING]็š„ไฝ็ฝฎ","_is [OPERAND1] identical to [OPERAND2]?":"[OPERAND1]===[OPERAND2]","_is [STRING] [TEXTCASE]?":"[STRING]ๆ˜ฏ[TEXTCASE]๏ผŸ","_item [ITEM] of [STRING] matched by regex /[REGEX]/[FLAGS]":"ไฝฟ็”จๆญฃๅˆ™่กจ่พพๅผ /[REGEX]/[FLAGS]ๅŒน้…[STRING]็š„็ฌฌ[ITEM]ไธชๅ†…ๅฎน","_item [ITEM] of [STRING] split by [SPLIT]":"ไปฅ[SPLIT]ๅˆ†ๅ‰ฒ[STRING]ๅŽ็š„็ฌฌ[ITEM]้กน","_letters [LETTER1] to [LETTER2] of [STRING]":"[STRING]็š„็ฌฌ[LETTER1]ๅˆฐ็ฌฌ[LETTER2]ไฝ","_lowercase":"ๅฐๅ†™","_repeat [STRING] [REPEAT] times":"้‡ๅค[REPEAT]ไธช[STRING]","_replace [SUBSTRING] in [STRING] with [REPLACE]":"ๆ›ฟๆข[STRING]ไธญ็š„[SUBSTRING]ไธบ[REPLACE]","_replace regex /[REGEX]/[FLAGS] in [STRING] with [REPLACE]":"ไฝฟ็”จๆญฃๅˆ™่กจ่พพๅผ /[REGEX]/[FLAGS] ๅœจ [STRING]ไธญๆ›ฟๆขไธบ [REPLACE]","_unicode [NUM] as letter":"unicode[NUM]ๅฏนๅบ”็š„ๅญ—็ฌฆ","_unicode of [STRING]":"[STRING]็š„unicode"}});/* end generated l10n code */(function (Scratch) { + "use strict"; + + const CaseParam = { + LOWERCASE: "lowercase", + UPPERCASE: "uppercase", + MIXEDCASE: "mixedcase", + TITLECASE: "titlecase", + EXACTTITLECASE: "exacttitlecase", + }; + + let splitCache; + let matchCache; + + class StringsExt { + constructor() {} + + _initCaseMenu() { + return [ + { + text: Scratch.translate({ + default: "lowercase", + description: "If your language has lowercase, style it accordingly", + }), + value: CaseParam.LOWERCASE, + }, + { + text: Scratch.translate({ + default: "UPPERCASE", + description: "If your language has uppercase, style it accordingly", + }), + value: CaseParam.UPPERCASE, + }, + { + text: Scratch.translate({ + default: "Title Case", + description: + "If your language has Title Case, style it accordingly. 'Abc' is title case and exactly title case but 'ABC' is only title case.", + }), + value: CaseParam.TITLECASE, + }, + { + text: Scratch.translate({ + default: "Exactly Title Case", + description: + "If your language has Title Case, style it accordingly. 'Abc' is title case and exactly title case but 'ABC' is only title case.", + }), + value: CaseParam.EXACTTITLECASE, + }, + { + text: Scratch.translate({ + default: "MiXeD CaSe", + description: + "If your language has mixed case, style it accordingly", + }), + value: CaseParam.MIXEDCASE, + }, + ]; + } + + getInfo() { + return { + // id "text" would conflict with Scratch Lab's Animated Text (lab/text.js) + id: "strings", + name: Scratch.translate("Text"), + blocks: [ + { + opcode: "letters_of", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate( + "letters [LETTER1] to [LETTER2] of [STRING]" + ), + arguments: { + LETTER1: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 2, + }, + LETTER2: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 4, + }, + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + }, + }, + { + opcode: "split", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("item [ITEM] of [STRING] split by [SPLIT]"), + arguments: { + ITEM: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 3, + }, + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + SPLIT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "p", + }, + }, + }, + { + opcode: "count", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + default: "count [SUBSTRING] in [STRING]", + description: + "Counts how many time [SUBSTRING] appears in [STRING]", + }), + arguments: { + SUBSTRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "p", + }, + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + }, + }, + { + opcode: "indexof", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + default: "index of [SUBSTRING] in [STRING]", + description: "Reports where [SUBSTRING] appears in [STRING]", + }), + arguments: { + SUBSTRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "p", + }, + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + }, + }, + + "---", + + { + opcode: "replace", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate( + "replace [SUBSTRING] in [STRING] with [REPLACE]" + ), + arguments: { + SUBSTRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "world", + }, + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello world!", + }, + REPLACE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "fellow Scratchers", + }, + }, + }, + { + opcode: "repeat", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("repeat [STRING] [REPEAT] times"), + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple ", + }, + REPEAT: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 3, + }, + }, + }, + + "---", + + { + opcode: "unicodeof", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("unicode of [STRING]"), + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "A", + }, + }, + }, + { + opcode: "unicodefrom", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("unicode [NUM] as letter"), + arguments: { + NUM: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 65, + }, + }, + }, + + "---", + { + opcode: "replaceRegex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate( + "replace regex /[REGEX]/[FLAGS] in [STRING] with [REPLACE]" + ), + arguments: { + REGEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: ".", + }, + FLAGS: { + type: Scratch.ArgumentType.STRING, + defaultValue: "g", + }, + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello world!", + }, + REPLACE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "$&$&", + }, + }, + }, + { + opcode: "matchRegex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + default: + "item [ITEM] of [STRING] matched by regex /[REGEX]/[FLAGS]", + description: + "/[REGEX]/ is supposed to match the syntax that some actual programming languages used for regular expressions.", + }), + arguments: { + ITEM: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 1, + }, + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello world!", + }, + REGEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "(.) (.{2})", + }, + FLAGS: { + type: Scratch.ArgumentType.STRING, + defaultValue: "g", + }, + }, + }, + { + opcode: "countRegex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + default: "count regex /[REGEX]/[FLAGS] in [STRING]", + description: + "/[REGEX]/ is supposed to match the syntax that some actual programming languages used for regular expressions.", + }), + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello world!", + }, + REGEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "[AEIOU]", + }, + FLAGS: { + type: Scratch.ArgumentType.STRING, + defaultValue: "i", + }, + }, + }, + { + opcode: "testRegex", + blockType: Scratch.BlockType.BOOLEAN, + text: Scratch.translate({ + default: "[STRING] matches regex /[REGEX]/[FLAGS]?", + description: + "/[REGEX]/ is supposed to match the syntax that some actual programming languages used for regular expressions.", + }), + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello world!", + }, + REGEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "hello", + }, + FLAGS: { + type: Scratch.ArgumentType.STRING, + defaultValue: "i", + }, + }, + }, + + "---", + + { + opcode: "identical", + blockType: Scratch.BlockType.BOOLEAN, + text: Scratch.translate("is [OPERAND1] identical to [OPERAND2]?"), + arguments: { + OPERAND1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "A", + }, + OPERAND2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "a", + }, + }, + }, + + "---", + + { + opcode: "isCase", + blockType: Scratch.BlockType.BOOLEAN, + text: Scratch.translate({ + default: "is [STRING] [TEXTCASE]?", + description: "Example block context: ", + }), + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + TEXTCASE: { + type: Scratch.ArgumentType.STRING, + menu: "textCase", + defaultValue: CaseParam.LOWERCASE, + }, + }, + }, + { + opcode: "toCase", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate({ + default: "convert [STRING] to [TEXTCASE]", + description: + "Example block context: (convert [HELLO] to [lowercase])", + }), + arguments: { + STRING: { + type: Scratch.ArgumentType.STRING, + defaultValue: "apple", + }, + TEXTCASE: { + type: Scratch.ArgumentType.STRING, + menu: "textCase", + defaultValue: CaseParam.UPPERCASE, + }, + }, + }, + ], + menus: { + textCase: { + acceptReporters: true, + items: this._initCaseMenu(), + }, + }, + }; + } + + identical(args, util) { + // Purposefully no casting, because + // types ARE differentiated in this block + return args.OPERAND1 === args.OPERAND2; + } + + unicodeof(args, util) { + const chars = Array.from(args.STRING.toString()); + return chars.map((char) => char.charCodeAt(0)).join(" "); + } + + unicodefrom(args, util) { + return String.fromCharCode(Number(args.NUM) || 0); + } + + letters_of(args, util) { + args.STRING = args.STRING.toString(); + args.LETTER1 = Number(args.LETTER1) || 0; + args.LETTER2 = Number(args.LETTER2) || 0; + return args.STRING.substring(args.LETTER1 - 1, args.LETTER2); + } + + _caseInsensitiveRegex(str) { + return new RegExp(str.replaceAll(/[^a-zA-Z0-9]/g, "\\$&"), "gi"); + } + + split(args, util) { + args.STRING = (args.STRING ?? "").toString(); + args.SPLIT = (args.SPLIT ?? "").toString(); + args.ITEM = Number(args.ITEM) || 0; + + // Cache the last split + if ( + !( + splitCache && + splitCache.string === args.STRING && + splitCache.split === args.SPLIT + ) + ) { + const regex = this._caseInsensitiveRegex(args.SPLIT); + + splitCache = { + string: args.STRING, + split: args.SPLIT, + arr: args.STRING.split(regex), + }; + } + return splitCache.arr[args.ITEM - 1] || ""; + } + + count(args, util) { + // Fill cache + this.split( + { + SPLIT: args.SUBSTRING, + STRING: args.STRING, + ITEM: 0, + }, + util + ); + return splitCache.arr.length - 1 || 0; + } + + replace(args, util) { + args.STRING = args.STRING.toString(); + args.SUBSTRING = args.SUBSTRING.toString(); + + args.REPLACE = args.REPLACE.toString(); + + const regex = this._caseInsensitiveRegex(args.SUBSTRING); + + return args.STRING.replace(regex, args.REPLACE); + } + + indexof(args, util) { + // .toLowerCase() for case insensitivity + args.STRING = (args.STRING ?? "").toString().toLowerCase(); + args.SUBSTRING = (args.SUBSTRING ?? "").toString().toLowerCase(); + + // Since both arguments are casted to strings beforehand, + // we don't have to worry about type differences + // like in the item number of in list block + const found = args.STRING.indexOf(args.SUBSTRING); + + // indexOf returns -1 when no matches are found, we can just +1 + return found + 1; + } + + repeat(args, util) { + args.STRING = args.STRING.toString(); + args.REPEAT = Number(args.REPEAT) || 0; + return args.STRING.repeat(args.REPEAT); + } + + replaceRegex(args, util) { + try { + args.STRING = args.STRING.toString(); + args.REPLACE = args.REPLACE.toString(); + args.REGEX = args.REGEX.toString(); + args.FLAGS = args.FLAGS.toString(); + + return args.STRING.replace( + new RegExp(args.REGEX, args.FLAGS), + args.REPLACE + ); + } catch (e) { + console.error(e); + return ""; + } + } + + matchRegex(args, util) { + try { + args.STRING = (args.STRING ?? "").toString(); + args.REGEX = (args.REGEX ?? "").toString(); + args.FLAGS = (args.FLAGS ?? "").toString(); + args.ITEM = Number(args.ITEM) || 0; + + // Cache the last matched string + if ( + !( + matchCache && + matchCache.string === args.STRING && + matchCache.regex === args.REGEX && + matchCache.flags === args.FLAGS + ) + ) { + const newFlags = args.FLAGS.includes("g") + ? args.FLAGS + : args.FLAGS + "g"; + const regex = new RegExp(args.REGEX, newFlags); + + matchCache = { + string: args.STRING, + regex: args.REGEX, + flags: args.FLAGS, + arr: args.STRING.match(regex) || [], + }; + } + return matchCache.arr[args.ITEM - 1] || ""; + } catch (e) { + console.error(e); + return ""; + } + } + + countRegex(args, util) { + // Fill cache + // (ITEM is casted into 0, + // but we don't care about the return value) + this.matchRegex(args, util); + return matchCache.arr.length || 0; + } + + testRegex(args, util) { + try { + args.STRING = args.STRING.toString(); + args.REGEX = args.REGEX.toString(); + args.FLAGS = args.FLAGS.toString(); + + return new RegExp(args.REGEX, args.FLAGS).test(args.STRING); + } catch (e) { + console.error(e); + return false; + } + } + + isCase(args, util) { + const string = args.STRING.toString(); + const textCase = args.TEXTCASE.toString(); + switch (textCase) { + case CaseParam.LOWERCASE: + return string.toLowerCase() === string; + case CaseParam.UPPERCASE: + return string.toUpperCase() === string; + case CaseParam.MIXEDCASE: + return !( + string.toUpperCase() === string || string.toLowerCase() === string + ); + case CaseParam.TITLECASE: + return string.split(/\b/g).every((word) => { + if (!word) return true; + const titleCased = word[0].toUpperCase() + word.substring(1); + return word === titleCased; + }); + case CaseParam.EXACTTITLECASE: + return string.split(/\b/g).every((word) => { + if (!word) return true; + const titleCased = + word[0].toUpperCase() + word.substring(1).toLowerCase(); + return word === titleCased; + }); + default: + return false; + } + } + + toCase(args, util) { + const string = args.STRING.toString(); + const textCase = args.TEXTCASE.toString(); + switch (textCase) { + case CaseParam.LOWERCASE: + return string.toLowerCase(); + case CaseParam.UPPERCASE: + return string.toUpperCase(); + case CaseParam.MIXEDCASE: + return Array.from(string) + .map((char, index) => + index % 2 === 0 ? char.toUpperCase() : char.toLowerCase() + ) + .join(""); + case CaseParam.TITLECASE: + return string + .split(/\b/g) + .map((word) => { + if (!word) return ""; + return word[0].toUpperCase() + word.substring(1); + }) + .join(""); + case CaseParam.EXACTTITLECASE: + return string + .split(/\b/g) + .map((word) => { + if (!word) return ""; + return word[0].toUpperCase() + word.substring(1).toLowerCase(); + }) + .join(""); + default: + return string; + } + } + } + + Scratch.extensions.register(new StringsExt()); +})(Scratch); \ No newline at end of file From e85e18d1d19b3911ceb6dd3fe32aeaf0bfadd53d Mon Sep 17 00:00:00 2001 From: HanHanDeYaYa Date: Tue, 11 Jun 2024 20:32:11 +0800 Subject: [PATCH 7/8] Test timers --- {Better timer => TurboWarp/Better timer}/Better timer.js | 0 {Test => TurboWarp/Test}/Test.js | 0 2 files changed, 0 insertions(+), 0 deletions(-) rename {Better timer => TurboWarp/Better timer}/Better timer.js (100%) rename {Test => TurboWarp/Test}/Test.js (100%) diff --git a/Better timer/Better timer.js b/TurboWarp/Better timer/Better timer.js similarity index 100% rename from Better timer/Better timer.js rename to TurboWarp/Better timer/Better timer.js diff --git a/Test/Test.js b/TurboWarp/Test/Test.js similarity index 100% rename from Test/Test.js rename to TurboWarp/Test/Test.js From d8968ed86e437fa797e202010325acb0264fb447 Mon Sep 17 00:00:00 2001 From: HanHanDeYaYa Date: Tue, 11 Jun 2024 22:05:11 +0800 Subject: [PATCH 8/8] Update Better timer.js --- TurboWarp/Better timer/Better timer.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/TurboWarp/Better timer/Better timer.js b/TurboWarp/Better timer/Better timer.js index f3f8c5de..659814c8 100644 --- a/TurboWarp/Better timer/Better timer.js +++ b/TurboWarp/Better timer/Better timer.js @@ -144,7 +144,7 @@ opcode: "changeTimer", blockType: Scratch.BlockType.COMMAND, extensions: ["colours_sensing"], - text: "ๆ›ดๆ”น่ฎกๆ—ถๅ™จ[TIMER]็”ฑ[NUM]", + text: "ๆ›ดๆ”น่ฎกๆ—ถๅ™จ[TIMER]ๅฐ†ๅ…ถๅขžๅŠ [NUM]", arguments: { TIMER: { type: Scratch.ArgumentType.STRING,