mirror of
https://github.com/DanilaFe/abacus
synced 2024-11-13 14:19:53 -08:00
Merge branch 'master' of https://github.com/DanilaFe/abacus
This commit is contained in:
commit
37d5c04911
|
@ -1,9 +1,34 @@
|
|||
package org.nwapw.abacus;
|
||||
|
||||
import org.nwapw.abacus.plugin.PluginManager;
|
||||
import org.nwapw.abacus.plugin.StandardPlugin;
|
||||
import org.nwapw.abacus.window.Window;
|
||||
|
||||
import javax.swing.*;
|
||||
|
||||
public class Abacus {
|
||||
|
||||
private Window mainUi;
|
||||
private PluginManager manager;
|
||||
|
||||
public Abacus(){
|
||||
init();
|
||||
}
|
||||
|
||||
private void init() {
|
||||
try {
|
||||
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
|
||||
} catch (ClassNotFoundException | InstantiationException | UnsupportedLookAndFeelException | IllegalAccessException e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
manager = new PluginManager();
|
||||
manager.addInstantiated(new StandardPlugin(manager));
|
||||
mainUi = new Window();
|
||||
mainUi.setVisible(true);
|
||||
}
|
||||
|
||||
public static void main(String[] args){
|
||||
System.out.println("Hello world!");
|
||||
new Abacus();
|
||||
}
|
||||
|
||||
}
|
||||
|
|
47
src/org/nwapw/abacus/function/Function.java
Executable file
47
src/org/nwapw/abacus/function/Function.java
Executable file
|
@ -0,0 +1,47 @@
|
|||
package org.nwapw.abacus.function;
|
||||
|
||||
import org.nwapw.abacus.number.NaiveNumber;
|
||||
import org.nwapw.abacus.number.NumberInterface;
|
||||
|
||||
import java.util.HashMap;
|
||||
|
||||
/**
|
||||
* A function that operates on one or more
|
||||
* inputs and returns a single number.
|
||||
*/
|
||||
public abstract class Function {
|
||||
|
||||
/**
|
||||
* A map to correctly promote different number implementations to each other.
|
||||
*/
|
||||
private static final HashMap<Class<? extends NumberInterface>, Integer> priorityMap =
|
||||
new HashMap<Class<? extends NumberInterface>, Integer>() {{
|
||||
put(NaiveNumber.class, 0);
|
||||
}};
|
||||
|
||||
/**
|
||||
* Checks whether the given params will work for the given function.
|
||||
* @param params the given params
|
||||
* @return true if the params can be used with this function.
|
||||
*/
|
||||
protected abstract boolean matchesParams(NumberInterface[] params);
|
||||
|
||||
/**
|
||||
* Internal apply implementation, which already receives appropriately promoted
|
||||
* parameters that have bee run through matchesParams
|
||||
* @param params the promoted parameters.
|
||||
* @return the return value of the function.
|
||||
*/
|
||||
protected abstract NumberInterface applyInternal(NumberInterface[] params);
|
||||
|
||||
/**
|
||||
* Function to check, promote arguments and run the function.
|
||||
* @param params the raw input parameters.
|
||||
* @return the return value of the function, or null if an error occurred.
|
||||
*/
|
||||
public NumberInterface apply(NumberInterface...params) {
|
||||
if(!matchesParams(params)) return null;
|
||||
return applyInternal(params);
|
||||
}
|
||||
|
||||
}
|
|
@ -9,19 +9,42 @@ import java.util.ArrayList;
|
|||
import java.util.Comparator;
|
||||
import java.util.HashSet;
|
||||
|
||||
/**
|
||||
* A lexer that can generate tokens of a given type given a list of regular expressions
|
||||
* to operate on.
|
||||
* @param <T> the type used to identify which match belongs to which pattern.
|
||||
*/
|
||||
public class Lexer<T> {
|
||||
|
||||
/**
|
||||
* The registered patterns.
|
||||
*/
|
||||
private ArrayList<Pattern<T>> patterns;
|
||||
|
||||
/**
|
||||
* Creates a new lexer with no registered patterns.
|
||||
*/
|
||||
public Lexer(){
|
||||
patterns = new ArrayList<>();
|
||||
}
|
||||
|
||||
/**
|
||||
* Registers a single pattern.
|
||||
* @param pattern the pattern regex
|
||||
* @param id the ID by which to identify the pattern.
|
||||
*/
|
||||
public void register(String pattern, T id){
|
||||
Pattern<T> compiledPattern = new Pattern<>(pattern, id);
|
||||
if(compiledPattern.getHead() != null) patterns.add(compiledPattern);
|
||||
}
|
||||
|
||||
/**
|
||||
* Reads one token from the given string.
|
||||
* @param from the string to read from
|
||||
* @param startAt the index to start at
|
||||
* @param compare the comparator used to sort tokens by their ID.
|
||||
* @return the best match.
|
||||
*/
|
||||
public Match<T> lexOne(String from, int startAt, Comparator<T> compare){
|
||||
ArrayList<Match<T>> matches = new ArrayList<>();
|
||||
HashSet<PatternNode<T>> currentSet = new HashSet<>();
|
||||
|
@ -53,6 +76,13 @@ public class Lexer<T> {
|
|||
return matches.isEmpty() ? null : matches.get(matches.size() - 1);
|
||||
}
|
||||
|
||||
/**
|
||||
* Reads all tokens from a string.
|
||||
* @param from the string to start from.
|
||||
* @param startAt the index to start at.
|
||||
* @param compare the comparator used to sort matches by their IDs.
|
||||
* @return the resulting list of matches, in order, or null on error.
|
||||
*/
|
||||
public ArrayList<Match<T>> lexAll(String from, int startAt, Comparator<T> compare){
|
||||
int index = startAt;
|
||||
ArrayList<Match<T>> matches = new ArrayList<>();
|
||||
|
|
|
@ -1,5 +1,9 @@
|
|||
package org.nwapw.abacus.lexing.pattern;
|
||||
|
||||
/**
|
||||
* A pattern node that matches any character.
|
||||
* @param <T> the type that's used to tell which pattern this node belongs to.
|
||||
*/
|
||||
public class AnyNode<T> extends PatternNode<T> {
|
||||
|
||||
@Override
|
||||
|
|
|
@ -1,13 +1,28 @@
|
|||
package org.nwapw.abacus.lexing.pattern;
|
||||
|
||||
/**
|
||||
* A node that represents a successful match.
|
||||
* @param <T> the type that's used to tell which pattern this node belongs to.
|
||||
*/
|
||||
public class EndNode<T> extends PatternNode<T> {
|
||||
|
||||
/**
|
||||
* The ID of the pattenr that has been matched.
|
||||
*/
|
||||
private T patternId;
|
||||
|
||||
/**
|
||||
* Creates a new end node with the given ID.
|
||||
* @param patternId the pattern ID.
|
||||
*/
|
||||
public EndNode(T patternId){
|
||||
this.patternId = patternId;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the pattern ID.
|
||||
* @return the pattern ID.
|
||||
*/
|
||||
public T getPatternId(){
|
||||
return patternId;
|
||||
}
|
||||
|
|
|
@ -3,6 +3,10 @@ package org.nwapw.abacus.lexing.pattern;
|
|||
import java.util.ArrayList;
|
||||
import java.util.Collection;
|
||||
|
||||
/**
|
||||
* A node that is used as structural glue in pattern compilation.
|
||||
* @param <T> the type that's used to tell which pattern this node belongs to.
|
||||
*/
|
||||
public class LinkNode<T> extends PatternNode<T> {
|
||||
|
||||
@Override
|
||||
|
|
|
@ -1,25 +1,56 @@
|
|||
package org.nwapw.abacus.lexing.pattern;
|
||||
|
||||
/**
|
||||
* A match that has been generated by the lexer.
|
||||
* @param <T> the type used to represent the ID of the pattern this match belongs to.
|
||||
*/
|
||||
public class Match<T> {
|
||||
|
||||
/**
|
||||
* The bottom range of the string, inclusive.
|
||||
*/
|
||||
private int from;
|
||||
/**
|
||||
* The top range of the string, exclusive.
|
||||
*/
|
||||
private int to;
|
||||
/**
|
||||
* The pattern type this match matched.
|
||||
*/
|
||||
private T type;
|
||||
|
||||
/**
|
||||
* Creates a new match with the given parameters.
|
||||
* @param from the bottom range of the string.
|
||||
* @param to the top range of the string.
|
||||
* @param type the type of the match.
|
||||
*/
|
||||
public Match(int from, int to, T type){
|
||||
this.from = from;
|
||||
this.to = to;
|
||||
this.type = type;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the bottom range bound of the string.
|
||||
* @return the bottom range bound of the string.
|
||||
*/
|
||||
public int getFrom() {
|
||||
return from;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the top range bound of the string.
|
||||
* @return the top range bound of the string.
|
||||
*/
|
||||
public int getTo() {
|
||||
return to;
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the pattern type of the node.
|
||||
* @return the ID of the pattern that this match matched.
|
||||
*/
|
||||
public T getType() {
|
||||
return type;
|
||||
}
|
||||
|
|
|
@ -5,13 +5,33 @@ import java.util.HashMap;
|
|||
import java.util.Stack;
|
||||
import java.util.function.Function;
|
||||
|
||||
/**
|
||||
* A pattern that can be compiled from a string and used in lexing.
|
||||
* @param <T> the type that is used to identify and sort this pattern.
|
||||
*/
|
||||
public class Pattern<T> {
|
||||
|
||||
/**
|
||||
* The ID of this pattern.
|
||||
*/
|
||||
private T id;
|
||||
/**
|
||||
* The head of this pattern.
|
||||
*/
|
||||
private PatternNode<T> head;
|
||||
/**
|
||||
* The source string of this pattern.
|
||||
*/
|
||||
private String source;
|
||||
/**
|
||||
* The index at which the compilation has stopped.
|
||||
*/
|
||||
private int index;
|
||||
|
||||
/**
|
||||
* A map of regex operator to functions that modify a PatternChain
|
||||
* with the appropriate operation.
|
||||
*/
|
||||
private HashMap<Character, Function<PatternChain<T>, PatternChain<T>>> operations =
|
||||
new HashMap<Character, Function<PatternChain<T>, PatternChain<T>>>() {{
|
||||
put('+', Pattern.this::transformPlus);
|
||||
|
@ -19,11 +39,23 @@ public class Pattern<T> {
|
|||
put('?', Pattern.this::transformQuestion);
|
||||
}};
|
||||
|
||||
/**
|
||||
* A regex operator function that turns the chain
|
||||
* into a one-or-more chain.
|
||||
* @param chain the chain to transform.
|
||||
* @return the modified chain.
|
||||
*/
|
||||
private PatternChain<T> transformPlus(PatternChain<T> chain){
|
||||
chain.tail.outputStates.add(chain.head);
|
||||
return chain;
|
||||
}
|
||||
|
||||
/**
|
||||
* A regex operator function that turns the chain
|
||||
* into a zero-or-more chain.
|
||||
* @param chain the chain to transform.
|
||||
* @return the modified chain.
|
||||
*/
|
||||
private PatternChain<T> transformStar(PatternChain<T> chain){
|
||||
LinkNode<T> newTail = new LinkNode<>();
|
||||
LinkNode<T> newHead = new LinkNode<>();
|
||||
|
@ -36,6 +68,12 @@ public class Pattern<T> {
|
|||
return chain;
|
||||
}
|
||||
|
||||
/**
|
||||
* A regex operator function that turns the chain
|
||||
* into a zero-or-one chain.
|
||||
* @param chain the chain to transform.
|
||||
* @return the modified chain.
|
||||
*/
|
||||
private PatternChain<T> transformQuestion(PatternChain<T> chain){
|
||||
LinkNode<T> newTail = new LinkNode<>();
|
||||
LinkNode<T> newHead = new LinkNode<>();
|
||||
|
@ -47,6 +85,11 @@ public class Pattern<T> {
|
|||
return chain;
|
||||
}
|
||||
|
||||
/**
|
||||
* Combines a collection of chains into one OR chain.
|
||||
* @param collection the collection of chains to combine.
|
||||
* @return the resulting OR chain.
|
||||
*/
|
||||
private PatternChain<T> combineChains(Collection<PatternChain<T>> collection){
|
||||
LinkNode<T> head = new LinkNode<>();
|
||||
LinkNode<T> tail = new LinkNode<>();
|
||||
|
@ -58,6 +101,10 @@ public class Pattern<T> {
|
|||
return newChain;
|
||||
}
|
||||
|
||||
/**
|
||||
* Parses a single value from the input into a chain.
|
||||
* @return the resulting chain, or null on error.
|
||||
*/
|
||||
private PatternChain<T> parseValue(){
|
||||
if(index >= source.length()) return null;
|
||||
if(source.charAt(index) == '\\'){
|
||||
|
@ -66,6 +113,10 @@ public class Pattern<T> {
|
|||
return new PatternChain<>(new ValueNode<>(source.charAt(index++)));
|
||||
}
|
||||
|
||||
/**
|
||||
* Parses a [] range from the input into a chain.
|
||||
* @return the resulting chain, or null on error.
|
||||
*/
|
||||
private PatternChain<T> parseOr(){
|
||||
Stack<PatternChain<T>> orStack = new Stack<>();
|
||||
index++;
|
||||
|
@ -88,6 +139,12 @@ public class Pattern<T> {
|
|||
return (orStack.size() == 1) ? orStack.pop() : combineChains(orStack);
|
||||
}
|
||||
|
||||
/**
|
||||
* Parses a repeatable segment from the input into a chain
|
||||
* @param isSubsegment whether the segment is a sub-expression "()", and therefore
|
||||
* whether to expect a closing brace.
|
||||
* @return the resulting chain, or null on error.
|
||||
*/
|
||||
private PatternChain<T> parseSegment(boolean isSubsegment){
|
||||
if(index >= source.length() || ((source.charAt(index) != '(') && isSubsegment)) return null;
|
||||
if(isSubsegment) index++;
|
||||
|
@ -152,6 +209,11 @@ public class Pattern<T> {
|
|||
return fullChain;
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates / compiles a new pattern with the given id from the given string.
|
||||
* @param from the string to compile a pattern from.
|
||||
* @param id the ID to use.
|
||||
*/
|
||||
public Pattern(String from, T id){
|
||||
this.id = id;
|
||||
index = 0;
|
||||
|
@ -166,6 +228,10 @@ public class Pattern<T> {
|
|||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Gets the head PatternNode, for use in matching
|
||||
* @return the pattern node.
|
||||
*/
|
||||
public PatternNode<T> getHead() {
|
||||
return head;
|
||||
}
|
||||
|
|
|
@ -1,23 +1,52 @@
|
|||
package org.nwapw.abacus.lexing.pattern;
|
||||
|
||||
/**
|
||||
* A chain of nodes that can be treated as a single unit.
|
||||
* Used during pattern compilation.
|
||||
* @param <T> the type used to identify which pattern has been matched.
|
||||
*/
|
||||
public class PatternChain<T> {
|
||||
|
||||
/**
|
||||
* The head node of the chain.
|
||||
*/
|
||||
public PatternNode<T> head;
|
||||
/**
|
||||
* The tail node of the chain.
|
||||
*/
|
||||
public PatternNode<T> tail;
|
||||
|
||||
/**
|
||||
* Creates a new chain with the given start and end.
|
||||
* @param head the start of the chain.
|
||||
* @param tail the end of the chain.
|
||||
*/
|
||||
public PatternChain(PatternNode<T> head, PatternNode<T> tail){
|
||||
this.head = head;
|
||||
this.tail = tail;
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates a chain that starts and ends with the same node.
|
||||
* @param node the node to use.
|
||||
*/
|
||||
public PatternChain(PatternNode<T> node){
|
||||
this(node, node);
|
||||
}
|
||||
|
||||
/**
|
||||
* Creates an empty chain.
|
||||
*/
|
||||
public PatternChain(){
|
||||
this(null);
|
||||
}
|
||||
|
||||
/**
|
||||
* Appends the other chain to this one. This modifies
|
||||
* the nodes, as well.
|
||||
* If this chain is empty, it is set to the other.
|
||||
* @param other the other chain to append.
|
||||
*/
|
||||
public void append(PatternChain<T> other){
|
||||
if(other.head == null || tail == null) {
|
||||
this.head = other.head;
|
||||
|
@ -28,6 +57,12 @@ public class PatternChain<T> {
|
|||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Appends a single node to this chain. This modifies
|
||||
* the nodes, as well.
|
||||
* If this chain is empty, it is set to the node.
|
||||
* @param node the node to append to this chain.
|
||||
*/
|
||||
public void append(PatternNode<T> node){
|
||||
if(tail == null){
|
||||
head = tail = node;
|
||||
|
|
|
@ -4,26 +4,58 @@ import java.util.ArrayList;
|
|||
import java.util.Collection;
|
||||
import java.util.HashSet;
|
||||
|
||||
/**
|
||||
* A base class for a pattern node. Provides all functions
|
||||
* necessary for matching, and is constructed by a Pattern instance
|
||||
* from a string.
|
||||
* @param <T> the type that's used to tell which pattern this node belongs to.
|
||||
*/
|
||||
public class PatternNode<T> {
|
||||
|
||||
/**
|
||||
* The set of states to which the lexer should continue
|
||||
* should this node be correctly matched.
|
||||
*/
|
||||
protected HashSet<PatternNode<T>> outputStates;
|
||||
|
||||
/**
|
||||
* Creates a new pattern node.
|
||||
*/
|
||||
public PatternNode(){
|
||||
outputStates = new HashSet<>();
|
||||
}
|
||||
|
||||
/**
|
||||
* Determines whether the current input character can
|
||||
* be matched by this node.
|
||||
* @param other the character being matched.
|
||||
* @return true if the character can be matched, false otherwise.
|
||||
*/
|
||||
public boolean matches(char other){
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* If this node can be used as part of a range, returns that value.
|
||||
* @return a NULL terminator if this character cannot be converted
|
||||
* into a range bound, or the appropriate range bound if it can.
|
||||
*/
|
||||
public char range(){
|
||||
return '\0';
|
||||
}
|
||||
|
||||
/**
|
||||
* Adds this node in a collection of other nodes.
|
||||
* @param into the collection to add into.
|
||||
*/
|
||||
public void addInto(Collection<PatternNode<T>> into){
|
||||
into.add(this);
|
||||
}
|
||||
|
||||
/**
|
||||
* Adds the node's children into a collection of other nodes.
|
||||
* @param into the collection to add into.
|
||||
*/
|
||||
public void addOutputsInto(Collection<PatternNode<T>> into){
|
||||
outputStates.forEach(e -> e.addInto(into));
|
||||
}
|
||||
|
|
|
@ -1,10 +1,25 @@
|
|||
package org.nwapw.abacus.lexing.pattern;
|
||||
|
||||
/**
|
||||
* A node that matches a range of characters.
|
||||
* @param <T> the type that's used to tell which pattern this node belongs to.
|
||||
*/
|
||||
public class RangeNode<T> extends PatternNode<T> {
|
||||
|
||||
/**
|
||||
* The bottom bound of the range, inclusive.
|
||||
*/
|
||||
private char from;
|
||||
/**
|
||||
* The top bound of the range, inclusive.
|
||||
*/
|
||||
private char to;
|
||||
|
||||
/**
|
||||
* Creates a new range node from the given range.
|
||||
* @param from the bottom bound of the range.
|
||||
* @param to the top bound of hte range.
|
||||
*/
|
||||
public RangeNode(char from, char to){
|
||||
this.from = from;
|
||||
this.to = to;
|
||||
|
|
|
@ -1,9 +1,20 @@
|
|||
package org.nwapw.abacus.lexing.pattern;
|
||||
|
||||
/**
|
||||
* A node that matches a single value.
|
||||
* @param <T> the type that's used to tell which pattern this node belongs to.
|
||||
*/
|
||||
public class ValueNode<T> extends PatternNode<T> {
|
||||
|
||||
/**
|
||||
* The value this node matches.
|
||||
*/
|
||||
private char value;
|
||||
|
||||
/**
|
||||
* Creates a new node that matches the given character.
|
||||
* @param value
|
||||
*/
|
||||
public ValueNode(char value){
|
||||
this.value = value;
|
||||
}
|
||||
|
|
|
@ -1,20 +0,0 @@
|
|||
package org.nwapw.abacus.number;
|
||||
|
||||
import java.util.HashMap;
|
||||
|
||||
public abstract class Function {
|
||||
|
||||
private static final HashMap<Class<? extends NumberInterface>, Integer> priorityMap =
|
||||
new HashMap<Class<? extends NumberInterface>, Integer>() {{
|
||||
put(NaiveNumber.class, 0);
|
||||
}};
|
||||
|
||||
protected abstract boolean matchesParams(NumberInterface[] params);
|
||||
protected abstract NumberInterface applyInternal(NumberInterface[] params);
|
||||
|
||||
public NumberInterface apply(NumberInterface...params) {
|
||||
if(!matchesParams(params)) return null;
|
||||
return applyInternal(params);
|
||||
}
|
||||
|
||||
}
|
|
@ -1,14 +1,30 @@
|
|||
package org.nwapw.abacus.number;
|
||||
|
||||
/**
|
||||
* An implementation of NumberInterface using a double.
|
||||
*/
|
||||
public class NaiveNumber implements NumberInterface {
|
||||
|
||||
/**
|
||||
* The value of this number.
|
||||
*/
|
||||
private double value;
|
||||
|
||||
/**
|
||||
* Creates a new NaiveNumber with the given value.
|
||||
* @param value the value to use.
|
||||
*/
|
||||
public NaiveNumber(double value) {
|
||||
this.value = value;
|
||||
}
|
||||
|
||||
/**
|
||||
* The number zero.
|
||||
*/
|
||||
public static final NaiveNumber ZERO = new NaiveNumber(0);
|
||||
/**
|
||||
* The number one.
|
||||
*/
|
||||
public static final NaiveNumber ONE = new NaiveNumber(1);
|
||||
|
||||
@Override
|
||||
|
|
|
@ -1,17 +1,77 @@
|
|||
package org.nwapw.abacus.number;
|
||||
|
||||
/**
|
||||
* An interface used to represent a number.
|
||||
*/
|
||||
public interface NumberInterface {
|
||||
|
||||
/**
|
||||
* The precision to which this number operates.
|
||||
* @return the precision.
|
||||
*/
|
||||
int precision();
|
||||
|
||||
/**
|
||||
* Multiplies this number by another, returning
|
||||
* a new number instance.
|
||||
* @param multiplier the multiplier
|
||||
* @return the result of the multiplication.
|
||||
*/
|
||||
NumberInterface multiply(NumberInterface multiplier);
|
||||
/**
|
||||
* Divides this number by another, returning
|
||||
* a new number instance.
|
||||
* @param divisor the divisor
|
||||
* @return the result of the division.
|
||||
*/
|
||||
NumberInterface divide(NumberInterface divisor);
|
||||
/**
|
||||
* Adds this number to another, returning
|
||||
* a new number instance.
|
||||
* @param summand the summand
|
||||
* @return the result of the summation.
|
||||
*/
|
||||
NumberInterface add(NumberInterface summand);
|
||||
/**
|
||||
* Subtracts another number from this number,
|
||||
* a new number instance.
|
||||
* @param subtrahend the subtrahend.
|
||||
* @return the result of the subtraction.
|
||||
*/
|
||||
NumberInterface subtract(NumberInterface subtrahend);
|
||||
|
||||
/**
|
||||
* Returns a new instance of this number with
|
||||
* the sign flipped.
|
||||
* @return the new instance.
|
||||
*/
|
||||
NumberInterface negate();
|
||||
|
||||
/**
|
||||
* Raises this number to an integer power.
|
||||
* @param exponent
|
||||
* @return
|
||||
*/
|
||||
NumberInterface intPow(int exponent);
|
||||
|
||||
/**
|
||||
* Compares this number to another.
|
||||
* @param number the number to compare to.
|
||||
* @return same as Integer.compare();
|
||||
*/
|
||||
int compareTo(NumberInterface number);
|
||||
|
||||
/**
|
||||
* Same as Math.signum().
|
||||
* @return 1 if this number is positive, -1 if this number is negative, 0 if this number is 0.
|
||||
*/
|
||||
int signum();
|
||||
|
||||
/**
|
||||
* Promotes this class to another number class.
|
||||
* @param toClass the class to promote to.
|
||||
* @return the resulting new instance.
|
||||
*/
|
||||
NumberInterface promoteTo(Class<? extends NumberInterface> toClass);
|
||||
|
||||
}
|
||||
|
|
|
@ -1,28 +0,0 @@
|
|||
package org.nwapw.abacus.number;
|
||||
|
||||
import java.util.HashMap;
|
||||
|
||||
public abstract class externalFunction {
|
||||
|
||||
private HashMap<String, Function> functions;
|
||||
|
||||
public externalFunction(){
|
||||
functions=new HashMap<>();
|
||||
}
|
||||
|
||||
public boolean hasFunction(Function x){
|
||||
return functions.containsKey(x);
|
||||
}
|
||||
public Function getFunction(String x){
|
||||
return functions.get(x);
|
||||
}
|
||||
public boolean registerFunction(String x, Function y){
|
||||
if(!functions.containsKey(x))
|
||||
return functions.put(x,y)==null;
|
||||
return false;
|
||||
}
|
||||
public Function functionFor(String x){
|
||||
return null;
|
||||
}
|
||||
public abstract void load();
|
||||
}
|
43
src/org/nwapw/abacus/plugin/Plugin.java
Normal file
43
src/org/nwapw/abacus/plugin/Plugin.java
Normal file
|
@ -0,0 +1,43 @@
|
|||
package org.nwapw.abacus.plugin;
|
||||
|
||||
import org.nwapw.abacus.function.Function;
|
||||
|
||||
import java.util.HashMap;
|
||||
|
||||
public abstract class Plugin {
|
||||
|
||||
private HashMap<String, Function> functions;
|
||||
private PluginManager manager;
|
||||
|
||||
private Plugin(){ }
|
||||
|
||||
public Plugin(PluginManager manager) {
|
||||
this.manager = manager;
|
||||
functions = new HashMap<>();
|
||||
}
|
||||
|
||||
public final boolean hasFunction(String functionName) {
|
||||
return functions.containsKey(functionName);
|
||||
}
|
||||
|
||||
public final Function getFunction(String functionName) {
|
||||
return functions.get(functionName);
|
||||
}
|
||||
|
||||
protected final boolean registerFunction(String name, Function toRegister) {
|
||||
if(functionFor(name) == null){
|
||||
functions.put(name, toRegister);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
protected final Function functionFor(String name) {
|
||||
Plugin ownerPlugin = manager.pluginForFunction(name);
|
||||
if(ownerPlugin == null) return null;
|
||||
return ownerPlugin.getFunction(name);
|
||||
}
|
||||
|
||||
public abstract void load();
|
||||
|
||||
}
|
49
src/org/nwapw/abacus/plugin/PluginManager.java
Normal file
49
src/org/nwapw/abacus/plugin/PluginManager.java
Normal file
|
@ -0,0 +1,49 @@
|
|||
package org.nwapw.abacus.plugin;
|
||||
|
||||
import java.lang.reflect.InvocationTargetException;
|
||||
import java.util.ArrayList;
|
||||
import java.util.HashMap;
|
||||
|
||||
public class PluginManager {
|
||||
|
||||
private ArrayList<Plugin> plugins;
|
||||
private HashMap<String, Plugin> pluginsForFunctions;
|
||||
|
||||
public PluginManager(){
|
||||
plugins = new ArrayList<>();
|
||||
pluginsForFunctions = new HashMap<>();
|
||||
}
|
||||
|
||||
public Plugin pluginForFunction(String name){
|
||||
if(pluginsForFunctions.containsKey(name)) {
|
||||
return pluginsForFunctions.get(name);
|
||||
}
|
||||
|
||||
Plugin foundPlugin = null;
|
||||
for(Plugin plugin : plugins){
|
||||
if(plugin.hasFunction(name)) {
|
||||
foundPlugin = plugin;
|
||||
break;
|
||||
}
|
||||
}
|
||||
pluginsForFunctions.put(name, foundPlugin);
|
||||
|
||||
return foundPlugin;
|
||||
}
|
||||
|
||||
public void addInstantiated(Plugin plugin){
|
||||
plugin.load();
|
||||
pluginsForFunctions.clear();
|
||||
plugins.add(plugin);
|
||||
}
|
||||
|
||||
public void addClass(Class<?> newClass){
|
||||
if(!Plugin.class.isAssignableFrom(newClass)) return;
|
||||
try {
|
||||
addInstantiated((Plugin) newClass.getConstructor(PluginManager.class).newInstance(this));
|
||||
} catch (InstantiationException | IllegalAccessException | NoSuchMethodException | InvocationTargetException e) {
|
||||
e.printStackTrace();
|
||||
}
|
||||
}
|
||||
|
||||
}
|
34
src/org/nwapw/abacus/number/FunctionDatabase.java → src/org/nwapw/abacus/plugin/StandardPlugin.java
Executable file → Normal file
34
src/org/nwapw/abacus/number/FunctionDatabase.java → src/org/nwapw/abacus/plugin/StandardPlugin.java
Executable file → Normal file
|
@ -1,13 +1,18 @@
|
|||
package org.nwapw.abacus.number;
|
||||
package org.nwapw.abacus.plugin;
|
||||
|
||||
import java.util.HashMap;
|
||||
import org.nwapw.abacus.function.Function;
|
||||
import org.nwapw.abacus.number.NaiveNumber;
|
||||
import org.nwapw.abacus.number.NumberInterface;
|
||||
|
||||
public class FunctionDatabase {
|
||||
public class StandardPlugin extends Plugin {
|
||||
|
||||
private HashMap<String, Function> functions;
|
||||
public StandardPlugin(PluginManager manager) {
|
||||
super(manager);
|
||||
}
|
||||
|
||||
private void registerDefault(){
|
||||
functions.put("+", new Function() {
|
||||
@Override
|
||||
public void load() {
|
||||
registerFunction("+", new Function() {
|
||||
@Override
|
||||
protected boolean matchesParams(NumberInterface[] params) {
|
||||
return params.length >= 1;
|
||||
|
@ -23,7 +28,7 @@ public class FunctionDatabase {
|
|||
}
|
||||
});
|
||||
|
||||
functions.put("-", new Function() {
|
||||
registerFunction("-", new Function() {
|
||||
@Override
|
||||
protected boolean matchesParams(NumberInterface[] params) {
|
||||
return params.length == 2;
|
||||
|
@ -35,7 +40,7 @@ public class FunctionDatabase {
|
|||
}
|
||||
});
|
||||
|
||||
functions.put("*", new Function() {
|
||||
registerFunction("*", new Function() {
|
||||
@Override
|
||||
protected boolean matchesParams(NumberInterface[] params) {
|
||||
return params.length >= 1;
|
||||
|
@ -51,7 +56,7 @@ public class FunctionDatabase {
|
|||
}
|
||||
});
|
||||
|
||||
functions.put("/", new Function() {
|
||||
registerFunction("/", new Function() {
|
||||
@Override
|
||||
protected boolean matchesParams(NumberInterface[] params) {
|
||||
return params.length == 2;
|
||||
|
@ -63,7 +68,7 @@ public class FunctionDatabase {
|
|||
}
|
||||
});
|
||||
|
||||
functions.put("!", new Function() {
|
||||
registerFunction("!", new Function() {
|
||||
@Override
|
||||
protected boolean matchesParams(NumberInterface[] params) {
|
||||
return params.length == 1;
|
||||
|
@ -82,13 +87,4 @@ public class FunctionDatabase {
|
|||
});
|
||||
}
|
||||
|
||||
public FunctionDatabase(){
|
||||
functions = new HashMap<>();
|
||||
registerDefault();
|
||||
}
|
||||
|
||||
public Function getFunction(String name){
|
||||
return functions.get(name);
|
||||
}
|
||||
|
||||
}
|
80
src/org/nwapw/abacus/window/Window.java
Normal file
80
src/org/nwapw/abacus/window/Window.java
Normal file
|
@ -0,0 +1,80 @@
|
|||
package org.nwapw.abacus.window;
|
||||
|
||||
import javax.swing.*;
|
||||
import java.awt.*;
|
||||
|
||||
public class Window extends JFrame {
|
||||
|
||||
private String history;
|
||||
private String lastOutput;
|
||||
|
||||
private JPanel outputPanel;
|
||||
private JTextArea lastOutputArea;
|
||||
private JTextArea historyArea;
|
||||
private JScrollPane historyAreaScroll;
|
||||
|
||||
private JPanel inputPanel;
|
||||
private JTextField inputField;
|
||||
private JButton inputEnterButton;
|
||||
|
||||
private JPanel sidePanel;
|
||||
private JPanel numberSystemPanel;
|
||||
private JComboBox<String> numberSystemList;
|
||||
private JButton functionSelectButton;
|
||||
private JPanel functionSelectPanel;
|
||||
private JComboBox<String> functionList;
|
||||
|
||||
public Window() {
|
||||
super();
|
||||
|
||||
|
||||
history = "";
|
||||
lastOutput = "";
|
||||
|
||||
setSize(640, 480);
|
||||
|
||||
inputField = new JTextField();
|
||||
inputEnterButton = new JButton("Calculate");
|
||||
|
||||
inputPanel = new JPanel();
|
||||
inputPanel.setLayout(new BorderLayout());
|
||||
inputPanel.add(inputField, BorderLayout.CENTER);
|
||||
inputPanel.add(inputEnterButton, BorderLayout.EAST);
|
||||
|
||||
historyArea = new JTextArea(history);
|
||||
historyAreaScroll = new JScrollPane(historyArea);
|
||||
lastOutputArea = new JTextArea(lastOutput);
|
||||
lastOutputArea.setEditable(false);
|
||||
lastOutputArea.setText(":)");
|
||||
|
||||
outputPanel = new JPanel();
|
||||
outputPanel.setLayout(new BorderLayout());
|
||||
outputPanel.add(historyAreaScroll, BorderLayout.CENTER);
|
||||
outputPanel.add(lastOutputArea, BorderLayout.SOUTH);
|
||||
|
||||
numberSystemList = new JComboBox<>();
|
||||
|
||||
numberSystemPanel = new JPanel();
|
||||
numberSystemPanel.setLayout(new BorderLayout());
|
||||
numberSystemPanel.add(new JLabel("Number Type:"), BorderLayout.NORTH);
|
||||
numberSystemPanel.add(numberSystemList, BorderLayout.CENTER);
|
||||
|
||||
functionList = new JComboBox<>();
|
||||
functionSelectButton = new JButton("Select");
|
||||
|
||||
functionSelectPanel = new JPanel();
|
||||
functionSelectPanel.setLayout(new BorderLayout());
|
||||
functionSelectPanel.add(new JLabel("Functions:"), BorderLayout.NORTH);
|
||||
functionSelectPanel.add(functionList, BorderLayout.CENTER);
|
||||
functionSelectPanel.add(functionSelectButton, BorderLayout.SOUTH);
|
||||
|
||||
sidePanel = new JPanel();
|
||||
sidePanel.setLayout(new BorderLayout());
|
||||
sidePanel.add(numberSystemPanel, BorderLayout.NORTH);
|
||||
sidePanel.add(functionSelectPanel, BorderLayout.SOUTH);
|
||||
|
||||
add(outputPanel, BorderLayout.CENTER);
|
||||
add(sidePanel, BorderLayout.EAST);
|
||||
add(inputPanel, BorderLayout.SOUTH);
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user