All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.anrisoftware.globalpom.math.format.measurement.AbstractValueParserWorker Maven / Gradle / Ivy

/*
 * Copyright 2013-2025 Erwin Müller 
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.anrisoftware.globalpom.math.format.measurement;

/*-
 * #%L
 * Global POM Utilities :: Math
 * %%
 * Copyright (C) 2013 - 2018 Advanced Natural Research Institute
 * %%
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *      http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 * #L%
 */

import static com.anrisoftware.globalpom.math.math.MathUtils.calculateValue;
import static org.apache.commons.lang3.StringUtils.split;

import java.math.BigInteger;
import java.text.DecimalFormat;
import java.text.DecimalFormatSymbols;
import java.text.ParseException;
import java.util.ArrayList;
import java.util.Locale;

import org.apache.commons.lang3.StringUtils;

import com.anrisoftware.globalpom.math.measurement.Value;
import com.anrisoftware.globalpom.math.measurement.ValueFactory;

public abstract class AbstractValueParserWorker implements ValueParserWorker {

    protected final DecimalFormatSymbols symbols;

    protected final Locale locale;

    protected final Integer significant;

    protected final Integer decimal;

    protected final ValueFactory valueFactory;

    protected final ArrayList valueSplit;

    protected final char decimalSeparator;

    protected final String exponentSeparator;

    protected final char minusSign;

    protected final boolean negative;

    protected String valueStr;

    protected String exponentStr;

    protected String uncString;

    protected AbstractValueParserWorker(DecimalFormatSymbols symbols,
            Locale locale, Integer significant, Integer decimal,
            ValueFactory valueFactory, String valueStr, String exponentStr,
            String uncString) {
        this.symbols = symbols;
        this.locale = locale;
        this.significant = significant;
        this.decimal = decimal;
        this.valueFactory = valueFactory;
        this.valueSplit = new ArrayList(3);
        this.decimalSeparator = symbols.getDecimalSeparator();
        this.exponentSeparator = symbols.getExponentSeparator();
        this.minusSign = symbols.getMinusSign();
        this.valueStr = valueStr;
        this.exponentStr = exponentStr;
        this.uncString = uncString;
        this.negative = valueStr.charAt(0) == minusSign;
    }

    protected final void parseExponent(String exstr) {
        if (exstr == null || valueSplit.size() == 3) {
            return;
        }
        String[] vsplit = split(exstr, exponentSeparator);
        if (vsplit.length == 1) {
            valueSplit.add(vsplit[0]);
        }
    }

    protected final void splitValue(String value) {
        String[] vsplit = split(value, exponentSeparator);
        String[] dsplit = split(vsplit[0], decimalSeparator);
        // number
        valueSplit.add(dsplit[0].substring(negative ? 1 : 0));
        // decimal
        if (dsplit.length == 2) {
            valueSplit.add(dsplit[1]);
        } else {
            valueSplit.add(null);
        }
        // exponent
        if (vsplit.length == 2) {
            valueSplit.add(vsplit[1]);
        }
    }

    protected final Value createValue(String mantissa, Double unc, int order,
            int sig, int dec) {
        BigInteger man = new BigInteger(mantissa);
        man = negative ? man.negate() : man;
        if (unc == null) {
            unc = Double.NaN;
            if (uncString != null) {
                unc = parseUncertainty(uncString, calculateValue(man, dec));
            }
        }
        return valueFactory.create(man, order, sig, dec, unc);
    }

    protected final String setupDecimal(String str, Integer dec)
            throws ParseException {
        if (dec == null) {
            return str;
        }
        StringBuilder pattern = new StringBuilder("0");
        if (dec > 0) {
            pattern.append('.');
            for (int i = 0; i < dec; i++) {
                pattern.append('0');
            }
        }
        Number parse = DecimalFormat.getInstance(locale).parse(str);
        double value = parse.doubleValue();
        this.exponentStr = null;
        return new DecimalFormat(pattern.toString(), symbols).format(value);
    }

    protected final double parseUncertainty(String str, double value) {
        char percent = symbols.getPercent();
        str = StringUtils.substringBetween(str, "(", ")");
        if (StringUtils.contains(str, percent)) {
            int idx = str.indexOf(percent);
            str = str.substring(0, idx);
            return Double.valueOf(str) / 100 * value;
        } else {
            return Double.valueOf(str);
        }
    }

    protected final int parseOrder() {
        int order = 0;
        // exponent order
        if (valueSplit.size() > 2) {
            String exp = valueSplit.get(2);
            order = Integer.valueOf(exp);
        }
        // number order
        if (valueSplit.size() > 0) {
            String number = valueSplit.get(0);
            boolean firstNotNull = false;
            for (int i = 0; i < number.length(); i++) {
                char c = number.charAt(i);
                switch (c) {
                case '0':
                    if (firstNotNull) {
                        order++;
                    }
                    break;
                default:
                    firstNotNull = true;
                    order++;
                    break;
                }
            }
        }
        // decimal order
        if (order < 1 && valueSplit.size() == 2 && valueSplit.get(1) != null) {
            String decimal = valueSplit.get(1);
            boolean onlyNull = true;
            int oorder = 0;
            for (int i = 0; i < decimal.length(); i++) {
                char c = decimal.charAt(i);
                switch (c) {
                case '0':
                    oorder--;
                    break;
                default:
                    onlyNull = false;
                    i = decimal.length();
                    break;
                }
            }
            if (!onlyNull) {
                order += oorder;
            }
        }
        return order;
    }

    protected final String parseMantissa() {
        StringBuilder buff = new StringBuilder();
        appendNumber(buff);
        if (valueSplit.size() > 1) {
            appendDecimal(buff.length() == 0, buff);
        }
        if (buff.length() == 0) {
            buff.append('0');
        }
        return buff.toString();
    }

    private StringBuilder appendNumber(StringBuilder buff) {
        String number = valueSplit.get(0);
        if (number.length() == 1 && number.charAt(0) == '0') {
            return buff;
        }
        StringBuilder b = new StringBuilder();
        boolean firstNotNull = valueSplit.size() > 1
                && valueSplit.get(1) != null;
        for (int i = number.length() - 1; i >= 0; i--) {
            char c = number.charAt(i);
            switch (c) {
            case '0':
                if (firstNotNull) {
                    b.append(c);
                }
                break;
            default:
                firstNotNull = true;
                b.append(c);
                break;
            }
        }
        buff.append(b.reverse());
        return buff;
    }

    private StringBuilder appendDecimal(boolean isNull, StringBuilder buff) {
        String decimal = valueSplit.get(1);
        if (decimal == null) {
            return buff;
        }
        StringBuilder b = new StringBuilder();
        for (int i = 0; i < decimal.length(); i++) {
            char c = decimal.charAt(i);
            switch (c) {
            case '0':
                if (!isNull) {
                    b.append(c);
                }
                break;
            default:
                isNull = false;
                b.append(c);
                break;
            }
        }
        buff.append(b);
        return buff;
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy