From 61e7d27f309638382067deea4cbd1fa6e9e3ed3c Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Wed, 13 Dec 2023 20:18:40 -0800 Subject: [PATCH 01/25] Create Display-Text.js --- extensions/SharkPool/Display-Text.js | 676 +++++++++++++++++++++++++++ 1 file changed, 676 insertions(+) create mode 100644 extensions/SharkPool/Display-Text.js diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js new file mode 100644 index 0000000000..f8cb552284 --- /dev/null +++ b/extensions/SharkPool/Display-Text.js @@ -0,0 +1,676 @@ +// Name: Display Text +// ID: SPdisText +// Description: Display Text in Your Projects! +// By: SharkPool + +// Version V.1.0.0 + +(function (Scratch) { + "use strict"; + if (!Scratch.extensions.unsandboxed) throw new Error("Display Text Extension must run unsandboxed!"); + + const menuIconURI = +""; + + const blockIconURI = +""; + const render = Scratch.vm.renderer; + let allText = []; + let lastRecdVals = {}; + const fontMenu = [ + "Scratch", "Sans Serif", + "Serif", "Handwriting", + "Marker", "Curly", "Pixel" + ]; + + class SPdisText { + getInfo() { + return { + id: "SPdisText", + name: "Display Text", + color1: "#778899", + color2: "#5E707F", + color3: "#3C4E5C", + menuIconURI, + blockIconURI, + blocks: [ + { + opcode: "printTxt", + blockType: Scratch.BlockType.COMMAND, + text: "print text [TXT] with ID [ID]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + TXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "hello world" + } + }, + }, + { + opcode: "replaceTxt", + blockType: Scratch.BlockType.COMMAND, + text: "replace text of ID [ID] with [TXT]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + TXT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "lorem ipsum" + } + }, + }, + { + opcode: "removeTxt", + blockType: Scratch.BlockType.COMMAND, + text: "remove text with ID [ID]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } + }, + }, + { + opcode: "removeAllTxt", + blockType: Scratch.BlockType.COMMAND, + text: "remove all text" + }, + "---", + { + opcode: "displayedTexts", + blockType: Scratch.BlockType.REPORTER, + text: "printed texts from ID [ID]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } + }, + }, + { + opcode: "existingID", + blockType: Scratch.BlockType.BOOLEAN, + text: "does text with ID [ID] exist?", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } + }, + }, + { + opcode: "allIDs", + blockType: Scratch.BlockType.REPORTER, + text: "all text IDs" + }, + { blockType: Scratch.BlockType.LABEL, text: "Formatting" }, + { + opcode: "setTextFont", + blockType: Scratch.BlockType.COMMAND, + text: "set font of ID [ID] to [FONT]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + FONT: { type: Scratch.ArgumentType.STRING, menu: "FONTS" } + }, + }, + { + opcode: "setFontSize", + blockType: Scratch.BlockType.COMMAND, + text: "set font size of ID [ID] to [SIZE]px", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + SIZE: { type: Scratch.ArgumentType.NUMBER, defaultValue: 30 } + }, + }, + { + opcode: "setTextAlignment", + blockType: Scratch.BlockType.COMMAND, + text: "set text alignment of ID [ID] to [ALIGNMENT]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + ALIGNMENT: { type: Scratch.ArgumentType.STRING, menu: "ALIGNMENTS" } + }, + }, + { + opcode: "setMargins", + blockType: Scratch.BlockType.COMMAND, + text: "set margins of ID [ID] to width [WIDTH] height [HEIGHT]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + WIDTH: { type: Scratch.ArgumentType.NUMBER, defaultValue: 100 }, + HEIGHT: { type: Scratch.ArgumentType.NUMBER, defaultValue: 30 } + }, + }, + { + opcode: "setTextSpacing", + blockType: Scratch.BlockType.COMMAND, + text: "set [ATT] spacing of ID [ID] to [SPACING] px", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + SPACING: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 }, + ATT: { type: Scratch.ArgumentType.STRING, menu: "TEXT_ATT" } + }, + }, + "---", + { + opcode: "attOfText", + blockType: Scratch.BlockType.REPORTER, + text: "[ATT] of text with ID [ID]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + ATT: { type: Scratch.ArgumentType.STRING, menu: "FORMATS" } + }, + }, + { blockType: Scratch.BlockType.LABEL, text: "Text Visuals" }, + { + opcode: "setTextColor", + blockType: Scratch.BlockType.COMMAND, + text: "set text color of ID [ID] to [COLOR]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + COLOR: { type: Scratch.ArgumentType.COLOR, defaultValue: "#000000" } + }, + }, + { + opcode: "setTextDropShadow", + blockType: Scratch.BlockType.COMMAND, + text: "set text shadow of ID [ID] to x [x] y [y] z [z] color [COLOR]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + x: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }, + y: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }, + z: { type: Scratch.ArgumentType.NUMBER, defaultValue: 2 }, + COLOR: { type: Scratch.ArgumentType.COLOR, defaultValue: "#ff0000" } + }, + }, + { + opcode: "setTextOutline", + blockType: Scratch.BlockType.COMMAND, + text: "set outline of ID [ID] to [COLOR] with thickness [THICKNESS]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + COLOR: { type: Scratch.ArgumentType.COLOR, defaultValue: "#ff0000" }, + THICKNESS: { type: Scratch.ArgumentType.NUMBER, defaultValue: 3 } + }, + }, + "---", + { + opcode: "makeGradient", + blockType: Scratch.BlockType.REPORTER, + text: "make a [TYPE] gradient with [COLOR1] and [COLOR2] at angle [ANGLE]", + arguments: { + COLOR1: { type: Scratch.ArgumentType.COLOR, defaultValue: "#ff0000" }, + COLOR2: { type: Scratch.ArgumentType.COLOR, defaultValue: "#00ff00" }, + ANGLE: { type: Scratch.ArgumentType.ANGLE, defaultValue: 90 }, + TYPE: { type: Scratch.ArgumentType.NUMBER, menu: "GRADIENTS" } + }, + }, + "---", + { + opcode: "setTextCurve", + blockType: Scratch.BlockType.COMMAND, + text: "set text curve of ID [ID] to [ARC]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + ARC: { type: Scratch.ArgumentType.STRING, defaultValue: "" } + }, + }, + { + opcode: "presetCurve", + blockType: Scratch.BlockType.REPORTER, + text: "preset curve [ARC]", + disableMonitor: true, + arguments: { + ARC: { type: Scratch.ArgumentType.STRING, menu: "ARCS" } + }, + }, + { blockType: Scratch.BlockType.LABEL, text: "Positioning" }, + { + opcode: "setTextPosition", + blockType: Scratch.BlockType.COMMAND, + text: "set position of ID [ID] to x: [X] y: [Y]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + X: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }, + Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 } + }, + }, + { + opcode: "setTextZIndex", + blockType: Scratch.BlockType.COMMAND, + text: "set layer of ID [ID] to [Z_INDEX]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + Z_INDEX: { type: Scratch.ArgumentType.NUMBER, defaultValue: 1 } + }, + }, + { + opcode: "textPosition", + blockType: Scratch.BlockType.REPORTER, + text: "[ATT] of ID [ID]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + ATT: { type: Scratch.ArgumentType.NUMBER, menu: "POS" } + }, + }, + { blockType: Scratch.BlockType.LABEL, text: "Effects" }, + { + opcode: "setEffect", + blockType: Scratch.BlockType.COMMAND, + text: "set [EFFECT] of ID [ID] to [VALUE]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + VALUE: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 }, + EFFECT: { type: Scratch.ArgumentType.STRING, menu: "EFFECTS" } + }, + }, + { + opcode: "amtOfEffect", + blockType: Scratch.BlockType.REPORTER, + text: "current [EFFECT] of ID [ID]", + arguments: { + EFFECT: { type: Scratch.ArgumentType.STRING, menu: "EFFECTS" }, + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } + }, + }, + { + opcode: "resetEffect", + blockType: Scratch.BlockType.COMMAND, + text: "reset effects of ID [ID]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } + }, + }, + ], + menus: { + FONTS: { + acceptReporters: true, + items: "allFonts" + }, + ALIGNMENTS: { + acceptReporters: true, + items: ["left", "right", "center"] + }, + TEXT_ATT: ["letter", "line"], + GRADIENTS: { + acceptReporters: true, // conic gradients exist but are complex + items: ["linear", "radial"] + }, + POS: ["x position", "y position", "z layer"], + ARCS: ["circle", "hill", "dip", "wave"], + FORMATS: { + acceptReporters: true, + items: [ + { text: "font", value: "fontFamily" }, + { text: "font size", value: "fontSize" }, + { text: "alignment", value: "textAlign" }, + { text: "margin width", value: "width" }, + { text: "margin height", value: "height" }, + { text: "letter spacing", value: "letterSpacing" }, + { text: "text spacing", value: "lineHeight" } + ] + }, + EFFECTS: { + acceptReporters: true, + items: [ + { text: "blur", value: "blur" }, + { text: "saturation", value: "saturate" }, + { text: "contrast", value: "contrast" }, + { text: "brightness", value: "brightness" }, + { text: "hue", value: "hue-rotate" }, + { text: "opacity", value: "opacity" }, + { text: "sepia", value: "sepia" }, + { text: "invert", value: "invert" }, + { text: "direction", value: "rotate" }, + { text: "scale x", value: "scaleX" }, + { text: "scale y", value: "scaleY" }, + { text: "skew x", value: "skewX" }, + { text: "skew y", value: "skewY" }, + ] + } + } + }; + } + + allFonts() { + const customFonts = Scratch.vm.runtime.fontManager ? Scratch.vm.runtime.fontManager.getFonts().map((i) => ({ text: i.name, value: i.family })) : []; + return [...fontMenu, ...customFonts]; + } + + printTxt(args) { + const newTextElement = document.createElement("div"); + newTextElement.textContent = args.TXT; + newTextElement.id = `SP_Text-Ext-${args.ID}`; + newTextElement.classList.add(args.ID); + render.addOverlay(newTextElement, "scale-centered"); + allText.push(`#SP_Text-Ext-${args.ID}`); + + // add formatting (if any) + if (lastRecdVals["textCLR"]) this.setTextColor(lastRecdVals["textCLR"].inputs); + if (lastRecdVals["textOUT"]) this.setTextOutline(lastRecdVals["textOUT"].inputs); + if (lastRecdVals["textSHA"]) this.setTextDropShadow(lastRecdVals["textSHA"].inputs); + if (lastRecdVals["txtFont"]) this.setTextFont(lastRecdVals["txtFont"].inputs); + if (lastRecdVals["txtFontSZ"]) this.setFontSize(lastRecdVals["txtFontSZ"].inputs); + if (lastRecdVals["txtALI"]) this.setTextAlignment(lastRecdVals["txtALI"].inputs); + if (lastRecdVals["lineDIS"]) this.setTextSpacing(lastRecdVals["lineDIS"].inputs); + if (lastRecdVals["letDIS"]) this.setTextSpacing(lastRecdVals["letDIS"].inputs); + + const box = newTextElement.getBoundingClientRect(); + this.setMargins({ ID : args.ID, WIDTH : box.width / 2, HEIGHT : box.height }); + } + + replaceTxt(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { element.textContent = args.TXT }); + } + + removeTxt(args) { + const elementsToRemove = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elementsToRemove.forEach((element) => { + render.removeOverlay(element) + const index = allText.indexOf(`#SP_Text-Ext-${args.ID}`); + if (index !== -1) allText.splice(index, 1); + }); + } + + removeAllTxt() { + for (let i = 0; i < allText.length; i++) { + const elementsToRemove = document.querySelectorAll(allText[i]); + elementsToRemove.forEach((element) => { render.removeOverlay(element) }); + } + allText = []; + } + + displayedTexts(args) { + let texts = []; + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { texts.push(element.textContent) }); + return JSON.stringify(texts); + } + + allIDs() { + const cleanedIDs = allText.map((item) => item.replace(/^#SP_Text-Ext-/, "")); + return JSON.stringify(cleanedIDs); + } + + existingID(args) { + const index = allText.indexOf(`#SP_Text-Ext-${args.ID}`); + return Scratch.Cast.toBoolean(allText[index]); + } + + makeGradient(args) { return `${args.TYPE}-gradient(${args.ANGLE}deg, ${args.COLOR1}, ${args.COLOR2})` } + + setTextColor(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { + element.style.color = args.COLOR; + if (args.COLOR.includes("gradient")) { + element.style.background = args.COLOR; + element.style.color = "transparent"; + element.style.webkitTextFillColor = "transparent"; + element.style.webkitBackgroundClip = "text"; + } else { + element.style.color = args.COLOR; + element.style.background = "none"; + element.style.webkitTextFillColor = "initial"; + element.style.webkitBackgroundClip = "initial"; + } + }); + lastRecdVals["textCLR"] = {inputs: args}; + } + + setTextDropShadow(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { + element.style.textShadow = `${args.x}px ${args.y * -1}px ${args.z}px ${args.COLOR}`; + }); + lastRecdVals["textSHA"] = {inputs: args}; + } + + setTextOutline(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { + element.style.webkitTextStrokeColor = args.COLOR; + element.style.webkitTextStrokeWidth = `${args.THICKNESS}px`; + //multi-platform support cuz we cant have nice things + element.style.textStrokeColor = args.COLOR; + element.style.textStrokeWidth = `${args.THICKNESS}px`; + element.style.mozTextStrokeColor = args.COLOR; + element.style.mozTextStrokeWidth = `${args.THICKNESS}px`; + }); + lastRecdVals["textOUT"] = {inputs: args}; + } + + setMargins(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { + element.style.width = `${args.WIDTH}px`; + element.style.height = `${args.HEIGHT}px`; + }); + } + + setTextAlignment(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { element.style.textAlign = args.ALIGNMENT }); + lastRecdVals["txtALI"] = {inputs: args}; + } + + setTextFont(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { element.style.fontFamily = args.FONT }); + lastRecdVals["txtFont"] = {inputs: args}; + } + + setFontSize(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { element.style.fontSize = `${args.SIZE}px` }); + lastRecdVals["txtFontSZ"] = {inputs: args}; + } + + setTextSpacing(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { + element.style[args.ATT === "letter" ? "letterSpacing" : "lineHeight"] = `${args.SPACING}px`; + }); + lastRecdVals["letDIS"] = {inputs: {ID : args.ID, SPACING : args.SPACING, ATT : "letter"}}; + lastRecdVals["lineDIS"] = {inputs: {ID : args.ID, SPACING : args.SPACING, ATT : "line"}}; + } + + setTextPosition(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { + const box = element.getBoundingClientRect(); + element.style.position = "absolute"; + element.style.left = `${args.X - (box.width / 2)}px`; + element.style.top = `${(args.Y * -1) - (box.height / 2)}px`; + }); + } + + textPosition(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + let value; + elements.forEach((element) => { + const box = element.getBoundingClientRect(); + if (args.ATT === "z layer") { + value = element.parentNode.style.zIndex; + } else { + value = parseFloat(element.style[args.ATT.includes("x") ? "left" : "top"]); + value = ((box[args.ATT.includes("x") ? "width" : "height"] / 2) + value) * (args.ATT.includes("x") ? 1 : -1); + } + }); + return value || 0; + } + + attOfText(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + let value; + elements.forEach((element) => { value = element.style[args.ATT] }); + value = args.ATT === "fontFamily" || args.ATT === "textAlign" ? value : parseFloat(value); + return value || ""; + } + + setTextZIndex(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { element.parentNode.style.zIndex = Math.round(args.Z_INDEX) }); + } + + setEffect(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { + const currentStyleArray = [ + element.style.filter || "", + element.parentNode.style.transform || "", + element.style.opacity || "" + ]; + const regex = new RegExp(`${args.EFFECT}\\([^)]+\\)`, "g"); + currentStyleArray.forEach((currentStyle, index) => { + currentStyleArray[index] = currentStyle.replace(regex, ""); + }); + switch (args.EFFECT) { + case "saturate": + element.style.filter = `${currentStyleArray[0]} saturate(${args.VALUE}%)`; + break; + case "contrast": + element.style.filter = `${currentStyleArray[0]} contrast(${args.VALUE + 100}%)`; + break; + case "brightness": + element.style.filter = `${currentStyleArray[0]} brightness(${args.VALUE + 100}%)`; + break; + case "hue-rotate": + element.style.filter = `${currentStyleArray[0]} hue-rotate(${args.VALUE}deg)`; + break; + case "opacity": + element.style.opacity = (100 - args.VALUE) / 100; + break; + case "sepia": + element.style.filter = `${currentStyleArray[0]} sepia(${args.VALUE}%)`; + break; + case "invert": + element.style.filter = `${currentStyleArray[0]} invert(${args.VALUE}%)`; + break; + case "rotate": + element.parentNode.style.transform = `${currentStyleArray[1]} rotate(${args.VALUE - 90}deg)`; + break; + case "scaleX": + element.parentNode.style.transform = `${currentStyleArray[1]} scaleX(${args.VALUE / 100})`; + break; + case "scaleY": + element.parentNode.style.transform = `${currentStyleArray[1]} scaleY(${args.VALUE / 100})`; + break; + case "skewX": + element.parentNode.style.transform = `${currentStyleArray[1]} skewX(${args.VALUE}deg)`; + break; + case "skewY": + element.parentNode.style.transform = `${currentStyleArray[1]} skewY(${args.VALUE}deg)`; + break; + default: + element.style.filter = `${currentStyleArray[0]} blur(${args.VALUE}px)`; + break; + } + }); + } + + resetEffect(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { + const currentTransform = element.parentNode.style.transform || ""; + const updatedTransform = currentTransform + .replace(/rotate\([^)]*\)/, "rotate(0deg)") + .replace(/scaleX\([^)]*\)/, "scaleX(1)") + .replace(/scaleY\([^)]*\)/, "scaleY(1)") + .replace(/skewX\([^)]*\)/, "skewX(0deg)") + .replace(/skewY\([^)]*\)/, "skewY(0deg)"); + element.parentNode.style.transform = updatedTransform; + element.style.filter = ""; + element.style.opacity = 1; + }); + } + + amtOfEffect(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + let effectValue = args.EFFECT === "rotate" ? 90 : args.EFFECT.includes("scale") ? 100 : 0; + elements.forEach((element) => { + if (args.EFFECT === "rotate" || args.EFFECT.includes("scale") || args.EFFECT.includes("skew")) { + const transformV = element.parentNode.style.transform.split(" "); + const effectIndex = transformV.findIndex(value => value.includes(args.EFFECT)); + if (effectIndex !== -1) { + effectValue = transformV[effectIndex].replace(/[^\d.-]/g, ""); + effectValue = args.EFFECT === "rotate" ? parseFloat(effectValue) + 90 : + parseFloat(effectValue) * (args.EFFECT.includes("scale") ? 100 : 1); + } + } else if (args.EFFECT === "opacity") { + effectValue = 100 - (parseFloat(element.style.opacity || 0) * 100); + } else { + const filterV = element.style.filter.split(" "); + const effectIndex = filterV.findIndex((value) => value.includes(args.EFFECT)); + if (effectIndex !== -1) { + effectValue = + args.EFFECT === "contrast" || args.EFFECT === "brightness" + ? parseFloat(filterV[effectIndex].replace(/[^\d.-]/g, "")) - 100 + : parseFloat(filterV[effectIndex].replace(/[^\d.-]/g, "")); + } + } + }); + return effectValue; + } + + setTextCurve(args) { + const regex = args.ARC.includes("]*d="([^"]*)"/ : /]*d="([^"]*)"/; + const match = args.ARC.match(regex); + const outline = lastRecdVals["textOUT"] !== undefined ? lastRecdVals["textOUT"] : ""; + if (match && match[1]) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { + const existingSvg = element.querySelector("svg"); + if (existingSvg) { + const path = existingSvg.querySelector("path"); + path.setAttribute("d", match[1]); + const textFill = existingSvg.querySelector("text"); + textFill.setAttribute("fill", element.style.color === "transparent" ? "#000000" : element.style.color); + const textPathFill = existingSvg.querySelector("textPath"); + textPathFill.setAttribute("href", `#MyPath-${args.ID}`); + textPathFill.textContent = element.textContent; + + const textStroke = existingSvg.querySelector("text"); + textStroke.setAttribute("fill", outline ? outline.inputs.COLOR ?? "#00000000" : "#00000000"); + textStroke.setAttribute("stroke", outline ? outline.inputs.COLOR ?? "#00000000" : "#00000000"); + textStroke.setAttribute("stroke-width", outline ? outline.inputs.THICKNESS ?? 1 : 1); + const textPathStroke = existingSvg.querySelector("textPath"); + textPathStroke.setAttribute("href", `#MyPath-${args.ID}`); + textPathStroke.textContent = element.textContent; + } else { + const svg = document.createElementNS("http://www.w3.org/2000/svg", "svg"); + const defs = document.createElementNS("http://www.w3.org/2000/svg", "defs"); + const path = document.createElementNS("http://www.w3.org/2000/svg", "path"); + path.setAttribute("id", `MyPath-${args.ID}`); + path.setAttribute("d", match[1]); + defs.appendChild(path); + + const textStroke = document.createElementNS("http://www.w3.org/2000/svg", "text"); + textStroke.setAttribute("fill", outline ? outline.inputs.COLOR : "#00000000"); + textStroke.setAttribute("stroke", outline ? outline.inputs.COLOR : "#00000000"); + textStroke.setAttribute("stroke-width", outline ? outline.inputs.THICKNESS : 1); + const textPathStroke = document.createElementNS("http://www.w3.org/2000/svg", "textPath"); + textPathStroke.setAttribute("href", `#MyPath-${args.ID}`); + textPathStroke.textContent = element.textContent; + textStroke.appendChild(textPathStroke); + + svg.appendChild(defs); + svg.appendChild(textStroke); + const textFill = document.createElementNS("http://www.w3.org/2000/svg", "text"); + textFill.setAttribute("fill", element.style.color === "transparent" ? "#000000" : element.style.color); + const textPathFill = document.createElementNS("http://www.w3.org/2000/svg", "textPath"); + textPathFill.setAttribute("href", `#MyPath-${args.ID}`); + textPathFill.textContent = element.textContent; + textFill.appendChild(textPathFill); + svg.appendChild(textFill); + element.innerHTML = ""; + element.appendChild(svg); + } + }); + } + } + + presetCurve(args) { + const presets = { + circle : ``, + wave : `` + }; + return presets[args.ARC]; + } + } + + Scratch.extensions.register(new SPdisText()); +})(Scratch); From fa82f66148ffc4c6eba3455354702ded2dc26289 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Wed, 13 Dec 2023 20:20:15 -0800 Subject: [PATCH 02/25] Create Display-Text.svg --- images/SharkPool/Display-Text.svg | 1 + 1 file changed, 1 insertion(+) create mode 100644 images/SharkPool/Display-Text.svg diff --git a/images/SharkPool/Display-Text.svg b/images/SharkPool/Display-Text.svg new file mode 100644 index 0000000000..8fc4cbca5e --- /dev/null +++ b/images/SharkPool/Display-Text.svg @@ -0,0 +1 @@ + From dc11c36fcbbba524a3581f96400179acdf36351a Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Fri, 15 Dec 2023 14:45:25 -0800 Subject: [PATCH 03/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 58 +++++++++++++++++++++++++--- 1 file changed, 52 insertions(+), 6 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index f8cb552284..ea79a75773 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.0.0 +// Version V.1.0.1 (function (Scratch) { "use strict"; @@ -141,6 +141,15 @@ ATT: { type: Scratch.ArgumentType.STRING, menu: "TEXT_ATT" } }, }, + { + opcode: "setOverflow", + blockType: Scratch.BlockType.COMMAND, + text: "set text overflow of ID [ID] to [TYPE]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + TYPE: { type: Scratch.ArgumentType.STRING, menu: "OVERFLOW" } + }, + }, "---", { opcode: "attOfText", @@ -192,7 +201,7 @@ COLOR1: { type: Scratch.ArgumentType.COLOR, defaultValue: "#ff0000" }, COLOR2: { type: Scratch.ArgumentType.COLOR, defaultValue: "#00ff00" }, ANGLE: { type: Scratch.ArgumentType.ANGLE, defaultValue: 90 }, - TYPE: { type: Scratch.ArgumentType.NUMBER, menu: "GRADIENTS" } + TYPE: { type: Scratch.ArgumentType.STRING, menu: "GRADIENTS" } }, }, "---", @@ -215,6 +224,16 @@ }, }, { blockType: Scratch.BlockType.LABEL, text: "Positioning" }, + { + opcode: "presetTextPosition", + blockType: Scratch.BlockType.COMMAND, + text: "preset position of ID [ID] to x: [X] y: [Y]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + X: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }, + Y: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 } + }, + }, { opcode: "setTextPosition", blockType: Scratch.BlockType.COMMAND, @@ -240,7 +259,7 @@ text: "[ATT] of ID [ID]", arguments: { ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, - ATT: { type: Scratch.ArgumentType.NUMBER, menu: "POS" } + ATT: { type: Scratch.ArgumentType.STRING, menu: "POS" } }, }, { blockType: Scratch.BlockType.LABEL, text: "Effects" }, @@ -287,6 +306,7 @@ items: ["linear", "radial"] }, POS: ["x position", "y position", "z layer"], + OVERFLOW: ["visible", "hidden"], ARCS: ["circle", "hill", "dip", "wave"], FORMATS: { acceptReporters: true, @@ -297,7 +317,8 @@ { text: "margin width", value: "width" }, { text: "margin height", value: "height" }, { text: "letter spacing", value: "letterSpacing" }, - { text: "text spacing", value: "lineHeight" } + { text: "text spacing", value: "lineHeight" }, + { text: "overflow type", value: "overflow" } ] }, EFFECTS: { @@ -344,6 +365,8 @@ if (lastRecdVals["txtALI"]) this.setTextAlignment(lastRecdVals["txtALI"].inputs); if (lastRecdVals["lineDIS"]) this.setTextSpacing(lastRecdVals["lineDIS"].inputs); if (lastRecdVals["letDIS"]) this.setTextSpacing(lastRecdVals["letDIS"].inputs); + if (lastRecdVals["textOVR"]) this.setOverflow(lastRecdVals["textOVR"].inputs); + if (lastRecdVals["preTxt1"]) this.presetTextPosition(lastRecdVals["preTxt1"].inputs,); const box = newTextElement.getBoundingClientRect(); this.setMargins({ ID : args.ID, WIDTH : box.width / 2, HEIGHT : box.height }); @@ -351,7 +374,11 @@ replaceTxt(args) { const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); - elements.forEach((element) => { element.textContent = args.TXT }); + if (elements.length > 0) { + elements.forEach((element) => { element.textContent = args.TXT }); + } else { + this.printTxt(args); + } } removeTxt(args) { @@ -417,6 +444,12 @@ lastRecdVals["textSHA"] = {inputs: args}; } + setOverflow(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { element.style.overflow = args.TYPE }); + lastRecdVals["textOVR"] = {inputs: args}; + } + setTextOutline(args) { const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); elements.forEach((element) => { @@ -466,6 +499,19 @@ lastRecdVals["lineDIS"] = {inputs: {ID : args.ID, SPACING : args.SPACING, ATT : "line"}}; } + presetTextPosition(args) { + if (args.isPrint === undefined) { + lastRecdVals["preTxt1"] = {inputs: {...args, isPrint: true}}; + } else { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const element = elements[elements.length - 1]; + const box = element.getBoundingClientRect(); + element.style.position = "absolute"; + element.style.left = `${(args.X - (box.width / 2)) + (parseFloat(Scratch.vm.renderer.canvas.style.width) / 2)}px`; + element.style.top = `${(args.Y * -1) - (box.height / 2)}px`; + } + } + setTextPosition(args) { const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); elements.forEach((element) => { @@ -495,7 +541,7 @@ const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); let value; elements.forEach((element) => { value = element.style[args.ATT] }); - value = args.ATT === "fontFamily" || args.ATT === "textAlign" ? value : parseFloat(value); + value = args.ATT === "fontFamily" || args.ATT === "textAlign" || args.ATT === "overflow" ? value : parseFloat(value); return value || ""; } From 47d1c2e9db7c3a8a34eef05a13c44f20103a9f52 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Mon, 18 Dec 2023 19:01:05 -0800 Subject: [PATCH 04/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 30 +++++++++++++++++++++++----- 1 file changed, 25 insertions(+), 5 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index ea79a75773..217f69a22b 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.0.1 +// Version V.1.1.0 (function (Scratch) { "use strict"; @@ -160,6 +160,14 @@ ATT: { type: Scratch.ArgumentType.STRING, menu: "FORMATS" } }, }, + { + opcode: "lineCnt", + blockType: Scratch.BlockType.REPORTER, + text: "# of lines in text ID [ID]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } + }, + }, { blockType: Scratch.BlockType.LABEL, text: "Text Visuals" }, { opcode: "setTextColor", @@ -317,7 +325,7 @@ { text: "margin width", value: "width" }, { text: "margin height", value: "height" }, { text: "letter spacing", value: "letterSpacing" }, - { text: "text spacing", value: "lineHeight" }, + { text: "line spacing", value: "lineHeight" }, { text: "overflow type", value: "overflow" } ] }, @@ -349,8 +357,9 @@ } printTxt(args) { + args.ID = args.ID.replaceAll(" ", "_"); const newTextElement = document.createElement("div"); - newTextElement.textContent = args.TXT; + newTextElement.innerHTML = args.TXT.replace(/\n/g, "
"); newTextElement.id = `SP_Text-Ext-${args.ID}`; newTextElement.classList.add(args.ID); render.addOverlay(newTextElement, "scale-centered"); @@ -375,7 +384,7 @@ replaceTxt(args) { const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); if (elements.length > 0) { - elements.forEach((element) => { element.textContent = args.TXT }); + elements.forEach((element) => { element.innerHTML = args.TXT.replace(/\n/g, "
"); }); } else { this.printTxt(args); } @@ -439,7 +448,7 @@ setTextDropShadow(args) { const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); elements.forEach((element) => { - element.style.textShadow = `${args.x}px ${args.y * -1}px ${args.z}px ${args.COLOR}`; + element.style.textShadow = args.z === 0 ? "none" : `${args.x}px ${args.y * -1}px ${args.z}px ${args.COLOR}`; }); lastRecdVals["textSHA"] = {inputs: args}; } @@ -544,6 +553,17 @@ value = args.ATT === "fontFamily" || args.ATT === "textAlign" || args.ATT === "overflow" ? value : parseFloat(value); return value || ""; } + + lineCnt(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + let value = []; + elements.forEach((element) => { + value.push(element.querySelectorAll("br").length + 1 || 1); + }); + console.log(value); + value = value.length > 1 ? JSON.stringify(value) : (value[0] || 0); + return value; + } setTextZIndex(args) { const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); From 840b40221ce6849f1de3eef09663a0133d4fc27f Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Mon, 18 Dec 2023 19:04:10 -0800 Subject: [PATCH 05/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 1 - 1 file changed, 1 deletion(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 217f69a22b..d4b2092d4f 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -560,7 +560,6 @@ elements.forEach((element) => { value.push(element.querySelectorAll("br").length + 1 || 1); }); - console.log(value); value = value.length > 1 ? JSON.stringify(value) : (value[0] || 0); return value; } From d10594271a3a6b44d310cd19d52e4764c3e8d9b0 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Mon, 18 Dec 2023 19:19:21 -0800 Subject: [PATCH 06/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 17 +++++++++++++++-- 1 file changed, 15 insertions(+), 2 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index d4b2092d4f..b4a980e7b4 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -23,6 +23,19 @@ "Marker", "Curly", "Pixel" ]; + const xmlEscape = function (unsafe) { + unsafe = String(unsafe); + return unsafe.replace(/[<>&'"]/g, c => { + switch (c) { + case "<": return "<"; + case ">": return ">"; + case "&": return "&"; + case "'": return "'"; + case "\"": return """; + } + }); + }; + class SPdisText { getInfo() { return { @@ -359,7 +372,7 @@ printTxt(args) { args.ID = args.ID.replaceAll(" ", "_"); const newTextElement = document.createElement("div"); - newTextElement.innerHTML = args.TXT.replace(/\n/g, "
"); + newTextElement.innerHTML = xmlEscape(args.TXT).replace(/\n/g, "
"); newTextElement.id = `SP_Text-Ext-${args.ID}`; newTextElement.classList.add(args.ID); render.addOverlay(newTextElement, "scale-centered"); @@ -384,7 +397,7 @@ replaceTxt(args) { const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); if (elements.length > 0) { - elements.forEach((element) => { element.innerHTML = args.TXT.replace(/\n/g, "
"); }); + elements.forEach((element) => { element.innerHTML = xmlEscape(args.TXT).replace(/\n/g, "
") }); } else { this.printTxt(args); } From fe5f45e33e9c4a87aff91beb01da3e22b7cfcf56 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Tue, 19 Dec 2023 20:34:03 -0800 Subject: [PATCH 07/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index b4a980e7b4..09201eba08 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.1.0 +// Version V.1.1.1 (function (Scratch) { "use strict"; @@ -37,6 +37,10 @@ }; class SPdisText { + constructor() { + Scratch.vm.runtime.on("PROJECT_START", () => { this.removeAllTxt() }); + Scratch.vm.runtime.on("PROJECT_STOP_ALL", () => { this.removeAllTxt() }); + } getInfo() { return { id: "SPdisText", From de3c3dd1aaa9ca498aa240f2d12cd4a109ce75cb Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Wed, 27 Dec 2023 19:48:10 -0800 Subject: [PATCH 08/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 09201eba08..e349180aaf 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.1.1 +// Version V.1.1.2 (function (Scratch) { "use strict"; @@ -373,6 +373,11 @@ return [...fontMenu, ...customFonts]; } + isScreen() { + const values = [parseFloat(Scratch.vm.renderer.canvas.style.width), Scratch.vm.runtime.stageWidth]; + return values[0] > values[1] ? [3.78, 7] : [2, 2]; + } + printTxt(args) { args.ID = args.ID.replaceAll(" ", "_"); const newTextElement = document.createElement("div"); @@ -543,8 +548,8 @@ elements.forEach((element) => { const box = element.getBoundingClientRect(); element.style.position = "absolute"; - element.style.left = `${args.X - (box.width / 2)}px`; - element.style.top = `${(args.Y * -1) - (box.height / 2)}px`; + element.style.left = `${args.X - (box.width / this.isScreen()[0])}px`; + element.style.top = `${(args.Y * -1) - (box.height / this.isScreen()[1])}px`; }); } From 28c663b93e26482a3e87675306fbd556d09b4f84 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Fri, 5 Jan 2024 14:19:54 -0800 Subject: [PATCH 09/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 65 ++++++++++++++++++++++------ 1 file changed, 52 insertions(+), 13 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index e349180aaf..20d246ecaa 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.1.2 +// Version V.1.2.0 (function (Scratch) { "use strict"; @@ -14,6 +14,7 @@ const blockIconURI = ""; + const render = Scratch.vm.renderer; let allText = []; let lastRecdVals = {}; @@ -129,6 +130,15 @@ SIZE: { type: Scratch.ArgumentType.NUMBER, defaultValue: 30 } }, }, + { + opcode: "setThick", + blockType: Scratch.BlockType.COMMAND, + text: "set boldness of ID [ID] to [NUM]", + arguments: { + NUM : { type: Scratch.ArgumentType.NUMBER, defaultValue: 100 }, + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } + } + }, { opcode: "setTextAlignment", blockType: Scratch.BlockType.COMMAND, @@ -333,6 +343,11 @@ POS: ["x position", "y position", "z layer"], OVERFLOW: ["visible", "hidden"], ARCS: ["circle", "hill", "dip", "wave"], + THICK: [ + { text : "thick", value : "900" }, + { text : "medium", value : "600" }, + { text : "none", value : "1" }, + ], FORMATS: { acceptReporters: true, items: [ @@ -375,7 +390,7 @@ isScreen() { const values = [parseFloat(Scratch.vm.renderer.canvas.style.width), Scratch.vm.runtime.stageWidth]; - return values[0] > values[1] ? [3.78, 7] : [2, 2]; + return values[0] > values[1] || Scratch.vm.runtime.isPackaged; } printTxt(args) { @@ -397,8 +412,9 @@ if (lastRecdVals["lineDIS"]) this.setTextSpacing(lastRecdVals["lineDIS"].inputs); if (lastRecdVals["letDIS"]) this.setTextSpacing(lastRecdVals["letDIS"].inputs); if (lastRecdVals["textOVR"]) this.setOverflow(lastRecdVals["textOVR"].inputs); - if (lastRecdVals["preTxt1"]) this.presetTextPosition(lastRecdVals["preTxt1"].inputs,); + if (lastRecdVals["txtFontTK"]) this.setThick(lastRecdVals["txtFontTK"].inputs); + if (lastRecdVals["preTxt1"]) this.presetTextPosition(lastRecdVals["preTxt1"].inputs); const box = newTextElement.getBoundingClientRect(); this.setMargins({ ID : args.ID, WIDTH : box.width / 2, HEIGHT : box.height }); } @@ -521,6 +537,12 @@ lastRecdVals["txtFontSZ"] = {inputs: args}; } + setThick(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { element.style.fontWeight = args.NUM * 9 }); + lastRecdVals["txtFontTK"] = {inputs: args}; + } + setTextSpacing(args) { const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); elements.forEach((element) => { @@ -534,12 +556,19 @@ if (args.isPrint === undefined) { lastRecdVals["preTxt1"] = {inputs: {...args, isPrint: true}}; } else { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); - const element = elements[elements.length - 1]; - const box = element.getBoundingClientRect(); - element.style.position = "absolute"; - element.style.left = `${(args.X - (box.width / 2)) + (parseFloat(Scratch.vm.renderer.canvas.style.width) / 2)}px`; - element.style.top = `${(args.Y * -1) - (box.height / 2)}px`; + setTimeout(function() { + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const element = elements[elements.length - 1]; + const box = element.getBoundingClientRect(); + const transformValue = element.parentNode.style.transform; + const match = transformValue.match(/scale\(([^,]+),\s*([^)]+)\)/); + const scales = [parseFloat(match[1]), parseFloat(match[2])]; + const centerW = this.isScreen() ? box.width / 2 / scales[0] : box.width / 2; + const centerH = this.isScreen() ? box.height / 2 / (scales[1] * scales[1]): box.height / 2; + element.style.position = "absolute"; + element.style.left = `${args.X - centerW}px`; + element.style.top = `${(args.Y * -1) - centerH}px`; + }.bind(this), 0); } } @@ -547,9 +576,14 @@ const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); elements.forEach((element) => { const box = element.getBoundingClientRect(); + const transformValue = element.parentNode.style.transform; + const match = transformValue.match(/scale\(([^,]+),\s*([^)]+)\)/); + const scales = [parseFloat(match[1]), parseFloat(match[2])]; + const centerW = this.isScreen() ? box.width / 2 / scales[0] : box.width / 2; + const centerH = this.isScreen() ? box.height / 2 / (scales[1] * scales[1]): box.height / 2; element.style.position = "absolute"; - element.style.left = `${args.X - (box.width / this.isScreen()[0])}px`; - element.style.top = `${(args.Y * -1) - (box.height / this.isScreen()[1])}px`; + element.style.left = `${args.X - centerW}px`; + element.style.top = `${(args.Y * -1) - centerH}px`; }); } @@ -558,14 +592,19 @@ let value; elements.forEach((element) => { const box = element.getBoundingClientRect(); + const transformValue = element.parentNode.style.transform; + const match = transformValue.match(/scale\(([^,]+),\s*([^)]+)\)/); + const scales = [parseFloat(match[1]), parseFloat(match[2])]; + const centerW = this.isScreen() ? box.width / 2 / scales[0] : box.width / 2; + const centerH = this.isScreen() ? box.height / 2 / (scales[1] * scales[1]): box.height / 2; if (args.ATT === "z layer") { value = element.parentNode.style.zIndex; } else { value = parseFloat(element.style[args.ATT.includes("x") ? "left" : "top"]); - value = ((box[args.ATT.includes("x") ? "width" : "height"] / 2) + value) * (args.ATT.includes("x") ? 1 : -1); + value = ((args.ATT.includes("x") ? centerW : centerH) + value) * (args.ATT.includes("x") ? 1 : -1); } }); - return value || 0; + return Math.round(value * 100) / 100 || 0; } attOfText(args) { From b8e2349a4047ea2a885b084dd1aeab0559a98448 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Fri, 5 Jan 2024 14:25:08 -0800 Subject: [PATCH 10/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 2 ++ 1 file changed, 2 insertions(+) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 20d246ecaa..e02dc6e519 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -353,6 +353,7 @@ items: [ { text: "font", value: "fontFamily" }, { text: "font size", value: "fontSize" }, + { text: "boldness", value: "fontWeight" }, { text: "alignment", value: "textAlign" }, { text: "margin width", value: "width" }, { text: "margin height", value: "height" }, @@ -612,6 +613,7 @@ let value; elements.forEach((element) => { value = element.style[args.ATT] }); value = args.ATT === "fontFamily" || args.ATT === "textAlign" || args.ATT === "overflow" ? value : parseFloat(value); + value = args.ATT === "fontWeight" ? value / 9 : value return value || ""; } From 8c1023996f9bba1fc8ba9596e1ed7a955ced7292 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Sat, 6 Jan 2024 16:29:46 -0800 Subject: [PATCH 11/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 40 +++++++++++----------------- 1 file changed, 16 insertions(+), 24 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index e02dc6e519..1c079e179e 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.2.0 +// Version V.1.2.1 (function (Scratch) { "use strict"; @@ -389,11 +389,6 @@ return [...fontMenu, ...customFonts]; } - isScreen() { - const values = [parseFloat(Scratch.vm.renderer.canvas.style.width), Scratch.vm.runtime.stageWidth]; - return values[0] > values[1] || Scratch.vm.runtime.isPackaged; - } - printTxt(args) { args.ID = args.ID.replaceAll(" ", "_"); const newTextElement = document.createElement("div"); @@ -560,12 +555,11 @@ setTimeout(function() { const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); const element = elements[elements.length - 1]; - const box = element.getBoundingClientRect(); - const transformValue = element.parentNode.style.transform; - const match = transformValue.match(/scale\(([^,]+),\s*([^)]+)\)/); - const scales = [parseFloat(match[1]), parseFloat(match[2])]; - const centerW = this.isScreen() ? box.width / 2 / scales[0] : box.width / 2; - const centerH = this.isScreen() ? box.height / 2 / (scales[1] * scales[1]): box.height / 2; + const centerW = parseFloat(element.style.width) / 2; + const computedStyle = window.getComputedStyle(element); + const lineHeight = computedStyle.getPropertyValue("line-height"); + const fontSize = computedStyle.getPropertyValue("font-size"); + const centerH = (lineHeight === "normal" ? parseFloat(fontSize) * 1.2 : parseFloat(lineHeight)) / 2; element.style.position = "absolute"; element.style.left = `${args.X - centerW}px`; element.style.top = `${(args.Y * -1) - centerH}px`; @@ -576,12 +570,11 @@ setTextPosition(args) { const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); elements.forEach((element) => { - const box = element.getBoundingClientRect(); - const transformValue = element.parentNode.style.transform; - const match = transformValue.match(/scale\(([^,]+),\s*([^)]+)\)/); - const scales = [parseFloat(match[1]), parseFloat(match[2])]; - const centerW = this.isScreen() ? box.width / 2 / scales[0] : box.width / 2; - const centerH = this.isScreen() ? box.height / 2 / (scales[1] * scales[1]): box.height / 2; + const centerW = parseFloat(element.style.width) / 2; + const computedStyle = window.getComputedStyle(element); + const lineHeight = computedStyle.getPropertyValue("line-height"); + const fontSize = computedStyle.getPropertyValue("font-size"); + const centerH = (lineHeight === "normal" ? parseFloat(fontSize) * 1.2 : parseFloat(lineHeight)) / 2; element.style.position = "absolute"; element.style.left = `${args.X - centerW}px`; element.style.top = `${(args.Y * -1) - centerH}px`; @@ -592,12 +585,11 @@ const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); let value; elements.forEach((element) => { - const box = element.getBoundingClientRect(); - const transformValue = element.parentNode.style.transform; - const match = transformValue.match(/scale\(([^,]+),\s*([^)]+)\)/); - const scales = [parseFloat(match[1]), parseFloat(match[2])]; - const centerW = this.isScreen() ? box.width / 2 / scales[0] : box.width / 2; - const centerH = this.isScreen() ? box.height / 2 / (scales[1] * scales[1]): box.height / 2; + const centerW = parseFloat(element.style.width) / 2; + const computedStyle = window.getComputedStyle(element); + const lineHeight = computedStyle.getPropertyValue("line-height"); + const fontSize = computedStyle.getPropertyValue("font-size"); + const centerH = (lineHeight === "normal" ? parseFloat(fontSize) * 1.2 : parseFloat(lineHeight)) / 2; if (args.ATT === "z layer") { value = element.parentNode.style.zIndex; } else { From 6205857a8a49cfabc5d9af23193b8e11f5add774 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Sun, 14 Jan 2024 20:01:10 -0800 Subject: [PATCH 12/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 58 +++++++++++++++++++++++++++- 1 file changed, 56 insertions(+), 2 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 1c079e179e..de2323d958 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.2.1 +// Version V.1.3.0 (function (Scratch) { "use strict"; @@ -111,6 +111,15 @@ blockType: Scratch.BlockType.REPORTER, text: "all text IDs" }, + "---", + { + opcode: "debug", + blockType: Scratch.BlockType.COMMAND, + text: "toggle debug mode [TOGGLE]", + arguments: { + TOGGLE: { type: Scratch.ArgumentType.STRING, menu: "TOGGLE" } + }, + }, { blockType: Scratch.BlockType.LABEL, text: "Formatting" }, { opcode: "setTextFont", @@ -227,6 +236,18 @@ THICKNESS: { type: Scratch.ArgumentType.NUMBER, defaultValue: 3 } }, }, + { + opcode: "setLine", + blockType: Scratch.BlockType.COMMAND, + text: "set text line of ID [ID] to [TYPE1] [TYPE2] colored [COLOR] thickness [THICK]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + TYPE1: { type: Scratch.ArgumentType.STRING, menu: "STYLE" }, + TYPE2: { type: Scratch.ArgumentType.STRING, menu: "LINE_TYPE" }, + COLOR: { type: Scratch.ArgumentType.COLOR, defaultValue: "#ff0000" }, + THICK: { type: Scratch.ArgumentType.NUMBER, defaultValue: 2 } + }, + }, "---", { opcode: "makeGradient", @@ -327,10 +348,19 @@ }, ], menus: { + TOGGLE: ["on", "off"], FONTS: { acceptReporters: true, items: "allFonts" }, + STYLE: { + acceptReporters: true, + items: ["solid", "wavy", "dashed", "double"] + }, + LINE_TYPE: { + acceptReporters: true, + items: ["underline", "strikethrough"] + }, ALIGNMENTS: { acceptReporters: true, items: ["left", "right", "center"] @@ -389,6 +419,17 @@ return [...fontMenu, ...customFonts]; } + debug(args) { + const toggle = args.TOGGLE === "on" ? "solid" : "none"; + const elements = document.querySelectorAll(`[id^="SP_Text-Ext-"]`); + elements.forEach((element) => { + element.style.border = toggle; + element.style.borderWidth = "1px"; + const color = element.style.textAlign === "center" ? "#00ff00" : element.style.textAlign === "right" ? "#0000ff" : "#ff0000"; + element.style.borderColor = color; + }); + } + printTxt(args) { args.ID = args.ID.replaceAll(" ", "_"); const newTextElement = document.createElement("div"); @@ -409,6 +450,7 @@ if (lastRecdVals["letDIS"]) this.setTextSpacing(lastRecdVals["letDIS"].inputs); if (lastRecdVals["textOVR"]) this.setOverflow(lastRecdVals["textOVR"].inputs); if (lastRecdVals["txtFontTK"]) this.setThick(lastRecdVals["txtFontTK"].inputs); + if (lastRecdVals["textLIN"]) this.setLine(lastRecdVals["textLIN"].inputs); if (lastRecdVals["preTxt1"]) this.presetTextPosition(lastRecdVals["preTxt1"].inputs); const box = newTextElement.getBoundingClientRect(); @@ -427,7 +469,7 @@ removeTxt(args) { const elementsToRemove = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); elementsToRemove.forEach((element) => { - render.removeOverlay(element) + render.removeOverlay(element); const index = allText.indexOf(`#SP_Text-Ext-${args.ID}`); if (index !== -1) allText.splice(index, 1); }); @@ -507,6 +549,18 @@ lastRecdVals["textOUT"] = {inputs: args}; } + setLine(args) { + const lineType = args.TYPE2.replace("strike", "line-"); + const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + elements.forEach((element) => { + element.style.textDecorationLine = lineType; + element.style.textDecorationStyle = args.TYPE1; + element.style.textDecorationThickness = `${args.THICK}px`; + element.style.textDecorationColor = args.COLOR; + }); + lastRecdVals["textLIN"] = {inputs: args}; + } + setMargins(args) { const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); elements.forEach((element) => { From 41b9f7b811d05ef3d75aa937fa9a66ea68abe025 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Wed, 17 Jan 2024 20:26:35 -0800 Subject: [PATCH 13/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 142 ++++++++++++++++----------- 1 file changed, 84 insertions(+), 58 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index de2323d958..024daf5974 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.3.0 +// Version V.1.3.1 (function (Scratch) { "use strict"; @@ -101,7 +101,7 @@ { opcode: "existingID", blockType: Scratch.BlockType.BOOLEAN, - text: "does text with ID [ID] exist?", + text: "text with ID [ID] exist?", arguments: { ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } }, @@ -260,7 +260,7 @@ TYPE: { type: Scratch.ArgumentType.STRING, menu: "GRADIENTS" } }, }, - "---", + { blockType: Scratch.BlockType.LABEL, text: "↓ Will Lose Some Formatting ↓" }, { opcode: "setTextCurve", blockType: Scratch.BlockType.COMMAND, @@ -367,7 +367,7 @@ }, TEXT_ATT: ["letter", "line"], GRADIENTS: { - acceptReporters: true, // conic gradients exist but are complex + acceptReporters: true, items: ["linear", "radial"] }, POS: ["x position", "y position", "z layer"], @@ -419,6 +419,11 @@ return [...fontMenu, ...customFonts]; } + fixID(ID) { + return xmlEscape(Scratch.Cast.toString(ID).replaceAll(" ", "_") + .replaceAll(/[#%(),.{}[\]$@^*&'";:]/g, "-")); + } + debug(args) { const toggle = args.TOGGLE === "on" ? "solid" : "none"; const elements = document.querySelectorAll(`[id^="SP_Text-Ext-"]`); @@ -428,37 +433,55 @@ const color = element.style.textAlign === "center" ? "#00ff00" : element.style.textAlign === "right" ? "#0000ff" : "#ff0000"; element.style.borderColor = color; }); + const cross = document.getElementById(`SP_Text-Debug-Cross`); + if (toggle === "none" && cross) { + cross.parentNode.removeChild(cross); + } else if (toggle === "solid" && !cross) { + const imageUrl = + ""; + const crossElement = document.createElement("img"); + crossElement.src = imageUrl; + crossElement.id = "SP_Text-Debug-Cross"; + crossElement.style.position = "absolute"; + crossElement.style.width = "480px"; + crossElement.style.height = "360px"; + crossElement.style.transform = "translate(-50%, -50%)"; + crossElement.style.zIndex = "0"; + render.addOverlay(crossElement, "scale-centered"); + allText.push(`#SP_Text-Debug-Cross`); + } } printTxt(args) { - args.ID = args.ID.replaceAll(" ", "_"); + args.ID = this.fixID(args.ID); const newTextElement = document.createElement("div"); newTextElement.innerHTML = xmlEscape(args.TXT).replace(/\n/g, "
"); newTextElement.id = `SP_Text-Ext-${args.ID}`; newTextElement.classList.add(args.ID); render.addOverlay(newTextElement, "scale-centered"); allText.push(`#SP_Text-Ext-${args.ID}`); + const box = newTextElement.getBoundingClientRect(); + if (!lastRecdVals.textMAR) this.setMargins({ ID : args.ID, WIDTH : box.width / 2, HEIGHT : box.height }); // add formatting (if any) - if (lastRecdVals["textCLR"]) this.setTextColor(lastRecdVals["textCLR"].inputs); - if (lastRecdVals["textOUT"]) this.setTextOutline(lastRecdVals["textOUT"].inputs); - if (lastRecdVals["textSHA"]) this.setTextDropShadow(lastRecdVals["textSHA"].inputs); - if (lastRecdVals["txtFont"]) this.setTextFont(lastRecdVals["txtFont"].inputs); - if (lastRecdVals["txtFontSZ"]) this.setFontSize(lastRecdVals["txtFontSZ"].inputs); - if (lastRecdVals["txtALI"]) this.setTextAlignment(lastRecdVals["txtALI"].inputs); - if (lastRecdVals["lineDIS"]) this.setTextSpacing(lastRecdVals["lineDIS"].inputs); - if (lastRecdVals["letDIS"]) this.setTextSpacing(lastRecdVals["letDIS"].inputs); - if (lastRecdVals["textOVR"]) this.setOverflow(lastRecdVals["textOVR"].inputs); - if (lastRecdVals["txtFontTK"]) this.setThick(lastRecdVals["txtFontTK"].inputs); - if (lastRecdVals["textLIN"]) this.setLine(lastRecdVals["textLIN"].inputs); - - if (lastRecdVals["preTxt1"]) this.presetTextPosition(lastRecdVals["preTxt1"].inputs); - const box = newTextElement.getBoundingClientRect(); - this.setMargins({ ID : args.ID, WIDTH : box.width / 2, HEIGHT : box.height }); + const propertiesAndMethods = [ + ["textMAR", "setMargins"], ["preTxt1", "presetTextPosition"], + ["textCLR", "setTextColor"], ["textLIN", "setLine"], + ["textOUT", "setTextOutline"], ["textSHA", "setTextDropShadow"], + ["txtFont", "setTextFont"], ["txtFontSZ", "setFontSize"], + ["txtALI", "setTextAlignment"], ["lineDIS", "setTextSpacing"], + ["letDIS", "setTextSpacing"], ["textOVR", "setOverflow"], + ["txtFontTK", "setThick"] + ]; + for (const [property, method] of propertiesAndMethods) { + if (lastRecdVals[property] && lastRecdVals[property].inputs.ID === args.ID) { + this[method](lastRecdVals[property].inputs); + } + } } replaceTxt(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); if (elements.length > 0) { elements.forEach((element) => { element.innerHTML = xmlEscape(args.TXT).replace(/\n/g, "
") }); } else { @@ -467,6 +490,7 @@ } removeTxt(args) { + args.ID = this.fixID(args.ID); const elementsToRemove = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); elementsToRemove.forEach((element) => { render.removeOverlay(element); @@ -485,7 +509,7 @@ displayedTexts(args) { let texts = []; - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { texts.push(element.textContent) }); return JSON.stringify(texts); } @@ -496,14 +520,14 @@ } existingID(args) { - const index = allText.indexOf(`#SP_Text-Ext-${args.ID}`); + const index = allText.indexOf(`#SP_Text-Ext-${this.fixID(args.ID)}`); return Scratch.Cast.toBoolean(allText[index]); } makeGradient(args) { return `${args.TYPE}-gradient(${args.ANGLE}deg, ${args.COLOR1}, ${args.COLOR2})` } setTextColor(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { element.style.color = args.COLOR; if (args.COLOR.includes("gradient")) { @@ -522,7 +546,7 @@ } setTextDropShadow(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { element.style.textShadow = args.z === 0 ? "none" : `${args.x}px ${args.y * -1}px ${args.z}px ${args.COLOR}`; }); @@ -530,13 +554,13 @@ } setOverflow(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { element.style.overflow = args.TYPE }); lastRecdVals["textOVR"] = {inputs: args}; } setTextOutline(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { element.style.webkitTextStrokeColor = args.COLOR; element.style.webkitTextStrokeWidth = `${args.THICKNESS}px`; @@ -551,7 +575,7 @@ setLine(args) { const lineType = args.TYPE2.replace("strike", "line-"); - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { element.style.textDecorationLine = lineType; element.style.textDecorationStyle = args.TYPE1; @@ -562,39 +586,40 @@ } setMargins(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { element.style.width = `${args.WIDTH}px`; element.style.height = `${args.HEIGHT}px`; }); + lastRecdVals["textMAR"] = {inputs: args}; } setTextAlignment(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { element.style.textAlign = args.ALIGNMENT }); lastRecdVals["txtALI"] = {inputs: args}; } setTextFont(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { element.style.fontFamily = args.FONT }); lastRecdVals["txtFont"] = {inputs: args}; } setFontSize(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { element.style.fontSize = `${args.SIZE}px` }); lastRecdVals["txtFontSZ"] = {inputs: args}; } setThick(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { element.style.fontWeight = args.NUM * 9 }); lastRecdVals["txtFontTK"] = {inputs: args}; } setTextSpacing(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { element.style[args.ATT === "letter" ? "letterSpacing" : "lineHeight"] = `${args.SPACING}px`; }); @@ -607,7 +632,7 @@ lastRecdVals["preTxt1"] = {inputs: {...args, isPrint: true}}; } else { setTimeout(function() { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); const element = elements[elements.length - 1]; const centerW = parseFloat(element.style.width) / 2; const computedStyle = window.getComputedStyle(element); @@ -622,7 +647,7 @@ } setTextPosition(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { const centerW = parseFloat(element.style.width) / 2; const computedStyle = window.getComputedStyle(element); @@ -636,7 +661,7 @@ } textPosition(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); let value; elements.forEach((element) => { const centerW = parseFloat(element.style.width) / 2; @@ -655,7 +680,7 @@ } attOfText(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); let value; elements.forEach((element) => { value = element.style[args.ATT] }); value = args.ATT === "fontFamily" || args.ATT === "textAlign" || args.ATT === "overflow" ? value : parseFloat(value); @@ -664,7 +689,7 @@ } lineCnt(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); let value = []; elements.forEach((element) => { value.push(element.querySelectorAll("br").length + 1 || 1); @@ -674,68 +699,68 @@ } setTextZIndex(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { element.parentNode.style.zIndex = Math.round(args.Z_INDEX) }); } setEffect(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { - const currentStyleArray = [ + const curStyles = [ element.style.filter || "", element.parentNode.style.transform || "", element.style.opacity || "" ]; const regex = new RegExp(`${args.EFFECT}\\([^)]+\\)`, "g"); - currentStyleArray.forEach((currentStyle, index) => { - currentStyleArray[index] = currentStyle.replace(regex, ""); + curStyles.forEach((curStyle, index) => { + curStyles[index] = curStyle.replace(regex, ""); }); switch (args.EFFECT) { case "saturate": - element.style.filter = `${currentStyleArray[0]} saturate(${args.VALUE}%)`; + element.style.filter = `${curStyles[0]} saturate(${args.VALUE}%)`; break; case "contrast": - element.style.filter = `${currentStyleArray[0]} contrast(${args.VALUE + 100}%)`; + element.style.filter = `${curStyles[0]} contrast(${args.VALUE + 100}%)`; break; case "brightness": - element.style.filter = `${currentStyleArray[0]} brightness(${args.VALUE + 100}%)`; + element.style.filter = `${curStyles[0]} brightness(${args.VALUE + 100}%)`; break; case "hue-rotate": - element.style.filter = `${currentStyleArray[0]} hue-rotate(${args.VALUE}deg)`; + element.style.filter = `${curStyles[0]} hue-rotate(${args.VALUE}deg)`; break; case "opacity": element.style.opacity = (100 - args.VALUE) / 100; break; case "sepia": - element.style.filter = `${currentStyleArray[0]} sepia(${args.VALUE}%)`; + element.style.filter = `${curStyles[0]} sepia(${args.VALUE}%)`; break; case "invert": - element.style.filter = `${currentStyleArray[0]} invert(${args.VALUE}%)`; + element.style.filter = `${curStyles[0]} invert(${args.VALUE}%)`; break; case "rotate": - element.parentNode.style.transform = `${currentStyleArray[1]} rotate(${args.VALUE - 90}deg)`; + element.parentNode.style.transform = `${curStyles[1]} rotate(${args.VALUE - 90}deg)`; break; case "scaleX": - element.parentNode.style.transform = `${currentStyleArray[1]} scaleX(${args.VALUE / 100})`; + element.parentNode.style.transform = `${curStyles[1]} scaleX(${args.VALUE / 100})`; break; case "scaleY": - element.parentNode.style.transform = `${currentStyleArray[1]} scaleY(${args.VALUE / 100})`; + element.parentNode.style.transform = `${curStyles[1]} scaleY(${args.VALUE / 100})`; break; case "skewX": - element.parentNode.style.transform = `${currentStyleArray[1]} skewX(${args.VALUE}deg)`; + element.parentNode.style.transform = `${curStyles[1]} skewX(${args.VALUE}deg)`; break; case "skewY": - element.parentNode.style.transform = `${currentStyleArray[1]} skewY(${args.VALUE}deg)`; + element.parentNode.style.transform = `${curStyles[1]} skewY(${args.VALUE}deg)`; break; default: - element.style.filter = `${currentStyleArray[0]} blur(${args.VALUE}px)`; + element.style.filter = `${curStyles[0]} blur(${args.VALUE}px)`; break; } }); } resetEffect(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { const currentTransform = element.parentNode.style.transform || ""; const updatedTransform = currentTransform @@ -751,7 +776,7 @@ } amtOfEffect(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); let effectValue = args.EFFECT === "rotate" ? 90 : args.EFFECT.includes("scale") ? 100 : 0; elements.forEach((element) => { if (args.EFFECT === "rotate" || args.EFFECT.includes("scale") || args.EFFECT.includes("skew")) { @@ -779,6 +804,7 @@ } setTextCurve(args) { + args.ID = this.fixID(args.ID) const regex = args.ARC.includes("]*d="([^"]*)"/ : /]*d="([^"]*)"/; const match = args.ARC.match(regex); const outline = lastRecdVals["textOUT"] !== undefined ? lastRecdVals["textOUT"] : ""; From 327ae382dbc5a5e7f343d611fe144d68253e1bca Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Fri, 19 Jan 2024 22:05:38 -0800 Subject: [PATCH 14/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 27 ++++++++++++++++++++++----- 1 file changed, 22 insertions(+), 5 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 024daf5974..ab1bb1ea61 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.3.1 +// Version V.1.3.2 (function (Scratch) { "use strict"; @@ -387,6 +387,8 @@ { text: "alignment", value: "textAlign" }, { text: "margin width", value: "width" }, { text: "margin height", value: "height" }, + { text: "display width", value: "box2w" }, + { text: "display height", value: "box2h" }, { text: "letter spacing", value: "letterSpacing" }, { text: "line spacing", value: "lineHeight" }, { text: "overflow type", value: "overflow" } @@ -682,10 +684,25 @@ attOfText(args) { const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); let value; - elements.forEach((element) => { value = element.style[args.ATT] }); - value = args.ATT === "fontFamily" || args.ATT === "textAlign" || args.ATT === "overflow" ? value : parseFloat(value); - value = args.ATT === "fontWeight" ? value / 9 : value - return value || ""; + if (args.ATT.includes("box2")) { + const calcs = []; + elements.forEach((element) => { + const tempSpan = document.createElement("span"); + tempSpan.innerHTML = element.textContent; + tempSpan.style.fontSize = element.style.fontSize; + tempSpan.style.fontFamily = getComputedStyle(element).fontFamily; + tempSpan.style.display = "inline"; + document.body.appendChild(tempSpan); + calcs.push(tempSpan[`offset${args.ATT.includes("w") ? "Width" : "Height"}`]); + document.body.removeChild(tempSpan); + }); + return JSON.stringify(calcs); + } else { + elements.forEach((element) => { value = element.style[args.ATT] }); + value = args.ATT === "fontFamily" || args.ATT === "textAlign" || args.ATT === "overflow" ? value : parseFloat(value); + value = args.ATT === "fontWeight" ? value / 9 : value; + return value || ""; + } } lineCnt(args) { From b3c46a5fc43a25e08c4fcbf22fe9ae6234e96b25 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Tue, 27 Feb 2024 21:27:49 -0800 Subject: [PATCH 15/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index ab1bb1ea61..785b39c8bf 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.3.2 +// Version V.1.3.3 (function (Scratch) { "use strict"; @@ -19,8 +19,7 @@ let allText = []; let lastRecdVals = {}; const fontMenu = [ - "Scratch", "Sans Serif", - "Serif", "Handwriting", + "Sans Serif", "Serif", "Handwriting", "Marker", "Curly", "Pixel" ]; @@ -423,7 +422,7 @@ fixID(ID) { return xmlEscape(Scratch.Cast.toString(ID).replaceAll(" ", "_") - .replaceAll(/[#%(),.{}[\]$@^*&'";:]/g, "-")); + .replaceAll(/[#%(),.{}[\/\]$@^*&'";:]/g, "-")); } debug(args) { @@ -463,7 +462,7 @@ render.addOverlay(newTextElement, "scale-centered"); allText.push(`#SP_Text-Ext-${args.ID}`); const box = newTextElement.getBoundingClientRect(); - if (!lastRecdVals.textMAR) this.setMargins({ ID : args.ID, WIDTH : box.width / 2, HEIGHT : box.height }); + if (lastRecdVals.textMAR === undefined) this.setMargins({ ID : args.ID, WIDTH : box.width / 2, HEIGHT : box.height }); // add formatting (if any) const propertiesAndMethods = [ @@ -518,6 +517,8 @@ allIDs() { const cleanedIDs = allText.map((item) => item.replace(/^#SP_Text-Ext-/, "")); + let index = cleanedIDs.indexOf("#SP_Text-Debug-Cross"); + if (index !== -1) cleanedIDs.splice(index, 1); return JSON.stringify(cleanedIDs); } From 33fae31e345f9db045d7243cdd241e1ed98c9407 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Tue, 27 Feb 2024 21:29:29 -0800 Subject: [PATCH 16/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 785b39c8bf..dea4cec599 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -422,7 +422,7 @@ fixID(ID) { return xmlEscape(Scratch.Cast.toString(ID).replaceAll(" ", "_") - .replaceAll(/[#%(),.{}[\/\]$@^*&'";:]/g, "-")); + .replaceAll(/[#%(),.{}[/\]$@^*&'";:]/g, "-")); } debug(args) { From b5f9df79b32e3d152b4527d22734ae36d0378fee Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Wed, 13 Mar 2024 22:22:44 -0700 Subject: [PATCH 17/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 10 ++++++---- 1 file changed, 6 insertions(+), 4 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index dea4cec599..27f752dc0d 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.3.3 +// Version V.1.3.4 (function (Scratch) { "use strict"; @@ -517,8 +517,9 @@ allIDs() { const cleanedIDs = allText.map((item) => item.replace(/^#SP_Text-Ext-/, "")); - let index = cleanedIDs.indexOf("#SP_Text-Debug-Cross"); - if (index !== -1) cleanedIDs.splice(index, 1); + for (let i = cleanedIDs.length - 1; i >= 0; i--) { + if (cleanedIDs[i] === "#SP_Text-Debug-Cross") cleanedIDs.splice(i, 1); + } return JSON.stringify(cleanedIDs); } @@ -652,7 +653,8 @@ setTextPosition(args) { const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); elements.forEach((element) => { - const centerW = parseFloat(element.style.width) / 2; + let centerW = parseFloat(element.style.width) / 2; + if (isNaN(centerW)) centerW = (element.getBoundingClientRect().width ?? 0 ) / 2; const computedStyle = window.getComputedStyle(element); const lineHeight = computedStyle.getPropertyValue("line-height"); const fontSize = computedStyle.getPropertyValue("font-size"); From 61f6fd1c65e72ee3a7089474789b0959fb7af549 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Sun, 12 May 2024 00:18:50 -0700 Subject: [PATCH 18/25] Update Display-Text.js --- extensions/SharkPool/Display-Text.js | 97 +++++++++++++++++++++++----- 1 file changed, 80 insertions(+), 17 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 27f752dc0d..1c356867dc 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.3.4 +// Version V.1.4.0 (function (Scratch) { "use strict"; @@ -17,10 +17,11 @@ const render = Scratch.vm.renderer; let allText = []; + let clickedTxts = []; let lastRecdVals = {}; const fontMenu = [ - "Sans Serif", "Serif", "Handwriting", - "Marker", "Curly", "Pixel" + "Scratch", "Sans Serif", "Serif", + "Handwriting", "Marker", "Curly", "Pixel" ]; const xmlEscape = function (unsafe) { @@ -83,6 +84,14 @@ ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } }, }, + { + opcode: "resetTxt", + blockType: Scratch.BlockType.COMMAND, + text: "reset text settings with ID [ID]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } + }, + }, { opcode: "removeAllTxt", blockType: Scratch.BlockType.COMMAND, @@ -110,15 +119,6 @@ blockType: Scratch.BlockType.REPORTER, text: "all text IDs" }, - "---", - { - opcode: "debug", - blockType: Scratch.BlockType.COMMAND, - text: "toggle debug mode [TOGGLE]", - arguments: { - TOGGLE: { type: Scratch.ArgumentType.STRING, menu: "TOGGLE" } - }, - }, { blockType: Scratch.BlockType.LABEL, text: "Formatting" }, { opcode: "setTextFont", @@ -203,7 +203,7 @@ ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } }, }, - { blockType: Scratch.BlockType.LABEL, text: "Text Visuals" }, + "---", { opcode: "setTextColor", blockType: Scratch.BlockType.COMMAND, @@ -345,6 +345,32 @@ ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } }, }, + { blockType: Scratch.BlockType.LABEL, text: "Advanced" }, + { + opcode: "debug", + blockType: Scratch.BlockType.COMMAND, + text: "toggle debug mode [TOGGLE]", + arguments: { + TOGGLE: { type: Scratch.ArgumentType.STRING, menu: "TOGGLE" } + }, + }, + { + opcode: "makeClick", + blockType: Scratch.BlockType.COMMAND, + text: "toggle clicking for ID [ID] [TYPE]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + TYPE: { type: Scratch.ArgumentType.STRING, menu: "TOGGLE" } + }, + }, + { + opcode: "isClicked", + blockType: Scratch.BlockType.BOOLEAN, + text: "ID [ID] clicked?", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } + }, + } ], menus: { TOGGLE: ["on", "off"], @@ -408,7 +434,7 @@ { text: "scale x", value: "scaleX" }, { text: "scale y", value: "scaleY" }, { text: "skew x", value: "skewX" }, - { text: "skew y", value: "skewY" }, + { text: "skew y", value: "skewY" } ] } } @@ -455,11 +481,14 @@ printTxt(args) { args.ID = this.fixID(args.ID); + const textDiv = document.createElement("div"); + textDiv.style.transformOrigin = "left top"; const newTextElement = document.createElement("div"); newTextElement.innerHTML = xmlEscape(args.TXT).replace(/\n/g, "
"); newTextElement.id = `SP_Text-Ext-${args.ID}`; newTextElement.classList.add(args.ID); - render.addOverlay(newTextElement, "scale-centered"); + textDiv.appendChild(newTextElement); + render.addOverlay(textDiv, "scale-centered"); allText.push(`#SP_Text-Ext-${args.ID}`); const box = newTextElement.getBoundingClientRect(); if (lastRecdVals.textMAR === undefined) this.setMargins({ ID : args.ID, WIDTH : box.width / 2, HEIGHT : box.height }); @@ -494,18 +523,29 @@ args.ID = this.fixID(args.ID); const elementsToRemove = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); elementsToRemove.forEach((element) => { - render.removeOverlay(element); + render.removeOverlay(element.parentNode); + element.removeEventListener("click", this.handleClick); const index = allText.indexOf(`#SP_Text-Ext-${args.ID}`); if (index !== -1) allText.splice(index, 1); }); } + resetTxt(args) { + Object.keys(lastRecdVals).forEach(key => { + if (lastRecdVals[key].inputs.ID === args.ID) delete lastRecdVals[key]; + }); + } + removeAllTxt() { for (let i = 0; i < allText.length; i++) { const elementsToRemove = document.querySelectorAll(allText[i]); - elementsToRemove.forEach((element) => { render.removeOverlay(element) }); + elementsToRemove.forEach((element) => { + render.removeOverlay(element.parentNode) + element.removeEventListener("click", this.handleClick); + }); } allText = []; + clickedTxts = []; } displayedTexts(args) { @@ -890,6 +930,29 @@ }; return presets[args.ARC]; } + + makeClick(args) { + const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + elements.forEach((element) => { + if (args.TYPE === "on") { + element.style.pointerEvents = "auto"; + element.style.cursor = "pointer"; + } else { + element.style.pointerEvents = "none"; + element.style.cursor = "none"; + } + element.removeEventListener("click", this.handleClick); + if (args.TYPE === "on") element.addEventListener("click", this.handleClick); + }); + } + + isClicked(args) { return clickedTxts.indexOf(this.fixID(args.ID)) > -1 } + + handleClick(event) { + clickedTxts.push(event.target.className); + const index = clickedTxts.length - 1; + setTimeout(function() { clickedTxts.splice(index, 1) }, 100); + } } Scratch.extensions.register(new SPdisText()); From 850e4ab48d3338bbb0b91fc1fd4a4f372b233361 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Thu, 18 Jul 2024 21:04:06 -0700 Subject: [PATCH 19/25] Display-Text -- Boost Speed and Fix Positioning Forever --- extensions/SharkPool/Display-Text.js | 144 ++++++++++++--------------- 1 file changed, 62 insertions(+), 82 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 1c356867dc..1e04fb72cb 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,11 +3,11 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.4.0 +// Version V.1.4.1 (function (Scratch) { "use strict"; - if (!Scratch.extensions.unsandboxed) throw new Error("Display Text Extension must run unsandboxed!"); + if (!Scratch.extensions.unsandboxed) throw new Error("Display Text must run unsandboxed!"); const menuIconURI = ""; @@ -15,7 +15,9 @@ const blockIconURI = ""; - const render = Scratch.vm.renderer; + const vm = Scratch.vm; + const runtime = vm.runtime; + const render = vm.renderer; let allText = []; let clickedTxts = []; let lastRecdVals = {}; @@ -39,8 +41,8 @@ class SPdisText { constructor() { - Scratch.vm.runtime.on("PROJECT_START", () => { this.removeAllTxt() }); - Scratch.vm.runtime.on("PROJECT_STOP_ALL", () => { this.removeAllTxt() }); + runtime.on("PROJECT_START", () => { this.removeAllTxt() }); + runtime.on("PROJECT_STOP_ALL", () => { this.removeAllTxt() }); } getInfo() { return { @@ -58,10 +60,7 @@ text: "print text [TXT] with ID [ID]", arguments: { ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, - TXT: { - type: Scratch.ArgumentType.STRING, - defaultValue: "hello world" - } + TXT: { type: Scratch.ArgumentType.STRING, defaultValue: "hello world" } }, }, { @@ -70,10 +69,7 @@ text: "replace text of ID [ID] with [TXT]", arguments: { ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, - TXT: { - type: Scratch.ArgumentType.STRING, - defaultValue: "lorem ipsum" - } + TXT: { type: Scratch.ArgumentType.STRING, defaultValue: "lorem ipsum" } }, }, { @@ -374,10 +370,7 @@ ], menus: { TOGGLE: ["on", "off"], - FONTS: { - acceptReporters: true, - items: "allFonts" - }, + FONTS: { acceptReporters: true, items: "allFonts" }, STYLE: { acceptReporters: true, items: ["solid", "wavy", "dashed", "double"] @@ -447,13 +440,12 @@ } fixID(ID) { - return xmlEscape(Scratch.Cast.toString(ID).replaceAll(" ", "_") - .replaceAll(/[#%(),.{}[/\]$@^*&'";:]/g, "-")); + return xmlEscape(Scratch.Cast.toString(ID).replaceAll(" ", "_").replaceAll(/[#%(),.{}[/\]$@^*&'";:]/g, "-")); } debug(args) { const toggle = args.TOGGLE === "on" ? "solid" : "none"; - const elements = document.querySelectorAll(`[id^="SP_Text-Ext-"]`); + const elements = document.querySelectorAll(`div[id^="SP_Text-Ext-"]`); elements.forEach((element) => { element.style.border = toggle; element.style.borderWidth = "1px"; @@ -461,9 +453,8 @@ element.style.borderColor = color; }); const cross = document.getElementById(`SP_Text-Debug-Cross`); - if (toggle === "none" && cross) { - cross.parentNode.removeChild(cross); - } else if (toggle === "solid" && !cross) { + if (toggle === "none" && cross) cross.parentNode.removeChild(cross); + else if (toggle === "solid" && !cross) { const imageUrl = ""; const crossElement = document.createElement("img"); @@ -483,14 +474,15 @@ args.ID = this.fixID(args.ID); const textDiv = document.createElement("div"); textDiv.style.transformOrigin = "left top"; - const newTextElement = document.createElement("div"); - newTextElement.innerHTML = xmlEscape(args.TXT).replace(/\n/g, "
"); - newTextElement.id = `SP_Text-Ext-${args.ID}`; - newTextElement.classList.add(args.ID); - textDiv.appendChild(newTextElement); + const txtElement = document.createElement("div"); + txtElement.innerHTML = xmlEscape(args.TXT).replace(/\n/g, "
"); + txtElement.id = `SP_Text-Ext-${args.ID}`; + txtElement.classList.add(args.ID); + txtElement.setAttribute("sptxtpos", "120|-10"); + textDiv.appendChild(txtElement); render.addOverlay(textDiv, "scale-centered"); allText.push(`#SP_Text-Ext-${args.ID}`); - const box = newTextElement.getBoundingClientRect(); + const box = txtElement.getBoundingClientRect(); if (lastRecdVals.textMAR === undefined) this.setMargins({ ID : args.ID, WIDTH : box.width / 2, HEIGHT : box.height }); // add formatting (if any) @@ -511,17 +503,14 @@ } replaceTxt(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); - if (elements.length > 0) { - elements.forEach((element) => { element.innerHTML = xmlEscape(args.TXT).replace(/\n/g, "
") }); - } else { - this.printTxt(args); - } + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + if (elements.length > 0) elements.forEach((element) => { element.innerHTML = xmlEscape(args.TXT).replace(/\n/g, "
") }); + else this.printTxt(args); } removeTxt(args) { args.ID = this.fixID(args.ID); - const elementsToRemove = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elementsToRemove = document.querySelectorAll(`div[id="SP_Text-Ext-${args.ID}"]`); elementsToRemove.forEach((element) => { render.removeOverlay(element.parentNode); element.removeEventListener("click", this.handleClick); @@ -550,7 +539,7 @@ displayedTexts(args) { let texts = []; - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { texts.push(element.textContent) }); return JSON.stringify(texts); } @@ -571,7 +560,7 @@ makeGradient(args) { return `${args.TYPE}-gradient(${args.ANGLE}deg, ${args.COLOR1}, ${args.COLOR2})` } setTextColor(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { element.style.color = args.COLOR; if (args.COLOR.includes("gradient")) { @@ -590,7 +579,7 @@ } setTextDropShadow(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { element.style.textShadow = args.z === 0 ? "none" : `${args.x}px ${args.y * -1}px ${args.z}px ${args.COLOR}`; }); @@ -598,13 +587,13 @@ } setOverflow(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { element.style.overflow = args.TYPE }); lastRecdVals["textOVR"] = {inputs: args}; } setTextOutline(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { element.style.webkitTextStrokeColor = args.COLOR; element.style.webkitTextStrokeWidth = `${args.THICKNESS}px`; @@ -619,7 +608,7 @@ setLine(args) { const lineType = args.TYPE2.replace("strike", "line-"); - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { element.style.textDecorationLine = lineType; element.style.textDecorationStyle = args.TYPE1; @@ -630,7 +619,7 @@ } setMargins(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { element.style.width = `${args.WIDTH}px`; element.style.height = `${args.HEIGHT}px`; @@ -639,31 +628,31 @@ } setTextAlignment(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { element.style.textAlign = args.ALIGNMENT }); lastRecdVals["txtALI"] = {inputs: args}; } setTextFont(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { element.style.fontFamily = args.FONT }); lastRecdVals["txtFont"] = {inputs: args}; } setFontSize(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { element.style.fontSize = `${args.SIZE}px` }); lastRecdVals["txtFontSZ"] = {inputs: args}; } setThick(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { element.style.fontWeight = args.NUM * 9 }); lastRecdVals["txtFontTK"] = {inputs: args}; } setTextSpacing(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { element.style[args.ATT === "letter" ? "letterSpacing" : "lineHeight"] = `${args.SPACING}px`; }); @@ -672,11 +661,10 @@ } presetTextPosition(args) { - if (args.isPrint === undefined) { - lastRecdVals["preTxt1"] = {inputs: {...args, isPrint: true}}; - } else { - setTimeout(function() { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + if (args.isPrint === undefined) lastRecdVals["preTxt1"] = {inputs: {...args, isPrint: true}}; + else { + setTimeout(() => { + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); const element = elements[elements.length - 1]; const centerW = parseFloat(element.style.width) / 2; const computedStyle = window.getComputedStyle(element); @@ -684,14 +672,14 @@ const fontSize = computedStyle.getPropertyValue("font-size"); const centerH = (lineHeight === "normal" ? parseFloat(fontSize) * 1.2 : parseFloat(lineHeight)) / 2; element.style.position = "absolute"; - element.style.left = `${args.X - centerW}px`; - element.style.top = `${(args.Y * -1) - centerH}px`; - }.bind(this), 0); + element.style.transform = `translate(${args.X - centerW}px, ${(args.Y * -1) - centerH}px)`; + element.setAttribute("sptxtpos", `${args.X}|${args.Y}`); + }, 0); } } setTextPosition(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { let centerW = parseFloat(element.style.width) / 2; if (isNaN(centerW)) centerW = (element.getBoundingClientRect().width ?? 0 ) / 2; @@ -700,13 +688,13 @@ const fontSize = computedStyle.getPropertyValue("font-size"); const centerH = (lineHeight === "normal" ? parseFloat(fontSize) * 1.2 : parseFloat(lineHeight)) / 2; element.style.position = "absolute"; - element.style.left = `${args.X - centerW}px`; - element.style.top = `${(args.Y * -1) - centerH}px`; + element.style.transform = `translate(${args.X - centerW}px, ${(args.Y * -1) - centerH}px)`; + element.setAttribute("sptxtpos", `${args.X}|${args.Y}`); }); } textPosition(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); let value; elements.forEach((element) => { const centerW = parseFloat(element.style.width) / 2; @@ -714,18 +702,14 @@ const lineHeight = computedStyle.getPropertyValue("line-height"); const fontSize = computedStyle.getPropertyValue("font-size"); const centerH = (lineHeight === "normal" ? parseFloat(fontSize) * 1.2 : parseFloat(lineHeight)) / 2; - if (args.ATT === "z layer") { - value = element.parentNode.style.zIndex; - } else { - value = parseFloat(element.style[args.ATT.includes("x") ? "left" : "top"]); - value = ((args.ATT.includes("x") ? centerW : centerH) + value) * (args.ATT.includes("x") ? 1 : -1); - } + if (args.ATT === "z layer") value = element.parentNode.style.zIndex; + else value = element.getAttribute("sptxtpos")?.split("|", 2)[args.ATT.includes("x") ? 0 : 1]; }); - return Math.round(value * 100) / 100 || 0; + return value || 0; } attOfText(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); let value; if (args.ATT.includes("box2")) { const calcs = []; @@ -749,22 +733,20 @@ } lineCnt(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); let value = []; - elements.forEach((element) => { - value.push(element.querySelectorAll("br").length + 1 || 1); - }); + elements.forEach((element) => { value.push(element.querySelectorAll("br").length + 1 || 1) }); value = value.length > 1 ? JSON.stringify(value) : (value[0] || 0); return value; } setTextZIndex(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { element.parentNode.style.zIndex = Math.round(args.Z_INDEX) }); } setEffect(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { const curStyles = [ element.style.filter || "", @@ -772,9 +754,7 @@ element.style.opacity || "" ]; const regex = new RegExp(`${args.EFFECT}\\([^)]+\\)`, "g"); - curStyles.forEach((curStyle, index) => { - curStyles[index] = curStyle.replace(regex, ""); - }); + curStyles.forEach((curStyle, index) => { curStyles[index] = curStyle.replace(regex, "") }); switch (args.EFFECT) { case "saturate": element.style.filter = `${curStyles[0]} saturate(${args.VALUE}%)`; @@ -820,7 +800,7 @@ } resetEffect(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { const currentTransform = element.parentNode.style.transform || ""; const updatedTransform = currentTransform @@ -836,7 +816,7 @@ } amtOfEffect(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); let effectValue = args.EFFECT === "rotate" ? 90 : args.EFFECT.includes("scale") ? 100 : 0; elements.forEach((element) => { if (args.EFFECT === "rotate" || args.EFFECT.includes("scale") || args.EFFECT.includes("skew")) { @@ -869,7 +849,7 @@ const match = args.ARC.match(regex); const outline = lastRecdVals["textOUT"] !== undefined ? lastRecdVals["textOUT"] : ""; if (match && match[1]) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${args.ID}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${args.ID}"]`) elements.forEach((element) => { const existingSvg = element.querySelector("svg"); if (existingSvg) { @@ -932,7 +912,7 @@ } makeClick(args) { - const elements = document.querySelectorAll(`#SP_Text-Ext-${this.fixID(args.ID)}`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { if (args.TYPE === "on") { element.style.pointerEvents = "auto"; @@ -951,7 +931,7 @@ handleClick(event) { clickedTxts.push(event.target.className); const index = clickedTxts.length - 1; - setTimeout(function() { clickedTxts.splice(index, 1) }, 100); + setTimeout(() => { clickedTxts.splice(index, 1) }, 100); } } From fbffbd252f5249fe84803b2dcb17f0c2e57bb49f Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Thu, 18 Jul 2024 21:06:21 -0700 Subject: [PATCH 20/25] Display-Text -- Boost Speed and Fix Positioning Forever --- extensions/SharkPool/Display-Text.js | 5 ----- 1 file changed, 5 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 1e04fb72cb..24f99898ba 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -697,11 +697,6 @@ const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); let value; elements.forEach((element) => { - const centerW = parseFloat(element.style.width) / 2; - const computedStyle = window.getComputedStyle(element); - const lineHeight = computedStyle.getPropertyValue("line-height"); - const fontSize = computedStyle.getPropertyValue("font-size"); - const centerH = (lineHeight === "normal" ? parseFloat(fontSize) * 1.2 : parseFloat(lineHeight)) / 2; if (args.ATT === "z layer") value = element.parentNode.style.zIndex; else value = element.getAttribute("sptxtpos")?.split("|", 2)[args.ATT.includes("x") ? 0 : 1]; }); From 22de101d960237667c8368f1649439a1d99394a3 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Wed, 7 Aug 2024 22:48:25 -0700 Subject: [PATCH 21/25] layering fix --- extensions/SharkPool/Display-Text.js | 18 +++++++++--------- 1 file changed, 9 insertions(+), 9 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 24f99898ba..16fffbce4d 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,28 +3,28 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.4.1 +// Version V.1.4.2 (function (Scratch) { "use strict"; if (!Scratch.extensions.unsandboxed) throw new Error("Display Text must run unsandboxed!"); const menuIconURI = -""; +""; const blockIconURI = -""; +""; const vm = Scratch.vm; const runtime = vm.runtime; const render = vm.renderer; + const fontMenu = [ + "Scratch", "Sans Serif", "Serif", "Handwriting", "Marker", "Curly", "Pixel" + ]; + let allText = []; let clickedTxts = []; let lastRecdVals = {}; - const fontMenu = [ - "Scratch", "Sans Serif", "Serif", - "Handwriting", "Marker", "Curly", "Pixel" - ]; const xmlEscape = function (unsafe) { unsafe = String(unsafe); @@ -697,7 +697,7 @@ const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); let value; elements.forEach((element) => { - if (args.ATT === "z layer") value = element.parentNode.style.zIndex; + if (args.ATT === "z layer") value = element.parentNode.parentNode.style.zIndex; else value = element.getAttribute("sptxtpos")?.split("|", 2)[args.ATT.includes("x") ? 0 : 1]; }); return value || 0; @@ -737,7 +737,7 @@ setTextZIndex(args) { const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); - elements.forEach((element) => { element.parentNode.style.zIndex = Math.round(args.Z_INDEX) }); + elements.forEach((element) => { element.parentNode.parentNode.style.zIndex = Math.round(args.Z_INDEX) }); } setEffect(args) { From 1b655ecc3dcc197a24244c5941a0fa8f570468bc Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Mon, 26 Aug 2024 17:11:37 -0700 Subject: [PATCH 22/25] Display-Text -- Content Drop --- extensions/SharkPool/Display-Text.js | 197 +++++++++++++-------------- 1 file changed, 95 insertions(+), 102 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 16fffbce4d..bf4cec3823 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.4.2 +// Version V.1.4.3 (function (Scratch) { "use strict"; @@ -22,9 +22,7 @@ "Scratch", "Sans Serif", "Serif", "Handwriting", "Marker", "Curly", "Pixel" ]; - let allText = []; - let clickedTxts = []; - let lastRecdVals = {}; + let allText = [], clickedTxts = [], lastRecdVals = {}; const xmlEscape = function (unsafe) { unsafe = String(unsafe); @@ -366,31 +364,34 @@ arguments: { ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } }, - } - ], - menus: { - TOGGLE: ["on", "off"], - FONTS: { acceptReporters: true, items: "allFonts" }, - STYLE: { - acceptReporters: true, - items: ["solid", "wavy", "dashed", "double"] }, - LINE_TYPE: { - acceptReporters: true, - items: ["underline", "strikethrough"] + { + opcode: "whenClick", + blockType: Scratch.BlockType.EVENT, + isEdgeActivated: false, + text: "when any text clicked" }, - ALIGNMENTS: { - acceptReporters: true, - items: ["left", "right", "center"] + { + opcode: "clickedID", + blockType: Scratch.BlockType.REPORTER, + disableMonitor: true, + text: "clicked text ID" }, + ], + menus: { + FONTS: { acceptReporters: true, items: "allFonts" }, + TOGGLE: ["on", "off"], TEXT_ATT: ["letter", "line"], - GRADIENTS: { - acceptReporters: true, - items: ["linear", "radial"] - }, POS: ["x position", "y position", "z layer"], OVERFLOW: ["visible", "hidden"], ARCS: ["circle", "hill", "dip", "wave"], + STYLE: { + acceptReporters: true, + items: ["solid", "wavy", "dashed", "double"] + }, + LINE_TYPE: { acceptReporters: true, items: ["underline", "strikethrough"] }, + ALIGNMENTS: { acceptReporters: true, items: ["left", "right", "center"] }, + GRADIENTS: { acceptReporters: true, items: ["linear", "radial"] }, THICK: [ { text : "thick", value : "900" }, { text : "medium", value : "600" }, @@ -434,6 +435,7 @@ }; } + // Helper Funcs allFonts() { const customFonts = Scratch.vm.runtime.fontManager ? Scratch.vm.runtime.fontManager.getFonts().map((i) => ({ text: i.name, value: i.family })) : []; return [...fontMenu, ...customFonts]; @@ -443,6 +445,15 @@ return xmlEscape(Scratch.Cast.toString(ID).replaceAll(" ", "_").replaceAll(/[#%(),.{}[/\]$@^*&'";:]/g, "-")); } + handleClick(e) { + clickedTxts.push(e.target.className); + const index = clickedTxts.length - 1; + const threads = runtime.startHats("SPdisText_whenClick"); + for (let i = 0; i < threads.length; i++) threads[i].SPdisTxtClickID = e.target.className; + setTimeout(() => { clickedTxts.splice(index, 1) }, 100); + } + + // Block Funcs debug(args) { const toggle = args.TOGGLE === "on" ? "solid" : "none"; const elements = document.querySelectorAll(`div[id^="SP_Text-Ext-"]`); @@ -455,14 +466,12 @@ const cross = document.getElementById(`SP_Text-Debug-Cross`); if (toggle === "none" && cross) cross.parentNode.removeChild(cross); else if (toggle === "solid" && !cross) { - const imageUrl = - ""; const crossElement = document.createElement("img"); - crossElement.src = imageUrl; + crossElement.src = + ""; crossElement.id = "SP_Text-Debug-Cross"; crossElement.style.position = "absolute"; - crossElement.style.width = "480px"; - crossElement.style.height = "360px"; + crossElement.style.width = "480px"; crossElement.style.height = "360px"; crossElement.style.transform = "translate(-50%, -50%)"; crossElement.style.zIndex = "0"; render.addOverlay(crossElement, "scale-centered"); @@ -533,8 +542,7 @@ element.removeEventListener("click", this.handleClick); }); } - allText = []; - clickedTxts = []; + allText = []; clickedTxts = []; } displayedTexts(args) { @@ -663,7 +671,7 @@ presetTextPosition(args) { if (args.isPrint === undefined) lastRecdVals["preTxt1"] = {inputs: {...args, isPrint: true}}; else { - setTimeout(() => { + runtime.once("AFTER_EXECUTE", () => { const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); const element = elements[elements.length - 1]; const centerW = parseFloat(element.style.width) / 2; @@ -671,10 +679,15 @@ const lineHeight = computedStyle.getPropertyValue("line-height"); const fontSize = computedStyle.getPropertyValue("font-size"); const centerH = (lineHeight === "normal" ? parseFloat(fontSize) * 1.2 : parseFloat(lineHeight)) / 2; + + let transform = element.style.transform; + const string = `translate(${args.X - centerW}px, ${(args.Y * -1) - centerH}px)`; + if (transform.includes("translate")) transform = transform.replace(/translate\([^)]*\)/, string); + else transform += ` ${string}`; + element.style.transform = transform.trim(); element.style.position = "absolute"; - element.style.transform = `translate(${args.X - centerW}px, ${(args.Y * -1) - centerH}px)`; element.setAttribute("sptxtpos", `${args.X}|${args.Y}`); - }, 0); + }); } } @@ -687,8 +700,13 @@ const lineHeight = computedStyle.getPropertyValue("line-height"); const fontSize = computedStyle.getPropertyValue("font-size"); const centerH = (lineHeight === "normal" ? parseFloat(fontSize) * 1.2 : parseFloat(lineHeight)) / 2; + + let transform = element.style.transform; + const string = `translate(${args.X - centerW}px, ${(args.Y * -1) - centerH}px)`; + if (transform.includes("translate")) transform = transform.replace(/translate\([^)]*\)/, string); + else transform += ` ${string}`; + element.style.transform = transform.trim(); element.style.position = "absolute"; - element.style.transform = `translate(${args.X - centerW}px, ${(args.Y * -1) - centerH}px)`; element.setAttribute("sptxtpos", `${args.X}|${args.Y}`); }); } @@ -731,8 +749,7 @@ const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); let value = []; elements.forEach((element) => { value.push(element.querySelectorAll("br").length + 1 || 1) }); - value = value.length > 1 ? JSON.stringify(value) : (value[0] || 0); - return value; + return value.length > 1 ? JSON.stringify(value) : (value[0] || 0); } setTextZIndex(args) { @@ -742,7 +759,9 @@ setEffect(args) { const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const val = Scratch.Cast.toNumber(args.VALUE); elements.forEach((element) => { + const parent = element.parentNode; const curStyles = [ element.style.filter || "", element.parentNode.style.transform || "", @@ -751,45 +770,25 @@ const regex = new RegExp(`${args.EFFECT}\\([^)]+\\)`, "g"); curStyles.forEach((curStyle, index) => { curStyles[index] = curStyle.replace(regex, "") }); switch (args.EFFECT) { - case "saturate": - element.style.filter = `${curStyles[0]} saturate(${args.VALUE}%)`; - break; - case "contrast": - element.style.filter = `${curStyles[0]} contrast(${args.VALUE + 100}%)`; - break; - case "brightness": - element.style.filter = `${curStyles[0]} brightness(${args.VALUE + 100}%)`; - break; - case "hue-rotate": - element.style.filter = `${curStyles[0]} hue-rotate(${args.VALUE}deg)`; - break; - case "opacity": - element.style.opacity = (100 - args.VALUE) / 100; - break; - case "sepia": - element.style.filter = `${curStyles[0]} sepia(${args.VALUE}%)`; - break; - case "invert": - element.style.filter = `${curStyles[0]} invert(${args.VALUE}%)`; - break; - case "rotate": - element.parentNode.style.transform = `${curStyles[1]} rotate(${args.VALUE - 90}deg)`; - break; - case "scaleX": - element.parentNode.style.transform = `${curStyles[1]} scaleX(${args.VALUE / 100})`; - break; - case "scaleY": - element.parentNode.style.transform = `${curStyles[1]} scaleY(${args.VALUE / 100})`; - break; - case "skewX": - element.parentNode.style.transform = `${curStyles[1]} skewX(${args.VALUE}deg)`; - break; - case "skewY": - element.parentNode.style.transform = `${curStyles[1]} skewY(${args.VALUE}deg)`; - break; - default: - element.style.filter = `${curStyles[0]} blur(${args.VALUE}px)`; - break; + case "saturate": return element.style.filter = `${curStyles[0]} saturate(${val}%)`; + case "contrast": return element.style.filter = `${curStyles[0]} contrast(${val + 100}%)`; + case "brightness": return element.style.filter = `${curStyles[0]} brightness(${val + 100}%)`; + case "hue-rotate": return element.style.filter = `${curStyles[0]} hue-rotate(${val}deg)`; + case "opacity": return element.style.opacity = (100 - val) / 100; + case "sepia": return element.style.filter = `${curStyles[0]} sepia(${val}%)`; + case "invert": return element.style.filter = `${curStyles[0]} invert(${val}%)`; + case "scaleX": return parent.style.transform = `${curStyles[1]} scaleX(${val / 100})`; + case "scaleY": return parent.style.transform = `${curStyles[1]} scaleY(${val / 100})`; + case "skewX": return parent.style.transform = `${curStyles[1]} skewX(${val}deg)`; + case "skewY": return parent.style.transform = `${curStyles[1]} skewY(${val}deg)`; + case "rotate": { + let transform = element.style.transform; + if (transform.includes("rotate")) transform = transform.replace(/rotate\([^)]*\)/, `rotate(${val - 90}deg)`); + else transform += ` rotate(${val - 90}deg)`; + element.style.transform = transform.trim(); + return element.style.transformOrigin = "center"; + } + default: return element.style.filter = `${curStyles[0]} blur(${val}px)`; } }); } @@ -797,41 +796,39 @@ resetEffect(args) { const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); elements.forEach((element) => { - const currentTransform = element.parentNode.style.transform || ""; - const updatedTransform = currentTransform - .replace(/rotate\([^)]*\)/, "rotate(0deg)") - .replace(/scaleX\([^)]*\)/, "scaleX(1)") - .replace(/scaleY\([^)]*\)/, "scaleY(1)") - .replace(/skewX\([^)]*\)/, "skewX(0deg)") - .replace(/skewY\([^)]*\)/, "skewY(0deg)"); - element.parentNode.style.transform = updatedTransform; + element.parentNode.style.transform = (element.parentNode.style.transform || "") + .replace(/scaleX\([^)]*\)/, "scaleX(1)").replace(/scaleY\([^)]*\)/, "scaleY(1)") + .replace(/skewX\([^)]*\)/, "skewX(0deg)").replace(/skewY\([^)]*\)/, "skewY(0deg)"); + element.style.transform = (element.style.transform || "") + .replace(/rotate\([^)]*\)/, "rotate(0deg)"); element.style.filter = ""; element.style.opacity = 1; }); } amtOfEffect(args) { + const effect = args.EFFECT; const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); - let effectValue = args.EFFECT === "rotate" ? 90 : args.EFFECT.includes("scale") ? 100 : 0; + let effectValue = effect === "rotate" ? 90 : effect.includes("scale") ? 100 : 0; elements.forEach((element) => { - if (args.EFFECT === "rotate" || args.EFFECT.includes("scale") || args.EFFECT.includes("skew")) { - const transformV = element.parentNode.style.transform.split(" "); - const effectIndex = transformV.findIndex(value => value.includes(args.EFFECT)); - if (effectIndex !== -1) { - effectValue = transformV[effectIndex].replace(/[^\d.-]/g, ""); - effectValue = args.EFFECT === "rotate" ? parseFloat(effectValue) + 90 : - parseFloat(effectValue) * (args.EFFECT.includes("scale") ? 100 : 1); + if (effect === "rotate" || effect.includes("scale") || effect.includes("skew")) { + const thisElement = effect === "rotate" ? element : element.parentNode; + const transformV = element.style.transform.split(" "); + const index = transformV.findIndex(value => value.includes(effect)); + if (index !== -1) { + effectValue = transformV[index].replace(/[^\d.-]/g, ""); + effectValue = effect === "rotate" ? parseFloat(effectValue) + 90 : + parseFloat(effectValue) * (effect.includes("scale") ? 100 : 1); } - } else if (args.EFFECT === "opacity") { + } else if (effect === "opacity") { effectValue = 100 - (parseFloat(element.style.opacity || 0) * 100); } else { const filterV = element.style.filter.split(" "); - const effectIndex = filterV.findIndex((value) => value.includes(args.EFFECT)); - if (effectIndex !== -1) { - effectValue = - args.EFFECT === "contrast" || args.EFFECT === "brightness" - ? parseFloat(filterV[effectIndex].replace(/[^\d.-]/g, "")) - 100 - : parseFloat(filterV[effectIndex].replace(/[^\d.-]/g, "")); + const index = filterV.findIndex((value) => value.includes(effect)); + if (index !== -1) { + effectValue = effect === "contrast" || effect === "brightness" + ? parseFloat(filterV[index].replace(/[^\d.-]/g, "")) - 100 + : parseFloat(filterV[index].replace(/[^\d.-]/g, "")); } } }); @@ -923,11 +920,7 @@ isClicked(args) { return clickedTxts.indexOf(this.fixID(args.ID)) > -1 } - handleClick(event) { - clickedTxts.push(event.target.className); - const index = clickedTxts.length - 1; - setTimeout(() => { clickedTxts.splice(index, 1) }, 100); - } + clickedID(_, util) { return util.thread.SPdisTxtClickID ?? "" } } Scratch.extensions.register(new SPdisText()); From c538f743eb493af7ecaf474fcb38c4517ba1d6b2 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Mon, 26 Aug 2024 17:13:23 -0700 Subject: [PATCH 23/25] fix thing --- extensions/SharkPool/Display-Text.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index bf4cec3823..921ebe0278 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -813,7 +813,7 @@ elements.forEach((element) => { if (effect === "rotate" || effect.includes("scale") || effect.includes("skew")) { const thisElement = effect === "rotate" ? element : element.parentNode; - const transformV = element.style.transform.split(" "); + const transformV = thisElement.style.transform.split(" "); const index = transformV.findIndex(value => value.includes(effect)); if (index !== -1) { effectValue = transformV[index].replace(/[^\d.-]/g, ""); From 791cc6de857c092620d0575a0bb3d785c474dcdb Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Fri, 30 Aug 2024 23:36:12 -0700 Subject: [PATCH 24/25] Display-Text -- V1.4.4 --- extensions/SharkPool/Display-Text.js | 240 ++++++++++++++++----------- 1 file changed, 147 insertions(+), 93 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 921ebe0278..7c14f14568 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -3,7 +3,7 @@ // Description: Display Text in Your Projects! // By: SharkPool -// Version V.1.4.3 +// Version V.1.4.4 (function (Scratch) { "use strict"; @@ -22,7 +22,7 @@ "Scratch", "Sans Serif", "Serif", "Handwriting", "Marker", "Curly", "Pixel" ]; - let allText = [], clickedTxts = [], lastRecdVals = {}; + let allText = [], clickedTxts = [], txtSettings = {}; const xmlEscape = function (unsafe) { unsafe = String(unsafe); @@ -78,14 +78,6 @@ ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } }, }, - { - opcode: "resetTxt", - blockType: Scratch.BlockType.COMMAND, - text: "reset text settings with ID [ID]", - arguments: { - ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } - }, - }, { opcode: "removeAllTxt", blockType: Scratch.BlockType.COMMAND, @@ -348,6 +340,34 @@ TOGGLE: { type: Scratch.ArgumentType.STRING, menu: "TOGGLE" } }, }, + "---", + { + opcode: "resetTxt", + blockType: Scratch.BlockType.COMMAND, + text: "reset text settings with ID [ID]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" } + }, + }, + { + opcode: "reuseStyle", + blockType: Scratch.BlockType.COMMAND, + text: "reuse text settings in ID [ID] for ID [ID2]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + ID2: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text2" } + }, + }, + "---", + { + opcode: "toggleSelect", + blockType: Scratch.BlockType.COMMAND, + text: "toggle highlighting for ID [ID] [TYPE]", + arguments: { + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "my-text" }, + TYPE: { type: Scratch.ArgumentType.STRING, menu: "TOGGLE" } + }, + }, { opcode: "makeClick", blockType: Scratch.BlockType.COMMAND, @@ -437,12 +457,14 @@ // Helper Funcs allFonts() { - const customFonts = Scratch.vm.runtime.fontManager ? Scratch.vm.runtime.fontManager.getFonts().map((i) => ({ text: i.name, value: i.family })) : []; + const customFonts = runtime.fontManager ? runtime.fontManager.getFonts().map((i) => ({ text: i.name, value: i.family })) : []; return [...fontMenu, ...customFonts]; } fixID(ID) { - return xmlEscape(Scratch.Cast.toString(ID).replaceAll(" ", "_").replaceAll(/[#%(),.{}[/\]$@^*&'";:]/g, "-")); + ID = xmlEscape(Scratch.Cast.toString(ID).replaceAll(" ", "_").replaceAll(/[#%(),.{}[/\]$@^*&'";:]/g, "-")); + txtSettings[ID] = { ...txtSettings[ID] }; + return ID; } handleClick(e) { @@ -453,15 +475,34 @@ setTimeout(() => { clickedTxts.splice(index, 1) }, 100); } + updateStyles(txtSetting, optOverrideName) { + const propsd2Func = [ + ["textMAR", "setMargins"], ["preTxt1", "presetTextPosition"], + ["textCLR", "setTextColor"], ["textLIN", "setLine"], + ["textOUT", "setTextOutline"], ["textSHA", "setTextDropShadow"], + ["txtFont", "setTextFont"], ["txtFontSZ", "setFontSize"], + ["txtALI", "setTextAlignment"], ["lineDIS", "setTextSpacing"], + ["letDIS", "setTextSpacing"], ["textOVR", "setOverflow"], + ["txtFontTK", "setThick"] + ]; + if (optOverrideName) { + for (const [id, func] of propsd2Func) { + if (txtSetting[id]) this[func]({ ...txtSetting[id], ID: optOverrideName }); + } + } else { + for (const [id, func] of propsd2Func) { + if (txtSetting[id]) this[func](txtSetting[id]); + } + } + } + // Block Funcs debug(args) { const toggle = args.TOGGLE === "on" ? "solid" : "none"; const elements = document.querySelectorAll(`div[id^="SP_Text-Ext-"]`); elements.forEach((element) => { - element.style.border = toggle; - element.style.borderWidth = "1px"; - const color = element.style.textAlign === "center" ? "#00ff00" : element.style.textAlign === "right" ? "#0000ff" : "#ff0000"; - element.style.borderColor = color; + element.style.border = toggle; element.style.borderWidth = "1px"; + element.style.borderColor = element.style.textAlign === "center" ? "#00ff00" : element.style.textAlign === "right" ? "blue" : "red"; }); const cross = document.getElementById(`SP_Text-Debug-Cross`); if (toggle === "none" && cross) cross.parentNode.removeChild(cross); @@ -480,35 +521,22 @@ } printTxt(args) { - args.ID = this.fixID(args.ID); + const ID = this.fixID(args.ID); + const settings = txtSettings[ID]; const textDiv = document.createElement("div"); textDiv.style.transformOrigin = "left top"; const txtElement = document.createElement("div"); txtElement.innerHTML = xmlEscape(args.TXT).replace(/\n/g, "
"); - txtElement.id = `SP_Text-Ext-${args.ID}`; - txtElement.classList.add(args.ID); + txtElement.id = `SP_Text-Ext-${ID}`; + txtElement.classList.add(ID); + txtElement.style.userSelect = "none"; txtElement.setAttribute("sptxtpos", "120|-10"); textDiv.appendChild(txtElement); render.addOverlay(textDiv, "scale-centered"); - allText.push(`#SP_Text-Ext-${args.ID}`); + allText.push(`#SP_Text-Ext-${ID}`); const box = txtElement.getBoundingClientRect(); - if (lastRecdVals.textMAR === undefined) this.setMargins({ ID : args.ID, WIDTH : box.width / 2, HEIGHT : box.height }); - - // add formatting (if any) - const propertiesAndMethods = [ - ["textMAR", "setMargins"], ["preTxt1", "presetTextPosition"], - ["textCLR", "setTextColor"], ["textLIN", "setLine"], - ["textOUT", "setTextOutline"], ["textSHA", "setTextDropShadow"], - ["txtFont", "setTextFont"], ["txtFontSZ", "setFontSize"], - ["txtALI", "setTextAlignment"], ["lineDIS", "setTextSpacing"], - ["letDIS", "setTextSpacing"], ["textOVR", "setOverflow"], - ["txtFontTK", "setThick"] - ]; - for (const [property, method] of propertiesAndMethods) { - if (lastRecdVals[property] && lastRecdVals[property].inputs.ID === args.ID) { - this[method](lastRecdVals[property].inputs); - } - } + if (settings.textMAR === undefined) this.setMargins({ ID, WIDTH : box.width / 2, HEIGHT : box.height }); + this.updateStyles(settings); // add formatting (if any) } replaceTxt(args) { @@ -518,27 +546,21 @@ } removeTxt(args) { - args.ID = this.fixID(args.ID); - const elementsToRemove = document.querySelectorAll(`div[id="SP_Text-Ext-${args.ID}"]`); + const ID = this.fixID(args.ID); + const elementsToRemove = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); elementsToRemove.forEach((element) => { render.removeOverlay(element.parentNode); element.removeEventListener("click", this.handleClick); - const index = allText.indexOf(`#SP_Text-Ext-${args.ID}`); + const index = allText.indexOf(`#SP_Text-Ext-${ID}`); if (index !== -1) allText.splice(index, 1); }); } - resetTxt(args) { - Object.keys(lastRecdVals).forEach(key => { - if (lastRecdVals[key].inputs.ID === args.ID) delete lastRecdVals[key]; - }); - } - removeAllTxt() { for (let i = 0; i < allText.length; i++) { const elementsToRemove = document.querySelectorAll(allText[i]); elementsToRemove.forEach((element) => { - render.removeOverlay(element.parentNode) + render.removeOverlay(element.parentNode); element.removeEventListener("click", this.handleClick); }); } @@ -568,7 +590,8 @@ makeGradient(args) { return `${args.TYPE}-gradient(${args.ANGLE}deg, ${args.COLOR1}, ${args.COLOR2})` } setTextColor(args) { - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const ID = this.fixID(args.ID); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); elements.forEach((element) => { element.style.color = args.COLOR; if (args.COLOR.includes("gradient")) { @@ -583,25 +606,28 @@ element.style.webkitBackgroundClip = "initial"; } }); - lastRecdVals["textCLR"] = {inputs: args}; + txtSettings[ID]["textCLR"] = { ...args }; } setTextDropShadow(args) { - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const ID = this.fixID(args.ID); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); elements.forEach((element) => { element.style.textShadow = args.z === 0 ? "none" : `${args.x}px ${args.y * -1}px ${args.z}px ${args.COLOR}`; }); - lastRecdVals["textSHA"] = {inputs: args}; + txtSettings[ID]["textSHA"] = { ...args }; } setOverflow(args) { - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const ID = this.fixID(args.ID); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); elements.forEach((element) => { element.style.overflow = args.TYPE }); - lastRecdVals["textOVR"] = {inputs: args}; + txtSettings[ID]["textOVR"] = { ...args }; } setTextOutline(args) { - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const ID = this.fixID(args.ID); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); elements.forEach((element) => { element.style.webkitTextStrokeColor = args.COLOR; element.style.webkitTextStrokeWidth = `${args.THICKNESS}px`; @@ -611,68 +637,77 @@ element.style.mozTextStrokeColor = args.COLOR; element.style.mozTextStrokeWidth = `${args.THICKNESS}px`; }); - lastRecdVals["textOUT"] = {inputs: args}; + txtSettings[ID]["textOUT"] = { ...args }; } setLine(args) { + const ID = this.fixID(args.ID); const lineType = args.TYPE2.replace("strike", "line-"); - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); elements.forEach((element) => { element.style.textDecorationLine = lineType; element.style.textDecorationStyle = args.TYPE1; element.style.textDecorationThickness = `${args.THICK}px`; element.style.textDecorationColor = args.COLOR; }); - lastRecdVals["textLIN"] = {inputs: args}; + txtSettings[ID]["textLIN"] = { ...args }; } setMargins(args) { - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const ID = this.fixID(args.ID); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); elements.forEach((element) => { element.style.width = `${args.WIDTH}px`; element.style.height = `${args.HEIGHT}px`; }); - lastRecdVals["textMAR"] = {inputs: args}; + txtSettings[ID]["textMAR"] = { ...args }; } setTextAlignment(args) { - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const ID = this.fixID(args.ID); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); elements.forEach((element) => { element.style.textAlign = args.ALIGNMENT }); - lastRecdVals["txtALI"] = {inputs: args}; + txtSettings[ID]["txtALI"] = { ...args }; } setTextFont(args) { - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const ID = this.fixID(args.ID); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); elements.forEach((element) => { element.style.fontFamily = args.FONT }); - lastRecdVals["txtFont"] = {inputs: args}; + txtSettings[ID]["txtFont"] = { ...args }; } setFontSize(args) { - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const ID = this.fixID(args.ID); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); elements.forEach((element) => { element.style.fontSize = `${args.SIZE}px` }); - lastRecdVals["txtFontSZ"] = {inputs: args}; + txtSettings[ID]["txtFontSZ"] = { ...args }; } setThick(args) { - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const ID = this.fixID(args.ID); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); elements.forEach((element) => { element.style.fontWeight = args.NUM * 9 }); - lastRecdVals["txtFontTK"] = {inputs: args}; + txtSettings[ID]["txtFontTK"] = { ...args }; } setTextSpacing(args) { - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const ID = this.fixID(args.ID); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); elements.forEach((element) => { element.style[args.ATT === "letter" ? "letterSpacing" : "lineHeight"] = `${args.SPACING}px`; }); - lastRecdVals["letDIS"] = {inputs: {ID : args.ID, SPACING : args.SPACING, ATT : "letter"}}; - lastRecdVals["lineDIS"] = {inputs: {ID : args.ID, SPACING : args.SPACING, ATT : "line"}}; + txtSettings[ID]["letDIS"] = { ID, SPACING : args.SPACING, ATT : "letter" }; + txtSettings[ID]["lineDIS"] = { ID, SPACING : args.SPACING, ATT : "line" }; } presetTextPosition(args) { - if (args.isPrint === undefined) lastRecdVals["preTxt1"] = {inputs: {...args, isPrint: true}}; + const ID = this.fixID(args.ID); + if (args.isPrint === undefined) txtSettings[ID]["preTxt1"] = { ...args, isPrint: true }; else { + const pos = [Scratch.Cast.toNumber(args.X), Scratch.Cast.toNumber(args.Y)]; runtime.once("AFTER_EXECUTE", () => { - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`); const element = elements[elements.length - 1]; const centerW = parseFloat(element.style.width) / 2; const computedStyle = window.getComputedStyle(element); @@ -681,18 +716,19 @@ const centerH = (lineHeight === "normal" ? parseFloat(fontSize) * 1.2 : parseFloat(lineHeight)) / 2; let transform = element.style.transform; - const string = `translate(${args.X - centerW}px, ${(args.Y * -1) - centerH}px)`; + const string = `translate(${pos[0] - centerW}px, ${(pos[1] * -1) - centerH}px)`; if (transform.includes("translate")) transform = transform.replace(/translate\([^)]*\)/, string); else transform += ` ${string}`; element.style.transform = transform.trim(); element.style.position = "absolute"; - element.setAttribute("sptxtpos", `${args.X}|${args.Y}`); + element.setAttribute("sptxtpos", `${pos[0]}|${pos[1]}`); }); } } setTextPosition(args) { const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + const pos = [Scratch.Cast.toNumber(args.X), Scratch.Cast.toNumber(args.Y)]; elements.forEach((element) => { let centerW = parseFloat(element.style.width) / 2; if (isNaN(centerW)) centerW = (element.getBoundingClientRect().width ?? 0 ) / 2; @@ -702,12 +738,12 @@ const centerH = (lineHeight === "normal" ? parseFloat(fontSize) * 1.2 : parseFloat(lineHeight)) / 2; let transform = element.style.transform; - const string = `translate(${args.X - centerW}px, ${(args.Y * -1) - centerH}px)`; + const string = `translate(${pos[0] - centerW}px, ${(pos[1] * -1) - centerH}px)`; if (transform.includes("translate")) transform = transform.replace(/translate\([^)]*\)/, string); else transform += ` ${string}`; element.style.transform = transform.trim(); element.style.position = "absolute"; - element.setAttribute("sptxtpos", `${args.X}|${args.Y}`); + element.setAttribute("sptxtpos", `${pos[0]}|${pos[1]}`); }); } @@ -836,21 +872,24 @@ } setTextCurve(args) { - args.ID = this.fixID(args.ID) + // TODO clean this up and add more availiable text stylings + // Perhaps make the entire extension rely on svgs? + const ID = this.fixID(args.ID); + const settings = txtSettings[ID]; const regex = args.ARC.includes("]*d="([^"]*)"/ : /]*d="([^"]*)"/; const match = args.ARC.match(regex); - const outline = lastRecdVals["textOUT"] !== undefined ? lastRecdVals["textOUT"] : ""; + const outline = settings["textOUT"] !== undefined ? settings["textOUT"] : ""; if (match && match[1]) { - const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${args.ID}"]`) + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${ID}"]`) elements.forEach((element) => { const existingSvg = element.querySelector("svg"); if (existingSvg) { const path = existingSvg.querySelector("path"); path.setAttribute("d", match[1]); const textFill = existingSvg.querySelector("text"); - textFill.setAttribute("fill", element.style.color === "transparent" ? "#000000" : element.style.color); + textFill.setAttribute("fill", element.style.color === "transparent" ? "#000" : element.style.color); const textPathFill = existingSvg.querySelector("textPath"); - textPathFill.setAttribute("href", `#MyPath-${args.ID}`); + textPathFill.setAttribute("href", `#MyPath-${ID}`); textPathFill.textContent = element.textContent; const textStroke = existingSvg.querySelector("text"); @@ -858,13 +897,13 @@ textStroke.setAttribute("stroke", outline ? outline.inputs.COLOR ?? "#00000000" : "#00000000"); textStroke.setAttribute("stroke-width", outline ? outline.inputs.THICKNESS ?? 1 : 1); const textPathStroke = existingSvg.querySelector("textPath"); - textPathStroke.setAttribute("href", `#MyPath-${args.ID}`); + textPathStroke.setAttribute("href", `#MyPath-${ID}`); textPathStroke.textContent = element.textContent; } else { const svg = document.createElementNS("http://www.w3.org/2000/svg", "svg"); const defs = document.createElementNS("http://www.w3.org/2000/svg", "defs"); const path = document.createElementNS("http://www.w3.org/2000/svg", "path"); - path.setAttribute("id", `MyPath-${args.ID}`); + path.setAttribute("id", `MyPath-${ID}`); path.setAttribute("d", match[1]); defs.appendChild(path); @@ -873,16 +912,16 @@ textStroke.setAttribute("stroke", outline ? outline.inputs.COLOR : "#00000000"); textStroke.setAttribute("stroke-width", outline ? outline.inputs.THICKNESS : 1); const textPathStroke = document.createElementNS("http://www.w3.org/2000/svg", "textPath"); - textPathStroke.setAttribute("href", `#MyPath-${args.ID}`); + textPathStroke.setAttribute("href", `#MyPath-${ID}`); textPathStroke.textContent = element.textContent; textStroke.appendChild(textPathStroke); svg.appendChild(defs); svg.appendChild(textStroke); const textFill = document.createElementNS("http://www.w3.org/2000/svg", "text"); - textFill.setAttribute("fill", element.style.color === "transparent" ? "#000000" : element.style.color); + textFill.setAttribute("fill", element.style.color === "transparent" ? "#000" : element.style.color); const textPathFill = document.createElementNS("http://www.w3.org/2000/svg", "textPath"); - textPathFill.setAttribute("href", `#MyPath-${args.ID}`); + textPathFill.setAttribute("href", `#MyPath-${ID}`); textPathFill.textContent = element.textContent; textFill.appendChild(textPathFill); svg.appendChild(textFill); @@ -894,13 +933,28 @@ } presetCurve(args) { - const presets = { - circle : ``, - wave : `` - }; - return presets[args.ARC]; + return { + circle: ``, + hill: ``, + dip: ``, + wave: `` + }[args.ARC]; + } + + resetTxt(args) { delete txtSettings[this.fixID(args.ID)] } + + reuseStyle(args) { + const ID = this.fixID(args.ID2); + this.updateStyles(txtSettings[this.fixID(args.ID)], this.fixID(args.ID2)); + } + + toggleSelect(args) { + const type = args.TYPE === "on" ? "auto" : "none"; + const elements = document.querySelectorAll(`div[id="SP_Text-Ext-${this.fixID(args.ID)}"]`); + elements.forEach((element) => { + element.style.userSelect = type; element.style.webkitUserSelect = type; element.style.mozUserSelect = type; + element.style.pointerEvents = type; + }); } makeClick(args) { From 06605367ebdd614b099fa0e39ee6e87c09bd1322 Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Fri, 30 Aug 2024 23:41:06 -0700 Subject: [PATCH 25/25] Display-Text -- V1.4.4 --- extensions/SharkPool/Display-Text.js | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/extensions/SharkPool/Display-Text.js b/extensions/SharkPool/Display-Text.js index 7c14f14568..5a9e188115 100644 --- a/extensions/SharkPool/Display-Text.js +++ b/extensions/SharkPool/Display-Text.js @@ -697,8 +697,7 @@ elements.forEach((element) => { element.style[args.ATT === "letter" ? "letterSpacing" : "lineHeight"] = `${args.SPACING}px`; }); - txtSettings[ID]["letDIS"] = { ID, SPACING : args.SPACING, ATT : "letter" }; - txtSettings[ID]["lineDIS"] = { ID, SPACING : args.SPACING, ATT : "line" }; + txtSettings[ID][args.ATT === "letter" ? "letDIS" : "lineDIS"] = { ...args, ID }; } presetTextPosition(args) { @@ -944,7 +943,6 @@ resetTxt(args) { delete txtSettings[this.fixID(args.ID)] } reuseStyle(args) { - const ID = this.fixID(args.ID2); this.updateStyles(txtSettings[this.fixID(args.ID)], this.fixID(args.ID2)); }