diff --git a/extensions/SharkPool/Camera.js b/extensions/SharkPool/Camera.js new file mode 100644 index 0000000000..dbbfddc5aa --- /dev/null +++ b/extensions/SharkPool/Camera.js @@ -0,0 +1,781 @@ +// Name: Camera +// ID: SPcamera +// Description: Move the visible part of the stage. +// By: SharkPool +// License: MIT + +// Version V.1.0.02 + +(function (Scratch) { + "use strict"; + if (!Scratch.extensions.unsandboxed) + throw new Error("Camera must run unsandboxed!"); + + const menuIconURI = + ""; + const rightArrow = + ""; + const leftArrow = + ""; + + const Cast = Scratch.Cast; + const vm = Scratch.vm; + const runtime = vm.runtime; + const render = vm.renderer; + const isEditor = typeof scaffolding === "undefined"; + const cameraSymbol = Symbol("SPcameraData"); + + let allCameras = { + default: { + xy: [0, 0], + zoom: 1, + dir: 0, + binds: undefined, + }, + }; + + // TODO add support for interpolation at some point + // we need a api to allow pushing interpolation data + runtime.setInterpolation(false); + runtime.runtimeOptions.fencing = false; + render.offscreenTouching = true; + + // custom gui + function openModal(titleName, func) { + // in a Button Context, ScratchBlocks always exists + ScratchBlocks.prompt( + titleName, + "", + (value) => func(value), + Scratch.translate("Camera Manager"), + "broadcast_msg" + ); + } + + // camera utils + function setupState(drawable) { + drawable[cameraSymbol] = { + name: "default", + needsRefresh: false, + ogXY: [0, 0], + ogSZ: 1, + ogDir: 0, + }; + } + + function translatePosition(xy, invert, camData) { + if (invert) { + const invRads = (camData.ogDir / 180) * Math.PI; + const invSin = Math.sin(invRads), + invCos = Math.cos(invRads); + const scaledX = xy[0] / camData.ogSZ; + const scaledY = xy[1] / camData.ogSZ; + const invOffX = scaledX * invCos + scaledY * invSin; + const invOffY = -scaledX * invSin + scaledY * invCos; + return [invOffX - camData.ogXY[0], invOffY - camData.ogXY[1]]; + } else { + const rads = (camData.dir / 180) * Math.PI; + const sin = Math.sin(rads), + cos = Math.cos(rads); + const offX = xy[0] + camData.xy[0]; + const offY = xy[1] + camData.xy[1]; + return [ + camData.zoom * (offX * cos - offY * sin), + camData.zoom * (offX * sin + offY * cos), + ]; + } + } + + function bindDrawable(drawable, camera) { + if (!drawable[cameraSymbol]) setupState(drawable); + const camSystem = drawable[cameraSymbol]; + if (camSystem.name === camera) return; + + // invert camera transformations + const fixedPos = translatePosition(drawable._position, true, camSystem); + const fixedDir = drawable._direction + camSystem.ogDir; + const fixedScale = [ + drawable._scale[0] / camSystem.ogSZ, + drawable._scale[1] / camSystem.ogSZ, + ]; + + drawable[cameraSymbol] = { + name: camera, + needsRefresh: true, + ogXY: [0, 0], + ogSZ: 1, + ogDir: 0, + }; + + const id = drawable._id; + render.updateDrawablePosition(id, fixedPos); + render.updateDrawableDirection(id, fixedDir); + render.updateDrawableScale(id, fixedScale); + } + + function updateCamera(camera) { + for (let i = 0; i < render._allDrawables.length; i++) { + const drawable = render._allDrawables[i]; + if (!drawable || !drawable.getVisible() || !drawable.skin) continue; + if (!drawable[cameraSymbol]) setupState(drawable); + + const camSystem = drawable[cameraSymbol]; + if (camSystem.name === camera) { + camSystem.needsRefresh = true; + drawable.updatePosition(drawable._position); + drawable.updateDirection(drawable._direction); + drawable.updateScale(drawable._scale); + camSystem.needsRefresh = false; + } + } + } + + // camera system patches + const ogUpdatePosition = render.exports.Drawable.prototype.updatePosition; + render.exports.Drawable.prototype.updatePosition = function (position) { + if (!this[cameraSymbol]) setupState(this); + const camSystem = this[cameraSymbol]; + const thisCam = allCameras[camSystem.name]; + if (camSystem.needsRefresh) { + // invert camera transformations + position = translatePosition(position, true, camSystem); + } + + camSystem.ogXY = [...thisCam.xy]; + position = translatePosition(position, false, thisCam); + ogUpdatePosition.call(this, position); + }; + + const ogUpdateDirection = render.exports.Drawable.prototype.updateDirection; + render.exports.Drawable.prototype.updateDirection = function (direction) { + if (!this[cameraSymbol]) setupState(this); + const camSystem = this[cameraSymbol]; + const thisCam = allCameras[camSystem.name]; + if (camSystem.needsRefresh) { + // invert camera transformations + direction += camSystem.ogDir; + } + + camSystem.ogDir = thisCam.dir; + direction -= thisCam.dir; + ogUpdateDirection.call(this, direction); + }; + + const ogUpdateScale = render.exports.Drawable.prototype.updateScale; + render.exports.Drawable.prototype.updateScale = function (scale) { + if (!this[cameraSymbol]) setupState(this); + const camSystem = this[cameraSymbol]; + const thisCam = allCameras[camSystem.name]; + if (camSystem.needsRefresh) { + // invert camera transformations + scale[0] /= camSystem.ogSZ; + scale[1] /= camSystem.ogSZ; + } + + camSystem.ogSZ = thisCam.zoom; + scale[0] *= thisCam.zoom; + scale[1] *= thisCam.zoom; + ogUpdateScale.call(this, scale); + this.skin?.emitWasAltered(); + }; + + // Turbowarp Extension Storage + runtime.on("PROJECT_LOADED", () => { + const stored = runtime.extensionStorage["SPcamera"]; + if (stored) + stored.cams.forEach((cam) => { + allCameras[cam] = { + xy: [0, 0], + zoom: 1, + dir: 0, + binds: name === "default" ? undefined : [], + }; + }); + }); + + class SPcamera { + getInfo() { + return { + id: "SPcamera", + name: Scratch.translate("Camera"), + color1: "#517af5", + color2: "#3460e3", + color3: "#2851c9", + menuIconURI, + blocks: [ + { + func: "addCamera", + blockType: Scratch.BlockType.BUTTON, + text: Scratch.translate("Add Camera"), + }, + { + func: "removeCamera", + blockType: Scratch.BlockType.BUTTON, + text: Scratch.translate("Remove Camera"), + }, + "---", + { + opcode: "bindTarget", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("bind [TARGET] to camera [CAMERA]"), + arguments: { + TARGET: { type: Scratch.ArgumentType.STRING, menu: "OBJECTS" }, + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + }, + }, + { + opcode: "unbindTarget", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("unbind [TARGET] from camera [CAMERA]"), + arguments: { + TARGET: { type: Scratch.ArgumentType.STRING, menu: "OBJECTS" }, + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + }, + }, + { + opcode: "targetCamera", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("camera of [TARGET]"), + arguments: { + TARGET: { + type: Scratch.ArgumentType.STRING, + menu: "EXACT_OBJECTS", + }, + }, + }, + "---", + { + opcode: "setSpaceColor", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("set background color to [COLOR]"), + arguments: { + COLOR: { type: Scratch.ArgumentType.COLOR }, + }, + }, + { + opcode: "spaceColor", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("background color"), + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("Camera Controls"), + }, + { + opcode: "setXY", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("set [CAMERA] camera to x: [X] y: [Y]"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + X: { type: Scratch.ArgumentType.NUMBER }, + Y: { type: Scratch.ArgumentType.NUMBER }, + }, + }, + { + opcode: "goToObject", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("move [CAMERA] camera to [TARGET]"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" }, + }, + }, + "---", + { + opcode: "moveSteps", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("move [CAMERA] camera [NUM] steps"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + NUM: { type: Scratch.ArgumentType.NUMBER, defaultValue: 10 }, + }, + }, + { + opcode: "setX", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("set [CAMERA] camera x to [NUM]"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + NUM: { type: Scratch.ArgumentType.NUMBER }, + }, + }, + { + opcode: "changeX", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("change [CAMERA] camera x by [NUM]"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + NUM: { type: Scratch.ArgumentType.NUMBER, defaultValue: 10 }, + }, + }, + { + opcode: "setY", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("set [CAMERA] camera y to [NUM]"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + NUM: { type: Scratch.ArgumentType.NUMBER }, + }, + }, + { + opcode: "changeY", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("change [CAMERA] camera y by [NUM]"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + NUM: { type: Scratch.ArgumentType.NUMBER, defaultValue: 10 }, + }, + }, + "---", + { + opcode: "getX", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[CAMERA] camera x"), + disableMonitor: true, + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + }, + }, + { + opcode: "getY", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[CAMERA] camera y"), + disableMonitor: true, + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + }, + }, + "---", + { + opcode: "setDirection", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("set [CAMERA] camera direction to [NUM]"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + NUM: { type: Scratch.ArgumentType.ANGLE, defaultValue: 90 }, + }, + }, + { + opcode: "pointCamera", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("point [CAMERA] camera towards [TARGET]"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + TARGET: { type: Scratch.ArgumentType.STRING, menu: "TARGETS" }, + }, + }, + "---", + { + opcode: "turnCamRight", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("turn [CAMERA] camera [IMG] [NUM] degrees"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + IMG: { type: Scratch.ArgumentType.IMAGE, dataURI: rightArrow }, + NUM: { type: Scratch.ArgumentType.NUMBER, defaultValue: 15 }, + }, + }, + { + opcode: "turnCamLeft", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("turn [CAMERA] camera [IMG] [NUM] degrees"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + IMG: { type: Scratch.ArgumentType.IMAGE, dataURI: leftArrow }, + NUM: { type: Scratch.ArgumentType.NUMBER, defaultValue: 15 }, + }, + }, + { + opcode: "getDirection", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[CAMERA] camera direction"), + disableMonitor: true, + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + }, + }, + "---", + { + opcode: "setZoom", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("set [CAMERA] camera zoom to [NUM]%"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + NUM: { type: Scratch.ArgumentType.NUMBER, defaultValue: 100 }, + }, + }, + { + opcode: "changeZoom", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("change [CAMERA] camera zoom by [NUM]"), + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + NUM: { type: Scratch.ArgumentType.NUMBER, defaultValue: 10 }, + }, + }, + { + opcode: "getZoom", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[CAMERA] camera zoom"), + disableMonitor: true, + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + }, + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("Utility"), + }, + { + opcode: "fixedMouseX", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("mouse x in camera [CAMERA]"), + disableMonitor: true, + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + }, + }, + { + opcode: "fixedMouseY", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("mouse y in camera [CAMERA]"), + disableMonitor: true, + arguments: { + CAMERA: { type: Scratch.ArgumentType.STRING, menu: "CAMERAS" }, + }, + }, + ], + menus: { + CAMERAS: { acceptReporters: false, items: "getCameras" }, + TARGETS: { acceptReporters: true, items: "getTargets" }, + OBJECTS: { acceptReporters: true, items: "getObjects" }, + EXACT_OBJECTS: { + acceptReporters: true, + items: this.getObjects(false), + }, + BINDS: { + acceptReporters: true, + items: [ + { text: Scratch.translate("bind"), value: "bind" }, + { text: Scratch.translate("unbind"), value: "unbind" }, + ], + }, + }, + }; + } + + // Helper Funcs + getObjects(includeAll) { + const objectNames = [ + { text: Scratch.translate("myself"), value: "_myself_" }, + ]; + if (includeAll) + objectNames.push({ + text: Scratch.translate("all objects"), + value: "_all_", + }); + objectNames.push({ text: Scratch.translate("Stage"), value: "_stage_" }); + + if (runtime.ext_videoSensing) + objectNames.push({ + text: Scratch.translate("video layer"), + value: "_video_", + }); + if (runtime.ext_pen) + objectNames.push({ + 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) + objectNames.push({ + text: drawable.customDrawableName, + value: `${i}=SP-custLayer`, + }); + } + + // Sprites + const targets = runtime.targets; + for (let i = 1; i < targets.length; i++) { + const target = targets[i]; + if (target.isOriginal) + objectNames.push({ text: target.getName(), value: target.getName() }); + } + return objectNames.length > 0 ? objectNames : [""]; + } + + getTargets() { + const targetNames = [ + { text: Scratch.translate("myself"), value: "_myself_" }, + { text: Scratch.translate("Stage"), value: "_stage_" }, + ]; + const targets = runtime.targets; + for (let i = 1; i < targets.length; i++) { + const target = targets[i]; + if (target.isOriginal) + targetNames.push({ text: target.getName(), value: target.getName() }); + } + return targetNames.length > 0 ? targetNames : [""]; + } + + getCameras() { + return Object.keys(allCameras); + } + + refreshBlocks() { + if (isEditor) { + runtime.once("BEFORE_EXECUTE", () => { + runtime.requestBlocksUpdate(); + }); + runtime.extensionStorage["SPcamera"] = { + cams: Object.keys(allCameras), + }; + } + } + + addCamera() { + openModal(Scratch.translate("New Camera name:"), (name) => { + if (name) { + allCameras[name] = { + xy: [0, 0], + zoom: 1, + dir: 0, + binds: [], + }; + this.refreshBlocks(); + } + }); + } + + removeCamera() { + openModal(Scratch.translate("Remove Camera named:"), (name) => { + if (name) { + if (name === "default") return; // never delete the placeholder + delete allCameras[name]; + this.refreshBlocks(); + } + }); + } + + getTarget(name, util) { + if (name === "_all_") return "_all_"; + else if (name === "_stage_") return runtime.getTargetForStage(); + else if (name === "_myself_") return util.target; + const penLayer = runtime.ext_pen?._penDrawableId; + const videoLayer = runtime.ioDevices.video._drawable; + + if (name === "_pen_") + return penLayer ? { drawableID: penLayer } : undefined; + else if (name === "_video_") + return videoLayer !== -1 ? { drawableID: videoLayer } : undefined; + else if (name.includes("=SP-custLayer")) { + const drawableID = parseInt(name); + if (render._allDrawables[drawableID]?.customDrawableName !== undefined) + return { + drawableID, + }; + } + return runtime.getSpriteTargetByName(name); + } + + translateAngledMovement(xy, steps, direction) { + const radians = direction * (Math.PI / 180); + return [ + xy[0] + steps * Math.cos(radians), + xy[1] + steps * Math.sin(radians), + ]; + } + + // Block Funcs + bindTarget(args, util) { + if (!allCameras[args.CAMERA]) return; + const target = this.getTarget(args.TARGET, util); + if (!target) return; + if (target === "_all_") { + render._allDrawables.forEach((drawable) => { + bindDrawable(drawable, args.CAMERA); + }); + } else { + const drawable = render._allDrawables[target.drawableID]; + bindDrawable(drawable, args.CAMERA); + } + } + + unbindTarget(args, util) { + if (!allCameras[args.CAMERA]) return; + const target = this.getTarget(args.TARGET, util); + if (!target) return; + if (target === "_all_") { + render._allDrawables.forEach((drawable) => { + bindDrawable(drawable, "default"); + }); + } else { + const drawable = render._allDrawables[target.drawableID]; + bindDrawable(drawable, "default"); + } + } + + targetCamera(args, util) { + const target = this.getTarget(args.TARGET, util); + if (!target) return ""; + return render._allDrawables[target.drawableID][cameraSymbol].name; + } + + setSpaceColor(args) { + const rgb = Cast.toRgbColorList(args.COLOR); + render.setBackgroundColor(rgb[0] / 255, rgb[1] / 255, rgb[2] / 255); + } + + spaceColor() { + const rgb = render._backgroundColor3b; + let decimal = (rgb[0] << 16) + (rgb[1] << 8) + rgb[2]; + if (decimal < 0) decimal += 0xffffff + 1; + const hex = Number(decimal).toString(16); + return `#${"000000".substr(0, 6 - hex.length)}${hex}`; + } + + setXY(args) { + if (!allCameras[args.CAMERA]) return; + allCameras[args.CAMERA].xy = [ + Cast.toNumber(args.X) * -1, + Cast.toNumber(args.Y) * -1, + ]; + updateCamera(args.CAMERA); + } + + moveSteps(args) { + if (!allCameras[args.CAMERA]) return; + const cam = allCameras[args.CAMERA]; + const steps = Cast.toNumber(args.NUM) * -1; + cam.xy = this.translateAngledMovement(cam.xy, steps, cam.dir); + updateCamera(args.CAMERA); + } + + setX(args) { + if (!allCameras[args.CAMERA]) return; + allCameras[args.CAMERA].xy[0] = Cast.toNumber(args.NUM) * -1; + updateCamera(args.CAMERA); + } + + changeX(args) { + if (!allCameras[args.CAMERA]) return; + const cam = allCameras[args.CAMERA]; + const steps = Cast.toNumber(args.NUM) * -1; + cam.xy = this.translateAngledMovement(cam.xy, steps, 0); + updateCamera(args.CAMERA); + } + + setY(args) { + if (!allCameras[args.CAMERA]) return; + allCameras[args.CAMERA].xy[1] = Cast.toNumber(args.NUM) * -1; + updateCamera(args.CAMERA); + } + + changeY(args) { + if (!allCameras[args.CAMERA]) return; + const cam = allCameras[args.CAMERA]; + const steps = Cast.toNumber(args.NUM) * -1; + cam.xy = this.translateAngledMovement(cam.xy, steps, 90); + updateCamera(args.CAMERA); + } + + goToObject(args, util) { + if (!allCameras[args.CAMERA]) return; + const target = this.getTarget(args.TARGET, util); + if (target) { + allCameras[args.CAMERA].xy = [target.x, target.y]; + updateCamera(args.CAMERA); + } + } + + getX(args) { + if (!allCameras[args.CAMERA]) return 0; + return allCameras[args.CAMERA].xy[0] * -1; + } + + getY(args) { + if (!allCameras[args.CAMERA]) return 0; + return allCameras[args.CAMERA].xy[1] * -1; + } + + setDirection(args) { + if (!allCameras[args.CAMERA]) return; + allCameras[args.CAMERA].dir = Cast.toNumber(args.NUM) - 90; + updateCamera(args.CAMERA); + } + + turnCamRight(args) { + if (!allCameras[args.CAMERA]) return; + allCameras[args.CAMERA].dir -= Cast.toNumber(args.NUM); + updateCamera(args.CAMERA); + } + + turnCamLeft(args) { + if (!allCameras[args.CAMERA]) return; + allCameras[args.CAMERA].dir += Cast.toNumber(args.NUM); + updateCamera(args.CAMERA); + } + + pointCamera(args, util) { + if (!allCameras[args.CAMERA]) return; + const target = this.getTarget(args.TARGET, util); + if (target) { + allCameras[args.CAMERA].dir = target.direction - 90; + updateCamera(args.CAMERA); + } + } + + getDirection(args) { + if (!allCameras[args.CAMERA]) return 0; + return allCameras[args.CAMERA].dir + 90; + } + + setZoom(args) { + if (!allCameras[args.CAMERA]) return; + allCameras[args.CAMERA].zoom = Cast.toNumber(args.NUM) / 100; + updateCamera(args.CAMERA); + } + + changeZoom(args) { + if (!allCameras[args.CAMERA]) return; + allCameras[args.CAMERA].zoom += Cast.toNumber(args.NUM) / 100; + updateCamera(args.CAMERA); + } + + getZoom(args) { + if (!allCameras[args.CAMERA]) return 0; + return allCameras[args.CAMERA].zoom * 100; + } + + fixedMouseX(args, util) { + if (!allCameras[args.CAMERA]) return 0; + const camData = allCameras[args.CAMERA]; + return translatePosition( + [ + util.ioQuery("mouse", "getScratchX"), + util.ioQuery("mouse", "getScratchY"), + ], + false, + camData + )[0]; + } + + fixedMouseY(args, util) { + if (!allCameras[args.CAMERA]) return 0; + const camData = allCameras[args.CAMERA]; + return translatePosition( + [ + util.ioQuery("mouse", "getScratchX"), + util.ioQuery("mouse", "getScratchY"), + ], + false, + camData + )[1]; + } + } + + Scratch.extensions.register(new SPcamera()); +})(Scratch); diff --git a/extensions/extensions.json b/extensions/extensions.json index 9027c00ac9..97cd039e69 100644 --- a/extensions/extensions.json +++ b/extensions/extensions.json @@ -70,7 +70,7 @@ "mbw/xml", "numerical-encoding-2", "cs2627883/numericalencoding", - "DT/cameracontrols", + "SharkPool/Camera", "TheShovel/CanvasEffects", "Longboost/color_channels", "CST1229/zip", diff --git a/images/SharkPool/Camera.svg b/images/SharkPool/Camera.svg new file mode 100644 index 0000000000..10c8b34511 --- /dev/null +++ b/images/SharkPool/Camera.svg @@ -0,0 +1,32 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +