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 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +