diff --git a/src/lib/extensions.js b/src/lib/extensions.js index 827847f5d..5c5893e9a 100644 --- a/src/lib/extensions.js +++ b/src/lib/extensions.js @@ -570,4 +570,14 @@ export default [ isGitHub: true, notes: "Gallery banner by Dillon", }, + { + name: "Project Interfaces", + description: "Effortlessly create intuitive graphical user interfaces in your projects.", + code: "LordCat0/ProjectInterfaces.js", + banner: "LordCat0/ProjectInterfaces.png", + creator: "LordCat0", + creatorAlias: "Lord cat", + notes: "Gallery banner by Dillon", + isGitHub: true, +}, ]; diff --git a/static/extensions/LordCat0/ProjectInterfaces.js b/static/extensions/LordCat0/ProjectInterfaces.js new file mode 100644 index 000000000..a5b6c6ef6 --- /dev/null +++ b/static/extensions/LordCat0/ProjectInterfaces.js @@ -0,0 +1,426 @@ +// Name: Project Interfaces +// ID: lordcatprojectinterfaces +// 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"} + const extIcon = '' + const textIcon = '' + const imageIcon = '' + const videoIcon = '' + const inputIcon = '' + const vm = Scratch.vm + const elementbox = document.createElement('div') + elementbox.classList.add('LordCatInterfaces') + vm.renderer.addOverlay(elementbox, "scale") + 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) + }) + } + let elements = {} + let metadata = {} + 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", + 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: "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: "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: "InputValue", + text: "Value of input with id [id]", + blockType: Scratch.BlockType.REPORTER, + arguments: {id: {type: Scratch.ArgumentType.STRING, defaultValue: "My element"}}, + blockIconURI: inputIcon + } + ], + menus: {ElementType: {acceptReporters: false,items: ["Label", "Image", "Video", "Input", "Box"]}, + Inputs: {acceptReporters: false,items: ["Text", "Number", "Color", "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"]}} + }} + 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.. + } + ClearAll(){ + elements = {} + metadata = {} + elementbox.innerHTML = '' + } + Create(args){ + if(elements[args.id]) return + const element = document.createElement(lookup[args.type]) + 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) + 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] + 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] + element.style.transform = `rotate(${args.dir - 90}deg)` + metadata[args.id].direction = args.dir + } + 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 + case 'Source': + if(element.tagName != 'IMG' && element.tagName != 'VIDEO') return + return element.src + default: + return meta[args.attr.toLowerCase()] + } + } + IsHovered(args){ + if(!elements[args.id]){return ''} + return metadata[args.id].hovered + } + Delete(args){ + if(!elements[args.id]){return} + if(document.getElementById(`LCGuiStyle_${args.id}`)){document.getElementById(`LCGuiStyle_${args.id}`).remove()} + elements[args.id].remove() + delete elements[args.id] + delete metadata[args.id] + } + 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() + } + 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 + } + ImageUrl(args){ + if(!elements[args.id] || elements[args.id].tagName != "IMG"){return} + elements[args.id].src = args.url + this.FixPos(args.id) + } + ImageCostume(args, util){ + if(!elements[args.id] || elements[args.id].tagName != "IMG"){return} + const target = util.target + const costumeIndex = target.getCostumeIndexByName("costume1")//Scratch.Cast.toString(args.COSTUME)) + const costume = target.sprite.costumes[costumeIndex] + elements[args.id].src = costume.asset.encodeDataURI() + this.FixPos(args.id) + } + InputType(args){ + if(!elements[args.id] || elements[args.id].tagName != "INPUT"){return} + elements[args.id].type = args.input + if(args.input == 'File'){elements[args.id].value = null} + this.FixPos(args.id) + } + InputPlaceholder(args){ + if(!elements[args.id] || elements[args.id].tagName != "INPUT"){return} + elements[args.id].setAttribute('placeholder', args.placeholder) + } + async InputValue(args){ + const element = elements[args.id] + if(!element || element.tagName != "INPUT"){return} + return (element.type == 'file' ? await datauri(element.files[0]) : element.value) + } + 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; + } + } + 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; + } + } + VideoLoop(args){ + const element = elements[args.id] + if(!element || element.tagName != "VIDEO") return + element.loop = (args.toggle == "Enabled") + } + } + Scratch.extensions.register(new lordcatprojectinterfaces()) +})(Scratch) diff --git a/static/images/LordCat0/ProjectInterfaces.png b/static/images/LordCat0/ProjectInterfaces.png new file mode 100644 index 000000000..fb7421fc0 Binary files /dev/null and b/static/images/LordCat0/ProjectInterfaces.png differ