Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add Layer Control Extension #1253

Open
wants to merge 10 commits into
base: master
Choose a base branch
from
236 changes: 236 additions & 0 deletions extensions/Layer-Control.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,236 @@
// Name: Layer Control
// ID: SPlayering
// Description: Relayer Pen, Video Camera, Backdrops, Sprites and More
// By: SharkPool

// Version V.1.1.2

(function (Scratch) {
"use strict";
if (!Scratch.extensions.unsandboxed) throw new Error("Layer Control must run unsandboxed!");

const vm = Scratch.vm;
const runtime = vm.runtime;
const render = vm.renderer;
let allowRunBlocks = true;

const menuIconURI =
"";

/* Layer Repair (Ripped from "More Events")
The Renderer will Error when the Stage/Pen/Video is Relayered and when
loading a Project into the editor with that same project already open.

This Function Fixes it by Reseting Special Layers and Temporarily skipping
the "SPlayering" Blocks
*/
const beforeSave = () =>
new Promise((resolve) => {
allowRunBlocks = false;
const instanceClass = new SPlayering();
instanceClass["reset"]();
resolve();
});
const ogSaveProjectSb3 = vm.saveProjectSb3;
vm.saveProjectSb3 = async function (...args) {
await beforeSave();
const result = await ogSaveProjectSb3.apply(this, args);
allowRunBlocks = true;
return result;
};
const ogSaveProjectSb3Stream = vm.saveProjectSb3Stream;
vm.saveProjectSb3Stream = function (...args) {
let realStream = null;
const queuedCalls = [];
const whenStreamReady = (methodName, args) => {
if (realStream) return realStream[methodName].apply(realStream, args);
else return new Promise((resolve) => { queuedCalls.push({ resolve, methodName, args }) });
};
const streamWrapper = {
on: (...args) => void whenStreamReady("on", args),
pause: (...args) => void whenStreamReady("pause", args),
resume: (...args) => void whenStreamReady("resume", args),
accumulate: (...args) => whenStreamReady("accumulate", args)
};
beforeSave().then(() => {
realStream = ogSaveProjectSb3Stream.apply(this, args);
realStream.on("end", () => { allowRunBlocks = true });
for (const queued of queuedCalls) {
queued.resolve(realStream[queued.methodName].apply(realStream, queued.args));
}
queuedCalls.length = 0;
});
return streamWrapper;
};

// Also do this for Restore Points
const ogSaveProjectNonZIP = vm.saveProjectSb3DontZip;
vm.saveProjectSb3DontZip = function (...args) {
allowRunBlocks = false;
const instanceClass = new SPlayering();
instanceClass["reset"]();
allowRunBlocks = true;
return ogSaveProjectNonZIP.apply(this, args);
};

class SPlayering {
getInfo() {
return {
id: "SPlayering",
name: Scratch.translate("Layer Control"),
color1: "#bb71e6",
menuIconURI,
blocks: [
{
opcode: "getAllID",
blockType: Scratch.BlockType.REPORTER,
text: Scratch.translate("get all layer IDs")
},
{
opcode: "getID",
blockType: Scratch.BlockType.REPORTER,
text: Scratch.translate("get ID layer of [TARGET]"),
arguments: {
TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" }
}
},
{
opcode: "getOwner",
blockType: Scratch.BlockType.REPORTER,
text: Scratch.translate("get owner of layer ID [ID]"),
arguments: {
ID: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }
}
},
{
opcode: "moveID",
blockType: Scratch.BlockType.COMMAND,
text: Scratch.translate("relayer ID [TARGET] to layer [ORDER]"),
arguments: {
ORDER: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 },
TARGET: { type: Scratch.ArgumentType.NUMBER, defaultValue: 1 }
}
},
{
opcode: "swapID",
blockType: Scratch.BlockType.COMMAND,
text: Scratch.translate("swap layers of ID [TARGET] with ID [ORDER]"),
arguments: {
ORDER: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 },
TARGET: { type: Scratch.ArgumentType.NUMBER, defaultValue: 1 }
}
},
{
opcode: "reset",
blockType: Scratch.BlockType.COMMAND,
text: Scratch.translate("reset special layers")
}
],
menus: {
TARGETS: { acceptReporters: true, items: "_getTargets" }
}
};
}

_getTargets() {
const spriteNames = [
{ text: Scratch.translate("myself"), value: "_myself_" },
{ text: Scratch.translate("Stage"), value: "_stage_" },
{ text: Scratch.translate("Video Layer"), value: "_video_" },
{ text: Scratch.translate("Pen Layer"), value: "_pen_" }
];
// Custom Drawable Layer (CST's 3D or Simple3D Exts for Example)
for (var i = 0; i < render._allDrawables.length; i++) {
const drawable = render._allDrawables[i];
if (drawable !== undefined && drawable.customDrawableName !== undefined) spriteNames.push({
text: drawable.customDrawableName, value: `${i}=SP-custLayer`
});
}
// Sprites
const targets = runtime.targets;
for (let index = 1; index < targets.length; index++) {
const target = targets[index];
if (target.isOriginal) spriteNames.push({ text: target.getName(), value: target.getName() });
}
return spriteNames.length > 0 ? spriteNames : [""];
}

getAllID() { return JSON.stringify(render._drawList) }

getID(args, util) {
if (args.TARGET === "_myself_") return util.target.drawableID;
if (args.TARGET === "_stage_") return runtime.getTargetForStage().drawableID;
if (args.TARGET === "_pen_") return runtime.ext_pen?._penDrawableId || "";
const videoL = runtime.ioDevices.video._drawable;
if (args.TARGET === "_video_") return videoL !== -1 ? videoL : "";
if (args.TARGET.includes("=SP-custLayer")) {
const layerID = parseInt(args.TARGET);
if (render._allDrawables[layerID]?.customDrawableName !== undefined) return layerID;
}
const target = runtime.getSpriteTargetByName(args.TARGET);
return target ? target.drawableID : "";
}

getOwner(args, util) {
const ID = Scratch.Cast.toNumber(args.ID); // Empty Number Inputs are always 0?
if (ID < 0) return "";
const penID = runtime.ext_pen?._penDrawableId || "";
const videoL = runtime.ioDevices.video._drawable;
const vidID = videoL !== -1 ? videoL : "";
if (ID === penID) return "Pen Layer";
if (ID === vidID) return "Video Layer";
// Sprite Check
for (const target of runtime.targets) {
if (target.drawableID === ID) return `${target.getName()}${target.isOriginal ? "" : " (Clone)"}`;
}
// Custom Layer Check
for (var i = 0; i < render._allDrawables.length; i++) {
const drawable = render._allDrawables[i];
if (drawable.customDrawableName !== undefined && i === ID)
return drawable.customDrawableName;
}
return "";
}

reset() {
const allLay = render._drawList;
const stage = runtime.getTargetForStage().drawableID;
const pen = runtime.ext_pen?._penDrawableId || -1;
const video = runtime.ioDevices.video._drawable || -1;
const newDraw = [stage, ...allLay.filter(item => ![stage, pen, video].includes(item))];
if (video !== -1) newDraw.splice(1, 0, video);
if (pen !== -1) newDraw.splice(video !== -1 ? 2 : 1, 0, pen);
render._drawList = newDraw;
runtime.targets[0].updateAllDrawableProperties();
runtime.requestRedraw();
}

moveID(args) {
if (!allowRunBlocks) return;
const allLay = render._drawList;
const index = allLay.indexOf(args.TARGET);
const targetL = Math.max(0, Math.min(allLay.length - 1, Math.round(args.ORDER)));
if (index !== -1) {
const removedLayer = allLay.splice(index, 1)[0];
render._drawList.splice(targetL, 0, removedLayer);
runtime.targets[0].updateAllDrawableProperties();
runtime.requestRedraw();
}
}

swapID(args) {
if (!allowRunBlocks) return;
const allLay = render._drawList;
const target1 = allLay.indexOf(args.TARGET);
const target2 = allLay.indexOf(args.ORDER);
if (target1 === -1 || target2 === -1) return;
if (target1 !== target2 && target1 >= 0 && target1 < allLay.length && target2 >= 0 && target2 < allLay.length) {
[allLay[target1], allLay[target2]] = [allLay[target2], allLay[target1]];
runtime.targets[0].updateAllDrawableProperties();
runtime.requestRedraw();
}
}
}

Scratch.extensions.register(new SPlayering());
})(Scratch);
Loading
Loading