diff --git a/extensions/Kenay-With-a-Y-At-The-End/Complexify!.js b/extensions/Kenay-With-a-Y-At-The-End/Complexify!.js new file mode 100644 index 0000000000..319d89151c --- /dev/null +++ b/extensions/Kenay-With-a-Y-At-The-End/Complexify!.js @@ -0,0 +1,2112 @@ +// Name: Complexify! +// ID: kenayComplexify +// Description: Complex numbers and numerical operations in TurboWarp +// By: Kenay +// By: Clickertale_2 +// Original: Complexity! +// License: MIT AND MPL-2.0 + +/** Excracted from + +MIT License + +Copyright (c) 2024 Robert Eisele + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. + +**/ //Thanks Rawify + +/* +Thanks [@rawify] for Complex.js v2.4.2 11/5/2024 + (Copyrighted on 2024, from Robert Eisele's https://raw.org/, licensed under the MIT license btw) +Thanks [@scratchfoundation] for the Timer +Thanks [@Brackets-Coder] for telling me where I could be better. +Thanks [@yuri-kiss] for telling me some tips and in what I was wrong. +Thanks [@CubesterYT] and [@CubesterYT] for just passing by. +And thanks [@Clickertale2] for being always there, always helping. +*/ + +(function (Scratch) { + "use strict"; + if (!Scratch.extensions.unsandboxed) { + throw new Error("Complexify! must run unsandboxed"); + //Y'know, for the `util` part to make sense. + } + + class Timer { + //From Scratch's vm. + constructor(nowObj = Timer.nowObj) { + this.startTime = 0; + /**What is now?*/ + this.nowObj = nowObj; + } + static get legacyDateCode() { + return { + now: function () { + return new Date().getTime(); + }, + }; + } + static get nowObj() { + if ( + Timer.USE_PERFORMANCE && + typeof self !== "undefined" && + self.performance && + "now" in self.performance + ) { + return self.performance; + } else if (Date.now) { + return Date; + } + return Timer.legacyDateCode; + //What + } + start() { + this.startTime = this.nowObj.now(); + } + timeElapsed() { + return this.nowObj.now() - this.startTime; + } + } + + //The complex parser. + function l(a, b) { + let d; //I really really really HATE lint-ing. + if (void 0 === a || null === a) f.re = f.im = 0; + else if (void 0 !== b) (f.re = a), (f.im = b); + else + switch (typeof a) { + case "object": + if ("im" in a && "re" in a) (f.re = a.re), (f.im = a.im); + else if ("abs" in a && "arg" in a) { + if (!isFinite(a.abs) && isFinite(a.arg)) return c.INFINITY; + f.re = a.abs * Math.cos(a.arg); + f.im = a.abs * Math.sin(a.arg); + } else if ("r" in a && "phi" in a) { + if (!isFinite(a.r) && isFinite(a.phi)) return c.INFINITY; + f.re = a.r * Math.cos(a.phi); + f.im = a.r * Math.sin(a.phi); + } else 2 === a.length ? ((f.re = a[0]), (f.im = a[1])) : m(); + break; + case "string": + if (a == "Infinity") return c.INFINITY; + if (a == "NaN") return c.NAN; + f.im = f.re = 0; + a = a + .replace(/_/g, "") + .match(/\d+\.?\d*e[+-]?\d+|\d+\.?\d*|\.\d+|./g); //Magic? + b = 1; + d = 0; + null === a && m(); + for (let e = 0; e < a.length; e++) { + const g = a[e]; + " " !== g && + "\t" !== g && + "\n" !== g && + ("+" === g + ? b++ + : "-" === g + ? d++ + : ("i" === g || "I" === g + ? (0 === b + d && m(), + " " === a[e + 1] || isNaN(a[e + 1]) + ? (f.im += parseFloat((d % 2 ? "-" : "") + "1")) + : ((f.im += parseFloat( + (d % 2 ? "-" : "") + a[e + 1] + )), + e++)) + : ((0 === b + d || isNaN(g)) && m(), + "i" === a[e + 1] || "I" === a[e + 1] + ? ((f.im += parseFloat((d % 2 ? "-" : "") + g)), e++) + : (f.re += parseFloat((d % 2 ? "-" : "") + g))), + (b = d = 0))); + } + 0 < b + d && m(); + break; + case "number": + f.im = 0; + f.re = a; + break; + default: + m(); //Error + } + return f; + } + function m() { + throw SyntaxError("Invalid Param"); + } + + //Pythagoras! + function n(a, b) { + a = Math.abs(a); + b = Math.abs(b); + a < b && ([a, b] = [b, a]); + if (1e8 > a) return Math.sqrt(a * a + b * b); + b /= a; //Avoid overflow + return a * Math.sqrt(1 + b * b); + } + //Pythagoras but ln() + function p(a, b) { + const d = Math.abs(a), + e = Math.abs(b); + if (0 === a) return Math.log(e); + if (0 === b) return Math.log(d); + if (3e3 > d && 3e3 > e) return 0.5 * Math.log(a * a + b * b); + a *= 0.5; + b *= 0.5; + return 0.5 * Math.log(a * a + b * b) + Math.LN2; + } + //Complexity! + function c(a, b) { + if (!(this instanceof c)) return new c(a, b); + a = l(a, b); + this.re = a.re; + this.im = a.im; + } + //Alternative functions for better calculations: + const h = + Math.cosh || + function (a) { + return 1e-9 > Math.abs(a) ? 1 - a : 0.5 * (Math.exp(a) + Math.exp(-a)); + }, + k = + Math.sinh || + function (a) { + return 1e-9 > Math.abs(a) ? a : 0.5 * (Math.exp(a) - Math.exp(-a)); + }, + //Silly constant + f = { re: 0, im: 0 }; + c.prototype = { + re: 0, + im: 0, + sign: function () { + //Normalized complex + const a = n(this.re, this.im); + return new c(this.re / a, this.im / a); + }, + add: function (a, b) { + a = l(a, b); + b = this.isInfinite(); + const d = !(isFinite(a.re) && isFinite(a.im)); + return b || d + ? b && d + ? c.NAN + : c.INFINITY + : new c(this.re + a.re, this.im + a.im); + }, + sub: function (a, b) { + a = l(a, b); + b = this.isInfinite(); + const d = !(isFinite(a.re) && isFinite(a.im)); + return b || d + ? b && d + ? c.NAN + : c.INFINITY + : new c(this.re - a.re, this.im - a.im); + }, + mul: function (a, b) { + //Guess [a, b]*[c, d]=[ac - bd, ad + bc] now + a = l(a, b); + b = this.isInfinite(); + const d = !(isFinite(a.re) && isFinite(a.im)), + e = 0 === this.re && 0 === this.im, + g = 0 === a.re && 0 === a.im; + return (b && g) || (d && e) + ? c.NAN + : b || d + ? c.INFINITY + : 0 === a.im && 0 === this.im + ? new c(this.re * a.re, 0) + : new c( + this.re * a.re - this.im * a.im, + this.re * a.im + this.im * a.re + ); + }, + div: function (a, b) { + a = l(a, b); + b = this.isInfinite(); + const d = !(isFinite(a.re) && isFinite(a.im)), + e = 0 === this.re && 0 === this.im, + g = 0 === a.re && 0 === a.im; + if ((e && g) || (b && d)) return c.NAN; // 0/0 = Inf/Inf = NaN + if (g || b) return c.INFINITY; + if (e || d) return c.ZERO; + if (0 === a.im) return new c(this.re / a.re, this.im / a.re); + if (Math.abs(a.re) < Math.abs(a.im)) + //Workaound + return ( + (b = a.re / a.im), + (a = a.re * b + a.im), + new c((this.re * b + this.im) / a, (this.im * b - this.re) / a) + ); + b = a.im / a.re; + a = a.im * b + a.re; + return new c((this.re + this.im * b) / a, (this.im - this.re * b) / a); + }, + pow: function (a, b) { + a = l(a, b); + if (0 === a.re && 0 === a.im) return c.ONE; //New 0^0 just dropped. + if (0 === a.im) { + if (0 === this.im && 0 < this.re) + //If (-real)^real, raise it. + return new c(Math.pow(this.re, a.re), 0); + if (0 === this.re) + //If imag^real, raise it. + switch (((a.re % 4) + 4) % 4) { + case 0: + return new c(Math.pow(this.im, a.re)); + case 1: + return new c(0, Math.pow(this.im, a.re)); + case 2: + return new c(-Math.pow(this.im, a.re)); + case 3: + return new c(0, -Math.pow(this.im, a.re)); + } + } + b = this.isZero(); + if (b && 0 < a.re) return c.ZERO; //0^positive = 0 + const d = this.arg(), + e = p(this.re, this.im); //Now it's a (e*∠d)^(a.re + a.im*i) problem + b = Math.exp(a.re * e - a.im * d); + a = a.im * e + a.re * d; + return new c(b * Math.cos(a), b * Math.sin(a)); + }, + sqrt: function () { + //I don't understand this trick + const a = this.re, + b = this.im; + if (0 === b) + return 0 <= a ? new c(Math.sqrt(a), 0) : new c(0, Math.sqrt(-a)); + var d = n(a, b); + d = Math.sqrt(0.5 * (d + Math.abs(a))); + let e = Math.abs(b) / (2 * d); + return 0 <= a ? new c(d, 0 > b ? -e : e) : new c(e, 0 > b ? -d : d); + }, + exp: function () { + //e^(a+bi) = e^a * ∠b + const a = Math.exp(this.re); + return 0 === this.im + ? new c(a, 0) + : new c(a * Math.cos(this.im), a * Math.sin(this.im)); + }, + expm1: function () { + //For smaller inputs + const a = this.re, + b = this.im; + var d = Math.expm1(a) * Math.cos(b); + var e = Math.PI / 4; + -e > b || b > e + ? (e = Math.cos(b) - 1) + : ((e = b * b), + (e *= + e * + (e * + (e * + (e * + (e * + (e * (e / 20922789888e3 - 1 / 87178291200) + + 1 / 479001600) - + 1 / 3628800) + + 1 / 40320) - + 1 / 720) + + 1 / 24) - + 0.5)); + return new c(d + e, Math.exp(a) * Math.sin(b)); + }, + log: function () { + const a = this.re, + b = this.im; + return 0 === b && 0 < a + ? new c(Math.log(a), 0) //Positive inputs acct as normal + : new c(p(a, b), Math.atan2(b, a)); //ln(r*∠q) =? ln(r) + iq + }, + abs: function () { + return n(this.re, this.im); + }, + arg: function () { + return Math.atan2(this.im, this.re); + }, + sin: function () { + //I won't bother to comment on all the trigs. + const a = this.re, + b = this.im; + return new c(Math.sin(a) * h(b), Math.cos(a) * k(b)); + }, + cos: function () { + const a = this.re, + b = this.im; + return new c(Math.cos(a) * h(b), -Math.sin(a) * k(b)); + }, + tan: function () { + const a = 2 * this.re, + b = 2 * this.im, + d = Math.cos(a) + h(b); + return new c(Math.sin(a) / d, k(b) / d); + }, + cot: function () { + const a = 2 * this.re, + b = 2 * this.im, + d = Math.cos(a) - h(b); + return new c(-Math.sin(a) / d, k(b) / d); + }, + sec: function () { + const a = this.re, + b = this.im, + d = 0.5 * h(2 * b) + 0.5 * Math.cos(2 * a); + return new c((Math.cos(a) * h(b)) / d, (Math.sin(a) * k(b)) / d); + }, + csc: function () { + const a = this.re, + b = this.im, + d = 0.5 * h(2 * b) - 0.5 * Math.cos(2 * a); + return new c((Math.sin(a) * h(b)) / d, (-Math.cos(a) * k(b)) / d); + }, + asin: function () { + var a = this.re; + const b = this.im, + d = new c(b * b - a * a + 1, -2 * a * b).sqrt(); + a = new c(d.re - b, d.im + a).log(); + return new c(a.im, -a.re); + }, + acos: function () { + var a = this.re; + const b = this.im, + d = new c(b * b - a * a + 1, -2 * a * b).sqrt(); + a = new c(d.re - b, d.im + a).log(); + return new c(Math.PI / 2 - a.im, a.re); + }, + atan: function () { + var a = this.re; + const b = this.im; + if (0 === a) { + if (1 === b) return new c(0, Infinity); + if (-1 === b) return new c(0, -Infinity); + } + const d = a * a + (1 - b) * (1 - b); + a = new c((1 - b * b - a * a) / d, (-2 * a) / d).log(); + return new c(-0.5 * a.im, 0.5 * a.re); + }, + acot: function () { + const a = this.re, + b = this.im; + if (0 === b) return new c(Math.atan2(1, a), 0); + const d = a * a + b * b; + return 0 !== d + ? new c(a / d, -b / d).atan() + : new c(0 !== a ? a / 0 : 0, 0 !== b ? -b / 0 : 0).atan(); + }, + asec: function () { + const a = this.re, + b = this.im; + if (0 === a && 0 === b) return new c(0, Infinity); + const d = a * a + b * b; + return 0 !== d + ? new c(a / d, -b / d).acos() + : new c(0 !== a ? a / 0 : 0, 0 !== b ? -b / 0 : 0).acos(); + }, + acsc: function () { + const a = this.re, + b = this.im; + if (0 === a && 0 === b) return new c(Math.PI / 2, Infinity); + const d = a * a + b * b; + return 0 !== d + ? new c(a / d, -b / d).asin() + : new c(0 !== a ? a / 0 : 0, 0 !== b ? -b / 0 : 0).asin(); + }, + sinh: function () { + const a = this.re, + b = this.im; + return new c(k(a) * Math.cos(b), h(a) * Math.sin(b)); + }, + cosh: function () { + const a = this.re, + b = this.im; + return new c(h(a) * Math.cos(b), k(a) * Math.sin(b)); + }, + tanh: function () { + const a = 2 * this.re, + b = 2 * this.im, + d = h(a) + Math.cos(b); + return new c(k(a) / d, Math.sin(b) / d); + }, + coth: function () { + const a = 2 * this.re, + b = 2 * this.im, + d = h(a) - Math.cos(b); + return new c(k(a) / d, -Math.sin(b) / d); + }, + csch: function () { + const a = this.re, + b = this.im, + d = Math.cos(2 * b) - h(2 * a); + return new c((-2 * k(a) * Math.cos(b)) / d, (2 * h(a) * Math.sin(b)) / d); + }, + sech: function () { + const a = this.re, + b = this.im, + d = Math.cos(2 * b) + h(2 * a); + return new c((2 * h(a) * Math.cos(b)) / d, (-2 * k(a) * Math.sin(b)) / d); + }, + asinh: function () { + let a = this.im; + this.im = -this.re; + this.re = a; + const b = this.asin(); + this.re = -this.im; + this.im = a; + a = b.re; + b.re = -b.im; + b.im = a; + return b; + }, + acosh: function () { + const a = this.acos(); + if (0 >= a.im) { + var b = a.re; + a.re = -a.im; + a.im = b; + } else (b = a.im), (a.im = -a.re), (a.re = b); + return a; + }, + atanh: function () { + var a = this.re, + b = this.im; + const d = 1 < a && 0 === b, + e = 1 - a, + g = 1 + a, + q = e * e + b * b; + a = + 0 !== q + ? new c((g * e - b * b) / q, (b * e + g * b) / q) + : new c(-1 !== a ? a / 0 : 0, 0 !== b ? b / 0 : 0); + b = a.re; + a.re = p(a.re, a.im) / 2; + a.im = Math.atan2(a.im, b) / 2; + d && (a.im = -a.im); + return a; + }, + acoth: function () { + const a = this.re, + b = this.im; + if (0 === a && 0 === b) return new c(0, Math.PI / 2); + const d = a * a + b * b; + return 0 !== d + ? new c(a / d, -b / d).atanh() + : new c(0 !== a ? a / 0 : 0, 0 !== b ? -b / 0 : 0).atanh(); + }, + acsch: function () { + const a = this.re, + b = this.im; + if (0 === b) + return new c( + 0 !== a ? Math.log(a + Math.sqrt(a * a + 1)) : Infinity, + 0 + ); + const d = a * a + b * b; + return 0 !== d + ? new c(a / d, -b / d).asinh() + : new c(0 !== a ? a / 0 : 0, 0 !== b ? -b / 0 : 0).asinh(); + }, + asech: function () { + const a = this.re, + b = this.im; + if (this.isZero()) return c.INFINITY; + const d = a * a + b * b; + return 0 !== d + ? new c(a / d, -b / d).acosh() + : new c(0 !== a ? a / 0 : 0, 0 !== b ? -b / 0 : 0).acosh(); + }, + inverse: function () { + //If you hate division + if (this.isZero()) return c.INFINITY; + if (this.isInfinite()) return c.ZERO; + const a = this.re, + b = this.im, + d = a * a + b * b; + return new c(a / d, -b / d); + }, + conjugate: function () { + return new c(this.re, -this.im); + }, + neg: function () { + //If you hate subtraction + return new c(-this.re, -this.im); + }, + ceil: function (a) { + a = Math.pow(10, a || 0); + return new c(Math.ceil(this.re * a) / a, Math.ceil(this.im * a) / a); + }, + floor: function (a) { + a = Math.pow(10, a || 0); + return new c(Math.floor(this.re * a) / a, Math.floor(this.im * a) / a); + }, + round: function (a) { + a = Math.pow(10, a || 0); + return new c(Math.round(this.re * a) / a, Math.round(this.im * a) / a); + }, + equals: function (a, b) { + a = l(a, b); + return ( + Math.abs(a.re - this.re) <= c.EPSILON && //Small error just in case. + Math.abs(a.im - this.im) <= c.EPSILON + ); + }, + clone: function () { + //For the OG Complex.js + return new c(this.re, this.im); + }, + toString: function () { + /** Check later to see why */ + if (this.isNaN()) return "NaN"; + if (this.isInfinite()) return "Infinity"; + let a = this.re, + b = this.im, + d = ""; + Math.abs(a) < c.EPSILON && (a = 0); + Math.abs(b) < c.EPSILON && (b = 0); + if (0 === b) return d + a; + 0 !== a + ? ((d = d + a + " "), + 0 > b ? ((b = -b), (d += "-")) : (d += "+"), + (d += " ")) + : 0 > b && ((b = -b), (d += "-")); + 1 !== b && (d += b); + return d + "i"; + }, + toVector: function () { + //Arrays + return [this.re, this.im]; + }, + valueOf: function () { + return 0 === this.im ? this.re : null; + }, + isNaN: function () { + return isNaN(this.re) || isNaN(this.im); + }, + isZero: function () { + return 0 === this.im && 0 === this.re; + }, + isFinite: function () { + return isFinite(this.re) && isFinite(this.im); + }, + isInfinite: function () { + return !this.isFinite(); + }, + /** + The following are custom methods for this very job. Here are the: + */ + mod: function (a, b) { + //Short for modulus. Expect the remainder of their division. + a = new c(a, b); + b = this.div(a); + return this.sub(a.mul(Math.floor(b.re), Math.floor(b.im))); + }, + cis: function () { + //Extension of the parser's arg/phi. Expect the complex angle. + let cs = Math.exp(-this.im); + return new c(Math.cos(this.re) * cs, Math.sin(this.re) * cs); + }, + dir: function () { + //Short for Direction. Don't ask why. + return this.sign().log().mul(0, -1); + }, + /* + inverse: function () { + if (this.isZero()) return c.INFINITY; + if (this.isInfinite()) return c.ZERO; + const a = this.re, + b = this.im, + d = a * a + b * b; + return new c(a / d, -b / d); + },*/ + trunc: function (a) { + //Short for Truncation. It erases all decimals + a = Math.pow(10, a || 0); + return new c(Math.trunc(this.re * a) / a, Math.trunc(this.im * a) / a); + }, + }; + c.ZERO = new c(0, 0); + c.ONE = new c(1, 0); + c.INFINITY = new c(Infinity, Infinity); + c.NAN = new c(NaN, NaN); + c.EPSILON = 5e-16; + c.SQRT2PI = new c(Math.sqrt(2 * Math.PI)); + c.ROOT3 = new c(-0.5, 0.8660254037844386); + c.GOLDEN = new c(1.618033988749895); + c.SILVER = new c(-0.618033988749895); + const Complex = c; + + var P = [ + Complex(0.99999999999980993), + Complex(676.5203681218851), + Complex(-1259.1392167224028), + Complex(771.32342877765313), + Complex(-176.61502916214059), + Complex(12.507343278686905), + Complex(-0.13857109526572012), + Complex(9.9843695780195716e-6), + Complex(1.5056327351493116e-7), + ]; + + function gamma(a, b) { + var z = new c(a, b); + z = z.sub(1); + var x = P[0]; + var t = z.add(7.5); + for (var i = 1; i < P.length; i++) { + x = x.add(P[i].div(z.add(i))); + } + return c.SQRT2PI.mul(t.pow(z.add(0.5))) + .mul(t.neg().exp()) + .mul(x); + } + + function classicFac(x) { + // 1*2*3*4*...*x + if (x < 0) return Infinity; + if (x == 0 || x == 1) return 1; + return classicFac(x - 1) * x; + } + function factorial(z, t) { + if (Number.isInteger(z)) return classicFac(z); + for (let i = 1; i <= t; i++) { + const numerator = c(1 + 1 / i).pow(z); + const denominator = z.div(i).add(1); + if (denominator.equals(0)) { + throw new Error(`Division by zero at i=${i}`); + } + z = z.mul(numerator.div(denominator)); + } + return z; + } + const menuIconURI = + ""; + + const converter = { + _all: ["degs to rads", "π", "rads to degs"], + "degs to rads": 0.017453292519943295, + π: 3.141592653589793, + "rads to degs": 57.29577951308232, + }; + + class ComplexityExtension { + constructor() { + this.trig = [ + "sin", + "cos", + "tan", + "cot", + "sec", + "csc", + "asin", + "acos", + "atan", + "acot", + "asec", + "acsc", + "sinh", + "cosh", + "tanh", + "coth", + "sech", + "csch", + "asinh", + "acosh", + "atanh", + "acoth", + "asech", + "acsch", + ]; // Used to make the menu look more clean. + this.noSpacing = false; + this.advanced = false; + this.devTools = true; /**** DON'T SET TO FALSE ****/ + this.terms = 100000; /** ignore pls **/ + try { + console.log("Loading Complexify..."); + console.log(window); + if (typeof window.Complex !== "undefined") { + this.Complex = window.Complex; + console.log("Complexified Successfully! Hooray!"); + } else { + console.log("The complex is out of the window!"); + } + } catch (e) { + console.log("Complexify had a complex error!"); + console.log(e); + } + if (!this) { + console.error( + "`this` isn't bounded enough for the complex constructor" + ); + } + } + + getInfo() { + return { + id: "kenayComplexify", + name: Scratch.translate("Complexify!"), + docsURI: + "https://sites.google.com/view/complexity/complexity/complexify-js/", + color1: "#77549f", + menuIconURI, + blocks: [ + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("The protagonist"), + }, + { + opcode: "Complexity", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("complex [REAL] [IMAG]"), + arguments: { + REAL: { type: Scratch.ArgumentType.NUMBER, defaultValue: "3" }, + IMAG: { type: Scratch.ArgumentType.NUMBER, defaultValue: "4" }, + }, + }, + { + blockType: Scratch.BlockType.BUTTON, + func: "toggleAdvancedBlocks", + text: this.advanced + ? "Show Advanced Blocks" + : "Hide Advanced Blocks", + hideFromPalette: this.devTools, + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("Rectangular Tools"), + }, + { + opcode: "rectComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[REAL] + [IMAG] i"), + arguments: { + REAL: { type: Scratch.ArgumentType.STRING, defaultValue: "3" }, + IMAG: { type: Scratch.ArgumentType.STRING, defaultValue: "4" }, + }, + }, + { + opcode: "reComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("real part of [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1+2i", + }, + }, + }, + { + opcode: "imComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("imaginary part of [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1+2i", + }, + }, + }, + { + opcode: "conjugateComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("conjugate of [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("The Binary operations"), + }, + { + opcode: "addComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[COMPLEX1] + [COMPLEX2]"), + arguments: { + COMPLEX1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + COMPLEX2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1+2i", + }, + }, + }, + { + opcode: "subtractComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[COMPLEX1] - [COMPLEX2]"), + arguments: { + COMPLEX1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + COMPLEX2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1+2i", + }, + }, + }, + { + opcode: "multiplyComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[COMPLEX1] x [COMPLEX2]"), + arguments: { + COMPLEX1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + COMPLEX2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1+2i", + }, + }, + }, + { + opcode: "divideComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[COMPLEX1] / [COMPLEX2]"), + arguments: { + COMPLEX1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + COMPLEX2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1+2i", + }, + }, + }, + "---", + { + opcode: "moduloComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[COMPLEX1] mod [COMPLEX2]"), + arguments: { + COMPLEX1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + COMPLEX2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1+2i", + }, + }, + hideFromPalette: this.advanced, + }, + { + opcode: "powComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[COMPLEX1] ^ [COMPLEX2]"), + arguments: { + COMPLEX1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "2+i", + }, + COMPLEX2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3", + }, + }, + hideFromPalette: this.advanced, + }, + { + opcode: "rootComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[COMPLEX1] √ [COMPLEX2]"), + arguments: { + COMPLEX1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3", + }, + COMPLEX2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "2+11i", + }, + }, + hideFromPalette: this.advanced, + }, + { + opcode: "logComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("log [BASE] of [INPUT]"), + arguments: { + BASE: { type: Scratch.ArgumentType.STRING, defaultValue: "2+i" }, + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "2+11i", + }, + }, + hideFromPalette: this.advanced, + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("The Polar Tools"), + }, + { + opcode: "polarComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("polar [RADIUS] ∠ [ANGLE]"), + arguments: { + RADIUS: { type: Scratch.ArgumentType.STRING, defaultValue: 5 }, + ANGLE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0.9272952180016123", + }, + }, + }, + { + opcode: "absComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("magnitude of [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + opcode: "complexSign", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("sign of [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + opcode: "cisThingie", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("∠ [ANGLE]"), + arguments: { + ANGLE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0.9272952180016123", + }, + }, + }, + { + opcode: "argComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("argument of [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + hideFromPalette: this.advanced, + }, + "---", + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("The Spacing"), + hideFromPalette: this.advanced, + }, + { + opcode: "stateSpacing", + blockType: Scratch.BlockType.BOOLEAN, + text: Scratch.translate("spacing"), + hideFromPalette: this.advanced, + }, + { + opcode: "switchSpacing", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("[Sw] spacing"), + arguments: { + Sw: { type: Scratch.ArgumentType.STRING, menu: "spacing" }, + }, + hideFromPalette: this.advanced, + }, + "---", + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("The Unary Operations"), + }, + { + opcode: "negComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("negative [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + description: "Negates the Complex", + }, + { + opcode: "inverseComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("inverse [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + description: "Inverses the Complex", + }, + { + opcode: "expComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("exp [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + hideFromPalette: this.advanced, + }, + { + opcode: "sqrtComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("sqrt of [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + opcode: "lnComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("ln [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + hideFromPalette: this.advanced, + }, + "---", + { + filter: [Scratch.TargetType.SPRITE], //Just in case + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("The Motion"), + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "getPosition", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("complex Position"), + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "goToComplex", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("Go to [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "30+40i", + }, + }, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "goAddComplex", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("Move by [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "30+40i", + }, + }, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "GoMulComplex", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("Mul position by [COMPLEX]"), + arguments: { + COMPLEX: { type: Scratch.ArgumentType.STRING, defaultValue: 5 }, + }, + hideFromPalette: this.advanced, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "goToPolar", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("Go polar [RADII] ∠ [ANGLY]"), + arguments: { + RADII: { type: Scratch.ArgumentType.STRING, defaultValue: 50 }, + ANGLY: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0.9272952180016123", + }, + }, + hideFromPalette: this.advanced, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "glideComplex", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("Glide [SECS] secs to [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "30+40i", + }, + SECS: { type: Scratch.ArgumentType.NUMBER, defaultValue: 1 }, + }, + hideFromPalette: this.advanced, + }, + "---", + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("The Trigonometry"), + hideFromPalette: this.advanced, + }, + { + opcode: "trigOfComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[TRIG] of [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + TRIG: { type: Scratch.ArgumentType.STRING, menu: "trigs" }, + }, + hideFromPalette: this.advanced, + }, + { + opcode: "convertComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("find [ANGLE] [TOSMTH]"), + arguments: { + ANGLE: { type: Scratch.ArgumentType.STRING, defaultValue: "30" }, + TOSMTH: { type: Scratch.ArgumentType.STRING, menu: "angles" }, + }, + hideFromPalette: this.advanced, + }, + "---", + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("The Equalities"), + }, + { + opcode: "equalsComplex", + blockType: Scratch.BlockType.BOOLEAN, + text: Scratch.translate("[COMPLEX1] = [COMPLEX2]?"), + arguments: { + COMPLEX1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + COMPLEX2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + opcode: "isFiniteComplex", + blockType: Scratch.BlockType.BOOLEAN, + text: Scratch.translate("[COMPLEX] is finite?"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + opcode: "isNaNComplex", + blockType: Scratch.BlockType.BOOLEAN, + text: Scratch.translate("[COMPLEX] is NaN?"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + "---", + { + text: Scratch.translate("The Strings"), //For others extensions to use + blockType: Scratch.BlockType.LABEL, + hideFromPalette: this.advanced, + }, + { + opcode: "complexArray", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[COMPLEX] to array"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + hideFromPalette: this.advanced, + }, + { + opcode: "complexJSON", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("[COMPLEX] to JSON"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + hideFromPalette: this.advanced, + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("The LeftOvers"), + }, + { + opcode: "decimalComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate( + "[OPERATION] [COMPLEX] to [DECIMALS] decimals" + ), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3.000004+3.999996i", + }, + OPERATION: { + type: Scratch.ArgumentType.STRING, + menu: "decTools", + }, + DECIMALS: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 4, + }, + }, + hideFromPalette: this.advanced, + }, + { + opcode: "rootNo3", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("ω"), + hideFromPalette: this.advanced, + }, + { + opcode: "goldenComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("φ"), + hideFromPalette: this.advanced, + }, + "---", + { + opcode: "fibonacciComplex", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("Fibonacci [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + hideFromPalette: this.advanced, + }, + { + opcode: "mulVectorAroundPoint", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("mul [VECTOR] around [POINT] by [FACTOR]"), + arguments: { + VECTOR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1+0i", + }, + POINT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0+0i", + }, + FACTOR: { type: Scratch.ArgumentType.STRING, defaultValue: 2 }, + }, + hideFromPalette: this.advanced, + }, + { + opcode: "rotateVectorAroundPoint", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate( + "rotate [VECTOR] around [POINT] by angle [ANGLE]" + ), + arguments: { + VECTOR: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1+0i", + }, + POINT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0+0i", + }, + ANGLE: { + type: Scratch.ArgumentType.STRING, + defaultValue: 3.141592653589793, + }, + }, + hideFromPalette: this.advanced, + }, + { + opcode: "smallEnough", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("Pretty small"), + }, + "---", + { + opcode: "gammaAprox", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("aprox. Γ [COMPLEX]"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + hideFromPalette: this.advanced, + }, + { + opcode: "facOf", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("aprox. [COMPLEX]!"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + hideFromPalette: this.devTools, + }, + { + opcode: "Tfac", + blockType: Scratch.BlockType.REPORTER, + text: Scratch.translate("factorial terms"), + hideFromPalette: this.devTools, + }, + { + opcode: "setTfac", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("set [COMPLEX] factorial terms"), + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.NUMBER, + defaultValue: 2000, + }, + }, + hideFromPalette: this.devTools, + }, + { + opcode: "empty", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("rem [smth]"), + color1: "#FFF099", + arguments: { + smth: { + type: Scratch.ArgumentType.STRING, + defaultValue: "does nothing", + }, + }, + hideFromPalette: this.devTools, + }, + { + opcode: "empty2", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("all [smth] blocks"), + arguments: { + smth: { + type: Scratch.ArgumentType.STRING, + defaultValue: "type", + }, + }, + hideFromPalette: this.devTools, + }, + { + opcode: "empty3", + blockType: Scratch.BlockType.COMMAND, + text: Scratch.translate("and all [smth] blocks"), + arguments: { + smth: { + type: Scratch.ArgumentType.STRING, + defaultValue: "type", + }, + }, + hideFromPalette: this.devTools, + }, + ], + menus: { + spacing: { + //For switchSpacing + acceptReporters: true, + items: ["switch", "turn on", "turn off"], + }, + trigs: { + //For trigOfComplex + acceptReporters: true, + items: this.trig, + }, + angles: { + //For convertComplex + acceptReporters: true, + items: converter._all, + }, + decTools: { + //For decimalComplex + acceptReporters: true, + items: ["round", "ceil of", "floor of", "truncate"], + }, + }, + }; + } + + empty() {} + empty2() {} + empty3() {} + + /** + We'll use toString() to return the Complex number with the **desired** math notation. + + [@Brackets-Coder]: May I ask why this is the case? + + I see no problem. + If you remember Complex is a class, you know it'll returns objects. + Complex(-5, 1) might mean "-5+i", but it's actually a {re: -5, im: 1} in a really silly complex wrapper. + Rawify already made a .toString() method to get "-5+i", and we wanna use that. + + So, we don't want "{re: -5, im: 1}", "[-5,1]" or weird "[object c]" thingies. We want "-5+i" _as is_. + Its pre-built .toString() method is the best option for this goal, to this very day. + Thus, no Scratch.Cast.toString() or anything as silly as that, because **.toString() will always do**. + + Or, as [@yuri-kiss] said: We know the value it returns is going to be a string 'cause it's .toString(). +*/ + + Complexity({ REAL, IMAG }) { + REAL = Scratch.Cast.toNumber(REAL); + IMAG = Scratch.Cast.toNumber(IMAG); + try { + var r = Complex(REAL, IMAG).toString(); //Look! + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + rectComplex({ REAL, IMAG }) { + try { + var r = Complex(IMAG).mul("i").add(REAL).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + reComplex({ COMPLEX }) { + try { + return Complex(COMPLEX).re; + } catch (e) { + console.log(e); + return NaN; + } + } + + imComplex({ COMPLEX }) { + try { + return Complex(COMPLEX).im; + } catch (e) { + console.log(e); + return NaN; + } + } + + conjugateComplex({ COMPLEX }) { + try { + var r = Complex(COMPLEX).conjugate().toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + addComplex({ COMPLEX1, COMPLEX2 }) { + try { + var r = Complex(COMPLEX1).add(COMPLEX2).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + subtractComplex({ COMPLEX1, COMPLEX2 }) { + try { + var r = Complex(COMPLEX1).sub(COMPLEX2).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + multiplyComplex({ COMPLEX1, COMPLEX2 }) { + try { + var r = Complex(COMPLEX1).mul(COMPLEX2).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + divideComplex({ COMPLEX1, COMPLEX2 }) { + try { + var r = Complex(COMPLEX1).div(COMPLEX2).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + moduloComplex({ COMPLEX1, COMPLEX2 }) { + try { + var r = Complex(COMPLEX1).mod(COMPLEX2).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + negComplex({ COMPLEX }) { + try { + var r = Complex(COMPLEX).neg().toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + inverseComplex({ COMPLEX }) { + try { + var r = Complex(COMPLEX).inverse().toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + polarComplex({ RADIUS, ANGLE }) { + try { + var r = Complex(ANGLE).cis().mul(RADIUS).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + absComplex({ COMPLEX }) { + try { + return Complex(COMPLEX).abs(); + } catch (e) { + console.log(e); + return NaN; + } + } + + complexSign({ COMPLEX }) { + try { + var r = Complex(COMPLEX).sign().toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + cisThingie({ ANGLE }) { + try { + var r = Complex(ANGLE).cis().toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + argComplex({ COMPLEX }) { + try { + return Complex(COMPLEX).dir(); + } catch (e) { + console.log(e); + return NaN; + } + } + + powComplex({ COMPLEX1, COMPLEX2 }) { + try { + var r = Complex(COMPLEX1).pow(COMPLEX2).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + stateSpacing(args, util) { + return !this.noSpacing; + } + + switchSpacing({ Sw }) { + try { + if (typeof Sw == "boolean") this.noSpacing = !Sw; + if (typeof Sw == "string") { + Sw = Sw.toLowerCase(); + switch (Sw) { + case "switch": + this.noSpacing = !this.noSpacing; + break; + case "no": + case "turn off": + this.noSpacing = true; + break; + case "yes": + case "turn on": + this.noSpacing = false; + break; + default: + console.log(Sw); + break; + } + } + } catch (e) { + console.log(e); + return NaN; + } + } + + expComplex({ COMPLEX }) { + try { + var r = Complex(COMPLEX).exp().toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + rootComplex({ COMPLEX1, COMPLEX2 }) { + try { + var r = Complex(COMPLEX2).pow(Complex(COMPLEX1).inverse()).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + sqrtComplex({ COMPLEX }) { + try { + var r = Complex(COMPLEX).sqrt().toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + logComplex({ BASE, INPUT }) { + try { + var r = Complex(INPUT).log().div(Complex(BASE).log()).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + lnComplex({ COMPLEX }) { + try { + var r = Complex(COMPLEX).log().toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + getPosition(args, util) { + try { + var r = Complex(util.target.x, util.target.y).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + if (util.target.y < 0) { + let imPart = util.target.y + "i"; + return util.target.x + imPart; + } + return util.target.x + "+" + util.target.y + "i"; + } + } + + goToComplex(args, util) { + if (util.target.isStage) return; + const cInstance = Complex(args.COMPLEX); + util.target.setXY(cInstance.re, cInstance.im); + } + + goAddComplex(args, util) { + if (util.target.isStage) return; + const cInstance = Complex(util.target.x, util.target.y).add(args.COMPLEX); + util.target.setXY(cInstance.re, cInstance.im); + } + + goToPolar(args, util) { + if (util.target.isStage) return; + const cInstance = Complex(args.ANGLY).cis().mul(args.RADII); + util.target.setXY(cInstance.re, cInstance.im); + } + + GoMulComplex(args, util) { + if (util.target.isStage) return; + const cInstance = Complex(util.target.x, util.target.y).mul(args.COMPLEX); + util.target.setXY(cInstance.re, cInstance.im); + } + + glideComplex(args, util) { + if (util.target.isStage) return; + args.SECS = Scratch.Cast.toNumber(args.SECS); + if (args.SECS == 0) { + //We've run out of time! + this.goToComplex(args, util); + return; + } + if (util.stackFrame.timer) { + const timeElapsed = util.stackFrame.timer.timeElapsed(); + if (timeElapsed < util.stackFrame.duration * 1000) { + // We've moving! And we'll move again. + const frac = timeElapsed / (util.stackFrame.duration * 1000); + const d = util.stackFrame.finish.sub(util.stackFrame.start).mul(frac); + util.target.setXY( + util.stackFrame.start.re + d.re, + util.stackFrame.start.im + d.im + ); + console.log(util.stackFrame); + util.yield(); + } else { + // We're done! Now, lets end this + util.target.setXY( + util.stackFrame.finish.re, + util.stackFrame.finish.im + ); + } + } else { + // We're starting! So, new Timer! + util.stackFrame.timer = new Timer(); + util.stackFrame.timer.start(); + if (args.SECS > 0) { + util.stackFrame.start = new Complex(util.target.x, util.target.y); + util.stackFrame.finish = new Complex(args.COMPLEX); + } else { + args.SECS = -args.SECS; + util.stackFrame.finish = new Complex(util.target.x, util.target.y); + util.stackFrame.start = new Complex(args.COMPLEX); + } + util.stackFrame.duration = args.SECS; + console.log(util.stackFrame); + util.yield(); + } + } + + trigOfComplex({ COMPLEX, TRIG }) { + try { + const cInstance = Complex(COMPLEX); + const trigMethod = TRIG.toLowerCase(); + const validTrigFunctions = this.trig; + if ( + validTrigFunctions.includes(trigMethod) && + typeof cInstance[trigMethod] === "function" + ) { + const result = cInstance[trigMethod](); + if (result.isNaN()) { + if (cInstance.isNaN()) return "NaN"; + return "Infinity"; + } + var r = result.toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } else { + return NaN; + } + } catch (e) { + console.log(e); + return NaN; + } + } + + convertComplex({ ANGLE, TOSMTH }) { + try { + if (ANGLE == "") { + return 0; + } + var r = Complex(ANGLE).mul(converter[TOSMTH]).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return 0; + } + } + + decimalComplex({ COMPLEX, OPERATION, DECIMALS }) { + const complex = Complex(COMPLEX); + OPERATION = OPERATION.toLowerCase(); + let D = DECIMALS; + var r; + switch (OPERATION) { + case "round up": + case "round": + r = complex.round(D).toString(); + break; + case "ceil of": + case "ceil": + r = complex.ceil(D).toString(); + break; + case "floor of": + case "floor": + r = complex.floor(D).toString(); + break; + case "truncate": + case "trunc": + r = complex.trunc(D).toString(); + break; + default: + return NaN; + } + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } + + equalsComplex({ COMPLEX1, COMPLEX2 }) { + try { + return Complex(COMPLEX1).equals(COMPLEX2); + } catch (e) { + console.log(e); + return false; + } + } + + isNaNComplex({ COMPLEX }) { + try { + return COMPLEX == "NaN" || Complex(COMPLEX).isNaN(); + } catch (e) { + console.log(e); + return false; + } + } + + isFiniteComplex({ COMPLEX }) { + try { + return COMPLEX != "Infinity" || Complex(COMPLEX).isFinite(); + } catch (e) { + console.log(e); + return false; + } + } + + complexArray({ COMPLEX }) { + try { + return JSON.stringify(Complex(COMPLEX).toVector()); + } catch (e) { + console.log(e); + return NaN; + } + } + + complexJSON({ COMPLEX }) { + try { + return JSON.stringify(Complex(COMPLEX)); + } catch (e) { + console.log(e); + return NaN; + } + } + + mulVectorAroundPoint({ VECTOR, POINT, FACTOR }) { + try { + var r = Complex(VECTOR).sub(POINT).mul(FACTOR).add(POINT).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + rotateVectorAroundPoint({ VECTOR, POINT, ANGLE }) { + try { + var r = Complex(VECTOR) + .sub(POINT) + .mul(Complex(ANGLE).cis()) + .add(POINT) + .toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + + rootNo3() { + if (this.noSpacing) return "-0.5+0.8660254037844386i"; + return "-0.5 + 0.8660254037844386i"; + } + smallEnough() { + return c.EPSILON; + } + goldenComplex() { + return "1.618033988749895"; + } + + fibonacciComplex({ COMPLEX }) { + var r = c.GOLDEN.pow(COMPLEX) + .sub(c.SILVER.pow(COMPLEX)) + .div(2.23606797749979) + .toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } + + toggleAdvancedBlocks() { + this.advanced = !this.advanced; + this.devTools = true; + Scratch.vm.extensionManager.refreshBlocks(); + } + + /** + The following is the v3 of the Factorial function. + The day it's fast, efficient and right, it'll see the light of day. + */ + + gammaAprox({ COMPLEX }) { + //Used in the extension + var r = gamma(COMPLEX).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } + + facOf({ COMPLEX }) { + try { + var r = factorial(Complex(COMPLEX), this.terms).toString(); + if (this.noSpacing) r = r.replace(/\s+/g, ""); + return r; + } catch (e) { + console.log(e); + return NaN; + } + } + Tfac() { + try { + return this.terms; + } catch (e) { + console.log(e); + return NaN; + } + } + setTfac({ COMPLEX }) { + let ZED = Scratch.Cast.toNumber(COMPLEX); + if (ZED < 0) return; + if (Number.isInteger(ZED)) { + this.terms = ZED; + } else { + this.terms = Math.trunc(ZED); + } + } + } + + //Gradient Patch by 0znzw & SharkPool [thanks] + if (Scratch.gui) + Scratch.gui.getBlockly().then((SB) => { + const svg = document.createElement("div"); + svg.innerHTML = ` + + + `; //Remember the id! + document.body.appendChild(svg); + if (!SB?.SPgradients?.patched) + //{ + // Patching... + SB.SPgradients = { gradientUrls: {}, patched: false }; + const BSP = SB.BlockSvg.prototype, + BSPR = BSP.render; + BSP.render = function (...args) { + /* calls the global ReduxStore */ + const res = BSPR.apply(this, args); + let category; + if ( + this?.svgPath_ && + this?.category_ && + (category = this.type.slice(0, this.type.indexOf("_"))) && + SB.SPgradients.gradientUrls[category] + ) { + // Cool checks, could we add more? + this.svgPath_.setAttribute("fill", "url(#kenayComplexify-GRAD)"); //gradient id + } + return res; + }; + SB.SPgradients.patched = true; + //} + SB.SPgradients.gradientUrls["kenayComplexify"] = [ + "url(#kenayComplexify-GRAD)", + ]; //ext id, gradient id + }); + + Scratch.extensions.register(new ComplexityExtension()); +})(Scratch);