diff --git a/extensions/obviousAlexC/3DMath.js b/extensions/obviousAlexC/3DMath.js new file mode 100644 index 0000000000..7b22799d60 --- /dev/null +++ b/extensions/obviousAlexC/3DMath.js @@ -0,0 +1,946 @@ +// Name: 3D Math +// ID: obviousAlexCMath3d +// Description: Math for 3D projects. +// By: ObviousAlexC + +(function (Scratch) { + "use strict"; + const vm = Scratch.vm; + + const spriteData = {}; + let fov = 300; + + const d2r = 0.0174533; + + const camera = { + position: [0, 0, 0], + rotation: [0, 0, 0], + }; + + class extension { + getInfo() { + return { + blocks: [ + { + opcode: "__NOUSEOPCODE", + blockType: Scratch.BlockType.LABEL, + text: "Vector 3", + }, + { + disableMonitor: true, + opcode: "newV3", + blockType: Scratch.BlockType.REPORTER, + text: "vector 3 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 }, + }, + }, + { + disableMonitor: true, + opcode: "newV3fromValue", + blockType: Scratch.BlockType.REPORTER, + text: "vector 3 from [value]", + arguments: { + value: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }, + }, + }, + { + disableMonitor: true, + opcode: "getAxisOfV3", + blockType: Scratch.BlockType.REPORTER, + text: "get the [axis] axis of [vector]", + arguments: { + axis: { type: Scratch.ArgumentType.STRING, menu: "axisMenu" }, + vector: { + type: Scratch.ArgumentType.STRING, + defaultValue: "[0,0,0]", + }, + }, + }, + { + opcode: "__NOUSEOPCODE", + blockType: Scratch.BlockType.LABEL, + text: "Equations", + }, + { + disableMonitor: true, + opcode: "addV3", + blockType: Scratch.BlockType.REPORTER, + text: "V3: [a] + [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "subV3", + blockType: Scratch.BlockType.REPORTER, + text: "V3: [a] - [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "mulV3", + blockType: Scratch.BlockType.REPORTER, + text: "V3: [a] * [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "divV3", + blockType: Scratch.BlockType.REPORTER, + text: "V3: [a] / [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "dotProductOfV3", + blockType: Scratch.BlockType.REPORTER, + text: "V3: dot product between [a] and [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "crossProductOfV3", + blockType: Scratch.BlockType.REPORTER, + text: "V3: cross product between [a] and [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "magnitudeV3", + blockType: Scratch.BlockType.REPORTER, + text: "V3: magnitude of [a]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "distanceV3", + blockType: Scratch.BlockType.REPORTER, + text: "V3: distance between [a] and [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "rotateAroundPointV3", + blockType: Scratch.BlockType.REPORTER, + text: "V3: rotate [a] around [b] by yaw:[yaw] pitch:[pitch], and roll:[roll]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + yaw: { type: Scratch.ArgumentType.STRING, defaultValue: "0" }, + pitch: { type: Scratch.ArgumentType.STRING, defaultValue: "0" }, + roll: { type: Scratch.ArgumentType.STRING, defaultValue: "0" }, + }, + }, + { + disableMonitor: true, + opcode: "rotateAroundCenterV3", + blockType: Scratch.BlockType.REPORTER, + text: "V3: rotate [a] around the center by yaw:[yaw] pitch:[pitch], and roll:[roll]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + yaw: { type: Scratch.ArgumentType.STRING, defaultValue: "0" }, + pitch: { type: Scratch.ArgumentType.STRING, defaultValue: "0" }, + roll: { type: Scratch.ArgumentType.STRING, defaultValue: "0" }, + }, + }, + { + opcode: "__NOUSEOPCODE", + blockType: Scratch.BlockType.LABEL, + text: "Vector 2", + }, + { + disableMonitor: true, + opcode: "newV2", + blockType: Scratch.BlockType.REPORTER, + text: "vector 2 x:[x] y:[y]", + arguments: { + x: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }, + y: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }, + }, + }, + { + disableMonitor: true, + opcode: "newV2fromValue", + blockType: Scratch.BlockType.REPORTER, + text: "vector 2 from [value]", + arguments: { + value: { type: Scratch.ArgumentType.NUMBER, defaultValue: 0 }, + }, + }, + { + disableMonitor: true, + opcode: "getAxisOfV2", + blockType: Scratch.BlockType.REPORTER, + text: "V2: get the [axis] axis of [vector]", + arguments: { + axis: { type: Scratch.ArgumentType.STRING, menu: "axisMenu2D" }, + vector: { + type: Scratch.ArgumentType.STRING, + defaultValue: "[0,0]", + }, + }, + }, + { + disableMonitor: true, + opcode: "project2DFromCam", + blockType: Scratch.BlockType.REPORTER, + text: "get projected [a] to 2D from camera", + arguments: { + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "[0,0,100]", + }, + }, + }, + { + disableMonitor: true, + opcode: "project2DFromPos", + blockType: Scratch.BlockType.REPORTER, + text: "get projected [a] to 2D from [b] yaw:[yaw] pitch:[pitch] roll:[roll]", + arguments: { + a: { + type: Scratch.ArgumentType.STRING, + defaultValue: "[0,0,100]", + }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + yaw: { type: Scratch.ArgumentType.STRING, defaultValue: "0" }, + pitch: { type: Scratch.ArgumentType.STRING, defaultValue: "0" }, + roll: { type: Scratch.ArgumentType.STRING, defaultValue: "0" }, + }, + }, + { + opcode: "__NOUSEOPCODE", + blockType: Scratch.BlockType.LABEL, + text: "Equations", + }, + { + disableMonitor: true, + opcode: "addV2", + blockType: Scratch.BlockType.REPORTER, + text: "V2: [a] + [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "subV2", + blockType: Scratch.BlockType.REPORTER, + text: "V2: [a] - [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "mulV2", + blockType: Scratch.BlockType.REPORTER, + text: "V2: [a] * [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "divV2", + blockType: Scratch.BlockType.REPORTER, + text: "V2: [a] / [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "dotProductOfV3", + blockType: Scratch.BlockType.REPORTER, + text: "V2: dot product between [a] and [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "crossProductOfV2", + blockType: Scratch.BlockType.REPORTER, + text: "V2: cross product between [a] and [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "magnitudeV2", + blockType: Scratch.BlockType.REPORTER, + text: "V2: magnitude of [a]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "distanceV2", + blockType: Scratch.BlockType.REPORTER, + text: "V2: distance between [a] and [b]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "rotateAroundPointV2", + blockType: Scratch.BlockType.REPORTER, + text: "V2: rotate [a] around [b] by [yaw] degrees", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + b: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + yaw: { type: Scratch.ArgumentType.STRING, defaultValue: "0" }, + }, + }, + { + disableMonitor: true, + opcode: "rotateAroundCenterV2", + blockType: Scratch.BlockType.REPORTER, + text: "V2: rotate [a] around the center by [yaw] degrees", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0]" }, + yaw: { type: Scratch.ArgumentType.STRING, defaultValue: "0" }, + }, + }, + { + opcode: "__NOUSEOPCODE", + blockType: Scratch.BlockType.LABEL, + text: "camera", + }, + { + disableMonitor: true, + opcode: "cam3DsetPosition", + blockType: Scratch.BlockType.COMMAND, + text: "set camera position to [a]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "cam3DgetPosition", + blockType: Scratch.BlockType.REPORTER, + text: "get camera position", + arguments: {}, + }, + { + disableMonitor: true, + opcode: "cam3DsetRotation", + blockType: Scratch.BlockType.COMMAND, + text: "set camera rotation to [a]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + }, + }, + { + disableMonitor: true, + opcode: "cam3DgetRotation", + blockType: Scratch.BlockType.REPORTER, + text: "get camera rotation", + arguments: {}, + }, + { + opcode: "__NOUSEOPCODE", + blockType: Scratch.BlockType.LABEL, + text: "sprite 3D", + }, + { + disableMonitor: true, + opcode: "setFov", + blockType: Scratch.BlockType.COMMAND, + text: "set fov to [dist]", + arguments: { + dist: { type: Scratch.ArgumentType.NUMBER, defaultValue: 300 }, + }, + }, + { + disableMonitor: true, + opcode: "spr3DsetPosition", + blockType: Scratch.BlockType.COMMAND, + text: "set my position to [a]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + }, + filter: "sprite", + }, + { + disableMonitor: true, + opcode: "spr3DchangePosition", + blockType: Scratch.BlockType.COMMAND, + text: "change my position by [a]", + arguments: { + a: { type: Scratch.ArgumentType.STRING, defaultValue: "[0,0,0]" }, + }, + filter: "sprite", + }, + { + disableMonitor: true, + opcode: "spr3DgetPosition", + blockType: Scratch.BlockType.REPORTER, + text: "my 3d position", + arguments: {}, + filter: "sprite", + }, + { + disableMonitor: true, + opcode: "spr3D", + blockType: Scratch.BlockType.COMMAND, + text: "go to my position in 3D", + arguments: {}, + filter: "sprite", + }, + ], + menus: { + axisMenu: { + items: [ + { text: "x", value: "0" }, + { text: "y", value: "1" }, + { text: "z", value: "2" }, + ], + acceptReporters: false, + }, + axisMenu2D: { + items: [ + { text: "x", value: "0" }, + { text: "y", value: "1" }, + ], + acceptReporters: false, + }, + }, + name: "3D Math", + id: "obviousAlexCMath3d", + menuIconURI: + "", + blockIconURI: + "", + color1: "#ADC213", + color2: "#A0B312", + color3: "#697700", + }; + } + newV3({ x, y, z }) { + return JSON.stringify([ + Scratch.Cast.toNumber(x) || 0, + Scratch.Cast.toNumber(y) || 0, + Scratch.Cast.toNumber(z) || 0, + ]); + } + newV3fromValue({ value }) { + if (typeof value == "number") { + return JSON.stringify([value, value, value]); + } + return JSON.stringify([0, 0, 0]); + } + getAxisOfV3({ axis, vector }) { + axis = Scratch.Cast.toNumber(axis); + vector = JSON.parse(vector); + if (vector) { + return vector[axis]; + } + return 0; + } + addV3({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + if (a && b) { + return JSON.stringify([a[0] + b[0], a[1] + b[1], a[2] + b[2]]); + } + return "[0,0,0]"; + } + subV3({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + if (a && b) { + return JSON.stringify([a[0] - b[0], a[1] - b[1], a[2] - b[2]]); + } + return "[0,0,0]"; + } + mulV3({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + if (a && b) { + return JSON.stringify([a[0] * b[0], a[1] * b[1], a[2] * b[2]]); + } + return "[0,0,0]"; + } + divV3({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + if (a && b) { + const c = [0, 0, 0]; + c[0] = a[0] / b[0]; + c[1] = a[1] / b[1]; + c[2] = a[2] / b[2]; + if (isNaN(c[0])) { + c[0] = 0; + } + + if (isNaN(c[1])) { + c[1] = 0; + } + + if (isNaN(c[2])) { + c[2] = 0; + } + + return JSON.stringify(c); + } + return "[0,0,0]"; + } + dotProductOfV3({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + if (a && b) { + return a[0] * b[0] + a[1] * b[1]; + } + return 0; + } + crossProductOfV3({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + + if (a && b) { + const c = [0, 0, 0]; + + c[0] = a[1] * b[2] - a[2] * b[1]; + c[1] = a[2] * b[0] - a[0] * b[2]; + c[2] = a[0] * b[1] - a[1] * b[0]; + + return JSON.stringify(c); + } + return "[0,0,0]"; + } + magnitudeV3({ a }) { + a = JSON.parse(a); + if (a) { + return Math.sqrt( + Math.pow(a[0], 2) + Math.pow(a[1], 2) + Math.pow(a[2], 2) + ); + } + return 0; + } + distanceV3({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + if (a && b) { + return Math.sqrt( + Math.pow(a[0] - b[0], 2) + + Math.pow(a[1] - b[1], 2) + + Math.pow(a[2] - b[2], 2) + ); + } + return 0; + } + rotateAroundPointV3({ a, b, yaw, pitch, roll }) { + a = JSON.parse(a); + b = JSON.parse(b); + + if (a && b) { + a[0] -= b[0]; + a[1] -= b[1]; + a[2] -= b[2]; + + const sinAndCos = [ + Math.sin(yaw * d2r), + Math.cos(yaw * d2r), + Math.sin(pitch * d2r), + Math.cos(pitch * d2r), + Math.sin(roll * d2r), + Math.cos(roll * d2r), + ]; + + let temp = a[0]; + + a[0] = a[2] * sinAndCos[0] + a[0] * sinAndCos[1]; + a[2] = a[2] * sinAndCos[1] - temp * sinAndCos[0]; + + temp = a[1]; + + a[1] = a[2] * sinAndCos[2] + a[1] * sinAndCos[3]; + a[2] = a[2] * sinAndCos[3] - temp * sinAndCos[2]; + + temp = a[0]; + + a[0] = a[1] * sinAndCos[4] + a[0] * sinAndCos[5]; + a[1] = a[1] * sinAndCos[5] - temp * sinAndCos[4]; + + a[0] += b[0]; + a[1] += b[1]; + a[2] += b[2]; + + return JSON.stringify(a); + } + return "[0,0,0]"; + } + rotateAroundCenterV3({ a, yaw, pitch, roll }) { + a = JSON.parse(a); + + if (a) { + const sinAndCos = [ + Math.sin(yaw * d2r), + Math.cos(yaw * d2r), + Math.sin(pitch * d2r), + Math.cos(pitch * d2r), + Math.sin(roll * d2r), + Math.cos(roll * d2r), + ]; + + let temp = a[0]; + + a[0] = a[2] * sinAndCos[0] + a[0] * sinAndCos[1]; + a[2] = a[2] * sinAndCos[1] - temp * sinAndCos[0]; + + temp = a[1]; + + a[1] = a[2] * sinAndCos[2] + a[1] * sinAndCos[3]; + a[2] = a[2] * sinAndCos[3] - temp * sinAndCos[2]; + + temp = a[0]; + + a[0] = a[1] * sinAndCos[4] + a[0] * sinAndCos[5]; + a[1] = a[1] * sinAndCos[5] - temp * sinAndCos[4]; + + return JSON.stringify(a); + } + return "[0,0,0]"; + } + newV2({ x, y }) { + return JSON.stringify([ + Scratch.Cast.toNumber(x) || 0, + Scratch.Cast.toNumber(y) || 0, + ]); + } + newV2fromValue({ value }) { + if (typeof value == "number") { + return JSON.stringify([value, value]); + } + return JSON.stringify([0, 0]); + } + getAxisOfV2({ axis, vector }) { + axis = Scratch.Cast.toNumber(axis); + vector = JSON.parse(vector); + if (vector) { + return vector[axis]; + } + return 0; + } + project2DFromCam({ a }) { + a = JSON.parse(a); + + if (a) { + a[0] -= camera.position[0]; + a[1] -= camera.position[1]; + a[2] -= camera.position[2]; + + const sinAndCos = [ + Math.sin(-camera.rotation[0] * d2r), + Math.cos(-camera.rotation[0] * d2r), + Math.sin(-camera.rotation[1] * d2r), + Math.cos(-camera.rotation[1] * d2r), + Math.sin(-camera.rotation[2] * d2r), + Math.cos(-camera.rotation[2] * d2r), + ]; + + let temp = a[0]; + + a[0] = a[2] * sinAndCos[0] + a[0] * sinAndCos[1]; + a[2] = a[2] * sinAndCos[1] - temp * sinAndCos[0]; + + temp = a[1]; + + a[1] = a[2] * sinAndCos[2] + a[1] * sinAndCos[3]; + a[2] = a[2] * sinAndCos[3] - temp * sinAndCos[2]; + + temp = a[0]; + + a[0] = a[1] * sinAndCos[4] + a[0] * sinAndCos[5]; + a[1] = a[1] * sinAndCos[5] - temp * sinAndCos[4]; + + let project = fov / a[2]; + + return JSON.stringify([a[0] * project, a[1] * project]); + } + return "[0,0]"; + } + project2DFromPos({ a, b, yaw, pitch, roll }) { + a = JSON.parse(a); + b = JSON.parse(b); + + if (a && b) { + a[0] -= b[0]; + a[1] -= b[1]; + a[2] -= b[2]; + + const sinAndCos = [ + Math.sin(-yaw * d2r), + Math.cos(-yaw * d2r), + Math.sin(-pitch * d2r), + Math.cos(-pitch * d2r), + Math.sin(-roll * d2r), + Math.cos(-roll * d2r), + ]; + + let temp = a[0]; + + a[0] = a[2] * sinAndCos[0] + a[0] * sinAndCos[1]; + a[2] = a[2] * sinAndCos[1] - temp * sinAndCos[0]; + + temp = a[1]; + + a[1] = a[2] * sinAndCos[2] + a[1] * sinAndCos[3]; + a[2] = a[2] * sinAndCos[3] - temp * sinAndCos[2]; + + temp = a[0]; + + a[0] = a[1] * sinAndCos[4] + a[0] * sinAndCos[5]; + a[1] = a[1] * sinAndCos[5] - temp * sinAndCos[4]; + + let project = fov / a[2]; + + return JSON.stringify([a[0] * project, a[1] * project]); + } + return "[0,0]"; + } + addV2({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + if (a && b) { + return JSON.stringify([a[0] + b[0], a[1] + b[1]]); + } + return "[0,0]"; + } + subV2({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + if (a && b) { + return JSON.stringify([a[0] - b[0], a[1] - b[1], a[2] - b[2]]); + } + return "[0,0]"; + } + mulV2({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + if (a && b) { + return JSON.stringify([a[0] * b[0], a[1] * b[1]]); + } + return "[0,0]"; + } + divV2({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + if (a && b) { + const c = [0, 0]; + c[0] = a[0] / b[0]; + c[1] = a[1] / b[1]; + if (isNaN(c[0])) { + c[0] = 0; + } + + if (isNaN(c[1])) { + c[1] = 0; + } + + return JSON.stringify(c); + } + return "[0,0]"; + } + crossProductOfV2({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + + if (a && b) { + const c = [0, 0]; + + c[0] = a[1] - b[1]; + c[1] = b[0] - a[0]; + + return JSON.stringify(c); + } + return 0; + } + magnitudeV2({ a }) { + a = JSON.parse(a); + if (a) { + return Math.sqrt(Math.pow(a[0], 2) + Math.pow(a[1], 2)); + } + return 0; + } + distanceV2({ a, b }) { + a = JSON.parse(a); + b = JSON.parse(b); + if (a && b) { + return Math.sqrt(Math.pow(a[0] - b[0], 2) + Math.pow(a[1] - b[1], 2)); + } + return 0; + } + rotateAroundPointV2({ a, b, yaw }) { + a = JSON.parse(a); + b = JSON.parse(b); + + if (a && b) { + a[0] -= b[0]; + a[1] -= b[1]; + + const sinAndCos = [Math.sin(yaw * d2r), Math.cos(yaw * d2r)]; + + let temp = a[0]; + + a[0] = a[1] * sinAndCos[0] + a[0] * sinAndCos[1]; + a[1] = a[1] * sinAndCos[1] - temp * sinAndCos[0]; + + a[0] += b[0]; + a[1] += b[1]; + + return JSON.stringify(a); + } + return "[0,0]"; + } + rotateAroundCenterV2({ a, yaw }) { + a = JSON.parse(a); + + if (a) { + const sinAndCos = [Math.sin(yaw * d2r), Math.cos(yaw * d2r)]; + + let temp = a[0]; + + a[0] = a[1] * sinAndCos[0] + a[0] * sinAndCos[1]; + a[2] = a[1] * sinAndCos[1] - temp * sinAndCos[0]; + + return JSON.stringify(a); + } + return "[0,0]"; + } + cam3DsetPosition({ a }) { + a = JSON.parse(a); + + if (a) { + camera.position = a; + } + } + cam3DgetPosition() { + return JSON.stringify(camera.position); + } + cam3DsetRotation({ a }) { + a = JSON.parse(a); + + if (a) { + camera.rotation = a; + } + } + cam3DgetRotation() { + return JSON.stringify(camera.rotation); + } + setFov({ dist }) { + fov = dist; + } + checkFor3dPositionData(targetID) { + if (!spriteData[targetID]) { + spriteData[targetID] = [0, 0, fov]; + } + } + spr3DsetPosition({ a }, util) { + const target = util.target; + extension.checkFor3dPositionData(target.id); + + a = JSON.parse(a); + if (a) { + spriteData[target.id] = a; + } + } + spr3DchangePosition({ a }, util) { + const target = util.target; + extension.checkFor3dPositionData(target.id); + + a = JSON.parse(a); + + spriteData[target.id][0] += a[0]; + spriteData[target.id][1] += a[1]; + spriteData[target.id][2] += a[2]; + } + spr3DgetPosition(args, util) { + const target = util.target; + extension.checkFor3dPositionData(target.id); + return JSON.stringify(spriteData[target.id]); + } + spr3D(args, util) { + const target = util.target; + extension.checkFor3dPositionData(target.id); + const myData = JSON.parse(JSON.stringify(spriteData[target.id])); + + myData[0] -= camera.position[0]; + myData[1] -= camera.position[1]; + myData[2] -= camera.position[2]; + + const sinAndCos = [ + Math.sin(-camera.rotation[0] * d2r), + Math.cos(-camera.rotation[0] * d2r), + Math.sin(-camera.rotation[1] * d2r), + Math.cos(-camera.rotation[1] * d2r), + Math.sin(-camera.rotation[2] * d2r), + Math.cos(-camera.rotation[2] * d2r), + ]; + + let temp = myData[0]; + + myData[0] = myData[2] * sinAndCos[0] + myData[0] * sinAndCos[1]; + myData[2] = myData[2] * sinAndCos[1] - temp * sinAndCos[0]; + + temp = myData[1]; + + myData[1] = myData[2] * sinAndCos[2] + myData[1] * sinAndCos[3]; + myData[2] = myData[2] * sinAndCos[3] - temp * sinAndCos[2]; + + temp = myData[0]; + + myData[0] = myData[1] * sinAndCos[4] + myData[0] * sinAndCos[5]; + myData[1] = myData[1] * sinAndCos[5] - temp * sinAndCos[4]; + + let project = fov / myData[2]; + + if (myData[2] < 1) { + target.setVisible(false); + } else { + target.setVisible(true); + target.setSize(100 * project); + target.setXY(myData[0] * project, myData[1] * project); + } + } + } + Scratch.extensions.register(new extension()); +})(Scratch);