diff --git a/extensions/Complexity!.js b/extensions/Complexity!.js new file mode 100644 index 0000000000..731cf5de54 --- /dev/null +++ b/extensions/Complexity!.js @@ -0,0 +1,1007 @@ +/* +Name: Complexity! +ID: complexMath +Description: Play with Complex Numbers in TurboWarp! +By Clickertale_2 and Kenay +Cool License: MIT AND MPL-2.0 +Thanks "rawify" for the complexity +complexCode source: https://github.com/rawify/Complex.js/tree/main +*/ +if (!Scratch.extensions.unsandboxed) { + throw new Error("We don't like sand :("); +} + +const OGcomplexCode = ` +/* +Complex.js v2.4.2 11/5/2024 +https://raw.org/article/complex-numbers-in-javascript/ +Copyright (c) 2024, Robert Eisele (https://raw.org/) +Licensed under the MIT license. Function +*/ +'use strict';(function(r){function l(a,b){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":f.im=f.re=0;a=a.replace(/_/g,"").match(/\\d+\\.?\\d*e[+-]?\\d+|\\d+\\.?\\d*|\\.\\d+|./g);b=1;let d=0;null===a&&m();for(let e=0;ea)return Math.sqrt(a*a+b*b);b/=a;return a*Math.sqrt(1+b*b)}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.5*Math.log(a*a+b*b);a*=.5;b*=.5;return.5*Math.log(a*a+b*b)+Math.LN2}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}const h=Math.cosh||function(a){return 1E-9>Math.abs(a)?1-a:.5*(Math.exp(a)+Math.exp(-a))},k=Math.sinh||function(a){return 1E-9>Math.abs(a)?a:.5*(Math.exp(a)-Math.exp(-a))},f={re:0,im:0};c.prototype={re:0,im:0,sign:function(){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){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;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)b?-e:e):new c(e,0>b?-d:d)},exp:function(){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(){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)-.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.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=1b?(b=-b,d+="-"):d+="+",d+=" "):0>b&&(b=-b,d+="-");1!== +b&&(d+=b);return d+"i"},toVector:function(){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()}};c.ZERO=new c(0,0);c.ONE=new c(1,0);c.I=new c(0,1);c.PI=new c(Math.PI,0);c.E=new c(Math.E,0);c.INFINITY=new c(Infinity,Infinity);c.NAN=new c(NaN,NaN);c.EPSILON=1E-15; +"function"===typeof define&&define.amd?define([],function(){return c}):"object"===typeof exports?(Object.defineProperty(c,"__esModule",{value:!0}),c["default"]=c,c.Complex=c,module.exports=c):r.Complex=c})(window);`; + +(function (Scratch) { + "use strict"; + function coolCis(ANGLE) { + const result = Complex(ANGLE).mul("i").exp(); + return result; + } + 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", + ]; + + try { + eval(OGcomplexCode); + if (window.Complex) { + this.Complex = window.Complex; + } else { + console.log("Complex is not defined on window"); + } + } catch (e) { + console.log("Error evaluating ComplexNumbersCode:", e); + } + } + + getInfo() { + return { + id: "complexMath", + name: "Complexity!", + color1: "#8d3d73", + color2: "#4c213e", + color3: "#562747", + menuIconURI: + "", + blocks: [ + { + opcode: "Complexity", + blockType: Scratch.BlockType.REPORTER, + text: "New Complex [REAL] [IMAG]", + arguments: { + REAL: { type: Scratch.ArgumentType.NUMBER, defaultValue: "3" }, + IMAG: { type: Scratch.ArgumentType.NUMBER, defaultValue: "4" }, + }, + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("Rectangular Tools"), + }, + { + opcode: "rectComplex", + blockType: Scratch.BlockType.REPORTER, + text: "[REAL] + [IMAG] 𝙞", + arguments: { + REAL: { type: Scratch.ArgumentType.STRING, defaultValue: "3" }, + IMAG: { type: Scratch.ArgumentType.STRING, defaultValue: "4" }, + }, + }, + { + opcode: "reComplex", + blockType: Scratch.BlockType.REPORTER, + text: "Real Part of [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1+2i", + }, + }, + }, + { + opcode: "imComplex", + blockType: Scratch.BlockType.REPORTER, + text: "Im. Part of [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1+2i", + }, + }, + }, + { + opcode: "conjugateComplex", + blockType: Scratch.BlockType.REPORTER, + text: "Conjugate of [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("Binary operators"), + }, + { + opcode: "addComplex", + blockType: Scratch.BlockType.REPORTER, + text: "[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: "[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: "[COMPLEX1] × [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: "[COMPLEX1] / [COMPLEX2]", + arguments: { + COMPLEX1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + COMPLEX2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "1+2i", + }, + }, + }, + { + opcode: "powComplex", + blockType: Scratch.BlockType.REPORTER, + text: "[COMPLEX1] ^ [COMPLEX2]", + arguments: { + COMPLEX1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "2+i", + }, + COMPLEX2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3", + }, + }, + }, + { + opcode: "rootComplex", + blockType: Scratch.BlockType.REPORTER, + text: "[COMPLEX1] √ [COMPLEX2]", + arguments: { + COMPLEX1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3", + }, + COMPLEX2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "2+11i", + }, + }, + }, + { + opcode: "logComplex", + blockType: Scratch.BlockType.REPORTER, + text: "log [BASE] [INPUT]", + arguments: { + BASE: { type: Scratch.ArgumentType.STRING, defaultValue: "2+i" }, + INPUT: { + type: Scratch.ArgumentType.STRING, + defaultValue: "2+11i", + }, + }, + }, + { + opcode: "equalsComplex", + blockType: Scratch.BlockType.BOOLEAN, + text: "[COMPLEX1] =? [COMPLEX2]", + arguments: { + COMPLEX1: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + COMPLEX2: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("Polar Tools"), + }, + { + opcode: "polarComplex", + blockType: Scratch.BlockType.REPORTER, + text: "Polar [RADIUS] ∠ [ANGLE]", + arguments: { + RADIUS: { type: Scratch.ArgumentType.STRING, defaultValue: 5 }, + ANGLE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0.9272952180016122", + }, + }, + }, + { + opcode: "absComplex", + blockType: Scratch.BlockType.REPORTER, + text: "Magnitude of [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, //Magnitude = sqrt(Re^2 + Im^2) + }, + { + opcode: "complexSign", + blockType: Scratch.BlockType.REPORTER, + text: "∠ [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, //Sign = Complex/Magnitude + }, + { + opcode: "cisThingie", + blockType: Scratch.BlockType.REPORTER, + text: "CiS [ANGLE]", + arguments: { + ANGLE: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0.9272952180016122", + }, + }, //CiS = e^(i*Angle) + }, + { + opcode: "argComplex", + blockType: Scratch.BlockType.REPORTER, + text: "Argument of [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, //Angle = -i*ln(Sign) + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("Product tools"), + }, + { + opcode: "expComplex", + blockType: Scratch.BlockType.REPORTER, + text: "𝑒 [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + description: "Raises 𝑒 to the Complex", + }, + { + opcode: "lnComplex", + blockType: Scratch.BlockType.REPORTER, + text: "ln [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + opcode: "ComplexTimesI", + blockType: Scratch.BlockType.REPORTER, + text: "[COMPLEXY] 𝙞", + arguments: { + COMPLEXY: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + hideFromPalette: true, + }, + { + opcode: "sqrtComplex", + blockType: Scratch.BlockType.REPORTER, + text: "Square Root of [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + opcode: "negComplex", + blockType: Scratch.BlockType.REPORTER, + text: "Negative [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + opcode: "inverseComplex", + blockType: Scratch.BlockType.REPORTER, + text: "Inverse of [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + opcode: "getPI", + blockType: Scratch.BlockType.REPORTER, + text: "[NUM] 𝜋", + arguments: { + NUM: { type: Scratch.ArgumentType.STRING, defaultValue: "" }, + }, + }, + { + opcode: "toDegrees", + blockType: Scratch.BlockType.REPORTER, + text: "[ANGLE] degs to rads", + arguments: { + ANGLE: { type: Scratch.ArgumentType.STRING, defaultValue: "30" }, + }, + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("Vectors"), + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "getPosition", + blockType: Scratch.BlockType.REPORTER, + text: "Complex Position", + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "goToComplex", + blockType: Scratch.BlockType.COMMAND, + text: "Go to [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "30+40i", + }, + }, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "goAddComplex", + blockType: Scratch.BlockType.COMMAND, + text: "Move by [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "30+40i", + }, + }, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "GoMulComplex", + blockType: Scratch.BlockType.COMMAND, + text: "Mul position by [COMPLEX]", + arguments: { + COMPLEX: { type: Scratch.ArgumentType.STRING, defaultValue: 5 }, + }, + }, + { + filter: [Scratch.TargetType.SPRITE], + opcode: "goToPolar", + blockType: Scratch.BlockType.COMMAND, + text: "Go polar [RADII] ∠ [ANGLY]", + arguments: { + RADII: { type: Scratch.ArgumentType.STRING, defaultValue: 50 }, + ANGLY: { + type: Scratch.ArgumentType.STRING, + defaultValue: "0.9272952180016122", + }, + }, + }, + { + opcode: "mulVectorAroundPoint", + blockType: Scratch.BlockType.REPORTER, + text: "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 }, + }, + }, + { + opcode: "rotateVectorAroundPoint", + blockType: Scratch.BlockType.REPORTER, + text: "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.1415, + }, + }, + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("Trigonometry"), + }, + { + opcode: "trigOfComplex", + blockType: Scratch.BlockType.REPORTER, + text: "[TRIG] of [COMPLEX]", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + TRIG: { type: Scratch.ArgumentType.STRING, menu: "trigs" }, + }, + }, + { + blockType: Scratch.BlockType.LABEL, + text: Scratch.translate("Test the numbers"), + }, + { + opcode: "getEpsilon", + blockType: Scratch.BlockType.REPORTER, + text: "Small", + }, + { + opcode: "isNaNComplex", + blockType: Scratch.BlockType.BOOLEAN, + text: "[COMPLEX] is NaN?", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + opcode: "isFiniteComplex", + blockType: Scratch.BlockType.BOOLEAN, + text: "[COMPLEX] is finite?", + arguments: { + COMPLEX: { + type: Scratch.ArgumentType.STRING, + defaultValue: "3+4i", + }, + }, + }, + { + opcode: "gammaOf", + blockType: Scratch.BlockType.REPORTER, + text: "Γ ([ZED])", + arguments: { + ZED: { type: Scratch.ArgumentType.STRING, defaultValue: "3+4i" }, + }, + hideFromPalette: true, + }, + ], + menus: { + trigs: { acceptReporters: true, items: this.trig }, + }, + }; + } + + Complexity({ REAL, IMAG }) { + try { + const cInstance = new this.Complex(parseFloat(REAL), parseFloat(IMAG)); + return cInstance.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + rectComplex({ REAL, IMAG }) { + try { + const cInstance = Complex(IMAG).mul("i").add(Complex(REAL)); + return cInstance.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + reComplex({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + return cInstance.re; + } catch (e) { + return NaN; + } + } + + imComplex({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + return cInstance.im; + } catch (e) { + return NaN; + } + } + + polarComplex({ RADIUS, ANGLE }) { + try { + const cInstance = coolCis(Complex(ANGLE)).mul(Complex(RADIUS)); + return cInstance.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + cisThingie({ ANGLE }) { + try { + return coolCis(ANGLE).toString().replace(/\s+/g, ""); //Meet my creation! + } catch (e) { + return NaN; + } + } + + addComplex({ COMPLEX1, COMPLEX2 }) { + try { + const c1 = new this.Complex(COMPLEX1); + const c2 = new this.Complex(COMPLEX2); + const result = c1.add(c2); //Uses the structure a1.func(a2) + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + subtractComplex( + { COMPLEX1, COMPLEX2 } //Same + ) { + try { + const c1 = new this.Complex(COMPLEX1); + const c2 = new this.Complex(COMPLEX2); + const result = c1.sub(c2); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + multiplyComplex( + { COMPLEX1, COMPLEX2 } //Again + ) { + try { + const c1 = new this.Complex(COMPLEX1); + const c2 = new this.Complex(COMPLEX2); + const result = c1.mul(c2); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + ComplexTimesI( + { COMPLEXY } //Forever + ) { + try { + const c1 = new this.Complex(COMPLEXY); + const result = c1.mul(Complex.I); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + divideComplex( + { COMPLEX1, COMPLEX2 } //And ever + ) { + try { + const c1 = new this.Complex(COMPLEX1); + const c2 = new this.Complex(COMPLEX2); + const result = c1.div(c2); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + powComplex( + { COMPLEX1, COMPLEX2 } //ever... + ) { + try { + const c1 = new this.Complex(COMPLEX1); + const c2 = new this.Complex(COMPLEX2); + const result = c1.pow(c2); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + rootComplex({ COMPLEX1, COMPLEX2 }) { + try { + const c1 = new this.Complex(COMPLEX2); + const c2 = new this.Complex(COMPLEX1).inverse(); + const result = c1.pow(c2); + return result.toString().replace(/\s+/g, ""); //Meet my... other creations! + } catch (e) { + return NaN; + } + } + + sqrtComplex({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + const result = cInstance.sqrt(); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + logComplex({ BASE, INPUT }) { + try { + const cBase = new this.Complex(BASE); + const cInput = new this.Complex(INPUT); + const result = cInput.log().div(cBase.log()); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + expComplex({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + const result = cInstance.exp(); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + lnComplex({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + const result = cInstance.log(); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + conjugateComplex({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + const result = cInstance.conjugate(); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + complexSign({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + const result = cInstance.sign(); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + absComplex({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + const result = cInstance.abs(); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + argComplex({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + const result = cInstance.arg(); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + inverseComplex({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + const result = cInstance.inverse(); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + negComplex({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + const result = cInstance.neg(); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + mulVectorAroundPoint({ VECTOR, POINT, FACTOR }) { + try { + const p = new this.Complex(POINT); + const result = Complex(VECTOR).sub(p).mul(Complex(FACTOR)).add(p); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + rotateVectorAroundPoint({ VECTOR, POINT, ANGLE }) { + try { + const a = new this.Complex(ANGLE); + const p = new this.Complex(POINT); + const result = Complex(VECTOR).sub(p).mul(coolCis(a)).add(p); + return result.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + getPosition(args, util) { + try { + return Complex(util.target.x, util.target.y) + .toString() + .replace(/\s+/g, ""); + } 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) { + try { + const cInstance = new this.Complex(args.COMPLEX); + util.target.setXY(cInstance.re, cInstance.im); + } catch (e) { + util.target.setXY(util.target.x, util.target.y); + } + } + + goAddComplex(args, util) { + try { + const cInstance = new this.Complex(args.COMPLEX); + util.target.setXY( + util.target.x + cInstance.re, + util.target.y + cInstance.im + ); + } catch (e) { + util.target.setXY(util.target.x, util.target.y); + } + } + + goToPolar(args, util) { + try { + const cInstance = coolCis(args.ANGLY).mul(args.RADII); + util.target.setXY(cInstance.re, cInstance.im); + } catch (e) { + util.target.setXY(util.target.x, util.target.y); + } + } + + GoMulComplex(args, util) { + try { + const cIn = Complex(args.COMPLEX); + const cInstance = Complex(util.target.x, util.target.y).mul(cIn); + util.target.setXY(cInstance.re, cInstance.im); + } catch (e) { + util.target.setXY(util.target.x, util.target.y); + } + } + + trigOfComplex({ COMPLEX, TRIG }) { + try { + const cInstance = new this.Complex(COMPLEX); + const trigMethod = TRIG.toLowerCase(); + const validTrigFunctions = this.trig; + if ( + validTrigFunctions.includes(trigMethod) && + typeof cInstance[trigMethod] === "function" + ) { + const result = cInstance[trigMethod](); + return result.toString().replace(/\s+/g, ""); + } else { + return "0"; + } + } catch (e) { + return NaN; + } + } + + getPI({ NUM }) { + try { + if (NUM == "") { + return Complex.PI; + } + const n = Complex(NUM).mul(Complex.PI); + return n.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + toDegrees({ ANGLE }) { + try { + const cInstance = Complex(ANGLE).mul(0.017453292519943295); + return cInstance.toString().replace(/\s+/g, ""); + } catch (e) { + return NaN; + } + } + + equalsComplex({ COMPLEX1, COMPLEX2 }) { + try { + const c1 = new this.Complex(COMPLEX1); + const c2 = new this.Complex(COMPLEX2); + return c1.equals(c2); + } catch (e) { + return false; + } + } + + gammaOf({ ZED }) { + let c = [ + 0, //term 0 + 12660849.842047213, + -50335415.002908535, + 84454267.34858105, //3 + -77919883.76930115, + 43222228.65544386, + -14822605.274039656, + 3120357.332437802, //7 + -387970.92317872215, + 26862.929569096596, + -580.6487234445449, + 312.43962238445295, //11 + 300.96569466095724, + 301.00429242666866, + 301.00427867948963, + 301.00427867952084, //15 + ]; + const Z = Complex(ZED); + if (Z.re < 0.5) { + return Complex.PI.div( + z + .mul(3.1415926535897932) + .sin() + .mul(gammaOf(Complex(1).sub(Z))) + ); + } + + const z = Z.sub(1); + let x = Complex(2.5066282746310002); + let a; + + for (var i = 1; i < 15; i++) { + x = Complex(c[i]).div(z.add(i)).add(x).toString(); + } + const b = z.add(15).pow(z.add(0.5)).mul(z.neg().sub(15).exp()).mul(x); + return b.toString().replace(/\s+/g, ""); + } + + getEpsilon() { + try { + return this.Complex.EPSILON; + } catch (e) { + return 0.000000000000001; + } + } + + isNaNComplex({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + return cInstance.isNaN(); + } catch (e) { + return false; + } + } + + isFiniteComplex({ COMPLEX }) { + try { + const cInstance = new this.Complex(COMPLEX); + return cInstance.isFinite(); + } catch (e) { + return false; + } + } + } + Scratch.extensions.register(new ComplexityExtension()); +})(Scratch);