From 6c4d18dea25a29928142d70ced7b7b6a80631ebc Mon Sep 17 00:00:00 2001 From: SharkPool <139097378+SharkPool-SP@users.noreply.github.com> Date: Wed, 9 Apr 2025 11:10:03 -0700 Subject: [PATCH 01/14] Create Looks-Expanded.js --- extensions/SharkPool/Looks-Expanded.js | 966 +++++++++++++++++++++++++ 1 file changed, 966 insertions(+) create mode 100644 extensions/SharkPool/Looks-Expanded.js diff --git a/extensions/SharkPool/Looks-Expanded.js b/extensions/SharkPool/Looks-Expanded.js new file mode 100644 index 0000000000..a55713b4ae --- /dev/null +++ b/extensions/SharkPool/Looks-Expanded.js @@ -0,0 +1,966 @@ +// Name: Looks Expanded +// ID: SPlooksExpanded +// Description: Expansion of the Looks Category +// By: SharkPool +// By: CST1229 +// Licence: MIT + +// Version V.1.0.2 + +(function (Scratch) { + "use strict"; + if (!Scratch.extensions.unsandboxed) throw new Error("Looks Expanded must run unsandboxed!"); + + const menuIconURI = +""; + + const vm = Scratch.vm; + const Cast = Scratch.Cast; + const runtime = vm.runtime; + const looksCore = runtime.ext_scratch3_looks; + + const render = vm.renderer; + const twgl = render.exports.twgl; + const drawableKey = Symbol("SPlooksKey"); + const newUniforms = [ + "u_replaceColorFromSP", "u_replaceColorToSP", "u_replaceThresholdSP", "u_numReplacersSP", + "u_warpSP", "u_maskTextureSP", "u_shouldMaskSP", + "u_tintColorSP", "u_saturateSP", "u_opaqueSP", "u_contrastSP", + "u_posterizeSP", "u_sepiaSP", "u_bloomSP" + ]; + const defaultNewEffects = { + warp: [0.5, -0.5, -0.5, -0.5, -0.5, 0.5, 0.5, 0.5], + tint: [1, 1, 1, 1], + replacers: [], + maskTexture: "", + oldMask: "", + shouldMask: 0, + newEffects: { + saturation: 1, opaque: 0, contrast: 1, + posterize: 0, sepia: 0, bloom: 0 + } + }; + + /* patch for new effects */ + function initDrawable(drawable) { + if (!drawable[drawableKey]) drawable[drawableKey] = structuredClone(defaultNewEffects); + } + + const ogGetShader = render._shaderManager.getShader; + render._shaderManager.getShader = function (drawMode, effectBits) { + const shader = ogGetShader.call(this, drawMode, effectBits); + const gl = render._gl; + + // add uniforms to the existing shader + newUniforms.forEach(name => { + shader.uniformSetters[name] = gl.getUniformLocation(shader.program, name); + }); + return shader; + }; + + // Clear the renderer"s shader cache since we"re patching shaders + for (const cache of Object.values(render._shaderManager._shaderCache)) { + for (const programInfo of cache) { + if (programInfo) render.gl.deleteProgram(programInfo.program); + } + cache.length = 0; + } + + let patchShaders = false; + const ogCreateProgramInfo = twgl.createProgramInfo; + twgl.createProgramInfo = function (...args) { + // perform a string find-and-replace on the shader text + if (patchShaders && args[1] && args[1][0] && args[1][1]) { + args[1][0] = args[1][0] + // replace attribute properties with variables we can modify + .replaceAll("vec4(a_position", "vec4(positionSP") + .replace("v_texCoord = a_texCoord;", "") + .replace("#if !(defined(DRAW_MODE_line) || defined(DRAW_MODE_background))", "#if 1") + .replace( + `void main() {`, + `uniform vec2 u_warpSP[4]; + +void main() { + vec2 positionSP = a_position; + #ifndef DRAW_MODE_background + v_texCoord = a_texCoord; + #endif + + float u = v_texCoord.x; + float v = v_texCoord.y; + + // apply position warp (bilinear) + vec2 warpedPos = + (1.0 - u) * (1.0 - v) * u_warpSP[0] + u * (1.0 - v) * u_warpSP[1] + + u * v * u_warpSP[2] + (1.0 - u) * v * u_warpSP[3]; + + // compute w for perspective correction + float w = (1.0 - u) * (1.0 - v) + u * (1.0 - v) + u * v + (1.0 - u) * v; + + positionSP = warpedPos / max(w, 1e-5); + + #ifdef DRAW_MODE_background + gl_Position = vec4(positionSP * 2.0, 0, 1); + #else + gl_Position = u_projectionMatrix * u_modelMatrix * vec4(positionSP, 0, 1); + #endif` + ); + + args[1][1] = args[1][1].replace( + `uniform sampler2D u_skin;`, + `uniform sampler2D u_skin; +uniform sampler2D u_maskTextureSP; +uniform float u_shouldMaskSP; + +#define MAX_REPLACERS 15 +uniform vec3 u_replaceColorFromSP[MAX_REPLACERS]; +uniform vec4 u_replaceColorToSP[MAX_REPLACERS]; +uniform float u_replaceThresholdSP[MAX_REPLACERS]; +uniform int u_numReplacersSP; + +uniform vec4 u_tintColorSP; +uniform float u_saturateSP; +uniform float u_opaqueSP; +uniform float u_contrastSP; +uniform float u_posterizeSP; +uniform float u_sepiaSP; +uniform float u_bloomSP; + +vec3 spRGB2HSV(vec3 c) { + vec4 K = vec4(0.0, -1.0 / 3.0, 2.0 / 3.0, -1.0); + vec4 p = mix(vec4(c.bg, K.wz), vec4(c.gb, K.xy), step(c.b, c.g)); + vec4 q = mix(vec4(p.xyw, c.r), vec4(c.r, p.yzx), step(p.x, c.r)); + + float d = q.x - min(q.w, q.y); + float e = 1.0e-10; + return vec3(abs(q.z + (q.w - q.y) / (6.0 * d + e)), d / (q.x + e), q.x); +} +vec3 spHSV2RGB(vec3 c) { + vec4 K = vec4(1.0, 2.0 / 3.0, 1.0 / 3.0, 3.0); + vec3 p = abs(fract(c.xxx + K.xyz) * 6.0 - K.www); + return c.z * mix(K.xxx, clamp(p - K.xxx, 0.0, 1.0), c.y); +}` + ).replace( + `gl_FragColor.rgb = clamp(gl_FragColor.rgb / (gl_FragColor.a + epsilon), 0.0, 1.0);`, + `gl_FragColor.rgb = clamp(gl_FragColor.rgb / (gl_FragColor.a + epsilon), 0.0, 1.0); +vec3 finalColor = gl_FragColor.rgb; +float finalAlpha = gl_FragColor.a; + +if (u_shouldMaskSP > 0.5 && finalAlpha > 0.0) { + vec4 maskColor = texture2D(u_maskTextureSP, texcoord0); + maskColor.rgb = clamp(maskColor.rgb / (maskColor.a + epsilon), 0.0, 1.0); + finalAlpha *= maskColor.a; +} + +for (int i = 0; i < MAX_REPLACERS; i++) { + if (i >= u_numReplacersSP) break; + + float dist = distance(finalColor, u_replaceColorFromSP[i]); + if (dist <= u_replaceThresholdSP[i]) { + float strength = 1.0 - (dist / (u_replaceThresholdSP[i] + 1.0)); + finalColor = mix(finalColor, u_replaceColorToSP[i].rgb, strength); + if (u_replaceColorToSP[i].a < 1.0 && strength > 0.01) { + finalAlpha = clamp(mix(finalAlpha, u_replaceColorToSP[i].a, strength), 0.0, 1.0); + } + } +} + +vec3 hsv = spRGB2HSV(finalColor); +if (u_saturateSP < 0.0) { + hsv.x = mod(hsv.x + 0.5, 1.0); + hsv.y *= -u_saturateSP; +} else { + hsv.y *= u_saturateSP; +} +finalColor = spHSV2RGB(hsv); +finalColor = (finalColor - 0.5) * u_contrastSP + 0.5; +if (u_posterizeSP > 0.0) finalColor = floor(finalColor * u_posterizeSP) / u_posterizeSP; + +if (u_sepiaSP > 0.0) { + vec3 sepiaColor = vec3( + dot(finalColor, vec3(0.393, 0.769, 0.189)), + dot(finalColor, vec3(0.349, 0.686, 0.168)), + dot(finalColor, vec3(0.272, 0.534, 0.131)) + ); + finalColor = mix(finalColor, sepiaColor, u_sepiaSP); +} +if (u_bloomSP > 0.0) { + vec3 bloom = max(finalColor - 0.4, 0.0); + + bloom += texture2D(u_skin, v_texCoord + vec2( 0.001, 0.001)).rgb; + bloom += texture2D(u_skin, v_texCoord + vec2(-0.001, 0.001)).rgb; + bloom += texture2D(u_skin, v_texCoord + vec2( 0.001, -0.001)).rgb; + bloom += texture2D(u_skin, v_texCoord + vec2(-0.001, -0.001)).rgb; + bloom *= 0.25; + + finalColor += bloom * u_bloomSP; + finalColor = clamp(finalColor, 0.0, 1.0); +} + +gl_FragColor.rgb = finalColor * u_tintColorSP.rgb; +float baseAlpha = finalAlpha; +if (baseAlpha > 0.0 && baseAlpha < 1.0) baseAlpha = mix(baseAlpha, 1.0, u_opaqueSP); +gl_FragColor.a = baseAlpha;` + ).replaceAll( + // The unpremultiply code will now always run due to palette replacement stuff. + // This is a bit more inefficient, but whatever. + "#if defined(ENABLE_color) || defined(ENABLE_brightness)", + // i have no idea how webgl works, and i don"t want to have to remove the #endif somehow + // just do something that will always be true -CST + "#if defined(MAX_REPLACERS)" + ); + } + return ogCreateProgramInfo.apply(this, args); + }; + const ogBuildShader = render._shaderManager._buildShader; + render._shaderManager._buildShader = function (...args) { + try { + patchShaders = true; + return ogBuildShader.apply(this, args); + } finally { + patchShaders = false; + } + }; + + const MAX_REPLACERS = 15; + // Clipping and Blending Support + let toCorrectThing = null, active = false, flipY = false; + const canvas = render.canvas; + let width = 0, height = 0; + let scratchUnitWidth = 480, scratchUnitHeight = 360; + + render._drawThese = function (drawables, drawMode, projection, opts = {}) { + const gl = render._gl; + let currentShader = null; + + const framebufferSpaceScaleDiffers = ( + "framebufferWidth" in opts && "framebufferHeight" in opts && + opts.framebufferWidth !== render._nativeSize[0] && opts.framebufferHeight !== render._nativeSize[1] + ); + + const numDrawables = drawables.length; + for (let drawableIndex = 0; drawableIndex < numDrawables; ++drawableIndex) { + const drawableID = drawables[drawableIndex]; + if (opts.filter && !opts.filter(drawableID)) continue; + + const drawable = render._allDrawables[drawableID]; + if (!drawable.getVisible() && !opts.ignoreVisibility) continue; + + const drawableScale = framebufferSpaceScaleDiffers ? [ + drawable.scale[0] * opts.framebufferWidth / render._nativeSize[0], + drawable.scale[1] * opts.framebufferHeight / render._nativeSize[1] + ] : drawable.scale; + + if (!drawable.skin || !drawable.skin.getTexture(drawableScale)) continue; + if (opts.skipPrivateSkins && drawable.skin.private) continue; + + const uniforms = {}; + + let effectBits = drawable.enabledEffects; + effectBits &= Object.prototype.hasOwnProperty.call(opts, "effectMask") ? opts.effectMask : effectBits; + const newShader = render._shaderManager.getShader(drawMode, effectBits); + + if (render._regionId !== newShader) { + render._doExitDrawRegion(); + render._regionId = newShader; + + currentShader = newShader; + gl.useProgram(currentShader.program); + twgl.setBuffersAndAttributes(gl, currentShader, render._bufferInfo); + Object.assign(uniforms, { u_projectionMatrix: projection }); + } + + /* handle new effects */ + initDrawable(drawable); + const effectData = drawable[drawableKey]; + const replacers = effectData.replacers; + + const replaceFrom = new Float32Array(MAX_REPLACERS * 3).fill(0); + const replaceTo = new Float32Array(MAX_REPLACERS * 4).fill(0); + const replaceThresh = new Float32Array(MAX_REPLACERS).fill(1); + if (replacers.length > 0) { + for (let i = 0; i < Math.min(replacers.length, MAX_REPLACERS); i++) { + replaceFrom.set(replacers[i].targetVert, i * 3); + replaceTo.set(replacers[i].replaceVert, i * 4); + replaceThresh[i] = replacers[i].soft; + } + } + + if (effectData.shouldMask) { + gl.activeTexture(gl.TEXTURE30); + gl.bindTexture(gl.TEXTURE_2D, effectData._maskTexture); + gl.uniform1i(currentShader.uniformSetters["u_maskTextureSP"], 30); + gl.activeTexture(gl.TEXTURE0); + } + + const newUniformSetters = { + u_replaceColorFromSP: { method: "uniform3fv", value: replaceFrom }, + u_replaceColorToSP: { method: "uniform4fv", value: replaceTo }, + u_replaceThresholdSP: { method: "uniform1fv", value: replaceThresh }, + u_numReplacersSP: { method: "uniform1i", value: replacers ? Math.min(replacers.length, MAX_REPLACERS) : 0 }, + u_tintColorSP: { method: "uniform4fv", value: effectData.tint }, + u_warpSP: { method: "uniform2fv", value: effectData.warp }, + u_shouldMaskSP: { method: "uniform1f", value: effectData.shouldMask }, + u_saturateSP: { method: "uniform1f", value: effectData.newEffects.saturation }, + u_opaqueSP: { method: "uniform1f", value: effectData.newEffects.opaque }, + u_contrastSP: { method: "uniform1f", value: effectData.newEffects.contrast }, + u_posterizeSP: { method: "uniform1f", value: effectData.newEffects.posterize }, + u_sepiaSP: { method: "uniform1f", value: effectData.newEffects.sepia }, + u_bloomSP: { method: "uniform1f", value: effectData.newEffects.bloom } + }; + + Object.entries(newUniformSetters).forEach(([key, { method, value }]) => { + if (currentShader.uniformSetters[key]) gl[method](currentShader.uniformSetters[key], value); + }); + /* end of new effects */ + + Object.assign(uniforms, drawable.skin.getUniforms(drawableScale), drawable.getUniforms()); + if (opts.extraUniforms) Object.assign(uniforms, opts.extraUniforms); + + if (uniforms.u_skin) { + twgl.setTextureParameters(gl, uniforms.u_skin, { minMag: drawable.skin.useNearest(drawableScale, drawable) ? gl.NEAREST : gl.LINEAR }); + } + + twgl.setUniforms(currentShader, uniforms); + twgl.drawBufferInfo(gl, render._bufferInfo, gl.TRIANGLES); + } + render._regionId = null; + + // Clipping and Blending Support + gl.disable(gl.SCISSOR_TEST); + gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA); + active = false; + }; + + // reset on stop/start/clear + const ogClearEffects = vm.exports.RenderedTarget.prototype.clearEffects; + vm.exports.RenderedTarget.prototype.clearEffects = function() { + const drawable = render._allDrawables[this.drawableID]; + drawable[drawableKey] = structuredClone(defaultNewEffects); + ogClearEffects.call(this); + }; + + // manipulate bounds for warping + const radianConverter = Math.PI / 180; + function rotatePoint(x, y, cx, cy, rads) { + const cos = Math.cos(rads), sin = Math.sin(rads); + const dx = x - cx, dy = y - cy; + return { + x: cx + dx * cos - dy * sin, + y: cy + dx * sin + dy * cos + }; + } + function warpBounds(drawable, bounds) { + if (!drawable[drawableKey]) return bounds; + + let warpVals = drawable[drawableKey].warp; + if (warpVals.join(",") === defaultNewEffects.warp.join(",")) return bounds; + + // original getBounds already accounts for rotation, so we have to make our own system + // for getting the non-rotated scale and position + warpVals = warpVals.map((v, i) => i > 0 && i < 5 ? v * -1 : v); + const angle = (drawable._direction - 90) * radianConverter; + const [x, y] = drawable._position; + const width = drawable.skin.size[0] * (drawable.scale[0] / 200); + const height = drawable.skin.size[1] * (drawable.scale[1] / 200); + + const points = [ + { x: (warpVals[0] * 2) * -width + x, y: (warpVals[1] * -2) * height - y }, + { x: (warpVals[2] * 2) * width + x, y: (warpVals[3] * -2) * height - y }, + { x: (warpVals[4] * 2) * width + x, y: (warpVals[5] * -2) * -height - y }, + { x: (warpVals[6] * 2) * -width + x, y: (warpVals[7] * -2) * -height - y } + ]; + + const rotatedPoints = points.map(p => rotatePoint(p.x, p.y, x, -y, angle)); + const xs = rotatedPoints.map(p => p.x); + const ys = rotatedPoints.map(p => p.y); + + bounds.left = Math.min(...xs); + bounds.top = -Math.min(...ys); + bounds.right = Math.max(...xs); + bounds.bottom = -Math.max(...ys); + return bounds; + } + + const ogGetBounds = render.exports.Drawable.prototype.getBounds; + render.exports.Drawable.prototype.getBounds = function() { + return warpBounds(this, ogGetBounds.call(this)); + }; + const ogGetAABB = render.exports.Drawable.prototype.getAABB; + render.exports.Drawable.prototype.getAABB = function() { + return warpBounds(this, ogGetAABB.call(this)); + }; + + // update the pen shader + if (runtime.ext_pen && runtime.ext_pen._penSkinId > -1) { + const penSkin = render._allSkins[runtime.ext_pen._penSkinId]; + const gl = render.gl; + penSkin._lineShader = render._shaderManager.getShader("line", 0); + penSkin._drawTextureShader = render._shaderManager.getShader("default", 0); + penSkin.a_position_loc = gl.getAttribLocation(penSkin._lineShader.program, "a_position"); + penSkin.a_lineColor_loc = gl.getAttribLocation(penSkin._lineShader.program, "a_lineColor"); + penSkin.a_lineThicknessAndLength_loc = gl.getAttribLocation(penSkin._lineShader.program, "a_lineThicknessAndLength"); + penSkin.a_penPoints_loc = gl.getAttribLocation(penSkin._lineShader.program, "a_penPoints"); + } + + // Clipping and Blending Support + if (vm.extensionManager._loadedExtensions.has("xeltallivclipblend")) { + const gl = render._gl; + /* compressed code from Clipping and Blending.js */ + const bfb = gl.bindFramebuffer; + const ogGetUniforms = render.exports.Drawable.prototype.getUniforms; + gl.bindFramebuffer=function(e,i){if(e==gl.FRAMEBUFFER){if(null==i)toCorrectThing=!0,flipY=!1,width=canvas.width,height=canvas.height;else if(render._penSkinId){let f=render._allSkins[render._penSkinId]._framebuffer;i==f.framebuffer?(toCorrectThing=!0,flipY=!0,width=f.width,height=f.height):toCorrectThing=!1}else toCorrectThing=!1}bfb.call(this,e,i)}; + function setupModes(e,n,a){if(e){gl.enable(gl.SCISSOR_TEST);let E=(e.x_min/scratchUnitWidth+.5)*width|0,S=(e.y_min/scratchUnitHeight+.5)*height|0,N=(e.x_max/scratchUnitWidth+.5)*width|0,b,l=((e.y_max/scratchUnitHeight+.5)*height|0)-S;a&&(S=(-e.y_max/scratchUnitHeight+.5)*height|0),gl.scissor(E,S,N-E,l)}else gl.disable(gl.SCISSOR_TEST);switch(n){case"additive":gl.blendEquation(gl.FUNC_ADD),gl.blendFunc(gl.ONE,gl.ONE);break;case"subtract":gl.blendEquation(gl.FUNC_REVERSE_SUBTRACT),gl.blendFunc(gl.ONE,gl.ONE);break;case"multiply":gl.blendEquation(gl.FUNC_ADD),gl.blendFunc(gl.DST_COLOR,gl.ONE_MINUS_SRC_ALPHA);break;case"invert":gl.blendEquation(gl.FUNC_ADD),gl.blendFunc(gl.ONE_MINUS_DST_COLOR,gl.ONE_MINUS_SRC_COLOR);break;default:gl.blendEquation(gl.FUNC_ADD),gl.blendFunc(gl.ONE,gl.ONE_MINUS_SRC_ALPHA)}} + render.exports.Drawable.prototype.getUniforms=function(){return active&&toCorrectThing&&setupModes(this.clipbox,this.blendMode,flipY),ogGetUniforms.call(this)}; + } + + // this will allow clones to inherit parent effects + const ogInitDrawable = vm.exports.RenderedTarget.prototype.initDrawable; + vm.exports.RenderedTarget.prototype.initDrawable = function(layerGroup) { + ogInitDrawable.call(this, layerGroup); + if (this.isOriginal) return; + + const parentSprite = this.sprite.clones[0]; // clone[0] is always the original + const parentDrawable = render._allDrawables[parentSprite.drawableID]; + if (!parentDrawable[drawableKey]) return; + + const drawable = render._allDrawables[this.drawableID]; + drawable[drawableKey] = structuredClone(parentDrawable[drawableKey]); + }; + + /* patch for "when costume switches" event */ + const ogSetCoreCostume = looksCore.constructor.prototype._setCostume; + ogSetCoreCostume.constructor.prototype._setCostume = function (target, requestedCostume, optZeroIndex) { + ogSetCostume.call(this, target, requestedCostume, optZeroIndex); + runtime.startHats( + "SPlooksExpanded_whenCostumeSwitch", + { COSTUME: target.getCurrentCostume()?.name || "" } + ); + }; + const ogSetSpriteCostume = vm.exports.RenderedTarget.prototype.setCostume; + vm.exports.RenderedTarget.prototype.setCostume = function (index) { + ogSetSpriteCostume.call(this, index); + runtime.startHats( + "SPlooksExpanded_whenCostumeSwitch", + { COSTUME: this.getCurrentCostume()?.name || "" } + ); + }; + + class SPlooksExpanded { + getInfo() { + return { + id: "SPlooksExpanded", + name: "Looks Expanded", + color1: "#9966FF", + color2: "#855CD6", + color3: "#774DCB", + menuIconURI, + blocks: [ + { + opcode: "getSpeech", + blockType: Scratch.BlockType.REPORTER, + extensions: ["colours_looks"], + text: "speech from [TARGET]", + arguments: { + TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" } + } + }, + "---", + { + opcode: "costumeCnt", + blockType: Scratch.BlockType.REPORTER, + extensions: ["colours_looks"], + text: "# of costumes in [TARGET]", + arguments: { + TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" } + } + }, + { + opcode: "costumeInfo", + blockType: Scratch.BlockType.REPORTER, + extensions: ["colours_looks"], + text: "[INFO] of costume # [NUM] in [TARGET]", + arguments: { + INFO: { type: Scratch.ArgumentType.STRING, menu: "COSTUME_DATA" }, + NUM: { type: Scratch.ArgumentType.NUMBER, defaultValue: 1 }, + TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" } + } + }, + { + opcode: "setTargetCostume", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_looks"], + text: "switch costume of [TARGET] to [VALUE]", + arguments: { + TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" }, + VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: "..." }, + } + }, + { + opcode: "whenCostumeSwitch", + blockType: Scratch.BlockType.EVENT, + extensions: ["colours_event"], + isEdgeActivated: false, + text: "when costume switches to [COSTUME]", + arguments: { + COSTUME: { type: Scratch.ArgumentType.STRING, menu: "COSTUMES" }, + } + }, + "---", + { + opcode: "setSpriteEffect", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_looks"], + text: "set [EFFECT] of [TARGET] to [VALUE]", + arguments: { + EFFECT: { type: Scratch.ArgumentType.STRING, menu: "EFFECT_MENU" }, + TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" }, + VALUE: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 } + } + }, + { + opcode: "effectValue", + blockType: Scratch.BlockType.REPORTER, + extensions: ["colours_looks"], + text: "[EFFECT] effect of [TARGET]", + arguments: { + EFFECT: { type: Scratch.ArgumentType.STRING, menu: "EFFECT_MENU" }, + TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" } + }, + }, + { + opcode: "tintSprite", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_looks"], + text: "set tint of [TARGET] to [COLOR]", + arguments: { + TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" }, + COLOR: { type: Scratch.ArgumentType.COLOR } + } + }, + "---", + { + opcode: "replaceColor", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_looks"], + text: "replace [COLOR1] with [COLOR2] in [TARGET] softness [VALUE]", + arguments: { + COLOR1: { type: Scratch.ArgumentType.COLOR }, + COLOR2: { type: Scratch.ArgumentType.COLOR }, + TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" }, + VALUE: { type: Scratch.ArgumentType.NUMBER, defaultValue: 10 }, + } + }, + { + opcode: "resetColor", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_looks"], + text: "reset [COLOR1] replacer in [TARGET]", + arguments: { + COLOR1: { type: Scratch.ArgumentType.COLOR }, + TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" } + } + }, + { + opcode: "resetReplacers", + blockType: Scratch.BlockType.COMMAND, + extensions: ["colours_looks"], + text: "reset color replacers in [TARGET]", + arguments: { + TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" } + } + }, + { + blockType: Scratch.BlockType.XML, + xml: "