diff --git a/extensions/SharkPool/Animations.js b/extensions/SharkPool/Animations.js new file mode 100644 index 0000000000..15e0eb6139 --- /dev/null +++ b/extensions/SharkPool/Animations.js @@ -0,0 +1,748 @@ +// Name: Animations +// ID: SPanimations +// Description: Play Animations for your Sprites +// By: SharkPool +// Licence: MIT + +// Version V.2.0.02 + +(function (Scratch) { + "use strict"; + if (!Scratch.extensions.unsandboxed) + throw new Error("Animations must run unsandboxed"); + + const vm = Scratch.vm; + const runtime = vm.runtime; + const render = vm.renderer; + const specialID = `SPspecialID${Math.random()}`; + const playTypes = { + normally: 1, + "in reverse": 2, + "looped normally": 3, + "looped reversed": 4, + }; + + let allAnimations = {}; + + const menuIconURI = + ""; + const blockIconURI = + ""; + + const playIconURI = + ""; + const keyIconURI = + ""; + + function doForEachAnimation(func) { + const targets = Object.values(allAnimations); + for (let i = 0; i < targets.length; i++) { + const targetCache = targets[i]; + const targAnims = Object.values(targetCache); + for (let j = 0; j < targAnims.length; j++) func(targAnims[j]); + } + } + + class SPanimations { + constructor() { + runtime.on("PROJECT_STOP_ALL", () => { + doForEachAnimation((anim) => { + anim.playing = false; + }); + }); + runtime.on("PROJECT_START", () => { + doForEachAnimation((anim) => { + anim.playing = false; + }); + }); + runtime.on("AFTER_EXECUTE", () => { + if (runtime.ioDevices.clock._paused) return; + doForEachAnimation((anim) => { + if (anim.playing) { + const isReverse = anim.playType === 2 || anim.playType === 4; + anim.timer += 0.01; + if (anim.timer > anim.buffer) { + anim.buffer += anim.fps / 1000; + if (isReverse) anim.currentFrame--; + else anim.currentFrame++; + + // determine if this frame is a keyframe/costume + const thisFrame = anim.frames[anim.currentFrame] ?? ""; + if (thisFrame.startsWith(specialID)) { + const keyframe = anim.specialFrames[thisFrame]; + if (keyframe.type === "pause") anim.buffer += keyframe.secs; + else this.keyframeUpdate(keyframe, anim, true); + } else if (thisFrame) { + const index = anim.target.getCostumeIndexByName(thisFrame); + if (index > -1) anim.target.setCostume(index); + else + console.warn(`Animations -- Invalid Costume (${thisFrame})`); + } + } + // complete any unfinished keyframes + anim.keyBuffers.forEach((key) => { + this.keyframeUpdate(key, anim, false); + }); + + const frameCnt = anim.frames.length; + const frameCheck = isReverse + ? anim.currentFrame <= 1 + : anim.currentFrame >= frameCnt - 1; + if (frameCheck && anim.keyBuffers.length === 0) { + this.resetAnimPlayer(anim); + if (isReverse) anim.currentFrame = frameCnt; + if (anim.playType < 3) anim.playing = false; + } + } + }); + }); + } + getInfo() { + return { + id: "SPanimations", + name: "Animations", + color1: "#7a7a7a", + color2: "#444444", + color3: "#232323", + menuIconURI, + blockIconURI, + blocks: [ + { + opcode: "createAnimation", + blockType: Scratch.BlockType.COMMAND, + text: "make new animation named [NAME]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + }, + }, + { + opcode: "removeAnimation", + blockType: Scratch.BlockType.COMMAND, + text: "delete animation named [NAME]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + }, + }, + { + opcode: "removeAll", + blockType: Scratch.BlockType.COMMAND, + text: "delete all animations", + }, + { + opcode: "isExists", + blockType: Scratch.BlockType.BOOLEAN, + text: "animation [NAME] exists?", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + }, + }, + { + opcode: "allAnimationsX", + blockType: Scratch.BlockType.REPORTER, + text: "all [TYPE] animations", + arguments: { + TYPE: { type: Scratch.ArgumentType.STRING, menu: "pullTypes" }, + }, + }, + { blockType: Scratch.BlockType.LABEL, text: "Frames" }, + { + opcode: "addFrame", + blockType: Scratch.BlockType.COMMAND, + text: "add [COSTUME] to [NAME]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + COSTUME: { type: Scratch.ArgumentType.COSTUME }, + }, + }, + { + opcode: "removeFrame", + blockType: Scratch.BlockType.COMMAND, + text: "remove [COSTUME] from [NAME]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + COSTUME: { type: Scratch.ArgumentType.COSTUME }, + }, + }, + { + opcode: "addAllFrames", + blockType: Scratch.BlockType.COMMAND, + text: "add costumes from [COS1] to [COS2] to [NAME]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + COS1: { type: Scratch.ArgumentType.NUMBER, defaultValue: 1 }, + COS2: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 }, + }, + }, + { + opcode: "removeAllFrames", + blockType: Scratch.BlockType.COMMAND, + text: "remove frames [COS1] to [COS2] from [NAME]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + COS1: { type: Scratch.ArgumentType.NUMBER, defaultValue: 1 }, + COS2: { type: Scratch.ArgumentType.NUMBER, defaultValue: 5 }, + }, + }, + "---", + { + opcode: "addPause", + blockType: Scratch.BlockType.COMMAND, + text: "add a [SECOND] second pause to [NAME] with ID [ID]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + SECOND: { type: Scratch.ArgumentType.NUMBER, defaultValue: 3 }, + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "pause1" }, + }, + }, + { + opcode: "removePause", + blockType: Scratch.BlockType.COMMAND, + text: "remove pause frame from [NAME] with ID [ID]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "pause1" }, + }, + }, + "---", + { + opcode: "numFrames", + blockType: Scratch.BlockType.REPORTER, + text: "number of frames in [NAME]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + }, + }, + { + opcode: "frameNames", + blockType: Scratch.BlockType.REPORTER, + text: "all frames in [NAME]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + }, + }, + { + opcode: "frameName", + blockType: Scratch.BlockType.REPORTER, + text: "frame # [FRAME] in [NAME]", + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + FRAME: { type: Scratch.ArgumentType.NUMBER, defaultValue: 1 }, + }, + }, + { blockType: Scratch.BlockType.LABEL, text: "Playback" }, + { + opcode: "setFPS", + blockType: Scratch.BlockType.COMMAND, + text: "set FPS of [NAME] to [FPS]", + blockIconURI: playIconURI, + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + FPS: { type: Scratch.ArgumentType.NUMBER, defaultValue: 30 }, + }, + }, + { + opcode: "playBack", + blockType: Scratch.BlockType.COMMAND, + text: "play animation [NAME] [TYPE]", + blockIconURI: playIconURI, + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + TYPE: { type: Scratch.ArgumentType.STRING, menu: "playBack" }, + }, + }, + { + opcode: "playBackWait", + blockType: Scratch.BlockType.COMMAND, + text: "play animation [NAME] [TYPE] until done", + blockIconURI: playIconURI, + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + TYPE: { type: Scratch.ArgumentType.STRING, menu: "playBack" }, + }, + }, + { + opcode: "stopAnimation", + blockType: Scratch.BlockType.COMMAND, + text: "stop animation [NAME]", + blockIconURI: playIconURI, + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + }, + }, + "---", + { + opcode: "currentFPS", + blockType: Scratch.BlockType.REPORTER, + text: "FPS of [NAME]", + blockIconURI: playIconURI, + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + }, + }, + { + opcode: "isPlaying", + blockType: Scratch.BlockType.BOOLEAN, + text: "is [NAME] playing?", + blockIconURI: playIconURI, + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + }, + }, + { + opcode: "currentFrame", + blockType: Scratch.BlockType.REPORTER, + text: "current frame of [NAME]", + blockIconURI: playIconURI, + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + }, + }, + { blockType: Scratch.BlockType.LABEL, text: "Keyframes" }, + { + opcode: "addPosition", + blockType: Scratch.BlockType.COMMAND, + text: "add keyframe position to [NAME] with ID [ID] start x [x] y [y] end x [x2] y [y2]", + blockIconURI: keyIconURI, + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "key1" }, + x: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }, + y: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }, + x2: { type: Scratch.ArgumentType.NUMBER, defaultValue: 100 }, + y2: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }, + }, + }, + { + opcode: "addDirection", + blockType: Scratch.BlockType.COMMAND, + text: "add keyframe direction to [NAME] with ID [ID] start [DIR1] end [DIR2]", + blockIconURI: keyIconURI, + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "key1" }, + DIR1: { type: Scratch.ArgumentType.ANGLE, defaultValue: 90 }, + DIR2: { type: Scratch.ArgumentType.ANGLE, defaultValue: 0 }, + }, + }, + { + opcode: "addScale", + blockType: Scratch.BlockType.COMMAND, + text: "add keyframe scale to [NAME] with ID [ID] start [scale]% end [scale2]%", + blockIconURI: keyIconURI, + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "key1" }, + scale: { type: Scratch.ArgumentType.NUMBER, defaultValue: 100 }, + scale2: { type: Scratch.ArgumentType.NUMBER, defaultValue: 150 }, + }, + }, + { + opcode: "addStretch", + blockType: Scratch.BlockType.COMMAND, + text: "add keyframe stretch to [NAME] with ID [ID] start width [x] height [y] end width [x2] height [y2]", + blockIconURI: keyIconURI, + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "key1" }, + x: { type: Scratch.ArgumentType.NUMBER, defaultValue: 100 }, + y: { type: Scratch.ArgumentType.NUMBER, defaultValue: 100 }, + x2: { type: Scratch.ArgumentType.NUMBER, defaultValue: 200 }, + y2: { type: Scratch.ArgumentType.NUMBER, defaultValue: 50 }, + }, + }, + "---", + { + opcode: "deleteKeyframe", + blockType: Scratch.BlockType.COMMAND, + text: "remove keyframe with ID [ID] from [NAME]", + blockIconURI: keyIconURI, + arguments: { + NAME: { + type: Scratch.ArgumentType.STRING, + defaultValue: "animation-1", + }, + ID: { type: Scratch.ArgumentType.STRING, defaultValue: "key1" }, + }, + }, + ], + menus: { + playBack: Object.keys(playTypes), + pullTypes: { acceptReporters: true, items: ["existing", "playing"] }, + }, + }; + } + + // Helper Funcs + getAnim(name, util, forAll) { + if (forAll) { + let animSearch = undefined; + doForEachAnimation((anim) => { + if (anim.name === name) animSearch = anim; + }); + return animSearch; + } + const target = util.target; + let anim = allAnimations[target.id]?.[name]; + if (anim) return anim; + else if (!target.isOriginal) { + anim = allAnimations[target.sprite.clones[0].id]?.[name]; + if (anim) { + if (allAnimations[target.id] === undefined) + allAnimations[target.id] = {}; + const cloneCopy = { ...anim }; + cloneCopy.target = target; + + allAnimations[target.id][name] = cloneCopy; + return cloneCopy; + } + } + return this.createAnimation({ NAME: name, SECRET: true }, util); + } + + resetAnimPlayer(anim) { + anim.buffer = anim.fps / 1000; + anim.timer = 0; + anim.currentFrame = -1; + } + + keyframeUpdate(data, anim, init) { + const target = anim.target; + let { type, start, end, x1, x2, y1, y2 } = data; + if (data.startStamp === undefined) data.startStamp = Date.now(); + const elapsedTime = Date.now() - data.startStamp; + const progress = Math.min(elapsedTime / (anim.fps * 20), 1); + let delta1, delta2; + switch (type) { + case "pos": + delta1 = x2 - x1; + delta2 = y2 - y1; + target.setXY(x1 + delta1 * progress, y1 + delta2 * progress); + break; + case "dir": + delta1 = end - start; + target.setDirection(start + delta1 * progress); + break; + case "size": + delta1 = end - start; + target.setSize(start + delta1 * progress); + break; + case "stretch": + delta1 = x2 - x1; + delta2 = y2 - y1; + render._allDrawables[target.drawableID].updateScale([ + x1 + delta1 * progress, + y1 + delta2 * progress, + ]); + break; + } + if (progress === 1) { + delete data.startStamp; + anim.keyBuffers.splice(anim.keyBuffers.indexOf(data), 1); + } else if (init) anim.keyBuffers.push(data); + } + + // Block Funcs + createAnimation(args, util) { + const name = Scratch.Cast.toString(args.NAME); + const id = util.target.id; + if (allAnimations[id] === undefined) allAnimations[id] = {}; + const obj = { + name, + buffer: 0, + timer: 0, + playing: false, + playType: 0, + fps: 10, + target: util.target, + frames: [], + specialFrames: {}, + keyBuffers: [], + currentFrame: -1, + }; + allAnimations[id][name] = obj; + if (args.SECRET) return obj; + } + + removeAnimation(args, util) { + const name = Scratch.Cast.toString(args.NAME); + if (this.isExists(args, util)) delete allAnimations[util.target.id][name]; + } + + removeAll() { + allAnimations = {}; + } + + isExists(args, util) { + return Scratch.Cast.toBoolean(this.getAnim(args.NAME, "", true)); + } + + addFrame(args, util) { + const anim = this.getAnim(args.NAME, util); + anim.frames.push(Scratch.Cast.toString(args.COSTUME)); + } + + removeFrame(args, util) { + const anim = this.getAnim(args.NAME, util); + const ind = anim.frames.indexOf(Scratch.Cast.toString(args.COSTUME)); + if (ind > -1) anim.frames.splice(ind, 1); + } + + addAllFrames(args, util) { + const costumes = util.target.getCostumes(); + const start = Math.min( + costumes.length, + Math.max(1, Math.round(Scratch.Cast.toNumber(args.COS1))) + ); + const end = Math.min( + costumes.length, + Math.max(start, Math.round(Scratch.Cast.toNumber(args.COS2))) + ); + const anim = this.getAnim(args.NAME, util); + for (let i = start - 1; i < end; i++) anim.frames.push(costumes[i].name); + } + + removeAllFrames(args, util) { + const costumes = util.target.getCostumes(); + const start = Math.min( + costumes.length, + Math.max(1, Math.round(Scratch.Cast.toNumber(args.COS1))) + ); + const end = Math.min( + costumes.length, + Math.max(start, Math.round(Scratch.Cast.toNumber(args.COS2))) + ); + const anim = this.getAnim(args.NAME, util); + anim.frames.splice(start - 1, end - (start - 1)); + } + + addPause(args, util) { + const id = `${specialID}pause-${Scratch.Cast.toString(args.ID)}`; + const anim = this.getAnim(args.NAME, util); + anim.frames.push(id); + anim.specialFrames[id] = { + type: "pause", + secs: Scratch.Cast.toNumber(args.SECOND), + }; + } + + removePause(args, util) { + this.deleteKeyframe(args, util); // works the same way + } + + numFrames(args) { + const anim = this.getAnim(args.NAME, "", true); + return anim ? anim.frames.length : 0; + } + + frameNames(args) { + const anim = this.getAnim(args.NAME, "", true); + if (!anim) return "[]"; + const rawFrames = structuredClone(anim.frames); + rawFrames.forEach((frame, i) => { + rawFrames[i] = frame.replace(specialID, ""); + }); + return anim ? JSON.stringify(rawFrames) : "[]"; + } + + frameName(args) { + const anim = this.getAnim(args.NAME, "", true); + const ind = Scratch.Cast.toNumber(args.FRAME) - 1; + return anim ? anim.frames[ind].replace(specialID, "") ?? "" : ""; + } + + allAnimationsX(args) { + let array = []; + if (args.TYPE === "existing") + doForEachAnimation((anim) => { + array.push(anim.name); + }); + else { + doForEachAnimation((anim) => { + if (anim.playing) array.push(anim.name); + }); + } + return JSON.stringify(array); + } + + setFPS(args, util) { + const anim = this.getAnim(args.NAME, util); + anim.fps = Scratch.Cast.toNumber(args.FPS); + } + + playBack(args, util) { + const anim = this.getAnim(args.NAME, util); + this.resetAnimPlayer(anim); + anim.playType = playTypes[args.TYPE] ?? 1; + if (anim.playType === 2 || anim.playType === 4) + anim.currentFrame = anim.frames.length; + anim.playing = true; + } + + playBackWait(args, util) { + const anim = this.getAnim(args.NAME, util); + if (util.stackFrame.executed === undefined) { + this.resetAnimPlayer(anim); + anim.playType = playTypes[args.TYPE] ?? 1; + if (anim.playType === 2 || anim.playType === 4) + anim.currentFrame = anim.frames.length; + anim.playing = true; + util.stackFrame.executed = true; + util.yield(); + } else if (anim.playing) util.yield(); + } + + stopAnimation(args, util) { + const anim = this.getAnim(args.NAME, util); + anim.playing = false; + } + + currentFPS(args) { + const anim = this.getAnim(args.NAME, "", true); + return anim ? anim.fps : ""; + } + + isPlaying(args) { + const anim = this.getAnim(args.NAME, "", true); + return anim ? anim.playing : false; + } + + currentFrame(args) { + const anim = this.getAnim(args.NAME, "", true); + return anim ? anim.currentFrame + 2 : 0; // + 2 since we start at -1 instead of 0 + } + + addPosition(args, util) { + const id = `${specialID}pos-${Scratch.Cast.toString(args.ID)}`; + const anim = this.getAnim(args.NAME, util); + anim.frames.push(id); + anim.specialFrames[id] = { + type: "pos", + x1: Scratch.Cast.toNumber(args.x), + x2: Scratch.Cast.toNumber(args.x2), + y1: Scratch.Cast.toNumber(args.y), + y2: Scratch.Cast.toNumber(args.y2), + }; + } + + addDirection(args, util) { + const id = `${specialID}dir-${Scratch.Cast.toString(args.ID)}`; + const anim = this.getAnim(args.NAME, util); + anim.frames.push(id); + anim.specialFrames[id] = { + type: "dir", + start: Scratch.Cast.toNumber(args.DIR1), + end: Scratch.Cast.toNumber(args.DIR2), + }; + } + + addScale(args, util) { + const id = `${specialID}size-${Scratch.Cast.toString(args.ID)}`; + const anim = this.getAnim(args.NAME, util); + anim.frames.push(id); + anim.specialFrames[id] = { + type: "size", + start: Scratch.Cast.toNumber(args.scale), + end: Scratch.Cast.toNumber(args.scale2), + }; + } + + addStretch(args, util) { + const id = `${specialID}stretch-${Scratch.Cast.toString(args.ID)}`; + const anim = this.getAnim(args.NAME, util); + anim.frames.push(id); + anim.specialFrames[id] = { + type: "stretch", + x1: Scratch.Cast.toNumber(args.x), + x2: Scratch.Cast.toNumber(args.x2), + y1: Scratch.Cast.toNumber(args.y), + y2: Scratch.Cast.toNumber(args.y2), + }; + } + + deleteKeyframe(args, util) { + const id = Scratch.Cast.toString(args.ID); + const anim = this.getAnim(args.NAME, util); + const specialFrames = anim.frames.filter((frame) => { + return frame.startsWith(specialID); + }); + const frame = specialFrames.find((frame) => { + return frame.endsWith(id); + }); + if (frame !== undefined) { + anim.frames.splice(anim.frames.indexOf(frame), 1); + delete anim.specialFrames[frame]; + } + } + } + + Scratch.extensions.register(new SPanimations()); +})(Scratch); diff --git a/images/SharkPool/Animations.svg b/images/SharkPool/Animations.svg new file mode 100644 index 0000000000..59aed2e475 --- /dev/null +++ b/images/SharkPool/Animations.svg @@ -0,0 +1 @@ +