diff --git a/static/extensions/LordCat0/ProjectInterfaces.js b/static/extensions/LordCat0/ProjectInterfaces.js index 8f49a900f..4c91ab164 100644 --- a/static/extensions/LordCat0/ProjectInterfaces.js +++ b/static/extensions/LordCat0/ProjectInterfaces.js @@ -3,24 +3,41 @@ // Description: Easily create more interactive projects! // By: LordCat0 // Licence: MIT -(function(Scratch){ - if(!Scratch.extensions.unsandboxed){alert("This extension must run unsandboxed!"); return} - const lookup = {Label: "span", Video: "video", Image: "img", Input: "input", Box: "div", Button: "button"} - const extIcon = '' - const textIcon = '' - const imageIcon = '' - const videoIcon = '' - const buttonicon = '' - const inputIcon = '' - const vm = Scratch.vm - const elementbox = document.createElement('div') - elementbox.classList.add('LordCatInterfaces') - vm.renderer.addOverlay(elementbox, "scale") - let elements = {} - let metadata = {} - let inputhold = {} - let lastValues = {} - const css = document.createElement('style') + +(Scratch => { + if (!Scratch.extensions.unsandboxed) { + alert("This extension must run unsandboxed!"); + return; + } + const lookup = { + Label: "span", + Video: "video", + Image: "img", + Input: "input", + Box: "div", + Button: "button", + }; + const extIcon = + ""; + const textIcon = + ""; + const imageIcon = + ""; + const videoIcon = + ""; + const buttonicon = + ""; + const inputIcon = + ""; + const vm = Scratch.vm; + const elementbox = document.createElement("div"); + elementbox.classList.add("LordCatInterfaces"); + vm.renderer.addOverlay(elementbox, "scale"); + let elements = {}; + let metadata = {}; + let inputhold = {}; + let lastValues = {}; + const css = document.createElement("style"); css.textContent = ` .LordCatInterfaces svg{ vertical-align: top; @@ -28,577 +45,1244 @@ .LordCatInterfaces[hidden]{ display: none } - ` - css.classList.add('LordCatInterfaces-Style') - document.head.append(css) - + `; + css.classList.add("LordCatInterfaces-Style"); + document.head.append(css); + + const textDecoder = new TextDecoder("utf-8"); + const domParser = new DOMParser(); + const datauri = (file) => { return new Promise((resolve, reject) => { - if(!(file instanceof File)) resolve('') - const reader = new FileReader() - reader.readAsDataURL(file) - reader.onload = () => resolve(reader.result) - reader.onerror = () => reject(reader.error) - }) - } + if (!(file instanceof File)) resolve(""); + const reader = new FileReader(); + reader.readAsDataURL(file); + reader.onload = () => resolve(reader.result); + reader.onerror = () => reject(reader.error); + }); + }; + const datauriFromCostume = (costume, target) => { - let costumeIndex = target.getCostumeIndexByName(costume) - if(costumeIndex === -1){ - switch(costume){ - case 'next costume': - costumeIndex = target.currentCostume===target.sprite.costumes_.length-1?0:(target.currentCostume+1) - break - case 'previous costume': - costumeIndex = target.currentCostume===0?(target.sprite.costumes_.length-1):(target.currentCostume-1) - break - case 'random costume': - costumeIndex = Math.floor(Math.random() * target.sprite.costumes_.length) - break - } - } - return target.sprite.costumes[costumeIndex].asset.encodeDataURI() - } + let costumeIndex = target.getCostumeIndexByName(costume); + if (costumeIndex === -1) { + switch (costume) { + case "next costume": + costumeIndex = + target.currentCostume === + target.sprite.costumes_.length - 1 + ? 0 + : target.currentCostume + 1; + break; + case "previous costume": + costumeIndex = + target.currentCostume === 0 + ? target.sprite.costumes_.length - 1 + : target.currentCostume - 1; + break; + case "random costume": + costumeIndex = Math.floor( + Math.random() * target.sprite.costumes_.length, + ); + break; + } + } + return target.sprite.costumes[costumeIndex].asset.encodeDataURI(); + }; + const replaceElement = (oldElement, newElement, id) => { - newElement.dataset.id = id - newElement.setAttribute("style", oldElement.getAttribute("style")) - newElement.addEventListener("mouseover", () => metadata[id].hovered = true) - newElement.addEventListener("mouseout", () => metadata[id].hovered = false) - newElement.addEventListener("click", () => metadata[id].clicked = true) - if(oldElement.tagName==="INPUT" || oldElement.tagName==="TEXTAREA"){ - newElement.value = oldElement.value - newElement.addEventListener('input', () => metadata[id].inputdirty = true) - } - oldElement.replaceWith(newElement) - return newElement - } - const fonts = [] - document.fonts.ready.then(() => {document.fonts.forEach(font => fonts.push(font.family));}); - class lordcatprojectinterfaces{ - getInfo(){return{ - id: "lordcatprojectinterfaces", - name: "Project interfaces", - color1: "#707eff", - color2: "#6675fa", - docsURI: "https://extensions.penguinmod.com/docs/ProjectInterfaces", - menuIconURI: extIcon, - blocks: [{ - opcode: "ClearAll", - text: "Clear all elements", - blockType: Scratch.BlockType.COMMAND - },{ - opcode: "Create", - text: "Create [type] element with ID [id]", - blockType: Scratch.BlockType.COMMAND, - arguments: {type: {type: Scratch.ArgumentType.STRING, menu: 'ElementType'}, id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}} - },{ - opcode: "Delete", - text: "Delete element with ID [id]", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}} - },{ - opcode: "Visibility", - text: "[menu] element with ID [id]", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}, menu: {type: Scratch.ArgumentType.STRING, menu: "Visibility"}} - },{ - opcode: "ElementVisibility", - text: "element with ID [id] is [status]", - blockType: Scratch.BlockType.BOOLEAN, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}, status: {type: Scratch.ArgumentType.STRING, menu: "VisibilityStatus"}} - },{ - opcode: "AllElements", - text: "All elements", - blockType: Scratch.BlockType.REPORTER - },{blockType: Scratch.BlockType.LABEL, text: "Styling"},{ - opcode: "Position", - text: "Set position of ID [id] to x: [x] y: [y]", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}, x: {type: Scratch.ArgumentType.NUMBER, defaultValue: 0}, y: {type: Scratch.ArgumentType.NUMBER, defaultValue: 0}} - },{ - opcode: "Direction", - text: "Set direction of ID [id] to [dir]", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}, dir: {type: Scratch.ArgumentType.NUMBER, defaultValue: 90}} - },{ - opcode: "Scale", - text: "Set scale of ID [id] to width: [width]px height: [height]px", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}, width: {type: Scratch.ArgumentType.NUMBER, defaultValue: 100}, height: {type: Scratch.ArgumentType.NUMBER, defaultValue: 100}} - },{ - opcode: "Layer", - text: "Set layer of ID [id] to [layer]", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}, layer: {type: Scratch.ArgumentType.NUMBER, defaultValue: 1}} - },{ - opcode: "Cursor", - text: "Set hover cursor of ID [id] to [cursor]", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}, cursor: {type: Scratch.ArgumentType.STRING, menu: 'Cursors'}} - },{ - opcode: "Color", - text: "Set color of ID [id] to [color]", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}, color: {type: Scratch.ArgumentType.COLOR}} - },{ - opcode: "BackgroundColor", - text: "Set background color of ID [id] to [color]", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}, color: {type: Scratch.ArgumentType.COLOR}} - },{ - opcode: "CustomCSS", - text: "Set custom CSS of [id] to [css]", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}, css: {type: Scratch.ArgumentType.STRING, defaultValue: 'background-color: red'}} - },{ - opcode: "HtmlElement", - text: "Create html element [htmltag] with ID [id]", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}, htmltag: {type: Scratch.ArgumentType.STRING, defaultValue: 'h1'}} - },"---",{ - opcode: "WhenClicked", - text: "When ID [id] is clicked", - blockType: Scratch.BlockType.HAT, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}} - },{ - opcode: "Attribute", - text: "[attr] of ID [id]", - blockType: Scratch.BlockType.REPORTER, - arguments: {attr: {type: Scratch.ArgumentType.STRING, menu: "Attributes"}, id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}} - },{ - opcode: "IsHovered", - text: "[id] hovered?", - blockType: Scratch.BlockType.BOOLEAN, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}} - },{ - blockType: Scratch.BlockType.LABEL,text: "Labels"},{ - opcode: "LabelText", - text: "Set label text with ID [id] to [text]", - arguments: {text: {type: Scratch.ArgumentType.STRING, defaultValue: "Hello world!"}, id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}}, - blockIconURI: textIcon - },{ - opcode: "LabelAlign", - text: "Set label alignment with ID [id] to [align]", - arguments: {align: {type: Scratch.ArgumentType.STRING, menu: "Alignment"}, id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}}, - blockIconURI: textIcon - },{ - opcode: "LabelFontSize", - text: "Set label font size with ID [id] to [size]px", - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}, size: {type: Scratch.ArgumentType.NUMBER, defaultValue: 40}}, - blockIconURI: textIcon - },{ - opcode: "LabelFont", - text: "Set label font with ID [id] to [font]", - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}, font: {type: Scratch.ArgumentType.STRING, menu: 'Fonts'}}, - blockIconURI: textIcon - },{blockType: Scratch.BlockType.LABEL, text: "Images" - },{ - opcode: "ImageUrl", - text: "Set image with ID [id] to url [url]", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}, url: {type: Scratch.ArgumentType.STRING, defaultValue: 'https://extensions.turbowarp.org/dango.png'}}, - blockIconURI: imageIcon - },{ - opcode: "ImageCostume", - text: "Set image with ID [id] to costume [costume]", - blockType: Scratch.BlockType.COMMAND, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}, costume: {type: Scratch.ArgumentType.COSTUME}}, - blockIconURI: imageIcon - },{blockType: Scratch.BlockType.LABEL,text: "Videos" - },{ - opcode: "VideoSource", - text: "Set video with ID [id] to url [url]", - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}, url: {type: Scratch.ArgumentType.STRING, defaultValue: 'https://extensions.turbowarp.org/dango.png'}}, - blockIconURI: videoIcon - },{ - opcode: "VideoControl", - text: "[control] video with ID [id]", - arguments: {control: {type: Scratch.ArgumentType.STRING, menu: 'VideoControls'}, id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}}, - blockIconURI: videoIcon - },{ - opcode: "VideoVolume", - text: "Set volume of video [id] to [volume]%", - arguments: {volume: {type: Scratch.ArgumentType.NUMBER, defaultValue: 100}, id: {type: Scratch.ArgumentType.STRING, defaultValue: 'My element'}}, - blockIconURI: videoIcon - },{ - opcode: "VideoLoop", - text: "Set loop of video [id] to [toggle]", - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}, toggle: {type: Scratch.ArgumentType.STRING, menu: "EnableDisable"}}, - blockIconURI: videoIcon - },{ - opcode: "VideoHtmlControls", - text: "Set video controls of ID [id] to [toggle]", - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}, toggle: {type: Scratch.ArgumentType.STRING, menu: "EnableDisable"}}, - blockIconURI: videoIcon - },{blockType: Scratch.BlockType.LABEL, text: "Inputs" - },{ - opcode: "InputType", - text: "Set input type of ID [id] to [input]", - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}, input: {type: Scratch.ArgumentType.STRING, menu: 'Inputs'}}, - blockIconURI: inputIcon - },{ - opcode: "InputAccent", - text: "Set input accent color of ID [id] to [color]", - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}, color: {type: Scratch.ArgumentType.COLOR}}, - blockIconURI: inputIcon - },{ - opcode: "InputPlaceholder", - text: "Set placeholder of ID [id] to [placeholder]", - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}, placeholder: {type: Scratch.ArgumentType.STRING, defaultValue: "Hello world!"}}, - blockIconURI: inputIcon - },{ - opcode: "InputSetValue", - text: "Set value of ID [id] to [value]", - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}, value: {type: Scratch.ArgumentType.STRING, defaultValue: "Hello world!"}}, - blockIconURI: inputIcon - },{ - opcode: "InputValue", - text: "Value of input with ID [id]", - blockType: Scratch.BlockType.REPORTER, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}}, - blockIconURI: inputIcon - },{ - opcode: "WhenInputChanged", - text: "When input with ID [id] changed", - blockType: Scratch.BlockType.HAT, - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}}, - blockIconURI: inputIcon - },{ - blockType: Scratch.BlockType.LABEL, - text: "Buttons" - },{ - opcode: "ButtonText", - text: "Set text of button [id] to [text]", - arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}, text: {type: Scratch.ArgumentType.STRING, defaultValue: "Hello world!"}}, - blockIconURI: buttonicon - } - ], - menus: {ElementType: {acceptReporters: false,items: ["Label", "Image", "Video", "Input", "Box", "Button"]}, - Inputs: {acceptReporters: false,items: ["Text", "Text Area", "Number", "Color", "Checkbox", "File", "Email", "Range", "Image"]}, - Cursors: {acceptReporters: false,items: ["default", "pointer", "text", "wait", "move", "not-allowed", "crosshair","help", "progress", "grab", "grabbing"]}, - Fonts: {acceptReporters: true, items: fonts}, - Attributes: {acceptReporters: false, items: ['X', "Y", "Direction", "Width", "Height", "Cursor", "Source"]}, - VideoControls: {acceptReporters: false, items: ["Play", "Stop", "Pause"]}, - EnableDisable: {acceptReporters: false, items: ["Enabled", "Disabled"]}, - Alignment: {acceptReporters: false, items: ["Left", "Right", "Center"]}, - Visibility: {acceptReporters: false, items: ["Show", "Hide"]}, - VisibilityStatus: {acceptReporters: false, items: ["Shown", "Hidden"]}}, - }} - tutorial(){window.open("https://discord.com/channels/1033551490331197462/1390741725705797642/1390741725705797642")} - FixPos(elementid){ - setTimeout(() => { - if(!elements[elementid]){return} - this.Position({id: elementid, x: metadata[elementid].x, y: metadata[elementid].y}) - }, 1) // Timeout needed because for some reason it wont run otherwise.. - } - FixTransform(elementid){ - setTimeout(() => { - elements[elementid].style.transform = elements[elementid].tagName==='SVG'?`translate(-50%, -50%) rotate(${metadata[elementid]-90}deg)`:`rotate(${metadata[elementid]-90}deg)` - }, 1) // Timeout needed because for some reason it wont run otherwise.. - } - ClearAll(){ - Object.entries(elements).forEach(([id, element]) => { - if((element.tagName==="INPUT"||element.tagName==="TEXTAREA")) inputhold[id] = element.type==="checkbox"?element.checked:element.value - }) - elements = {} - metadata = {} - elementbox.innerHTML = '' - } - Create(args){ - if(elements[args.id]) return - const element = document.createElement(lookup[args.type]) - if(lookup[args.type] === 'button'){ - element.append(document.createElement('span')) - element.append(document.createElement('img')) - } - const boundingRect = element.getBoundingClientRect() - element.dataset.id = args.id - element.style.position = 'absolute' - element.style.pointerEvents = 'auto' - element.style.userSelect = 'none' - element.style.color = 'black' - if(args.type == 'Image'){element.draggable = false} - elements[args.id] = element - elementbox.append(element) - metadata[args.id] = {x: 0, y: 0, direction: 90, width: boundingRect.width, height: boundingRect.height, hovered: false, clicked: false} - this.FixPos(args.id) - if(args.type === 'Input'){ - metadata[args.id].inputdirty = false - element.addEventListener("input", () => metadata[args.id].inputdirty = true) - } - element.addEventListener("mouseover", () => metadata[args.id].hovered = true) - element.addEventListener("mouseout", () => metadata[args.id].hovered = false) - element.addEventListener("click", () => metadata[args.id].clicked = true) - } - Position(args){ - if(!elements[args.id]){return} - const element = elements[args.id] - if(element.tagName==='svg'){ - const bbox = element.getBBox() - element.style.left = `${(vm.runtime.stageWidth/2) + args.x - (bbox.width/2)}px` - element.style.top = `${(vm.runtime.stageHeight/2) - args.y - (bbox.height/2)}px` - }else{ - element.style.left = `${(vm.runtime.stageWidth/2) + args.x - (element.offsetWidth/2)}px` - element.style.top = `${(vm.runtime.stageHeight/2) - args.y - (element.offsetHeight/2)}px` - } - metadata[args.id].x = args.x - metadata[args.id].y = args.y - } - Direction(args){ - if(!elements[args.id]){return} - const element = elements[args.id] - metadata[args.id].direction = args.dir - element.style.transform = `rotate(${args.dir - 90}deg)` - } - Scale(args){ - if(!elements[args.id]){return} - const element = elements[args.id] - element.style.width = `${args.width}px` - element.style.height = `${args.height}px` - metadata[args.id].width = args.width + 'px' - metadata[args.id].height = args.height + 'px' - element.style.objectFit = 'fill' - this.FixPos(args.id) - } - Layer(args){ - if(!elements[args.id]){return} - const element = elements[args.id] - element.style.zIndex = args.layer - } - Cursor(args){ - if(!elements[args.id]){return} - const element = elements[args.id] - element.style.cursor = args.cursor - } - Color(args){ - if(!elements[args.id]){return} - if(elements[args.id].tagName == 'DIV'){elements[args.id].style.backgroundColor = args.color; return} // for people who are confused why 'color' doesnt work with div/box elements - elements[args.id].style.color = args.color - } - BackgroundColor(args){ - if(!elements[args.id]){return} - elements[args.id].style.backgroundColor = args.color - } - CustomCSS(args){ - if(!elements[args.id]){return} - const element = elements[args.id] - let style = document.getElementById(`LCGuiStyle_${args.id}`) - let lines = args.css.split(";") - if(!style){ - style = document.createElement('style') - style.id = `LCGuiStyle_${args.id}` - document.head.append(style) - } - style.textContent = `[data-id='${args.id}']{\n${lines.join(" !important;\n") + " !important"}\n}` - } - HtmlElement(args){ - if(elements[args.id]) return - const element = document.createElement(args.htmltag.toLowerCase()) - const boundingRect = element.getBoundingClientRect() - element.dataset.id = args.id - element.style.position = 'absolute' - element.style.pointerEvents = 'auto' - element.style.userSelect = 'none' - element.style.color = 'black' - elements[args.id] = element - elementbox.append(element) - metadata[args.id] = {x: 0, y: 0, direction: 90, width: boundingRect.width, height: boundingRect.height, hovered: false, clicked: false} - this.FixPos(args.id) - element.addEventListener("mouseover", () => metadata[args.id].hovered = true) - element.addEventListener("mouseout", () => metadata[args.id].hovered = false) - element.addEventListener("click", () => metadata[args.id].clicked = true) - } - Attribute(args){ - const element = elements[args.id] - if(!element) return - const meta = metadata[args.id] - switch(args.attr){ - case 'Cursor': - return element.style.cursor===""?"default":element.style.cursor - case 'Source': - if(element.tagName != 'IMG' && element.tagName != 'VIDEO' && element.tagName != "svg") return - return element.tagName==='svg'?element.outerHTML:element.src - case 'Width': - return element.getBBox().width - case 'Height': - return element.getBBox().height - default: - return meta[args.attr.toLowerCase()] + newElement.dataset.id = id; + newElement.setAttribute("style", oldElement.getAttribute("style")); + newElement.addEventListener( + "mouseover", + () => (metadata[id].hovered = true), + ); + newElement.addEventListener( + "mouseout", + () => (metadata[id].hovered = false), + ); + newElement.addEventListener( + "click", + () => (metadata[id].clicked = true), + ); + if ( + oldElement.tagName === "INPUT" || + oldElement.tagName === "TEXTAREA" + ) { + newElement.value = oldElement.value; + newElement.addEventListener( + "input", + () => (metadata[id].inputdirty = true), + ); } - } - IsHovered(args){ - if(!elements[args.id]){return ''} - return metadata[args.id].hovered - } - Delete(args){ - const element = elements[args.id] - if(!element){return} - if((element.tagName==="INPUT"||element.tagName==="TEXTAREA")) inputhold[args.id] = element.type==="checkbox"?element.checked:element.value - if(document.getElementById(`LCGuiStyle_${args.id}`)) document.getElementById(`LCGuiStyle_${args.id}`).remove() - element.remove() - delete elements[args.id] - delete metadata[args.id] - } - Visibility(args){ - if(!elements[args.id]){return} - elements[args.id].hidden = (args.menu==="Hide") - } - ElementVisibility(args){ - if(!elements[args.id]) return - return args.status==='Shown'?!elements[args.id].hidden:!!elements[args.id].hidden - } - AllElements(){return JSON.stringify(Object.keys(elements))} - LabelText(args){ - if(!elements[args.id] || elements[args.id].tagName != "SPAN"){return} - elements[args.id].textContent = args.text - this.FixPos(args.id) - } - LabelAlign(args){ - if(!elements[args.id] || elements[args.id].tagName != "SPAN"){return} - elements[args.id].style.textAlign = args.align.toLowerCase() - this.FixPos(args.id) - } - LabelFontSize(args){ - if(!elements[args.id] || elements[args.id].tagName != "SPAN"){return} - elements[args.id].style.fontSize = `${args.size}px` - this.FixPos(args.id) - } - LabelFont(args){ - if(!elements[args.id] || elements[args.id].tagName != "SPAN"){return} - elements[args.id].style.fontFamily = args.font - this.FixPos(args.id) - } - ImageUrl(args){ - if(!elements[args.id] || (elements[args.id].tagName != "IMG" && elements[args.id].tagName != "svg")){return} - if(elements[args.id].tagName==='svg') elements[args.id] = replaceElement(elements[args.id], document.createElement('img'), args.id) - elements[args.id].src = args.url - this.FixPos(args.id) - } - ImageCostume(args, util){ - if(!elements[args.id] || (elements[args.id].tagName != "IMG" && elements[args.id].tagName != "svg")){return} - if(util.target.getCostumes().find(costume => costume.name === args.costume).dataFormat === "svg"){ - const decoder = new TextDecoder('utf-8') - const htmlparser = new DOMParser() - elements[args.id] = replaceElement(elements[args.id], - htmlparser.parseFromString(decoder.decode(util.target.getCostumes().find(costume => costume.name === args.costume).asset.data), 'image/svg+xml') - .documentElement, args.id) //damn - }else{ - if(elements[args.id].tagName==='svg') elements[args.id] = replaceElement(elements[args.id], document.createElement('img'), args.id) - elements[args.id].src = datauriFromCostume(args.costume, util.target) - } - this.FixPos(args.id) - this.FixTransform(args.id) - } - InputType(args){ - if(!elements[args.id] || (elements[args.id].tagName != "INPUT" && elements[args.id].tagName != "TEXTAREA")){return} - if(args.input === 'Text Area'){ - elements[args.id] = replaceElement(elements[args.id], document.createElement('textarea'), args.id) - elements[args.id].style.resize = 'none' - }else{ - if(elements[args.id].tagName == "TEXTAREA") elements[args.id] = replaceElement(elements[args.id], document.createElement('input'), args.id) - elements[args.id].type = args.input - if(args.input == 'File'){elements[args.id].value = null} - } - this.FixPos(args.id) - } - InputAccent(args){ - if(!elements[args.id] || (elements[args.id].tagName != "INPUT" && elements[args.id].tagName != "TEXTAREA")){return} - elements[args.id].style.accentColor = args.color - } - InputPlaceholder(args){ - if(!elements[args.id] || (elements[args.id].tagName != "INPUT" && elements[args.id].tagName != "TEXTAREA")){return} - elements[args.id].setAttribute('placeholder', args.placeholder) - } - InputSetValue(args){ - if(!elements[args.id] || (elements[args.id].tagName != "INPUT" && elements[args.id].tagName != "TEXTAREA")){return} - if(elements[args.id].type === 'checkbox') elements[args.id].checked = Scratch.Cast.toBoolean(args.value) - elements[args.id].value = args.value - } - async InputValue(args){ - if(!elements[args.id] && inputhold[args.id]) return inputhold[args.id] - const element = elements[args.id] - if((!element) || (element.tagName != "INPUT" && element.tagName != "TEXTAREA")) return "" - if(!element && inputhold[args.id]) return inputhold[args.id] - if(element.type === 'checkbox') return element.checked - return (element.type === 'file' ? await datauri(element.files[0]) : element.value) - } - WhenInputChanged(args, util){ - if(!elements[args.id] || (elements[args.id].tagName!="INPUT"&&elements[args.id].tagName!="TEXTAREA")) return false - const element = elements[args.id] - const value = element.type==='checkbox'?element.checked:element.value - const blockId = util.thread.peekStack() - if(!lastValues[blockId]) - lastValues[blockId] = value.toString() - if(lastValues[blockId] !== value.toString()){ - lastValues[blockId] = value.toString() - return true - } - return false - - /* - if(!elements[args.id] || (elements[args.id].tagName!="INPUT"&&elements[args.id].tagName!="TEXTAREA")) return false - if(metadata[args.id].inputdirty){metadata[args.id].inputdirty = false; return true} - return false - */ + oldElement.replaceWith(newElement); + return newElement; + }; - } - WhenClicked(args){ - //This isnt ideal, but its basically the only option we have - if(!metadata[args.id]) return false - if(metadata[args.id].clicked){return new Promise((res, rej) => { - setTimeout(() => {metadata[args.id].clicked = false; res(true)}, 1) - })} - return false - } - VideoSource(args){ - const element = elements[args.id] - if(!element || element.tagName != "VIDEO") return - element.src = args.url - this.FixPos(args.id) - } - VideoControl(args){ - const element = elements[args.id] - if(!element || element.tagName != "VIDEO") return - switch(args.control){ - case 'Play': - element.play() - break; - case 'Stop': - element.pause() - element.currentTime = 0 - break; - case 'Pause': - element.pause() - break; + let fonts = []; + document.fonts.ready.then(() => { + fonts = Array.from(document.fonts.values()).map(f => f.family); + }); + + class ProjectInterfaces { + getInfo () { + return { + id: "lordcatprojectinterfaces", + name: "Project interfaces", + color1: "#707eff", + color2: "#6675fa", + docsURI: + "https://extensions.penguinmod.com/docs/ProjectInterfaces", + menuIconURI: extIcon, + blocks: [ + { + opcode: "ClearAll", + text: "clear all elements", + blockType: Scratch.BlockType.COMMAND, + }, + { + opcode: "Create", + text: "create [type] element with id [id]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + type: { + type: Scratch.ArgumentType.STRING, + menu: "ElementType", + }, + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + }, + }, + { + opcode: "Delete", + text: "delete element with id [id]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + }, + }, + { + opcode: "Visibility", + text: "[menu] element with id [id]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + menu: { + type: Scratch.ArgumentType.STRING, + menu: "Visibility", + }, + }, + }, + { + opcode: "ElementVisibility", + text: "element with id [id] is [status]", + blockType: Scratch.BlockType.BOOLEAN, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + status: { + type: Scratch.ArgumentType.STRING, + menu: "VisibilityStatus", + }, + }, + }, + { + opcode: "AllElements", + text: "All elements", + blockType: Scratch.BlockType.REPORTER, + }, + { blockType: Scratch.BlockType.LABEL, text: "Styling" }, + { + opcode: "Position", + text: "set position of id [id] to x [x] y [y]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + x: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 0, + }, + y: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 0, + }, + }, + }, + { + opcode: "Direction", + text: "set direction of id [id] to [dir]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + dir: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 90, + }, + }, + }, + { + opcode: "Scale", + text: "set scale of id [id] to width [width]px height [height]px", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + width: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 100, + }, + height: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 100, + }, + }, + }, + { + opcode: "Layer", + text: "set layer of id [id] to [layer]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + layer: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 1, + }, + }, + }, + { + opcode: "Cursor", + text: "set hover cursor of id [id] to [cursor]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + cursor: { + type: Scratch.ArgumentType.STRING, + menu: "Cursors", + }, + }, + }, + { + opcode: "Color", + text: "set color of id [id] to [color]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + color: { type: Scratch.ArgumentType.COLOR }, + }, + }, + { + opcode: "BackgroundColor", + text: "set background color of id [id] to [color]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + color: { type: Scratch.ArgumentType.COLOR }, + }, + }, + { + opcode: "CustomCSS", + text: "set custom css of id [id] to [css]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + css: { + type: Scratch.ArgumentType.STRING, + defaultValue: "background-color: red", + }, + }, + }, + { + opcode: "HtmlElement", + text: "create html element [htmltag] with id [id]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + htmltag: { + type: Scratch.ArgumentType.STRING, + defaultValue: "h1", + }, + }, + }, + "---", + { + opcode: "WhenClicked", + text: "when id [id] is clicked", + blockType: Scratch.BlockType.HAT, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + }, + }, + { + opcode: "Attribute", + text: "[attr] of id [id]", + blockType: Scratch.BlockType.REPORTER, + arguments: { + attr: { + type: Scratch.ArgumentType.STRING, + menu: "Attributes", + }, + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + }, + }, + { + opcode: "IsHovered", + text: "[id] hovered?", + blockType: Scratch.BlockType.BOOLEAN, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + }, + }, + { + blockType: Scratch.BlockType.LABEL, + text: "Labels", + }, + { + opcode: "LabelText", + text: "set label text with id [id] to [text]", + arguments: { + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello world!", + }, + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + }, + blockIconURI: textIcon, + }, + { + opcode: "LabelAlign", + text: "set label alignment with id [id] to [align]", + arguments: { + align: { + type: Scratch.ArgumentType.STRING, + menu: "Alignment", + }, + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + }, + blockIconURI: textIcon, + }, + { + opcode: "LabelFontSize", + text: "set label font size with id [id] to [size]px", + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + size: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 40, + }, + }, + blockIconURI: textIcon, + }, + { + opcode: "LabelFont", + text: "set label font with id [id] to [font]", + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + font: { + type: Scratch.ArgumentType.STRING, + menu: "Fonts", + }, + }, + blockIconURI: textIcon, + }, + { blockType: Scratch.BlockType.LABEL, text: "Images" }, + { + opcode: "ImageUrl", + text: "set image with id [id] to url [url]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + url: { + type: Scratch.ArgumentType.STRING, + defaultValue: + "https://extensions.turbowarp.org/dango.png", + }, + }, + blockIconURI: imageIcon, + }, + { + opcode: "ImageCostume", + text: "set image with id [id] to costume [costume]", + blockType: Scratch.BlockType.COMMAND, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + costume: { type: Scratch.ArgumentType.COSTUME }, + }, + blockIconURI: imageIcon, + }, + { blockType: Scratch.BlockType.LABEL, text: "Videos" }, + { + opcode: "VideoSource", + text: "set video with id [id] to url [url]", + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + url: { + type: Scratch.ArgumentType.STRING, + defaultValue: + "https://extensions.turbowarp.org/dango.png", + }, + }, + blockIconURI: videoIcon, + }, + { + opcode: "VideoControl", + text: "[control] video with id [id]", + arguments: { + control: { + type: Scratch.ArgumentType.STRING, + menu: "VideoControls", + }, + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + }, + blockIconURI: videoIcon, + }, + { + opcode: "VideoVolume", + text: "set volume of video [id] to [volume]%", + arguments: { + volume: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 100, + }, + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + }, + blockIconURI: videoIcon, + }, + { + opcode: "VideoLoop", + text: "set loop of video [id] to [toggle]", + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + toggle: { + type: Scratch.ArgumentType.STRING, + menu: "EnableDisable", + }, + }, + blockIconURI: videoIcon, + }, + { + opcode: "VideoHtmlControls", + text: "set video controls of id [id] to [toggle]", + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + toggle: { + type: Scratch.ArgumentType.STRING, + menu: "EnableDisable", + }, + }, + blockIconURI: videoIcon, + }, + { blockType: Scratch.BlockType.LABEL, text: "Inputs" }, + { + opcode: "InputType", + text: "set input type of id [id] to [input]", + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + input: { + type: Scratch.ArgumentType.STRING, + menu: "Inputs", + }, + }, + blockIconURI: inputIcon, + }, + { + opcode: "InputAccent", + text: "set input accent color of id [id] to [color]", + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + color: { type: Scratch.ArgumentType.COLOR }, + }, + blockIconURI: inputIcon, + }, + { + opcode: "InputPlaceholder", + text: "set placeholder of id [id] to [placeholder]", + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + placeholder: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello world!", + }, + }, + blockIconURI: inputIcon, + }, + { + opcode: "InputMinMax", + text: "set [type] of slider [id] to [value]", + arguments: { + type: { + type: Scratch.ArgumentType.STRING, + menu: 'MinMax', + defaultValue: 'min' + }, + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element" + }, + value: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 100 + } + }, + blockIconURI: inputIcon + }, + { + opcode: "InputSetValue", + text: "set value of id [id] to [value]", + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + value: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello world!", + }, + }, + blockIconURI: inputIcon, + }, + { + opcode: "InputValue", + text: "value of input with id [id]", + blockType: Scratch.BlockType.REPORTER, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + }, + blockIconURI: inputIcon, + }, + { + opcode: "WhenInputChanged", + text: "when input with id [id] changed", + blockType: Scratch.BlockType.HAT, + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + }, + blockIconURI: inputIcon, + }, + { + blockType: Scratch.BlockType.LABEL, + text: "Buttons", + }, + { + opcode: "ButtonText", + text: "set text of button [id] to [text]", + arguments: { + id: { + type: Scratch.ArgumentType.STRING, + defaultValue: "My element", + }, + text: { + type: Scratch.ArgumentType.STRING, + defaultValue: "Hello world!", + }, + }, + blockIconURI: buttonicon, + }, + ], + menus: { + ElementType: { + acceptReporters: false, + items: [ + "Label", + "Image", + "Video", + "Input", + "Box", + "Button", + ], + }, + Inputs: { + acceptReporters: false, + items: [ + "Text", + "Text Area", + "Number", + "Color", + "Checkbox", + "File", + "Email", + "Range", + "Image", + ], + }, + Cursors: { + acceptReporters: false, + items: [ + "default", + "pointer", + "text", + "wait", + "move", + "not-allowed", + "crosshair", + "help", + "progress", + "grab", + "grabbing", + ], + }, + Fonts: { acceptReporters: true, items: fonts }, + Attributes: { + acceptReporters: false, + items: [ + "X", + "Y", + "Direction", + "Width", + "Height", + "Cursor", + "Source", + ], + }, + VideoControls: { + acceptReporters: false, + items: ["Play", "Stop", "Pause"], + }, + EnableDisable: { + acceptReporters: false, + items: ["Enabled", "Disabled"], + }, + Alignment: { + acceptReporters: false, + items: ["Left", "Right", "Center"], + }, + Visibility: { + acceptReporters: false, + items: ["Show", "Hide"], + }, + VisibilityStatus: { + acceptReporters: false, + items: ["Shown", "Hidden"], + }, + MinMax: { + acceptReporters: false, + items: ['min', 'max'] + } + }, + }; } - } - VideoVolume(args){ - const element = elements[args.id] - if(!element || element.tagName != "VIDEO") return - element.volume = (args.volume / 100) - } - VideoHtmlControls(args){ - const element = elements[args.id] - if(!element || element.tagName != "VIDEO") return - switch(args.toggle){ - case 'Enabled': - element.setAttribute("controls", "true") - break; - case 'Disabled': - element.removeAttribute("controls") - break; + FixPos (elementid) { + setTimeout(() => { + if (!elements[elementid]) { + return; + } + this.Position({ + id: elementid, + x: metadata[elementid].x, + y: metadata[elementid].y, + }); + }, 1); // Timeout needed because for some reason it wont run otherwise.. + } + FixTransform (elementid) { + setTimeout(() => { + elements[elementid].style.transform = + elements[elementid].tagName === "SVG" + ? `translate(-50%, -50%) rotate(${metadata[elementid] - 90}deg)` + : `rotate(${metadata[elementid] - 90}deg)`; + }, 1); // Timeout needed because for some reason it wont run otherwise.. + } + ClearAll () { + Object.entries(elements).forEach(([id, element]) => { + if ( + element.tagName === "INPUT" || + element.tagName === "TEXTAREA" + ) + inputhold[id] = + element.type === "checkbox" + ? element.checked + : element.value; + }); + elements = {}; + metadata = {}; + elementbox.innerHTML = ""; + } + Create (args) { + if (elements[args.id]) return; + const element = document.createElement(lookup[args.type]); + if (lookup[args.type] === "button") { + element.append(document.createElement("span")); + element.append(document.createElement("img")); + } + const boundingRect = element.getBoundingClientRect(); + element.dataset.id = args.id; + element.style.position = "absolute"; + element.style.pointerEvents = "auto"; + element.style.userSelect = "none"; + element.style.color = "black"; + if (args.type === "Image") { + element.draggable = false; + } + elements[args.id] = element; + elementbox.append(element); + metadata[args.id] = { + x: 0, + y: 0, + direction: 90, + width: boundingRect.width, + height: boundingRect.height, + hovered: false, + clicked: false, + }; + this.FixPos(args.id); + if (args.type === "Input") { + metadata[args.id].inputdirty = false; + element.addEventListener( + "input", + () => (metadata[args.id].inputdirty = true), + ); + } + element.addEventListener( + "mouseover", + () => (metadata[args.id].hovered = true), + ); + element.addEventListener( + "mouseout", + () => (metadata[args.id].hovered = false), + ); + element.addEventListener( + "click", + () => (metadata[args.id].clicked = true), + ); + } + Position (args) { + const element = elements[args.id]; + if (!element) { + return; + } + if (element.tagName === "svg") { + const bbox = element.getBBox(); + element.style.left = `${vm.runtime.stageWidth / 2 + args.x - bbox.width / 2}px`; + element.style.top = `${vm.runtime.stageHeight / 2 - args.y - bbox.height / 2}px`; + } else { + element.style.left = `${vm.runtime.stageWidth / 2 + args.x - element.offsetWidth / 2}px`; + element.style.top = `${vm.runtime.stageHeight / 2 - args.y - element.offsetHeight / 2}px`; + } + metadata[args.id].x = args.x; + metadata[args.id].y = args.y; + } + Direction (args) { + const element = elements[args.id]; + if (!element) { + return; + } + metadata[args.id].direction = args.dir; + element.style.transform = `rotate(${args.dir - 90}deg)`; + } + Scale (args) { + const element = elements[args.id]; + if (!element) { + return; + } + element.style.width = `${args.width}px`; + element.style.height = `${args.height}px`; + metadata[args.id].width = args.width + "px"; + metadata[args.id].height = args.height + "px"; + element.style.objectFit = "fill"; + this.FixPos(args.id); + } + Layer (args) { + const element = elements[args.id]; + if (!element) { + return; + } + element.style.zIndex = args.layer; + } + Cursor (args) { + const element = elements[args.id]; + if (!element) { + return; + } + element.style.cursor = args.cursor; + } + Color (args) { + const element = elements[args.id]; + if (!element) { + return; + } + if (element.tagName === "DIV") { + // If the element is a box, set it's background color instead + // This probably avoids some confusion with users + element.style.backgroundColor = args.color; + } else { + element.style.color = args.color; + } + } + BackgroundColor (args) { + const element = elements[args.id]; + if (!element) { + return; + } + element.style.backgroundColor = args.color; + } + CustomCSS (args) { + if (!elements[args.id]) { + return; + } + let style = document.getElementById(`LCGuiStyle_${args.id}`); + let lines = args.css.split(";"); + if (!style) { + style = document.createElement("style"); + style.id = `LCGuiStyle_${args.id}`; + document.head.append(style); + } + style.textContent = `[data-id='${args.id}']{\n${lines.join(" !important;\n") + " !important"}\n}`; + } + HtmlElement (args) { + if (elements[args.id]) return; + const element = document.createElement(args.htmltag.toLowerCase()); + const boundingRect = element.getBoundingClientRect(); + element.dataset.id = args.id; + element.style.position = "absolute"; + element.style.pointerEvents = "auto"; + element.style.userSelect = "none"; + element.style.color = "black"; + elements[args.id] = element; + elementbox.append(element); + metadata[args.id] = { + x: 0, + y: 0, + direction: 90, + width: boundingRect.width, + height: boundingRect.height, + hovered: false, + clicked: false, + }; + this.FixPos(args.id); + element.addEventListener( + "mouseover", + () => (metadata[args.id].hovered = true), + ); + element.addEventListener( + "mouseout", + () => (metadata[args.id].hovered = false), + ); + element.addEventListener( + "click", + () => (metadata[args.id].clicked = true), + ); + } + Attribute (args) { + const element = elements[args.id]; + if (!element) return; + const meta = metadata[args.id]; + switch (args.attr) { + case "Cursor": + return element.style.cursor === "" + ? "default" + : element.style.cursor; + case "Source": + if ( + element.tagName != "IMG" && + element.tagName != "VIDEO" && + element.tagName != "svg" + ) + return; + return element.tagName === "svg" + ? element.outerHTML + : element.src; + case "Width": + if (element.tagName === 'SVG') { + return element.getBBox().width; + } else { + return element.getBoundingClientRect().width; + } + case "Height": + if (element.tagName === 'SVG') { + return element.getBBox().height; + } else { + return element.getBoundingClientRect().height; + } + default: + return meta[args.attr.toLowerCase()]; + } + } + IsHovered (args) { + if (!elements[args.id]) { + return ""; + } + return metadata[args.id].hovered; + } + Delete (args) { + const element = elements[args.id]; + if (!element) { + return; + } + if (element.tagName === "INPUT" || element.tagName === "TEXTAREA") + inputhold[args.id] = + element.type === "checkbox" + ? element.checked + : element.value; + if (document.getElementById(`LCGuiStyle_${args.id}`)) + document.getElementById(`LCGuiStyle_${args.id}`).remove(); + element.remove(); + delete elements[args.id]; + delete metadata[args.id]; + } + Visibility (args) { + const element = elements[args.id]; + if (!element) { + return; + } + element.hidden = args.menu === "Hide"; + } + ElementVisibility (args) { + const element = elements[args.id]; + if (!element) return; + return args.status === "Shown" + ? !element.hidden + : !!element.hidden; + } + AllElements () { + return JSON.stringify(Object.keys(elements)); + } + LabelText (args) { + const element = elements[args.id]; + if (!element || element.tagName != "SPAN") { + return; + } + element.textContent = args.text; + this.FixPos(args.id); + } + LabelAlign (args) { + const element = elements[args.id]; + if (!element || element.tagName !== "SPAN") { + return; + } + element.style.textAlign = args.align.toLowerCase(); + this.FixPos(args.id); + } + LabelFontSize (args) { + const element = elements[args.id]; + if (!element || element.tagName !== "SPAN") { + return; + } + element.style.fontSize = `${args.size}px`; + this.FixPos(args.id); + } + LabelFont (args) { + const element = elements[args.id]; + if (!element || element.tagName !== "SPAN") { + return; + } + element.style.fontFamily = args.font; + this.FixPos(args.id); + } + ImageUrl (args) { + const element = elements[args.id]; + if ( + !element || + (element.tagName !== "IMG" && + element.tagName !== "svg") + ) return; + + if (element.tagName === "svg") { + elements[args.id] = replaceElement( + elements[args.id], + document.createElement("img"), + args.id, + ); + } + elements[args.id].src = args.url; + this.FixPos(args.id); + } + ImageCostume (args, util) { + const element = elements[args.id]; + if ( + !element || + (element.tagName != "IMG" && + element.tagName != "svg") + ) return; + + const costume = util.target.getCostumes().find(c => c.name === args.costume); + if (!costume) return; + + if (costume.dataFormat === "svg") { + elements[args.id] = replaceElement( + element, + domParser.parseFromString( + textDecoder.decode(costume.asset.data), + "image/svg+xml", + ).documentElement, + args.id, + ); + } else { + if (element.tagName === "svg") { + elements[args.id] = replaceElement( + element, + document.createElement("img"), + args.id, + ); + } + elements[args.id].src = datauriFromCostume( + args.costume, + util.target, + ); + } + this.FixPos(args.id); + this.FixTransform(args.id); + } + InputType (args) { + const element = elements[args.id]; + if ( + !element || + (element.tagName != "INPUT" && + element.tagName != "TEXTAREA") + ) return; + + if (args.input === "Text Area") { + elements[args.id] = replaceElement( + element, + document.createElement("textarea"), + args.id, + ); + elements[args.id].style.resize = "none"; + } else { + if (element.tagName == "TEXTAREA") { + elements[args.id] = replaceElement( + element, + document.createElement("input"), + args.id, + ); + } + elements[args.id].type = args.input; + if (args.input == "File") { + elements[args.id].value = null; + } + } + this.FixPos(args.id); + } + InputAccent (args) { + const element = elements[args.id]; + if ( + !element || + (element.tagName != "INPUT" && + element.tagName != "TEXTAREA") + ) return; + element.style.accentColor = args.color; + } + InputPlaceholder (args) { + const element = elements[args.id]; + if ( + !element || + (element.tagName !== "INPUT" && + element.tagName !== "TEXTAREA") + ) return; + element.setAttribute("placeholder", args.placeholder); + } + InputSetValue (args) { + const element = elements[args.id]; + if ( + !element || + (element.tagName !== "INPUT" && + element.tagName !== "TEXTAREA") + ) return; + + if (element.type === "checkbox") { + element.checked = Scratch.Cast.toBoolean(args.value); + } else { + element.value = args.value; + } + } + InputValue (args) { + const element = elements[args.id]; + if (!element) { + if ( + inputhold[args.id] && (element.tagName !== "INPUT" && element.tagName !== "TEXTAREA") + ) { + return inputhold[args.id]; + } else { + return ""; + } + } + switch (element.type) { + case 'checkbox': + return element.checked; + case 'file': + return datauri(element.files[0]); + default: + return element.value; + } + } + WhenInputChanged (args, util) { + const element = elements[args.id]; + if ( + !element || + (element.tagName !== "INPUT" && + element.tagName !== "TEXTAREA") + ) return false; + + const value = Scratch.Cast.toString(element.type === "checkbox" ? element.checked : element.value); + const blockId = util.thread.peekStack(); + + if (lastValues[blockId] !== value) { + lastValues[blockId] = value; + return true; + } else { + return false; + } + } + InputMinMax (args) { + const element = elements[args.id]; + if ( + !element || + element.tagName !== 'INPUT' || + element.type !== 'range' + ) return; + + if (args.type === 'max') { + element.setAttribute('max', args.value); + } else { + element.setAttribute('min', args.value); + } + } + WhenClicked (args) { + if (!metadata[args.id]) return false; + + if (metadata[args.id].clicked) { + metadata[args.id].clicked = false; + return true; + } else { + return false; + } + } + VideoSource (args) { + const element = elements[args.id]; + if (!element || element.tagName !== "VIDEO") return; + element.src = args.url; + this.FixPos(args.id); + } + VideoControl (args) { + const element = elements[args.id]; + if (!element || element.tagName !== "VIDEO") return; + switch (args.control) { + case "Play": + element.play(); + break; + case "Stop": + element.pause(); + element.currentTime = 0; + break; + case "Pause": + element.pause(); + break; + } + } + VideoVolume (args) { + const element = elements[args.id]; + if (!element || element.tagName !== "VIDEO") return; + element.volume = args.volume / 100; + } + VideoHtmlControls (args) { + const element = elements[args.id]; + if (!element || element.tagName !== "VIDEO") return; + if (args.toggle === 'Enabled') { + element.setAttribute('controls', 'true'); + } else { + element.removeAttribute('controls'); + } + } + VideoLoop (args) { + const element = elements[args.id]; + if (!element || element.tagName !== "VIDEO") return; + element.loop = args.toggle == "Enabled"; + } + ButtonText (args) { + const element = elements[args.id]; + if (!element || element.tagName !== "BUTTON") return; + element.children[0].textContent = args.text; + this.FixPos(args.id); } } - VideoLoop(args){ - const element = elements[args.id] - if(!element || element.tagName != "VIDEO") return - element.loop = (args.toggle == "Enabled") - } - ButtonText(args){ - const element = elements[args.id] - if(!element || element.tagName != "BUTTON") return - document.querySelector(`.LordCatInterfaces button[data-id="${args.id}"] span`).textContent = args.text - this.FixPos(args.id) - } - } - Scratch.extensions.register(new lordcatprojectinterfaces()) -})(Scratch) + Scratch.extensions.register(new ProjectInterfaces()); +})(Scratch); \ No newline at end of file