diff --git a/extensions/AlephOmega/SDFs.js b/extensions/AlephOmega/SDFs.js
new file mode 100644
index 0000000000..81e540f409
--- /dev/null
+++ b/extensions/AlephOmega/SDFs.js
@@ -0,0 +1,591 @@
+(function (Scratch) {
+ "use strict";
+ if (!Scratch.extensions.unsandboxed) {
+ throw new Error("This extension must run unsandboxed");
+ }
+
+ const max = Math.max;
+ const min = Math.min;
+
+ class SDF {
+ getInfo() {
+ return {
+ color1: "#ad0090",
+ color2: "#6e145f",
+ color3: "#3b0932",
+ id: "AlephOmegaSDFS",
+ name: "SDFs",
+ blocks: [
+ {
+ opcode: "Sphere",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Get distance from [P] to sphere at [A] with radius [R]",
+ arguments: {
+ P: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ A: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ R: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Box",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Get distance from [P] to box at [A] with size [S]",
+ arguments: {
+ P: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ A: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ S: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ },
+ },
+ {
+ opcode: "Box2",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Get distance from [P] to rounded box at [A] with size [S] with bevel [R]",
+ arguments: {
+ P: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ A: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ S: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ R: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Capsule",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Get distance from [P] to capsule from [A] to [B] with Radius [R]",
+ arguments: {
+ P: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ A: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ B: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ R: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Plane",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Get distance from [P] to plane with normal [A] and height (Perpendicular) [B]",
+ arguments: {
+ P: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ A: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ B: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ },
+ },
+ {
+ opcode: "Union",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] union [B] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Intersection",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Intersection [B] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Subtraction",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Subtraction [B] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Lerp",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Interpolate [B] by [t] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ T: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Champfer",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Champfer [B] by [K] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ K: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "ChampferInt",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Champfer intersect [B] by [K] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ K: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "ChampferSub",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Champfer Subtract [B] by [K] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ K: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Hollow",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Hollowed out with mid radius of [B] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Groove",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Groove [B] by [K] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ K: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "round_merge",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] round merge [B] with radius [K] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ K: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "round_intersect",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] round intersect [B] with radius [K] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ K: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "round_subtract",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] round subtract [B] with radius [K] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ K: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Smoothunion",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Smootly merge [B] with a coefficient of [K] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ K: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Smoothsub",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Smootly subtract [B] with a coefficient of [K] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ K: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Smoothint",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Smootly intersect [B] with a coefficient of [K] ",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ K: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Scale",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Scale by [B]",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Rounded",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] round by radius of [B]",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Onioning",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] onion by [B]",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Edge",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "[A] Get edge with thickness [B]",
+ arguments: {
+ A: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ B: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ ],
+ };
+ }
+
+ vecSub(v1, v2) {
+ return { x: v1.x - v2.x, y: v1.y - v2.y, z: v1.z - v2.z };
+ }
+
+ vecAdd(v1, v2) {
+ return { x: v1.x + v2.x, y: v1.y + v2.y, z: v1.z + v2.z };
+ }
+
+ vecMult(v, s) {
+ return { x: v.x * s, y: v.y * s, z: v.z * s };
+ }
+
+ dot(v1, v2) {
+ return v1.x * v2.x + v1.y * v2.y + v1.z * v2.z;
+ }
+
+ vecDist(v1, v2) {
+ var dx = v1.x - v2.x;
+ var dy = v1.y - v2.y;
+ var dz = v1.z - v2.z;
+ return Math.sqrt(dx * dx + dy * dy + dz * dz);
+ }
+
+ max(a, b) {
+ return Math.max(a, b);
+ }
+
+ min(a, b) {
+ return Math.min(a, b);
+ }
+
+ clamp(a, b, c) {
+ return max(min(a, b), c);
+ }
+
+ Edge({ A, B }) {
+ return Math.abs(A + B) - B;
+ }
+
+ Onioning({ A, B }) {
+ return Math.abs(A) - B;
+ }
+
+ Round({ A, B }) {
+ return A - B;
+ }
+
+ Smoothint({ A, B, K }) {
+ return A * (1 - Math.exp(-B / K));
+ }
+
+ Smoothsub({ A, B, K }) {
+ return A * Math.exp(-B / K);
+ }
+
+ Smoothunion({ A, B, K }) {
+ return A + B - ((A * B) / (A + B)) * (1 - Math.exp(-(A + B) / K));
+ }
+
+ Scale({ A, B }) {
+ A = JSON.parse(A);
+ B = JSON.parse(B);
+ A.x = A.x * B.x;
+ A.y = A.y * B.y;
+ A.z = A.z * B.z;
+ return JSON.stringify(A);
+ }
+ Plane({ P, A, B }) {
+ var dot = A.x * P.x + A.y * P.y + A.z * P.z;
+ return dot - B;
+ }
+
+ round_subtract({ A, B, K }) {
+ return this.round_intersect({ A: A, B: 0 - B, K: K });
+ }
+
+ round_intersect({ A, B, K }) {
+ var ISX = min(A + K, 0);
+ var ISY = min(B + K, 0);
+ var outdist = Math.sqrt(ISX * ISX + ISY * ISY);
+ var intersect = max(A, B);
+ var indist = min(intersect, -K);
+ return indist + outdist;
+ }
+
+ round_merge({ A, B, K }) {
+ var ISX = max(A - K, 0);
+ var ISY = max(B - K, 0);
+ var indist = 0 - Math.sqrt(ISX * ISX + ISY * ISY);
+ var union = min(A, B);
+ var outdist = max(union, K);
+ return indist + outdist;
+ }
+
+ Groove({ A, B, K }) {
+ return max(A, 0 - (Math.abs(B) - K));
+ }
+
+ Hollow({ A, B }) {
+ return Math.abs(A) - B;
+ }
+
+ ChampferSub({ A, B, K }) {
+ var simpleMerge = max(A, 0 - B);
+ var champfer = (A + (0 - B)) * Math.sqrt(0.5);
+ champfer -= K;
+ return max(simpleMerge, champfer);
+ }
+
+ ChampferInt({ A, B, K }) {
+ var simpleMerge = max(A, B);
+ var champfer = (A + B) * Math.sqrt(0.5);
+ champfer -= K;
+ return max(simpleMerge, champfer);
+ }
+
+ Champfer({ A, B, K }) {
+ var simpleMerge = min(A, B);
+ var champfer = (A + B) * Math.sqrt(0.5);
+ champfer -= K;
+ return min(simpleMerge, champfer);
+ }
+
+ Lerp({ A, B, T }) {
+ return A * T + B * (1 - T);
+ }
+
+ Intersection({ A, B }) {
+ return max(A, B);
+ }
+
+ Subtraction({ A, B }) {
+ return min(A, 0 - B);
+ }
+
+ Union({ A, B }) {
+ return min(A, B);
+ }
+
+ Capsule({ P, A, B, R }) {
+ P = JSON.parse(P);
+ A = JSON.parse(A);
+ B = JSON.parse(B);
+ var AP = this.vecSub(P, A);
+ var AB = this.vecSub(B, A);
+ var t = Math.max(0, Math.min(1, this.dot(AP, AB) / this.dot(AB, AB)));
+ var closestPoint = this.vecAdd(A, this.vecMult(AB, t));
+ var distToClosest = this.vecDist(P, closestPoint);
+ if (t === 0 || t === 1) {
+ return distToClosest - R;
+ } else {
+ var distToAxis = this.vecDist(P, A) - R;
+ return Math.sqrt(
+ distToClosest * distToClosest + distToAxis * distToAxis
+ );
+ }
+ }
+
+ Sphere({ P, A, R }) {
+ P = JSON.parse(P);
+ A = JSON.parse(A);
+ var dx = P.x - A.x;
+ var dy = P.y - A.y;
+ var dz = P.z - A.z;
+ var dist = Math.sqrt(dx * dx + dy * dy + dz * dz);
+ return dist - R;
+ }
+
+ Box({ P, A, S }) {
+ P = JSON.parse(P);
+ A = JSON.parse(A);
+ var dx = P.x - A.x;
+ var dy = P.y - A.y;
+ var dz = P.z - A.z;
+ dx = Math.abs(dx) - S.x;
+ dy = Math.abs(dy) - S.y;
+ dz = Math.abs(dz) - S.z;
+ dx = max(dx, 0);
+ dy = max(dy, 0);
+ dz = max(dz, 0);
+ var dist = Math.sqrt(dx * dx + dy * dy + dz * dz);
+ return dist + min(max(dx, max(dy, dz)), 0);
+ }
+
+ Box2({ P, A, S, R }) {
+ P = JSON.parse(P);
+ A = JSON.parse(A);
+ var dx = P.x - A.x;
+ var dy = P.y - A.y;
+ var dz = P.z - A.z;
+ dx = Math.abs(dx) - S.x;
+ dy = Math.abs(dy) - S.y;
+ dz = Math.abs(dz) - S.z;
+ dx = max(dx, 0);
+ dy = max(dy, 0);
+ dz = max(dz, 0);
+ var dist = Math.sqrt(dx * dx + dy * dy + dz * dz);
+ return dist + min(max(dx, max(dy, dz)), 0) - R;
+ }
+ }
+
+ Scratch.extensions.register(new SDF());
+})(Scratch);
diff --git a/extensions/AlephOmega/VectorUtils.js b/extensions/AlephOmega/VectorUtils.js
new file mode 100644
index 0000000000..04a39f2bd5
--- /dev/null
+++ b/extensions/AlephOmega/VectorUtils.js
@@ -0,0 +1,653 @@
+(function (Scratch) {
+ "use strict";
+ if (!Scratch.extensions.unsandboxed) {
+ throw new Error("This Vector extension must run unsandboxed");
+ }
+ const menuIconURI =
+ "";
+ const blockIconURI =
+ "";
+ class VectorExtension {
+ getInfo() {
+ return {
+ id: "AlephOmegaVector",
+ name: "VectorUtils",
+ color1: "#4286f4",
+ color2: "#63a8f5",
+ color3: "#a3c9f7",
+ menuIconURI: menuIconURI,
+ blockIconURI: blockIconURI,
+ blocks: [
+ {
+ opcode: "getVector",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "create vector with x: [x] y: [y] z: [z]",
+ arguments: {
+ x: {
+ type: Scratch.ArgumentType.NUMBER,
+ defaultValue: 0,
+ },
+ y: {
+ type: Scratch.ArgumentType.NUMBER,
+ defaultValue: 0,
+ },
+ z: {
+ type: Scratch.ArgumentType.NUMBER,
+ defaultValue: 0,
+ },
+ },
+ },
+ {
+ opcode: "addVectors",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "add vector [vector1] and [vector2]",
+ arguments: {
+ vector1: {
+ type: Scratch.ArgumentType.STRING,
+ defaultValue: "",
+ },
+ vector2: {
+ type: Scratch.ArgumentType.STRING,
+ defaultValue: "",
+ },
+ },
+ },
+ {
+ opcode: "subtractVectors",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "subtract vector [vector1] from [vector2]",
+ arguments: {
+ vector1: {
+ type: Scratch.ArgumentType.STRING,
+ defaultValue: "",
+ },
+ vector2: {
+ type: Scratch.ArgumentType.STRING,
+ defaultValue: "",
+ },
+ },
+ },
+ {
+ opcode: "calculateMagnitude",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "magnitude of vector [vector]",
+ arguments: {
+ vector: {
+ type: Scratch.ArgumentType.STRING,
+ defaultValue: "",
+ },
+ },
+ },
+ {
+ opcode: "multiplyVector",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "multiply vector [vector] by scalar [scalar]",
+ arguments: {
+ vector: {
+ type: Scratch.ArgumentType.STRING,
+ defaultValue: "",
+ },
+ scalar: {
+ type: Scratch.ArgumentType.NUMBER,
+ defaultValue: 0,
+ },
+ },
+ },
+ {
+ opcode: "dotProduct",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "dot product of [vector1] and [vector2]",
+ arguments: {
+ vector1: {
+ type: Scratch.ArgumentType.STRING,
+ defaultValue: "",
+ },
+ vector2: {
+ type: Scratch.ArgumentType.STRING,
+ defaultValue: "",
+ },
+ },
+ },
+ {
+ opcode: "crossProduct",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "cross product of [vector1] and [vector2]",
+ arguments: {
+ vector1: {
+ type: Scratch.ArgumentType.STRING,
+ defaultValue: "",
+ },
+ vector2: {
+ type: Scratch.ArgumentType.STRING,
+ defaultValue: "",
+ },
+ },
+ },
+ {
+ opcode: "Rotate",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Rotate [VEC] by [Vector]",
+ arguments: {
+ VEC: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ Vector: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ },
+ },
+ {
+ opcode: "Normalize",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Normalize [VECTOR]",
+ arguments: {
+ VECTOR: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ },
+ },
+ {
+ opcode: "Angle",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Angle between [V1] and [V2]",
+ arguments: {
+ V1: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ V2: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ },
+ },
+ {
+ opcode: "ProjectionOn",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Project [V1] onto [V2]",
+ arguments: {
+ V1: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ V2: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ },
+ },
+ {
+ opcode: "ProjectionFrom",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Project [V1] From [V2]",
+ arguments: {
+ V1: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ V2: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ },
+ },
+ {
+ opcode: "get",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Get [VAL] from [VECTOR]",
+ arguments: {
+ VAL: {
+ type: Scratch.ArgumentType.STRING,
+ menu: "VAL",
+ },
+ VECTOR: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ },
+ },
+ {
+ opcode: "Reflect",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Reflect vector [Vector] onto plane with [Normal] as its normal",
+ arguments: {
+ Vector: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ type: Scratch.ArgumentType.STRING,
+ },
+ },
+ {
+ opcode: "Setmag",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Set magnitude of [VEC] to [MAG]",
+ arguments: {
+ VEC: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ MAG: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "lerp",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Interpolate [V1] towards [V2] with an alpha of [T]",
+ arguments: {
+ V1: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ V2: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ T: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Mat",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Multiply [VEC] by Matrix X [XV] Y [YV] Z [ZV]",
+ arguments: {
+ VEC: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ XV: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ YV: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ ZV: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ },
+ },
+ {
+ opcode: "Distance",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "distance between [V1] and [V2]",
+ arguments: {
+ V1: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ V2: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ },
+ },
+ {
+ opcode: "Set",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Set [SET] of [OF] to [TO]",
+ arguments: {
+ SET: {
+ type: Scratch.ArgumentType.STRING,
+ menu: "VAL",
+ },
+ OF: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ TO: {
+ type: Scratch.ArgumentType.NUMBER,
+ },
+ },
+ },
+ {
+ opcode: "Abs",
+ blockType: Scratch.BlockType.REPORTER,
+ text: "Absolute value of all axis in [V1]",
+ arguments: {
+ V1: {
+ type: Scratch.ArgumentType.STRING,
+ },
+ },
+ },
+ ],
+ menus: {
+ VAL: {
+ acceptReporters: true,
+ items: ["X", "Y", "Z"],
+ },
+ },
+ };
+ }
+
+ isValidJSON(jsonString) {
+ try {
+ const parsedJSON = JSON.parse(jsonString);
+ return (
+ typeof parsedJSON === "object" &&
+ !Array.isArray(parsedJSON) &&
+ this.areAllValuesNumbers(parsedJSON)
+ );
+ } catch (error) {
+ return false;
+ }
+ }
+
+ areAllValuesNumbers(obj) {
+ const requiredKeys = ["x", "y", "z"];
+ for (const key of requiredKeys) {
+ if (!(key in obj) || typeof obj[key] !== "number") {
+ return false;
+ }
+ }
+ return true;
+ }
+
+ Abs(args) {
+ if (!this.isValidJSON(args.V1)) {
+ return "Make sure all inputs are valid";
+ }
+ var V1 = JSON.parse(args.V1);
+ V1.X = Math.abs(V1.X);
+ V1.Y = Math.abs(V1.Y);
+ V1.Z = Math.abs(V1.Z);
+ return JSON.stringify(V1);
+ }
+
+ Set(args) {
+ if (
+ !(
+ this.isValidJSON(args.SET) &&
+ (args.OF == "X" || args.OF == "Y" || args.OF == "Z") &&
+ typeof args.OF == "number"
+ )
+ ) {
+ return "Make sure all inputs are valid";
+ }
+ var { SET, OF, TO } = args;
+ OF = JSON.parse(OF);
+ if (SET == "X") {
+ OF.X = TO;
+ }
+ if (SET == "Y") {
+ OF.Y = TO;
+ }
+ if (SET == "Z") {
+ OF.Z = TO;
+ }
+ return JSON.stringify(OF);
+ }
+
+ Distance(args) {
+ if (!(this.isValidJSON(args.V1) && this.isValidJSON(args.V2))) {
+ return "Make sure all inputs are valid";
+ }
+ var { V1, V2 } = args;
+ V1 = JSON.parse(V1);
+ V2 = JSON.parse(V2);
+
+ var dx = V2.x - V1.x;
+ var dy = V2.y - V1.y;
+ var dz = V2.z - V1.z;
+
+ return Math.sqrt(dx * dx + dy * dy + dz * dz);
+ }
+
+ Mat(args) {
+ var { VEC, XV, YV, ZV } = args;
+ if (
+ !(
+ this.isValidJSON(VEC) &&
+ this.isValidJSON(XV) &&
+ this.isValidJSON(YV) &&
+ this.isValidJSON(ZV)
+ )
+ ) {
+ return "Make sure all inputs are valid";
+ }
+ var x = VEC.x * XV.x + VEC.y * XV.y + VEC.y * XV.y;
+ var y = VEC.x * YV.x + VEC.y * YV.y + VEC.y * YV.y;
+ var z = VEC.z * ZV.x + VEC.y * ZV.y + VEC.y * ZV.y;
+ return JSON.stringify({ x: x, y: y, z: z });
+ }
+
+ lerp(args) {
+ var { V1, V2, T } = args;
+ if (
+ !(this.isValidJSON(V1) && this.isValidJSON(V2) && typeof t == "number")
+ ) {
+ return "Make sure all inputs are valid";
+ }
+ var scaled = V1;
+ scaled.x *= T;
+ scaled.y *= T;
+ scaled.z *= T;
+ var scaled2 = V2;
+ scaled2 *= 1 - T;
+ scaled2 *= 1 - T;
+ scaled2 *= 1 - T;
+ var end = V1;
+ end.x += scaled2.x;
+ end.y += scaled2.y;
+ end.z += scaled2.z;
+ return JSON.stringify(end);
+ }
+
+ Setmag(args) {
+ var { VEC, MAG } = args;
+ if (!(this.isValidJSON(VEC) && typeof MAG == "number")) {
+ return "Make sure all inputs are valid";
+ }
+ VEC = JSON.parse(VEC);
+ var Mag = Math.sqrt(VEC.x * VEC.x + VEC.y * VEC.y + VEC.z * VEC.z);
+ var scalingfactor = MAG / Mag;
+ VEC.x *= scalingfactor;
+ VEC.y *= scalingfactor;
+ VEC.z *= scalingfactor;
+ return JSON.stringify(VEC);
+ }
+ Reflect(args) {
+ var { Vector, Normal } = args;
+ if (!(this.isValidJSON(Vector) && this.isValidJSON(Normal))) {
+ return "Make sure all inputs are valid";
+ }
+ Vector = JSON.parse(Vector);
+ Normal = JSON.parse(Normal);
+ var dot = Vector.x * Normal.x + Vector.y * Normal.y + Vector.z * Normal.z;
+ var scaled = Normal;
+ scaled.x *= dot * 2;
+ scaled.y *= dot * 2;
+ scaled.z *= dot * 2;
+
+ var reflected = Vector;
+ reflected.x -= scaled.x;
+ reflected.y -= scaled.y;
+ reflected.z -= scaled.z;
+
+ return JSON.stringify(reflected);
+ }
+
+ get(args) {
+ var { VAL, VECTOR } = args;
+ if (!this.isValidJSON(VECTOR)) {
+ return "Make sure all inputs are valid";
+ }
+ VECTOR = JSON.parse(VECTOR);
+ if (VAL == "X") {
+ return VECTOR.x;
+ }
+ if (VAL == "Y") {
+ return VECTOR.y;
+ }
+ if (VAL == "Z") {
+ return VECTOR.z;
+ }
+ }
+
+ ProjectionFrom(args) {
+ if (!(this.isValidJSON(args.V1) && this.isValidJSON(args.V2))) {
+ return "Make sure all inputs are valid";
+ }
+ var V1 = JSON.parse(args.V1);
+ var V2 = JSON.parse(args.V2);
+
+ const dot = V2.x * V1.x + V2.y * V1.y + V2.z * V1.z;
+ var Pfrom = V1;
+ Pfrom.x *= dot;
+ Pfrom.y *= dot;
+ Pfrom.Z *= dot;
+ var from = V2;
+ from.x -= Pfrom.x;
+ from.y -= Pfrom.y;
+ from.z -= Pfrom.z;
+ return JSON.stringify(from);
+ }
+
+ ProjectionOn(args) {
+ var { V1, V2 } = args;
+ if (!(this.isValidJSON(V1) && this.isValidJSON(V2))) {
+ return "Make sure all inputs are valid";
+ }
+ V1 = JSON.parse(V1);
+ V2 = JSON.parse(V2);
+ const dot = V2.x * V1.x + V2.y * V1.y + V2.z * V1.z;
+ const Pfrom = V1;
+ Pfrom.x *= dot;
+ Pfrom.y *= dot;
+ Pfrom.Z *= dot;
+ return JSON.stringify(Pfrom);
+ }
+
+ Angle(args) {
+ var { V1, V2 } = args;
+ if (!(this.isValidJSON(V1) && this.isValidJSON(V2))) {
+ return "Make sure all inputs are valid";
+ }
+ V1 = JSON.parse(V1);
+ V2 = JSON.parse(V2);
+ const mag1 = Math.sqrt(V1.x * V1.x + V1.y * V1.y + V1.z * V1.z);
+ const mag2 = Math.sqrt(V2.x * V2.x + V2.y * V2.y + V2.z * V2.z);
+ const dot = V2.x * V1.x + V2.y * V1.y + V2.z * V1.z;
+ return Math.acos(dot / (mag1 * mag2)) * (180 / Math.PI);
+ }
+
+ Normalize(args) {
+ if (!this.isValidJSON(args.VECTOR)) {
+ return "Make sure all inputs are valid";
+ }
+ var vec = args.VECTOR;
+ vec = JSON.parse(vec);
+ const magnitude = Math.sqrt(
+ vec.x * vec.x + vec.y * vec.y + vec.z * vec.z
+ );
+ vec.x /= magnitude;
+ vec.y /= magnitude;
+ vec.z /= magnitude;
+ return JSON.stringify(vec);
+ }
+
+ Rotate(args) {
+ const { Vector, VEC } = args;
+ if (!(this.isValidJSON(Vector) && this.isValidJSON(VEC))) {
+ return "Make sure all inputs are valid";
+ }
+ var vector = JSON.parse(VEC);
+ var angle = JSON.parse(Vector);
+ angle.x *= Math.PI / 180;
+ angle.y *= Math.PI / 180;
+ angle.z *= Math.PI / 180;
+ var rx = vector.x * Math.cos(angle.y) + vector.z * Math.sin(angle.y);
+ vector.z = vector.z * Math.cos(angle.y) - vector.x * Math.sin(angle.y);
+ vector.x = rx;
+ var ry = vector.y * Math.cos(angle.y) + vector.z * Math.sin(angle.y);
+ vector.z = vector.z * Math.cos(angle.y) - vector.y * Math.sin(angle.y);
+ vector.y = ry;
+ var rz = vector.x * Math.cos(angle.z) - vector.y * Math.sin(angle.z);
+ vector.y = vector.y * Math.cos(angle.z) + vector.x * Math.sin(angle.z);
+ vector.x = rz;
+ return JSON.stringify(vector);
+ }
+
+ getVector(args) {
+ const { x, y, z } = args;
+ if (
+ !(typeof x == "number" && typeof y == "number" && typeof z == "number")
+ ) {
+ return "Make sure all inputs are valid";
+ }
+ const vector = { x, y, z };
+ return JSON.stringify(vector);
+ }
+
+ addVectors(args) {
+ const { vector1, vector2 } = args;
+ if (!(this.isValidJSON(vector1) && this.isValidJSON(vector2))) {
+ return "Make sure all inputs are valid";
+ }
+ const vec1 = JSON.parse(vector1);
+ const vec2 = JSON.parse(vector2);
+ const result = {
+ x: Number(vec1.x) + Number(vec2.x),
+ y: Number(vec1.y) + Number(vec2.y),
+ z: Number(vec1.z) + Number(vec2.z),
+ };
+ return JSON.stringify(result);
+ }
+
+ subtractVectors(args) {
+ const { vector1, vector2 } = args;
+ if (!(this.isValidJSON(vector1) && this.isValidJSON(vector2))) {
+ return "Make sure all inputs are valid";
+ }
+ const vec1 = JSON.parse(vector1);
+ const vec2 = JSON.parse(vector2);
+ const result = {
+ x: Number(vec2.x) - Number(vec1.x),
+ y: Number(vec2.y) - Number(vec1.y),
+ z: Number(vec2.z) - Number(vec1.z),
+ };
+ return JSON.stringify(result);
+ }
+
+ calculateMagnitude(args) {
+ const { vector } = args;
+ if (!this.isValidJSON(vector)) {
+ return "Make sure all inputs are valid";
+ }
+ const vec = JSON.parse(vector);
+ const magnitude = Math.sqrt(
+ vec.x * vec.x + vec.y * vec.y + vec.z * vec.z
+ );
+ return magnitude;
+ }
+
+ multiplyVector(args) {
+ const { vector, scalar } = args;
+ if (!(this.isValidJSON(vector) && typeof scalar == "number")) {
+ return "Make sure all inputs are valid";
+ }
+ const vec = JSON.parse(vector);
+ const result = {
+ x: vec.x * scalar,
+ y: vec.y * scalar,
+ z: vec.z * scalar,
+ };
+ return JSON.stringify(result);
+ }
+
+ dotProduct(args) {
+ const { vector1, vector2 } = args;
+ if (!(this.isValidJSON(vector1) && this.isValidJSON(vector2))) {
+ return "Make sure all inputs are valid";
+ }
+ const vec1 = JSON.parse(vector1);
+ const vec2 = JSON.parse(vector2);
+ const result = vec1.x * vec2.x + vec1.y * vec2.y + vec1.z * vec2.z;
+ return result;
+ }
+
+ crossProduct(args) {
+ const { vector1, vector2 } = args;
+ if (!(this.isValidJSON(vector1) && this.isValidJSON(vector2))) {
+ return "Make sure all inputs are valid";
+ }
+ const vec1 = JSON.parse(vector1);
+ const vec2 = JSON.parse(vector2);
+ const result = {
+ x: vec1.y * vec2.z - vec1.z * vec2.y,
+ y: vec1.z * vec2.x - vec1.x * vec2.z,
+ z: vec1.x * vec2.y - vec1.y * vec2.x,
+ };
+ return JSON.stringify(result);
+ }
+ }
+ Scratch.extensions.register(new VectorExtension());
+})(Scratch);
diff --git a/images/AlephOmega/SDF.svg b/images/AlephOmega/SDF.svg
new file mode 100644
index 0000000000..dbee4acb4e
--- /dev/null
+++ b/images/AlephOmega/SDF.svg
@@ -0,0 +1,17 @@
+
+
diff --git a/images/AlephOmega/VectorUtils.svg b/images/AlephOmega/VectorUtils.svg
new file mode 100644
index 0000000000..51c3845c6a
--- /dev/null
+++ b/images/AlephOmega/VectorUtils.svg
@@ -0,0 +1,66 @@
+
+