From d67d498625482aba12275ce2ff2df1fe7bb8815a Mon Sep 17 00:00:00 2001 From: Riley Jones Date: Tue, 8 Aug 2017 14:00:35 -0700 Subject: [PATCH] Add variables --- src/main/java/org/nwapw/abacus/Abacus.java | 14 ++- .../org/nwapw/abacus/fx/AbacusController.java | 25 ++++- .../org/nwapw/abacus/number/NaiveNumber.java | 13 +-- .../nwapw/abacus/number/NumberInterface.java | 6 ++ .../nwapw/abacus/number/PreciseNumber.java | 10 +- .../nwapw/abacus/plugin/StandardPlugin.java | 98 +++++++++---------- .../org/nwapw/abacus/tree/NumberReducer.java | 3 + .../nwapw/abacus/tests/CalculationTests.java | 2 +- .../nwapw/abacus/tests/TokenizerTests.java | 2 +- 9 files changed, 103 insertions(+), 70 deletions(-) diff --git a/src/main/java/org/nwapw/abacus/Abacus.java b/src/main/java/org/nwapw/abacus/Abacus.java index fa5f76a..c8f7559 100644 --- a/src/main/java/org/nwapw/abacus/Abacus.java +++ b/src/main/java/org/nwapw/abacus/Abacus.java @@ -2,6 +2,8 @@ package org.nwapw.abacus; import org.nwapw.abacus.config.Configuration; import org.nwapw.abacus.fx.AbacusApplication; +import org.nwapw.abacus.fx.AbacusController; +import org.nwapw.abacus.number.NaiveNumber; import org.nwapw.abacus.number.NumberInterface; import org.nwapw.abacus.parsing.LexerTokenizer; import org.nwapw.abacus.parsing.ShuntingYardParser; @@ -12,6 +14,8 @@ import org.nwapw.abacus.plugin.StandardPlugin; import org.nwapw.abacus.tree.NumberReducer; import org.nwapw.abacus.tree.TreeNode; +import java.util.HashMap; + /** * The main calculator class. This is responsible * for piecing together all of the components, allowing @@ -43,13 +47,13 @@ public class Abacus { * from a string. */ private TreeBuilder treeBuilder; - + private AbacusController controller; /** * Creates a new instance of the Abacus calculator. * * @param configuration the configuration object for this Abacus instance. */ - public Abacus(Configuration configuration) { + public Abacus(Configuration configuration,AbacusController controller) { pluginManager = new PluginManager(this); numberReducer = new NumberReducer(this); this.configuration = new Configuration(configuration); @@ -59,8 +63,12 @@ public class Abacus { pluginManager.addListener(shuntingYardParser); pluginManager.addListener(lexerTokenizer); - } + this.controller =controller; + } + public NumberInterface getVar(String variable){ + return controller.getVar(variable); + } public static void main(String[] args) { AbacusApplication.launch(AbacusApplication.class, args); } diff --git a/src/main/java/org/nwapw/abacus/fx/AbacusController.java b/src/main/java/org/nwapw/abacus/fx/AbacusController.java index 733560c..ea81b6f 100644 --- a/src/main/java/org/nwapw/abacus/fx/AbacusController.java +++ b/src/main/java/org/nwapw/abacus/fx/AbacusController.java @@ -1,8 +1,6 @@ package org.nwapw.abacus.fx; import javafx.application.Platform; -import javafx.beans.value.ChangeListener; -import javafx.beans.value.ObservableValue; import javafx.collections.FXCollections; import javafx.collections.ObservableList; import javafx.fxml.FXML; @@ -14,12 +12,14 @@ import javafx.util.StringConverter; import org.nwapw.abacus.Abacus; import org.nwapw.abacus.config.Configuration; import org.nwapw.abacus.number.ComputationInterruptedException; +import org.nwapw.abacus.number.NaiveNumber; import org.nwapw.abacus.number.NumberInterface; import org.nwapw.abacus.plugin.*; import org.nwapw.abacus.tree.TreeNode; import java.io.File; import java.io.IOException; +import java.util.ArrayList; import java.util.Set; @@ -125,6 +125,17 @@ public class AbacusController implements PluginListener { * The alert shown when a press to "apply" is needed. */ private Alert reloadAlert; + private ArrayList plugins; + public NumberInterface getVar(String variable){ + for(Plugin plugin:plugins){ + if(plugin instanceof VariablePlugin){ + if(((VariablePlugin)plugin).getValue(variable)!=null) + return ((VariablePlugin)plugin).getValue(variable); + return NaiveNumber.ZERO; + } + } + return null; + } /** * The runnable that takes care of killing computations that take too long. */ @@ -229,11 +240,15 @@ public class AbacusController implements PluginListener { if (oldValue.equals(settingsTab)) alertIfApplyNeeded(true); }); - abacus = new Abacus(new Configuration(CONFIG_FILE)); + abacus = new Abacus(new Configuration(CONFIG_FILE),this); PluginManager abacusPluginManager = abacus.getPluginManager(); abacusPluginManager.addListener(this); - abacusPluginManager.addInstantiated(new StandardPlugin(abacus.getPluginManager())); - abacusPluginManager.addInstantiated(new VariablePlugin(abacus.getPluginManager())); + plugins = new ArrayList<>(); + plugins.add(new StandardPlugin(abacus.getPluginManager())); + plugins.add(new VariablePlugin(abacus.getPluginManager())); + for(Plugin plugin: plugins){ + abacusPluginManager.addInstantiated(plugin); + } try { ClassFinder.loadJars("plugins").forEach(abacusPluginManager::addClass); } catch (IOException | ClassNotFoundException e) { diff --git a/src/main/java/org/nwapw/abacus/number/NaiveNumber.java b/src/main/java/org/nwapw/abacus/number/NaiveNumber.java index 48beb1a..1e4e5f0 100755 --- a/src/main/java/org/nwapw/abacus/number/NaiveNumber.java +++ b/src/main/java/org/nwapw/abacus/number/NaiveNumber.java @@ -35,7 +35,6 @@ public class NaiveNumber extends NumberInterface { public NaiveNumber(double value) { this.value = value; } - @Override public int getMaxPrecision() { return 18; @@ -43,22 +42,22 @@ public class NaiveNumber extends NumberInterface { @Override public NumberInterface multiplyInternal(NumberInterface multiplier) { - return new NaiveNumber(value * ((NaiveNumber) multiplier).value); + return new NaiveNumber(value * ((NaiveNumber) multiplier.number()).value); } @Override public NumberInterface divideInternal(NumberInterface divisor) { - return new NaiveNumber(value / ((NaiveNumber) divisor).value); + return new NaiveNumber(value / ((NaiveNumber) divisor.number()).value); } @Override public NumberInterface addInternal(NumberInterface summand) { - return new NaiveNumber(value + ((NaiveNumber) summand).value); + return new NaiveNumber(value + ((NaiveNumber) summand.number()).value); } @Override public NumberInterface subtractInternal(NumberInterface subtrahend) { - return new NaiveNumber(value - ((NaiveNumber) subtrahend).value); + return new NaiveNumber(value - ((NaiveNumber) subtrahend.number()).value); } @Override @@ -85,7 +84,7 @@ public class NaiveNumber extends NumberInterface { @Override public int compareTo(NumberInterface number) { - NaiveNumber num = (NaiveNumber) number; + NaiveNumber num = (NaiveNumber) number.number(); return Double.compare(value, num.value); } @@ -119,6 +118,8 @@ public class NaiveNumber extends NumberInterface { if (toClass == this.getClass()) return this; else if (toClass == PreciseNumber.class) { return new PreciseNumber(Double.toString(value)); + }else if(toClass == Variable.class){ + return this; } return null; } diff --git a/src/main/java/org/nwapw/abacus/number/NumberInterface.java b/src/main/java/org/nwapw/abacus/number/NumberInterface.java index c77f531..317f5f4 100755 --- a/src/main/java/org/nwapw/abacus/number/NumberInterface.java +++ b/src/main/java/org/nwapw/abacus/number/NumberInterface.java @@ -13,6 +13,12 @@ public abstract class NumberInterface { if(Thread.currentThread().isInterrupted()) throw new ComputationInterruptedException(); } + public NumberInterface number(){ + return this; + } + public Class getClassVal(){ + return this.getClass(); + } /** * The maximum precision to which this number operates. * diff --git a/src/main/java/org/nwapw/abacus/number/PreciseNumber.java b/src/main/java/org/nwapw/abacus/number/PreciseNumber.java index abc657f..60045f5 100755 --- a/src/main/java/org/nwapw/abacus/number/PreciseNumber.java +++ b/src/main/java/org/nwapw/abacus/number/PreciseNumber.java @@ -53,22 +53,22 @@ public class PreciseNumber extends NumberInterface { @Override public NumberInterface multiplyInternal(NumberInterface multiplier) { - return new PreciseNumber(this.value.multiply(((PreciseNumber) multiplier).value)); + return new PreciseNumber(this.value.multiply(((PreciseNumber) multiplier.number()).value)); } @Override public NumberInterface divideInternal(NumberInterface divisor) { - return new PreciseNumber(value.divide(((PreciseNumber) divisor).value, this.getMaxPrecision(), RoundingMode.HALF_UP)); + return new PreciseNumber(value.divide(((PreciseNumber) divisor.number()).value, this.getMaxPrecision(), RoundingMode.HALF_UP)); } @Override public NumberInterface addInternal(NumberInterface summand) { - return new PreciseNumber(value.add(((PreciseNumber) summand).value)); + return new PreciseNumber(value.add(((PreciseNumber) summand.number()).value)); } @Override public NumberInterface subtractInternal(NumberInterface subtrahend) { - return new PreciseNumber(value.subtract(((PreciseNumber) subtrahend).value)); + return new PreciseNumber(value.subtract(((PreciseNumber) subtrahend.number()).value)); } @Override @@ -90,7 +90,7 @@ public class PreciseNumber extends NumberInterface { @Override public int compareTo(NumberInterface number) { - return value.compareTo(((PreciseNumber) number).value); + return value.compareTo(((PreciseNumber) number.number()).value); } @Override diff --git a/src/main/java/org/nwapw/abacus/plugin/StandardPlugin.java b/src/main/java/org/nwapw/abacus/plugin/StandardPlugin.java index 363af7e..b6109b2 100755 --- a/src/main/java/org/nwapw/abacus/plugin/StandardPlugin.java +++ b/src/main/java/org/nwapw/abacus/plugin/StandardPlugin.java @@ -94,7 +94,7 @@ public class StandardPlugin extends Plugin { public static final Operator OP_DIVIDE = new Operator(OperatorAssociativity.LEFT, OperatorType.BINARY_INFIX, 1, new Function() { @Override protected boolean matchesParams(NumberInterface[] params) { - return params.length == 2 && params[1].compareTo(NaiveNumber.ZERO.promoteTo(params[1].getClass())) != 0; + return params.length == 2 && params[1].compareTo(NaiveNumber.ZERO.promoteTo(params[1].getClassVal())) != 0; } @Override @@ -110,26 +110,26 @@ public class StandardPlugin extends Plugin { @Override protected boolean matchesParams(NumberInterface[] params) { return params.length == 1 - && params[0].fractionalPart().compareTo(NaiveNumber.ZERO.promoteTo(params[0].getClass())) == 0 + && params[0].fractionalPart().compareTo(NaiveNumber.ZERO.promoteTo(params[0].getClassVal())) == 0 && params[0].signum() >= 0; } @Override protected NumberInterface applyInternal(NumberInterface[] params) { if (params[0].signum() == 0) { - return fromInt(params[0].getClass(), 1); + return fromInt(params[0].getClassVal(), 1); } NumberInterface factorial = params[0]; NumberInterface multiplier = params[0]; //It is necessary to later prevent calls of factorial on anything but non-negative integers. - while ((multiplier = multiplier.subtract(NaiveNumber.ONE.promoteTo(multiplier.getClass()))).signum() == 1) { + while ((multiplier = multiplier.subtract(NaiveNumber.ONE.promoteTo(multiplier.getClassVal()))).signum() == 1) { factorial = factorial.multiply(multiplier); } return factorial; - /*if(!storedList.containsKey(params[0].getClass())){ - storedList.put(params[0].getClass(), new ArrayList()); - storedList.get(params[0].getClass()).add(NaiveNumber.ONE.promoteTo(params[0].getClass())); - storedList.get(params[0].getClass()).add(NaiveNumber.ONE.promoteTo(params[0].getClass())); + /*if(!storedList.containsKey(params[0].getClassVal())){ + storedList.put(params[0].getClassVal(), new ArrayList()); + storedList.get(params[0].getClassVal()).add(NaiveNumber.ONE.promoteTo(params[0].getClassVal())); + storedList.get(params[0].getClassVal()).add(NaiveNumber.ONE.promoteTo(params[0].getClassVal())); }*/ } }); @@ -144,7 +144,7 @@ public class StandardPlugin extends Plugin { @Override protected NumberInterface applyInternal(NumberInterface[] params) { - return params[0].multiply((new NaiveNumber(params[0].signum())).promoteTo(params[0].getClass())); + return params[0].multiply((new NaiveNumber(params[0].signum())).promoteTo(params[0].getClassVal())); } }; /** @@ -153,31 +153,31 @@ public class StandardPlugin extends Plugin { public static final Function FUNCTION_LN = new Function() { @Override protected boolean matchesParams(NumberInterface[] params) { - return params.length == 1 && params[0].compareTo(NaiveNumber.ZERO.promoteTo(params[0].getClass())) > 0; + return params.length == 1 && params[0].compareTo(NaiveNumber.ZERO.promoteTo(params[0].getClassVal())) > 0; } @Override protected NumberInterface applyInternal(NumberInterface[] params) { NumberInterface param = params[0]; int powersOf2 = 0; - while (FUNCTION_ABS.apply(param.subtract(NaiveNumber.ONE.promoteTo(param.getClass()))).compareTo(new NaiveNumber(0.1).promoteTo(param.getClass())) >= 0) { - if (param.subtract(NaiveNumber.ONE.promoteTo(param.getClass())).signum() == 1) { - param = param.divide(fromInt(param.getClass(), 2)); + while (FUNCTION_ABS.apply(param.subtract(NaiveNumber.ONE.promoteTo(param.getClassVal()))).compareTo(new NaiveNumber(0.1).promoteTo(param.getClassVal())) >= 0) { + if (param.subtract(NaiveNumber.ONE.promoteTo(param.getClassVal())).signum() == 1) { + param = param.divide(fromInt(param.getClassVal(), 2)); powersOf2++; - if (param.subtract(NaiveNumber.ONE.promoteTo(param.getClass())).signum() != 1) { + if (param.subtract(NaiveNumber.ONE.promoteTo(param.getClassVal())).signum() != 1) { break; //No infinite loop for you. } } else { - param = param.multiply(fromInt(param.getClass(), 2)); + param = param.multiply(fromInt(param.getClassVal(), 2)); powersOf2--; - if (param.subtract(NaiveNumber.ONE.promoteTo(param.getClass())).signum() != -1) { + if (param.subtract(NaiveNumber.ONE.promoteTo(param.getClassVal())).signum() != -1) { break; //No infinite loop for you. } } } - return getLog2(param).multiply((new NaiveNumber(powersOf2)).promoteTo(param.getClass())).add(getLogPartialSum(param)); + return getLog2(param).multiply((new NaiveNumber(powersOf2)).promoteTo(param.getClassVal())).add(getLogPartialSum(param)); } /** @@ -189,13 +189,13 @@ public class StandardPlugin extends Plugin { private NumberInterface getLogPartialSum(NumberInterface x) { NumberInterface maxError = getMaxError(x); - x = x.subtract(NaiveNumber.ONE.promoteTo(x.getClass())); //Terms used are for log(x+1). + x = x.subtract(NaiveNumber.ONE.promoteTo(x.getClassVal())); //Terms used are for log(x+1). NumberInterface currentNumerator = x, currentTerm = x, sum = x; int n = 1; while (FUNCTION_ABS.apply(currentTerm).compareTo(maxError) > 0) { n++; currentNumerator = currentNumerator.multiply(x).negate(); - currentTerm = currentNumerator.divide(new NaiveNumber(n).promoteTo(x.getClass())); + currentTerm = currentNumerator.divide(new NaiveNumber(n).promoteTo(x.getClassVal())); sum = sum.add(currentTerm); } return sum; @@ -208,18 +208,18 @@ public class StandardPlugin extends Plugin { */ private NumberInterface getLog2(NumberInterface number) { NumberInterface maxError = getMaxError(number); - //NumberInterface errorBound = fromInt(number.getClass(), 1); + //NumberInterface errorBound = fromInt(number.getClassVal(), 1); //We'll use the series \sigma_{n >= 1) ((1/3^n + 1/4^n) * 1/n) //In the following, a=1/3^n, b=1/4^n, c = 1/n. //a is also an error bound. - NumberInterface a = fromInt(number.getClass(), 1), b = a, c = a; - NumberInterface sum = NaiveNumber.ZERO.promoteTo(number.getClass()); + NumberInterface a = fromInt(number.getClassVal(), 1), b = a, c = a; + NumberInterface sum = NaiveNumber.ZERO.promoteTo(number.getClassVal()); int n = 0; while (a.compareTo(maxError) >= 1) { n++; - a = a.divide(fromInt(number.getClass(), 3)); - b = b.divide(fromInt(number.getClass(), 4)); - c = NaiveNumber.ONE.promoteTo(number.getClass()).divide((new NaiveNumber(n)).promoteTo(number.getClass())); + a = a.divide(fromInt(number.getClassVal(), 3)); + b = b.divide(fromInt(number.getClassVal(), 4)); + c = NaiveNumber.ONE.promoteTo(number.getClassVal()).divide((new NaiveNumber(n)).promoteTo(number.getClassVal())); sum = sum.add(a.add(b).multiply(c)); } return sum; @@ -236,7 +236,7 @@ public class StandardPlugin extends Plugin { @Override protected NumberInterface applyInternal(NumberInterface[] params) { - return OP_CARET.getFunction().apply(params[0], ((new NaiveNumber(0.5)).promoteTo(params[0].getClass()))); + return OP_CARET.getFunction().apply(params[0], ((new NaiveNumber(0.5)).promoteTo(params[0].getClassVal()))); } }; /** @@ -304,25 +304,25 @@ public class StandardPlugin extends Plugin { NumberInterface maxError = getMaxError(params[0]); int n = 0; if (params[0].signum() <= 0) { - NumberInterface currentTerm = NaiveNumber.ONE.promoteTo(params[0].getClass()), sum = currentTerm; + NumberInterface currentTerm = NaiveNumber.ONE.promoteTo(params[0].getClassVal()), sum = currentTerm; while (FUNCTION_ABS.apply(currentTerm).compareTo(maxError) > 0) { n++; - currentTerm = currentTerm.multiply(params[0]).divide((new NaiveNumber(n)).promoteTo(params[0].getClass())); + currentTerm = currentTerm.multiply(params[0]).divide((new NaiveNumber(n)).promoteTo(params[0].getClassVal())); sum = sum.add(currentTerm); } return sum; } else { //We need n such that x^(n+1) * 3^ceil(x) <= maxError * (n+1)!. //right and left refer to lhs and rhs in the above inequality. - NumberInterface sum = NaiveNumber.ONE.promoteTo(params[0].getClass()); + NumberInterface sum = NaiveNumber.ONE.promoteTo(params[0].getClassVal()); NumberInterface nextNumerator = params[0]; - NumberInterface left = params[0].multiply(fromInt(params[0].getClass(), 3).intPow(params[0].ceiling().intValue())), right = maxError; + NumberInterface left = params[0].multiply(fromInt(params[0].getClassVal(), 3).intPow(params[0].ceiling().intValue())), right = maxError; do { - sum = sum.add(nextNumerator.divide(factorial(params[0].getClass(), n + 1))); + sum = sum.add(nextNumerator.divide(factorial(params[0].getClassVal(), n + 1))); n++; nextNumerator = nextNumerator.multiply(params[0]); left = left.multiply(params[0]); - NumberInterface nextN = (new NaiveNumber(n + 1)).promoteTo(params[0].getClass()); + NumberInterface nextN = (new NaiveNumber(n + 1)).promoteTo(params[0].getClassVal()); right = right.multiply(nextN); //System.out.println(left + ", " + right); } @@ -339,16 +339,16 @@ public class StandardPlugin extends Plugin { @Override protected boolean matchesParams(NumberInterface[] params) { return params.length == 2 - && !(params[0].compareTo(NaiveNumber.ZERO.promoteTo(params[0].getClass())) == 0 - && params[1].compareTo(NaiveNumber.ZERO.promoteTo(params[1].getClass())) == 0); + && !(params[0].compareTo(NaiveNumber.ZERO.promoteTo(params[0].getClassVal())) == 0 + && params[1].compareTo(NaiveNumber.ZERO.promoteTo(params[1].getClassVal())) == 0); } @Override protected NumberInterface applyInternal(NumberInterface[] params) { - if (params[0].compareTo(NaiveNumber.ZERO.promoteTo(params[0].getClass())) == 0) - return NaiveNumber.ZERO.promoteTo(params[0].getClass()); - else if (params[1].compareTo(NaiveNumber.ZERO.promoteTo(params[0].getClass())) == 0) - return NaiveNumber.ONE.promoteTo(params[1].getClass()); + if (params[0].compareTo(NaiveNumber.ZERO.promoteTo(params[0].getClassVal())) == 0) + return NaiveNumber.ZERO.promoteTo(params[0].getClassVal()); + else if (params[1].compareTo(NaiveNumber.ZERO.promoteTo(params[0].getClassVal())) == 0) + return NaiveNumber.ONE.promoteTo(params[1].getClassVal()); return FUNCTION_EXP.apply(FUNCTION_LN.apply(FUNCTION_ABS.apply(params[0])).multiply(params[1])); } }); @@ -363,13 +363,13 @@ public class StandardPlugin extends Plugin { @Override protected NumberInterface applyInternal(NumberInterface[] params) { - NumberInterface pi = piFor(params[0].getClass()); - NumberInterface twoPi = pi.multiply(fromInt(pi.getClass(), 2)); + NumberInterface pi = piFor(params[0].getClassVal()); + NumberInterface twoPi = pi.multiply(fromInt(pi.getClassVal(), 2)); NumberInterface theta = getSmallAngle(params[0], pi); //System.out.println(theta); - if (theta.compareTo(pi.multiply(new NaiveNumber(1.5).promoteTo(twoPi.getClass()))) >= 0) { + if (theta.compareTo(pi.multiply(new NaiveNumber(1.5).promoteTo(twoPi.getClassVal()))) >= 0) { theta = theta.subtract(twoPi); - } else if (theta.compareTo(pi.divide(fromInt(pi.getClass(), 2))) > 0) { + } else if (theta.compareTo(pi.divide(fromInt(pi.getClassVal(), 2))) > 0) { theta = pi.subtract(theta); } //System.out.println(theta); @@ -387,7 +387,7 @@ public class StandardPlugin extends Plugin { @Override protected NumberInterface applyInternal(NumberInterface[] params) { - return functionSin.apply(piFor(params[0].getClass()).divide(fromInt(params[0].getClass(), 2)) + return functionSin.apply(piFor(params[0].getClassVal()).divide(fromInt(params[0].getClassVal(), 2)) .subtract(params[0])); } }; @@ -416,7 +416,7 @@ public class StandardPlugin extends Plugin { @Override protected NumberInterface applyInternal(NumberInterface[] params) { - return NaiveNumber.ONE.promoteTo(params[0].getClass()).divide(functionCos.apply(params[0])); + return NaiveNumber.ONE.promoteTo(params[0].getClassVal()).divide(functionCos.apply(params[0])); } }; /** @@ -430,7 +430,7 @@ public class StandardPlugin extends Plugin { @Override protected NumberInterface applyInternal(NumberInterface[] params) { - return NaiveNumber.ONE.promoteTo(params[0].getClass()).divide(functionSin.apply(params[0])); + return NaiveNumber.ONE.promoteTo(params[0].getClassVal()).divide(functionSin.apply(params[0])); } }; /** @@ -461,7 +461,7 @@ public class StandardPlugin extends Plugin { * @return the value of the partial sum that has the same class as x. */ private static NumberInterface sumSeries(NumberInterface x, BiFunction nthTermFunction, int n) { - NumberInterface sum = NaiveNumber.ZERO.promoteTo(x.getClass()); + NumberInterface sum = NaiveNumber.ZERO.promoteTo(x.getClassVal()); for (int i = 0; i <= n; i++) { sum = sum.add(nthTermFunction.apply(i, x)); } @@ -475,7 +475,7 @@ public class StandardPlugin extends Plugin { * @return the maximum error. */ private static NumberInterface getMaxError(NumberInterface number) { - return fromInt(number.getClass(), 10).intPow(-number.getMaxPrecision()); + return fromInt(number.getClassVal(), 10).intPow(-number.getMaxPrecision()); } /** @@ -514,7 +514,7 @@ public class StandardPlugin extends Plugin { do { n += 2; power = power.multiply(multiplier); - currentTerm = power.divide(factorial(x.getClass(), n)); + currentTerm = power.divide(factorial(x.getClassVal(), n)); sum = sum.add(currentTerm); } while (FUNCTION_ABS.apply(currentTerm).compareTo(maxError) > 0); return sum; @@ -527,7 +527,7 @@ public class StandardPlugin extends Plugin { * @return theta in [0, 2pi) that differs from phi by a multiple of 2pi. */ private static NumberInterface getSmallAngle(NumberInterface phi, NumberInterface pi) { - NumberInterface twoPi = pi.multiply(new NaiveNumber("2").promoteTo(phi.getClass())); + NumberInterface twoPi = pi.multiply(new NaiveNumber("2").promoteTo(phi.getClassVal())); NumberInterface theta = FUNCTION_ABS.apply(phi).subtract(twoPi .multiply(FUNCTION_ABS.apply(phi).divide(twoPi).floor())); //Now theta is in [0, 2pi). if (phi.signum() < 0) { diff --git a/src/main/java/org/nwapw/abacus/tree/NumberReducer.java b/src/main/java/org/nwapw/abacus/tree/NumberReducer.java index 8584231..9699363 100644 --- a/src/main/java/org/nwapw/abacus/tree/NumberReducer.java +++ b/src/main/java/org/nwapw/abacus/tree/NumberReducer.java @@ -3,6 +3,7 @@ package org.nwapw.abacus.tree; import org.nwapw.abacus.Abacus; import org.nwapw.abacus.function.Function; import org.nwapw.abacus.number.NumberInterface; +import org.nwapw.abacus.number.Variable; /** * A reducer implementation that turns a tree into a single number. @@ -47,6 +48,8 @@ public class NumberReducer implements Reducer { Function function = abacus.getPluginManager().functionFor(((FunctionNode) node).getFunction()); if (function == null) return null; return function.apply(convertedChildren); + } else if (node instanceof VariableNode){ + return (NumberInterface)new Variable(abacus.getVar(((VariableNode)node).getVariable()),((VariableNode)node).getVariable()); } return null; } diff --git a/src/test/java/org/nwapw/abacus/tests/CalculationTests.java b/src/test/java/org/nwapw/abacus/tests/CalculationTests.java index e451732..e8e505e 100644 --- a/src/test/java/org/nwapw/abacus/tests/CalculationTests.java +++ b/src/test/java/org/nwapw/abacus/tests/CalculationTests.java @@ -11,7 +11,7 @@ import org.nwapw.abacus.tree.TreeNode; public class CalculationTests { - private static Abacus abacus = new Abacus(new Configuration(0, "precise", new String[]{})); + private static Abacus abacus = new Abacus(new Configuration(0, "precise", new String[]{}),null); @BeforeClass public static void prepareTests(){ diff --git a/src/test/java/org/nwapw/abacus/tests/TokenizerTests.java b/src/test/java/org/nwapw/abacus/tests/TokenizerTests.java index 56515e3..e98d6dc 100644 --- a/src/test/java/org/nwapw/abacus/tests/TokenizerTests.java +++ b/src/test/java/org/nwapw/abacus/tests/TokenizerTests.java @@ -19,7 +19,7 @@ import java.util.List; public class TokenizerTests { - private static Abacus abacus = new Abacus(new Configuration(0, "precise", new String[]{})); + private static Abacus abacus = new Abacus(new Configuration(0, "precise", new String[]{}),null); private static LexerTokenizer lexerTokenizer = new LexerTokenizer(); private static Function subtractFunction = new Function() { @Override