From f8b313fb0a809e84e8136c20aa331392d477a6ad Mon Sep 17 00:00:00 2001 From: AlphaTechNinja <114359889+AlphaTechNinja@users.noreply.github.com> Date: Mon, 25 Dec 2023 18:48:58 -0600 Subject: [PATCH 01/12] Create images.js --- static/extensions/AlphaTechNinja/images.js | 295 +++++++++++++++++++++ 1 file changed, 295 insertions(+) create mode 100644 static/extensions/AlphaTechNinja/images.js diff --git a/static/extensions/AlphaTechNinja/images.js b/static/extensions/AlphaTechNinja/images.js new file mode 100644 index 00000000..d5496ef4 --- /dev/null +++ b/static/extensions/AlphaTechNinja/images.js @@ -0,0 +1,295 @@ +(function (Scratch) { + 'use strict'; + const extIcon = ""; + let saved = {}; +function hexToRGB(hex, alpha) { + var r = parseInt(hex.slice(1, 3), 16), + g = parseInt(hex.slice(3, 5), 16), + b = parseInt(hex.slice(5, 7), 16); + + if (alpha) { + return "rgba(" + r + ", " + g + ", " + b + ", " + alpha + ")"; + } else { + return "rgb(" + r + ", " + g + ", " + b + ")"; + } +} +function RGBAToHex(rgba, forceRemoveAlpha = false) { + return "#" + rgba.replace(/^rgba?\(|\s+|\)$/g, '') // Get's rgba / rgb string values + .split(',') // splits them at "," + .filter((string, index) => !forceRemoveAlpha || index !== 3) + .map(string => parseFloat(string)) // Converts them to numbers + .map((number, index) => index === 4 ? Math.round(number * 255) : number) // Converts alpha to 255 number + .map(number => number.toString(16)) // Converts numbers to hex + .map(string => string.length === 1 ? "0" + string : string) // Adds 0 when length of one number is 1 + .join("") // Puts the array to together to a string + } + + class images { + canvas; + ctx; + current; + constructor() { + this.canvas = document.createElement("canvas"); + this.ctx = this.canvas.getContext("2d"); + this.current = ""; + } + getInfo() { + return { + id:"images", + name:"Images", + blockIconURI: extIcon, + color1:"#327da8", + color2:"#1f475e", + color3:"#235c7d", + blocks:[ + { + opcode:"newImage", + text:"new image [name] with size [width], [height]", + blockType:Scratch.BlockType.COMMAND, + arguments:{ + name:{ + type:Scratch.ArgumentType.STRING, + defaultValue: "image" + }, + width:{ + type:Scratch.ArgumentType.NUMBER, + defaultValue: 128 + }, + height:{ + type:Scratch.ArgumentType.NUMBER, + defaultValue: 128 + } + } + }, + { + opcode:"removeImage", + text:"delete image [name]", + blockType:Scratch.BlockType.COMMAND, + arguments:{ + name:{ + type:Scratch.ArgumentType.STRING, + defaultValue:"image" + } + } + }, + { + opcode:"setCurrent", + text:"set current image to [name]", + blockType:Scratch.BlockType.COMMAND, + arguments:{ + name:{ + type:Scratch.ArgumentType.STRING, + defaultValue: "image" + } + } + }, + { + opcode:"writePixel", + text:"write pixel [color] at [x], [y]", + blockType:Scratch.BlockType.COMMAND, + arguments:{ + color:{ + type:Scratch.ArgumentType.COLOR, + defaultValue: "#ffffff" + }, + x:{ + type:Scratch.ArgumentType.NUMBER, + defaultValue: 0 + }, + y:{ + type:Scratch.ArgumentType.NUMBER, + defaultValue: 0 + } + } + }, + { + opcode:"readPixel", + text:"read pixel [x], [y]", + blockType:Scratch.BlockType.REPORTER, + arguments:{ + x:{ + type:Scratch.ArgumentType.NUMBER, + defaultValue: 0 + }, + y:{ + type:Scratch.ArgumentType.NUMBER, + defaultValue: 0 + } + } + }, + { + opcode:"importImage", + text:"import image [uri] as [name]", + blockType:Scratch.BlockType.COMMAND, + arguments:{ + uri:{ + type:Scratch.ArgumentType.STRING, + defaultValue:"" + }, + name:{ + type:Scratch.ArgumentType.STRING, + defaultValue:"image" + } + } + }, + { + opcode:"exportImage", + text:"export current image as [format]", + disableMonitor:true, + blockType:Scratch.BlockType.REPORTER, + arguments:{ + format:{ + type:Scratch.ArgumentType.STRING, + menu:"exportTypes", + defaultValue:"png" + } + } + }, + { + opcode:"listImages", + text:"list images", + disableMonitor:true, + blockType:Scratch.BlockType.REPORTER + }, + { + opcode:"getCurrent", + text:"get current", + disableMonitor:true, + blockType:Scratch.BlockType.REPORTER + }, + { + opcode:"getWidth", + text:"get width", + blockType:Scratch.BlockType.REPORTER + }, + { + opcode:"getHeight", + text:"get height", + blockType:Scratch.BlockType.REPORTER + }, + { + opcode:"grabRegion", + text:"grab region at [x], [y] of size [width], [height]", + blockType:Scratch.BlockType.REPORTER, + arguments:{ + x:{ + type:Scratch.ArgumentType.NUMBER, + defaultValue:0 + }, + y:{ + type:Scratch.ArgumentType.NUMBER, + defaultValue:0 + }, + width:{ + type:Scratch.ArgumentType.NUMBER, + defaultValue:16 + }, + height:{ + type:Scratch.ArgumentType.NUMBER, + defaultValue:16 + } + } + } + ], + menus:{ + exportTypes:{ + items:[ + "png", + "jpeg", + "bitmap" + ], + acceptReporters: false + } + } + }; + }; + newImage({name,width,height}){ + let image = new Image(width,height); + this.canvas.width = width; + this.canvas.height = height; + // fill canvas + this.ctx.fillStyle = "white"; + this.ctx.fillRect(0,0,this.canvas.width,this.canvas.height); + image.src = this.canvas.toDataURL('image/png'); + saved[name] = image; + }; + removeImage({name}) { + delete saved[name]; + } + setCurrent({name}) { + this.current = name; + }; + writePixel({color,x,y}) { + if (!saved[this.current]) { + throw new Error("image does not exists!"); + } + let image = saved[this.current]; + this.canvas.width = image.width; + this.canvas.height = image.height; + this.ctx.drawImage(image,0,0); + this.ctx.fillStyle = hexToRGB(color); + this.ctx.fillRect(x + 1,y + 1,x + 1,y + 1); + image.src = this.canvas.toDataURL('image/png'); + saved[this.current] = image; + }; + importImage({uri,name}) { + let image = document.createElement("img"); + image.src = uri; + saved[name] = image; + }; + readPixel({x,y}) { + if (!saved[this.current]) { + throw new Error("image does not exists!"); + } + let image = saved[this.current]; + this.canvas.width = image.width; + this.canvas.height = image.height; + this.ctx.drawImage(image,0,0); + let pixel = this.ctx.getImageData(x + 1,y + 1,1,1); + // reformat to array + let newArray = RGBAToHex("rgba(" + pixel.data["0"] + ", " + pixel.data["1"] + ", " + pixel.data["2"] + ", " + pixel.data["3"] + ")"); + return newArray; + }; + listImages() { + return JSON.stringify(Object.keys(saved)); + }; + getCurrent() { + return this.current; + }; + getWidth() { + return saved[this.current].width; + }; + getHeight() { + return saved[this.current].height; + }; + // a harder function + grabRegion({x,y,width,height}) { + if (!saved[this.current]) { + throw new Error("image does not exists!"); + } + let image = saved[this.current]; + this.canvas.width = image.width; + this.canvas.height = image.height; + this.ctx.drawImage(image,0,0); + let pixels = this.ctx.getImageData(x + 1,y + 1,width,height); + console.log(pixels); + //start formating + let array = []; + function fp(i) { + return pixels.data[i + ""]; + } + for (let i = 0;i < pixels.data.length;i += 4) { + array.push(RGBAToHex("rgba(" + fp(i) + ", " + fp(i + 1) + ", " + fp(i + 2) + ", " + fp(i + 3) + ")")); + } + return JSON.stringify(array); + }; + exportImage({format}) { + let image = saved[this.current]; + this.canvas.width = image.width; + this.canvas.height = image.height; + this.ctx.drawImage(image,0,0); + return this.canvas.toDataURL("image/" + format); + }; + } + Scratch.extensions.register(new images()); +})(Scratch) From 1f97c5ceee4f290e605ab013ce3fa7b79b6281ad Mon Sep 17 00:00:00 2001 From: AlphaTechNinja <114359889+AlphaTechNinja@users.noreply.github.com> Date: Sun, 31 Dec 2023 11:51:02 -0600 Subject: [PATCH 02/12] Update extensions.js MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Added “images” extension --- src/lib/extensions.js | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/src/lib/extensions.js b/src/lib/extensions.js index 491518c4..2095cf9a 100644 --- a/src/lib/extensions.js +++ b/src/lib/extensions.js @@ -240,4 +240,12 @@ export default [ creator: "MrRedstonia", isGitHub: true, }, + { + name: "Images", + description: "Manipulate images on the spot!", + code: "AlphaTechNinja/images.js", + banner: "AlphaTechNinja/images.svg", + creator: "AlphaTechNinja", + isGitHub: true, + }, ]; From 3657a8f6f6e5fcc799a534738013ea6c63b46249 Mon Sep 17 00:00:00 2001 From: AlphaTechNinja <114359889+AlphaTechNinja@users.noreply.github.com> Date: Sun, 11 Feb 2024 21:56:16 -0600 Subject: [PATCH 03/12] Update images.js half of fix will fix later --- static/extensions/AlphaTechNinja/images.js | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/static/extensions/AlphaTechNinja/images.js b/static/extensions/AlphaTechNinja/images.js index d5496ef4..ba991d81 100644 --- a/static/extensions/AlphaTechNinja/images.js +++ b/static/extensions/AlphaTechNinja/images.js @@ -124,7 +124,7 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { arguments:{ uri:{ type:Scratch.ArgumentType.STRING, - defaultValue:"" + defaultValue:"data:" }, name:{ type:Scratch.ArgumentType.STRING, @@ -214,6 +214,9 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { saved[name] = image; }; removeImage({name}) { + if (!saved[name]) { + return null; + }; delete saved[name]; } setCurrent({name}) { @@ -224,6 +227,9 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { throw new Error("image does not exists!"); } let image = saved[this.current]; + if (!image) { + return null; + } this.canvas.width = image.width; this.canvas.height = image.height; this.ctx.drawImage(image,0,0); @@ -242,6 +248,9 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { throw new Error("image does not exists!"); } let image = saved[this.current]; + if (!image) { + return -1; + } this.canvas.width = image.width; this.canvas.height = image.height; this.ctx.drawImage(image,0,0); From 83675b1dabfdf56f3eaac30662b7ec2dee6bd0c9 Mon Sep 17 00:00:00 2001 From: AlphaTechNinja <114359889+AlphaTechNinja@users.noreply.github.com> Date: Mon, 12 Feb 2024 11:24:13 -0600 Subject: [PATCH 04/12] Update images.js instead of throwing errors when we go to an invalid index return -1 --- static/extensions/AlphaTechNinja/images.js | 19 ++++++++++++++----- 1 file changed, 14 insertions(+), 5 deletions(-) diff --git a/static/extensions/AlphaTechNinja/images.js b/static/extensions/AlphaTechNinja/images.js index ba991d81..aad944c9 100644 --- a/static/extensions/AlphaTechNinja/images.js +++ b/static/extensions/AlphaTechNinja/images.js @@ -245,12 +245,9 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { }; readPixel({x,y}) { if (!saved[this.current]) { - throw new Error("image does not exists!"); - } - let image = saved[this.current]; - if (!image) { return -1; } + let image = saved[this.current]; this.canvas.width = image.width; this.canvas.height = image.height; this.ctx.drawImage(image,0,0); @@ -263,18 +260,27 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { return JSON.stringify(Object.keys(saved)); }; getCurrent() { + if (!saved[this.current]) { + return -1; + }; return this.current; }; getWidth() { + if (!saved[this.current]) { + return -1; + }; return saved[this.current].width; }; getHeight() { + if (!saved[this.current]) { + return -1; + }; return saved[this.current].height; }; // a harder function grabRegion({x,y,width,height}) { if (!saved[this.current]) { - throw new Error("image does not exists!"); + return -1; } let image = saved[this.current]; this.canvas.width = image.width; @@ -293,6 +299,9 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { return JSON.stringify(array); }; exportImage({format}) { + if (!saved[this.current]) { + return -1; + }; let image = saved[this.current]; this.canvas.width = image.width; this.canvas.height = image.height; From 95c71922fe80e385768bb1c716cfbab0c0757003 Mon Sep 17 00:00:00 2001 From: AlphaTechNinja <114359889+AlphaTechNinja@users.noreply.github.com> Date: Mon, 12 Feb 2024 11:48:01 -0600 Subject: [PATCH 05/12] Update images.js now import waits for the image to fully load before it continues running the script --- static/extensions/AlphaTechNinja/images.js | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/static/extensions/AlphaTechNinja/images.js b/static/extensions/AlphaTechNinja/images.js index aad944c9..8b756fb2 100644 --- a/static/extensions/AlphaTechNinja/images.js +++ b/static/extensions/AlphaTechNinja/images.js @@ -238,10 +238,15 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { image.src = this.canvas.toDataURL('image/png'); saved[this.current] = image; }; - importImage({uri,name}) { - let image = document.createElement("img"); - image.src = uri; - saved[name] = image; + async importImage({uri,name}) { + let image; + const ImportPromise = new Promise(resolve => { + image = document.createElement("img"); + image.onload = resolve; + image.src = uri; + saved[name] = image; + }) + await ImportPromise; }; readPixel({x,y}) { if (!saved[this.current]) { From 65c77d0903d92b5b9d65f003076a6eeba97a2e3b Mon Sep 17 00:00:00 2001 From: AlphaTechNinja <114359889+AlphaTechNinja@users.noreply.github.com> Date: Mon, 12 Feb 2024 11:52:56 -0600 Subject: [PATCH 06/12] Update images.js hopefully fixed the import function --- static/extensions/AlphaTechNinja/images.js | 9 ++++----- 1 file changed, 4 insertions(+), 5 deletions(-) diff --git a/static/extensions/AlphaTechNinja/images.js b/static/extensions/AlphaTechNinja/images.js index 8b756fb2..47fa1f85 100644 --- a/static/extensions/AlphaTechNinja/images.js +++ b/static/extensions/AlphaTechNinja/images.js @@ -241,12 +241,11 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { async importImage({uri,name}) { let image; const ImportPromise = new Promise(resolve => { - image = document.createElement("img"); + image = new Image(); image.onload = resolve; - image.src = uri; - saved[name] = image; - }) - await ImportPromise; + image.src = imageUrl; + }); + return ImportPromise; }; readPixel({x,y}) { if (!saved[this.current]) { From ae811f6dfad0296f339c7d243988d874f92aa93a Mon Sep 17 00:00:00 2001 From: AlphaTechNinja <114359889+AlphaTechNinja@users.noreply.github.com> Date: Mon, 12 Feb 2024 12:00:21 -0600 Subject: [PATCH 07/12] Update images.js i forgot something --- static/extensions/AlphaTechNinja/images.js | 1 + 1 file changed, 1 insertion(+) diff --git a/static/extensions/AlphaTechNinja/images.js b/static/extensions/AlphaTechNinja/images.js index 47fa1f85..7dd931bb 100644 --- a/static/extensions/AlphaTechNinja/images.js +++ b/static/extensions/AlphaTechNinja/images.js @@ -244,6 +244,7 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { image = new Image(); image.onload = resolve; image.src = imageUrl; + saved[name] = image; }); return ImportPromise; }; From b8c379e4481667f27268bc5695da0dfece5bf704 Mon Sep 17 00:00:00 2001 From: AlphaTechNinja <114359889+AlphaTechNinja@users.noreply.github.com> Date: Mon, 12 Feb 2024 12:05:19 -0600 Subject: [PATCH 08/12] Update images.js I'm so sorry for changing this so many times I had to reread the documentation on how to use promises with Turbowarp and PenguinMod extensions. --- static/extensions/AlphaTechNinja/images.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/static/extensions/AlphaTechNinja/images.js b/static/extensions/AlphaTechNinja/images.js index 7dd931bb..a8538b23 100644 --- a/static/extensions/AlphaTechNinja/images.js +++ b/static/extensions/AlphaTechNinja/images.js @@ -238,7 +238,7 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { image.src = this.canvas.toDataURL('image/png'); saved[this.current] = image; }; - async importImage({uri,name}) { + importImage({uri,name}) { let image; const ImportPromise = new Promise(resolve => { image = new Image(); From 081b0dcdd76bb81fb862b81eb7c607ddc7e760ad Mon Sep 17 00:00:00 2001 From: AlphaTechNinja <114359889+AlphaTechNinja@users.noreply.github.com> Date: Mon, 12 Feb 2024 12:14:22 -0600 Subject: [PATCH 09/12] Update images.js attempting to fix it with RedMan13's idea --- static/extensions/AlphaTechNinja/images.js | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/static/extensions/AlphaTechNinja/images.js b/static/extensions/AlphaTechNinja/images.js index a8538b23..20a30c9f 100644 --- a/static/extensions/AlphaTechNinja/images.js +++ b/static/extensions/AlphaTechNinja/images.js @@ -240,13 +240,13 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { }; importImage({uri,name}) { let image; - const ImportPromise = new Promise(resolve => { - image = new Image(); + image = new Image(); + saved[name] = image; + return new Promise((resolve, reject) => { image.onload = resolve; - image.src = imageUrl; - saved[name] = image; + image.onerror = reject; + image.src = uri; }); - return ImportPromise; }; readPixel({x,y}) { if (!saved[this.current]) { From 5fcc1b0acbfa86fc07aefd9653ae202e66ef8388 Mon Sep 17 00:00:00 2001 From: AlphaTechNinja <114359889+AlphaTechNinja@users.noreply.github.com> Date: Mon, 12 Feb 2024 12:19:04 -0600 Subject: [PATCH 10/12] Update images.js might be fixed --- static/extensions/AlphaTechNinja/images.js | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/static/extensions/AlphaTechNinja/images.js b/static/extensions/AlphaTechNinja/images.js index 20a30c9f..8f718ab3 100644 --- a/static/extensions/AlphaTechNinja/images.js +++ b/static/extensions/AlphaTechNinja/images.js @@ -223,9 +223,6 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { this.current = name; }; writePixel({color,x,y}) { - if (!saved[this.current]) { - throw new Error("image does not exists!"); - } let image = saved[this.current]; if (!image) { return null; @@ -239,8 +236,7 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { saved[this.current] = image; }; importImage({uri,name}) { - let image; - image = new Image(); + let image = new Image("data:"); saved[name] = image; return new Promise((resolve, reject) => { image.onload = resolve; From d801710042e87f7c1bfc72fcda047e98b2587a0e Mon Sep 17 00:00:00 2001 From: AlphaTechNinja <114359889+AlphaTechNinja@users.noreply.github.com> Date: Mon, 12 Feb 2024 17:29:43 -0600 Subject: [PATCH 11/12] Update images.js Scraped promise system for import image --- static/extensions/AlphaTechNinja/images.js | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/static/extensions/AlphaTechNinja/images.js b/static/extensions/AlphaTechNinja/images.js index 8f718ab3..5c7c6299 100644 --- a/static/extensions/AlphaTechNinja/images.js +++ b/static/extensions/AlphaTechNinja/images.js @@ -236,13 +236,17 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { saved[this.current] = image; }; importImage({uri,name}) { - let image = new Image("data:"); + let image = new Image(); + image.src = uri; saved[name] = image; + /* + TODO return new Promise((resolve, reject) => { image.onload = resolve; image.onerror = reject; image.src = uri; }); + */ }; readPixel({x,y}) { if (!saved[this.current]) { From 07dd84a94a7b0a2868e9f8e28562977545ecf013 Mon Sep 17 00:00:00 2001 From: AlphaTechNinja <114359889+AlphaTechNinja@users.noreply.github.com> Date: Tue, 13 Feb 2024 11:30:39 -0600 Subject: [PATCH 12/12] Update images.js fixed the `writePixel` offset so that it is aligned to top-left --- static/extensions/AlphaTechNinja/images.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/static/extensions/AlphaTechNinja/images.js b/static/extensions/AlphaTechNinja/images.js index 5c7c6299..348af5a2 100644 --- a/static/extensions/AlphaTechNinja/images.js +++ b/static/extensions/AlphaTechNinja/images.js @@ -231,7 +231,7 @@ function RGBAToHex(rgba, forceRemoveAlpha = false) { this.canvas.height = image.height; this.ctx.drawImage(image,0,0); this.ctx.fillStyle = hexToRGB(color); - this.ctx.fillRect(x + 1,y + 1,x + 1,y + 1); + this.ctx.fillRect(x,y,x,y); image.src = this.canvas.toDataURL('image/png'); saved[this.current] = image; };