summaryrefslogtreecommitdiff
path: root/js/dojo/dojox/calc/_Executor.js
diff options
context:
space:
mode:
Diffstat (limited to 'js/dojo/dojox/calc/_Executor.js')
-rw-r--r--js/dojo/dojox/calc/_Executor.js149
1 files changed, 149 insertions, 0 deletions
diff --git a/js/dojo/dojox/calc/_Executor.js b/js/dojo/dojox/calc/_Executor.js
new file mode 100644
index 0000000..ca5033a
--- /dev/null
+++ b/js/dojo/dojox/calc/_Executor.js
@@ -0,0 +1,149 @@
+//>>built
+define("dojox/calc/_Executor", [
+ "dojo/_base/kernel",
+ "dojo/_base/declare",
+ "dojo/_base/lang",
+ "dojo/number",
+ "dijit/_base/manager",
+ "dijit/_WidgetBase",
+ "dijit/_TemplatedMixin",
+ "dojox/math/_base"
+], function(kernel, declare, lang, number, dijit, WidgetBase, TemplatedMixin, math){
+
+ kernel.experimental("dojox.calc");
+
+ var calcEnv, calc; // private
+
+ var magicBigInt = (1 << 30) - 35; // 2^30 - 35 is a prime that ensures approx(n/(2^k)) != n/(2^k) for k >= 1 and n < 2^k
+
+ /*=====
+ WidgetBase = dijit._WidgetBase;
+ TemplatedMixin = dijit._TemplatedMixin;
+ =====*/
+ var Executor = declare(
+ "dojox.calc._Executor",
+ [WidgetBase, TemplatedMixin],
+ {
+ // summary:
+ // A graphing, scientific calculator
+ //
+
+ templateString: '<iframe src="' + require.toUrl("dojox/calc/_ExecutorIframe.html") +
+ '" style="display:none;" onload="if(arguments[0] && arguments[0].Function)'+dijit._scopeName+'.byNode(this)._onLoad(arguments[0])"></iframe>',
+
+ _onLoad: function(env){
+ // summary
+ // prepare for communications between the user and the calculator by saving the calculator environment, storing the prompt function locally, and making dojox.math available
+ //
+ calcEnv = env;
+ env.outerPrompt = window.prompt; // for IE who can't execute the iframe's prompt method without notifying the user first
+ // let the user call dojo math functions
+ env.dojox = {math: {}};
+ for(var f in math){ env.dojox.math[f] = lang.hitch(math, f); }
+ if("toFrac" in calc){
+ env.toFracCall = lang.hitch(calc, 'toFrac');
+ this.Function('toFrac', 'x', "return toFracCall(x)");
+ }
+
+ env.isJavaScriptLanguage = number.format(1.5, {pattern:'#.#'}) == "1.5";
+ env.Ans = 0;
+ env.pi = Math.PI;
+ env.eps = Math.E;
+
+ env.powCall = lang.hitch(calc, 'pow');
+
+ // TODO add Degrees support to trig functions
+
+
+ //this.normalizedFunction('toString', 'number, radix', "return number.toString(radix)");
+ this.normalizedFunction('sqrt', 'x', "return Math.sqrt(x)");
+ this.normalizedFunction('sin', 'x', "return Math.sin(x)");
+ this.normalizedFunction('cos', 'x', "return Math.cos(x)");
+ this.normalizedFunction('tan', 'x', "return Math.tan(x)");
+ this.normalizedFunction('asin', 'x', "return Math.asin(x)");
+ this.normalizedFunction('acos', 'x', "return Math.acos(x)");
+ this.normalizedFunction('atan', 'x', "return Math.atan(x)");
+ this.normalizedFunction('atan2', 'y, x', "return Math.atan2(y, x)");
+ this.normalizedFunction('Round', 'x', "return Math.round(x)");
+ this.normalizedFunction('Int', 'x', "return Math.floor(x)");
+ this.normalizedFunction('Ceil', 'x', "return Math.ceil(x)");
+ this.normalizedFunction('ln', 'x', "return Math.log(x)");
+ this.normalizedFunction('log', 'x', "return Math.log(x)/Math.log(10)");
+ this.normalizedFunction('pow', 'x, y', "return powCall(x,y)");
+ this.normalizedFunction('permutations', 'n, r', "return dojox.math.permutations(n, r);");
+ this.normalizedFunction('P', 'n, r', "return dojox.math.permutations(n, r);");
+ this.normalizedFunction('combinations', 'n, r', "return dojox.math.combinations(n, r);");
+ this.normalizedFunction('C', 'n, r', "return dojox.math.combinations(n, r)");
+
+ this.normalizedFunction('toRadix', 'number, baseOut', "if(!baseOut){ baseOut = 10; } if(typeof number == 'string'){ number = parseFloat(number); }return number.toString(baseOut);");
+ this.normalizedFunction('toBin', 'number', "return toRadix(number, 2)");
+ this.normalizedFunction('toOct', 'number', "return toRadix(number, 8)");
+ this.normalizedFunction('toHex', 'number', "return toRadix(number, 16)");
+ this.onLoad();
+ },
+
+ onLoad: function(){
+ // summary:
+ // this should be overwritten and become a great place for making user predefined functions
+ //
+ },
+ Function: function(name, args, body){
+ // summary
+ // create an anonymous function to run the code the parser generates from the user input.
+ // params
+ // name: this argument is simply a String that represents the name of the function being evaluated. It can be undefined, but in that case the function is a one time use.
+ // args: the function arguments (a String)
+ // body: the function body, also a String
+ //
+ return lang.hitch(calcEnv, calcEnv.Function.apply(calcEnv, arguments));
+ },
+ normalizedFunction: function(name, args, body){
+ return lang.hitch(calcEnv, calcEnv.normalizedFunction.apply(calcEnv, arguments));
+ },
+ deleteFunction: function(name){
+ calcEnv[name] = undefined;
+ delete calcEnv[name];
+ },
+ eval: function(text){
+ // summary
+ // create an anonymous function to run the code the parser generates from the user input.
+ // params
+ // text, type String, is the user input that needs to be parsed
+ //
+ return calcEnv.eval.apply(calcEnv, arguments);
+ },
+ destroy: function(){
+ this.inherited(arguments);
+ calcEnv = null; // assist garbage collection
+ }
+ });
+
+ return calc = {
+ pow: function(/*Number*/ base, /*Number*/ exponent){
+ // summary:
+ // Computes base ^ exponent
+ // Wrapper to Math.pow(base, exponent) to handle (-27) ^ (1/3)
+
+ function isInt(n){
+ return Math.floor(n) == n;
+ }
+
+ if(base >= 0 || isInt(exponent)){
+ return Math.pow(base, exponent);
+ }else{ // e.g. (1/3) root of -27 = -3
+ var inv = 1 / exponent;
+ // e.g. 1 / (1/3) must be an odd integer
+ return (isInt(inv) && (inv & 1)) ? -Math.pow(-base, exponent) : NaN;
+ }
+ },
+ approx: function(/*Number*/ r){
+ // summary:
+ // Return a less exact approximation of r such that approx(r * (1 +- eps)) == approx(r)
+ if(typeof r == "number"){
+ return Math.round(r * magicBigInt) / magicBigInt;
+ }
+ return r;
+ },
+ _Executor: Executor
+ };
+});