WarpPI/core/src/main/java/it/cavallium/warppi/util/Utils.java

599 lines
19 KiB
Java

package it.cavallium.warppi.util;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.math.BigDecimal;
import java.math.BigInteger;
import java.math.RoundingMode;
import java.net.URISyntaxException;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import org.nevec.rjm.BigDecimalMath;
import org.nevec.rjm.Rational;
import it.cavallium.warppi.WarpPI;
import it.cavallium.warppi.StaticVars;
import it.cavallium.warppi.gui.graphicengine.BinaryFont;
import it.cavallium.warppi.math.Function;
import it.cavallium.warppi.math.FunctionOperator;
import it.cavallium.warppi.math.FunctionSingle;
import it.cavallium.warppi.math.functions.Division;
import it.cavallium.warppi.math.functions.Expression;
import it.cavallium.warppi.math.functions.Multiplication;
import it.cavallium.warppi.math.functions.Negative;
import it.cavallium.warppi.math.functions.Number;
import it.cavallium.warppi.math.functions.Subtraction;
import it.cavallium.warppi.math.functions.Sum;
import it.cavallium.warppi.math.functions.SumSubtraction;
import it.cavallium.warppi.math.functions.Variable;
import it.cavallium.warppi.math.functions.equations.Equation;
import it.cavallium.warppi.math.functions.equations.EquationsSystemPart;
import it.unimi.dsi.fastutil.objects.ObjectArrayList;
public class Utils {
public static final int scale = 24;
public static final int displayScale = 8;
public static final BigInteger maxFactor = BigInteger.valueOf(1000000L);
public static final int scaleMode = BigDecimal.ROUND_HALF_UP;
public static final RoundingMode scaleMode2 = RoundingMode.HALF_UP;
public static final int maxAutoFractionDigits = 5;
public static boolean newtMode = true;
public static <T> boolean isInArray(final T ch, final T[] a) {
return Arrays.stream(a).anyMatch(item -> ch.equals(item));
}
public static boolean isInArray(final char ch, final char[] a) {
for (final char c : a) {
if (c == ch) {
return true;
}
}
return false;
}
private static final String[] regexNormalSymbols = new String[] { "\\", ".", "[", "]", "{", "}", "(", ")", "*", "+", "-", "?", "^", "$", "|" };
public static String ArrayToRegex(final String[] array) {
String regex = null;
for (final String symbol : array) {
boolean contained = false;
for (final String smb : Utils.regexNormalSymbols) {
if (smb.equals(symbol)) {
contained = true;
break;
}
}
if (contained) {
if (regex != null) {
regex += "|\\" + symbol;
} else {
regex = "\\" + symbol;
}
} else if (regex != null) {
regex += "|" + symbol;
} else {
regex = symbol;
}
}
return regex;
}
public static String ArrayToRegex(final char[] array) {
String regex = null;
for (final char symbol : array) {
boolean contained = false;
for (final String smb : Utils.regexNormalSymbols) {
if (smb.equals(symbol + "")) {
contained = true;
break;
}
}
if (contained) {
if (regex != null) {
regex += "|\\" + symbol;
} else {
regex = "\\" + symbol;
}
} else if (regex != null) {
regex += "|" + symbol;
} else {
regex = symbol + "";
}
}
return regex;
}
public static String[] concat(final String[] a, final String[] b) {
final int aLen = a.length;
final int bLen = b.length;
final String[] c = new String[aLen + bLen];
System.arraycopy(a, 0, c, 0, aLen);
System.arraycopy(b, 0, c, aLen, bLen);
return c;
}
public static char[] concat(final char[] a, final char[] b) {
final int aLen = a.length;
final int bLen = b.length;
final char[] c = new char[aLen + bLen];
System.arraycopy(a, 0, c, 0, aLen);
System.arraycopy(b, 0, c, aLen, bLen);
return c;
}
public static String[] add(final String[] a, final String b) {
final int aLen = a.length;
final String[] c = new String[aLen + 1];
System.arraycopy(a, 0, c, 0, aLen);
c[aLen] = b;
return c;
}
public static char[] add(final char[] a, final char b) {
final int aLen = a.length;
final char[] c = new char[aLen + 1];
System.arraycopy(a, 0, c, 0, aLen);
c[aLen] = b;
return c;
}
public static boolean areThereOnlySettedUpFunctionsSumsEquationsAndSystems(final List<Function> fl) {
for (int i = 0; i < fl.size(); i++) {
if (!(fl.get(i) instanceof Number || fl.get(i) instanceof Variable || fl.get(i) instanceof Sum || fl.get(i) instanceof SumSubtraction || fl.get(i) instanceof Subtraction || fl.get(i) instanceof Equation || fl.get(i) instanceof EquationsSystemPart || fl.get(i) instanceof Expression)) {
if (fl.get(i) instanceof FunctionSingle) {
if (((FunctionSingle) fl.get(i)).getParameter() == null) {
return false;
}
} else if (fl.get(i) instanceof FunctionOperator) {
if (((FunctionOperator) fl.get(i)).getParameter1() == null || ((FunctionOperator) fl.get(i)).getParameter2() == null) {
return false;
}
} else {
return false;
}
}
}
return true;
}
public static boolean areThereOnlySettedUpFunctionsSumsMultiplicationsEquationsAndSystems(final List<Function> fl) {
for (int i = 0; i < fl.size(); i++) {
if (!(fl.get(i) instanceof Number || fl.get(i) instanceof Variable || fl.get(i) instanceof Multiplication || fl.get(i) instanceof Sum || fl.get(i) instanceof SumSubtraction || fl.get(i) instanceof Subtraction || fl.get(i) instanceof Equation || fl.get(i) instanceof EquationsSystemPart || fl.get(i) instanceof Expression)) {
if (fl.get(i) instanceof FunctionSingle) {
if (((FunctionSingle) fl.get(i)).getParameter() == null) {
return false;
}
} else if (fl.get(i) instanceof FunctionOperator) {
if (((FunctionOperator) fl.get(i)).getParameter1() == null || ((FunctionOperator) fl.get(i)).getParameter2() == null) {
return false;
}
} else {
return false;
}
}
}
return true;
}
public static boolean areThereOnlySettedUpFunctionsEquationsAndSystems(final List<Function> fl) {
for (int i = 0; i < fl.size(); i++) {
if (!(fl.get(i) instanceof Number || fl.get(i) instanceof Variable || fl.get(i) instanceof Equation || fl.get(i) instanceof EquationsSystemPart || fl.get(i) instanceof Expression)) {
if (fl.get(i) instanceof FunctionSingle) {
if (((FunctionSingle) fl.get(i)).getParameter() == null) {
return false;
}
} else if (fl.get(i) instanceof FunctionOperator) {
if (((FunctionOperator) fl.get(i)).getParameter1() == null || ((FunctionOperator) fl.get(i)).getParameter2() == null) {
return false;
}
} else {
return false;
}
}
}
return true;
}
public static boolean areThereOnlySettedUpFunctionsAndSystems(final List<Function> fl) {
for (int i = 0; i < fl.size(); i++) {
if (!(fl.get(i) instanceof Number || fl.get(i) instanceof Variable || fl.get(i) instanceof Equation || fl.get(i) instanceof EquationsSystemPart || fl.get(i) instanceof Expression)) {
if (fl.get(i) instanceof FunctionSingle) {
if (((FunctionSingle) fl.get(i)).getParameter() == null) {
return false;
}
} else if (fl.get(i) instanceof FunctionOperator) {
if (((FunctionOperator) fl.get(i)).getParameter1() == null || ((FunctionOperator) fl.get(i)).getParameter2() == null) {
return false;
}
} else {
return false;
}
}
}
return true;
}
public static boolean areThereOnlyEmptySNFunctions(final List<Function> fl) {
for (int i = 0; i < fl.size(); i++) {
if (fl.get(i) instanceof FunctionSingle) {
if (((FunctionSingle) fl.get(i)).getParameter() == null) {
return true;
}
}
}
return false;
}
public static boolean areThereOnlyEmptyNSNFunctions(final List<Function> fl) {
for (int i = 0; i < fl.size(); i++) {
if (fl.get(i) instanceof FunctionOperator && !(fl.get(i) instanceof Sum) && !(fl.get(i) instanceof SumSubtraction) && !(fl.get(i) instanceof Subtraction) && !(fl.get(i) instanceof Multiplication) && !(fl.get(i) instanceof Division)) {
if (((FunctionOperator) fl.get(i)).getParameter1() == null && ((FunctionOperator) fl.get(i)).getParameter2() == null) {
return true;
}
}
}
return false;
}
public static boolean areThereEmptyMultiplications(final List<Function> fl) {
for (int i = 0; i < fl.size(); i++) {
if (fl.get(i) instanceof Multiplication || fl.get(i) instanceof Division) {
if (((FunctionOperator) fl.get(i)).getParameter1() == null && ((FunctionOperator) fl.get(i)).getParameter2() == null) {
return true;
}
}
}
return false;
}
public static boolean areThereEmptySums(final List<Function> fl) {
for (int i = 0; i < fl.size(); i++) {
if (fl.get(i) instanceof Sum || fl.get(i) instanceof SumSubtraction || fl.get(i) instanceof Subtraction) {
if (((FunctionOperator) fl.get(i)).getParameter1() == null && ((FunctionOperator) fl.get(i)).getParameter2() == null) {
return true;
}
}
}
return false;
}
public static boolean areThereEmptySystems(final List<Function> fl) {
for (int i = 0; i < fl.size(); i++) {
if (fl.get(i) instanceof EquationsSystemPart) {
if (((EquationsSystemPart) fl.get(i)).getParameter() == null) {
return true;
}
}
}
return false;
}
public static boolean areThereOtherSettedUpFunctions(final List<Function> fl) {
for (int i = 0; i < fl.size(); i++) {
if (!(fl.get(i) instanceof Number || fl.get(i) instanceof Variable || fl.get(i) instanceof Sum || fl.get(i) instanceof SumSubtraction || fl.get(i) instanceof Expression || fl.get(i) instanceof FunctionSingle || fl.get(i) instanceof Multiplication || fl.get(i) instanceof Division)) {
if (fl.get(i) instanceof FunctionSingle) {
if (((FunctionSingle) fl.get(i)).getParameter() == null) {
return true;
}
} else if (fl.get(i) instanceof FunctionOperator) {
if (((FunctionOperator) fl.get(i)).getParameter1() == null || ((FunctionOperator) fl.get(i)).getParameter2() == null) {
return true;
}
} else {
return true;
}
}
}
return false;
}
public static Rational getRational(final BigDecimal str) {
try {
return Utils.getRational(str.toString());
} catch (final Error e) {
//E' IMPOSSIBILE CHE VENGA THROWATO UN ERRORE
return new Rational("0");
}
}
public static Rational getRational(String str) throws Error {
try {
return new Rational(str);
} catch (final NumberFormatException ex) {
if (new BigDecimal(str).compareTo(new BigDecimal(8000.0)) < 0 && new BigDecimal(str).compareTo(new BigDecimal(-8000.0)) > 0) {
if (str.equals("-")) {
str = "-1";
}
final long bits = Double.doubleToLongBits(Double.parseDouble(str));
final long sign = bits >>> 63;
final long exponent = (bits >>> 52 ^ sign << 11) - 1023;
final long fraction = bits << 12; // bits are "reversed" but that's
// not a problem
long a = 1L;
long b = 1L;
for (int i = 63; i >= 12; i--) {
a = a * 2 + (fraction >>> i & 1);
b *= 2;
}
if (exponent > 0) {
a *= 1 << exponent;
} else {
b *= 1 << -exponent;
}
if (sign == 1) {
a *= -1;
}
if (b == 0) {
a = 0;
b = 1;
}
return new Rational(new BigInteger(a + ""), new BigInteger(b + ""));
} else {
final BigDecimal original = new BigDecimal(str);
final BigInteger numerator = original.unscaledValue();
final BigInteger denominator = BigDecimalMath.pow(BigDecimal.TEN, new BigDecimal(original.scale())).toBigIntegerExact();
return new Rational(numerator, denominator);
}
}
}
public static BigDecimal rationalToIrrationalString(final Rational r) {
return BigDecimalMath.divideRound(new BigDecimal(r.numer()).setScale(Utils.scale, Utils.scaleMode), new BigDecimal(r.denom()).setScale(Utils.scale, Utils.scaleMode));
}
public static boolean equalsVariables(final List<Variable> variables, final List<Variable> variables2) {
if (variables.size() != variables2.size()) {
return false;
} else {
for (final Variable v : variables) {
if (!variables2.contains(v)) {
return false;
}
}
return true;
}
}
@Deprecated
public static void writeSquareRoot(final Function var, final int x, final int y, final boolean small) {
// var.setSmall(small);
// final int w1 = var.getWidth();
// final int h1 = var.getHeight();
// final int wsegno = 5;
// final int hsegno = h1 + 2;
//
// var.draw(x + wsegno, y + (hsegno - h1), null, null);
//
// Engine.INSTANCE.getHardwareDevice().getDisplayManager().renderer.glDrawLine(x + 1, y + hsegno - 3, x + 1, y + hsegno - 3);
// Engine.INSTANCE.getHardwareDevice().getDisplayManager().renderer.glDrawLine(x + 2, y + hsegno - 2, x + 2, y + hsegno - 2);
// Engine.INSTANCE.getHardwareDevice().getDisplayManager().renderer.glDrawLine(x + 3, y + hsegno - 1, x + 3, y + hsegno - 1);
// Engine.INSTANCE.getHardwareDevice().getDisplayManager().renderer.glDrawLine(x + 3, y + (hsegno - 1) / 2 + 1, x + 3, y + hsegno - 1);
// Engine.INSTANCE.getHardwareDevice().getDisplayManager().renderer.glDrawLine(x + 4, y, x + 4, y + (hsegno - 1) / 2);
// Engine.INSTANCE.getHardwareDevice().getDisplayManager().renderer.glDrawLine(x + 4, y, x + 4 + 1 + w1 + 1, y);
}
public static final int getFontHeight() {
return Utils.getFontHeight(false);
}
public static final BinaryFont getFont(final boolean small) {
return Utils.getFont(small, StaticVars.zoomed);
}
public static final BinaryFont getFont(final boolean small, final boolean zoomed) {
return WarpPI.INSTANCE.getHardwareDevice().getDisplayManager().fonts[Utils.getFontIndex(small, zoomed)];
}
public static final int getFontIndex(final boolean small, final boolean zoomed) {
if (small) {
if (zoomed) {
return 3;
} else {
return 1;
}
} else if (zoomed) {
return 2;
} else {
return 0;
}
}
public static final int getFontHeight(final boolean small) {
return Utils.getFontHeight(small, StaticVars.zoomed);
}
public static final int getFontHeight(final boolean small, final boolean zoomed) {
if (small) {
if (zoomed) {
return WarpPI.INSTANCE.getHardwareDevice().getDisplayManager().glyphsHeight[3];
} else {
return WarpPI.INSTANCE.getHardwareDevice().getDisplayManager().glyphsHeight[1];
}
} else if (zoomed) {
return WarpPI.INSTANCE.getHardwareDevice().getDisplayManager().glyphsHeight[2];
} else {
return WarpPI.INSTANCE.getHardwareDevice().getDisplayManager().glyphsHeight[0];
}
}
public static byte[] convertStreamToByteArray(final InputStream stream, final long size) throws IOException {
// check to ensure that file size is not larger than Integer.MAX_VALUE.
if (size > Integer.MAX_VALUE) {
return new byte[0];
}
final byte[] buffer = new byte[(int) size];
final ByteArrayOutputStream os = new ByteArrayOutputStream();
int line = 0;
// read bytes from stream, and store them in buffer
while ((line = stream.read(buffer)) != -1) {
// Writes bytes from byte array (buffer) into output stream.
os.write(buffer, 0, line);
}
stream.close();
os.flush();
os.close();
return os.toByteArray();
}
public static int[] realBytes(final byte[] bytes) {
final int len = bytes.length;
final int[] realbytes = new int[len];
for (int i = 0; i < len; i++) {
realbytes[i] = bytes[i] & 0xFF;
}
return realbytes;
}
public static Function[][] joinFunctionsResults(final List<Function> l1, final List<Function> l2) {
final int size1 = l1.size();
final int size2 = l2.size();
int cur1 = 0;
int cur2 = 0;
final int total = size1 * size2;
final Function[][] results = new Function[total][2];
for (int i = 0; i < total; i++) {
results[i] = new Function[] { l1.get(cur1), l2.get(cur2) };
if (i % size2 == 0) {
cur1 += 1;
}
if (i % size1 == 0) {
cur2 += 1;
}
if (cur1 >= size1) {
cur1 = 0;
}
if (cur2 >= size2) {
cur2 = 0;
}
}
return results;
}
public static Function[][] joinFunctionsResults(final ObjectArrayList<ObjectArrayList<Function>> ln) {
final int[] sizes = new int[ln.size()];
for (int i = 0; i < ln.size(); i++) {
sizes[i] = ln.get(i).size();
}
final int[] curs = new int[sizes.length];
int total = 0;
for (int i = 0; i < ln.size(); i++) {
if (i == 0) {
total = sizes[i];
} else {
total *= sizes[i];
}
}
final Function[][] results = new Function[total][sizes.length];
for (int i = 0; i < total; i++) {
results[i] = new Function[sizes.length];
for (int j = 0; j < sizes.length; j++) {
results[i][j] = ln.get(j).get(curs[j]);
}
for (int k = 0; k < sizes.length; k++) {
if (i % sizes[k] == 0) {
for (int l = 0; l < sizes.length; l++) {
if (l != k) {
curs[l] += 1;
}
}
}
}
for (int k = 0; k < sizes.length; k++) {
if (curs[k] >= sizes[k]) {
curs[k] = 0;
}
}
}
return results;
}
public static boolean isNegative(final Function b) {
if (b instanceof Negative) {
return true;
} else if (b instanceof Number && ((Number) b).getTerm().compareTo(BigDecimal.ZERO) < 0) {
return true;
}
return false;
}
public static CharSequence multipleChars(final String string, final int i) {
String result = "";
for (int j = 0; j < i; j++) {
result += string;
}
return result;
}
public static boolean isIntegerValue(final BigDecimal bd) {
return bd.signum() == 0 || bd.scale() <= 0 || bd.stripTrailingZeros().scale() <= 0;
}
@SafeVarargs
public static <T> String arrayToString(final T... data) {
String sdata = "";
for (final T o : data) {
sdata += "," + o;
}
return sdata.substring(1);
}
public static String arrayToString(final boolean... data) {
String sdata = "";
for (final boolean o : data) {
sdata += o ? 1 : 0;
}
return sdata;
}
public static boolean isWindows() {
return WarpPI.getPlatform().getOsName().indexOf("win") >= 0;
}
public static <T> ObjectArrayList<T> newArrayList(final T o) {
final ObjectArrayList<T> t = new ObjectArrayList<>();
t.add(o);
return t;
}
public static InputStream getResourceStreamSafe(final String string) throws IOException, URISyntaxException {
try {
return WarpPI.getPlatform().getPlatformStorage().getResourceStream(string);
} catch (final Exception ex) {
return null;
}
}
public static File getJarDirectory() {
return new File("").getAbsoluteFile();
}
public static <T, U> U getOrDefault(final Map<T, U> enginesList, final T key, final U object) {
if (enginesList.containsKey(key)) {
return enginesList.get(key);
} else {
return object;
}
}
}