com.anrisoftware.globalpom.math.format.degree.DegreeSexagesimalFormat 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.degree;
import static com.anrisoftware.globalpom.math.format.degree.Direction.N;
import static com.anrisoftware.globalpom.math.measurement.RoundToSignificantFigures.roundToDecimal;
import static com.google.inject.Guice.createInjector;
import static java.lang.Double.parseDouble;
import static java.util.regex.Pattern.compile;
import static javax.measure.unit.NonSI.DEGREE_ANGLE;
import static org.apache.commons.lang3.StringUtils.isEmpty;
import static org.apache.commons.math3.util.FastMath.abs;
import java.text.DecimalFormat;
import java.text.FieldPosition;
import java.text.Format;
import java.text.NumberFormat;
import java.text.ParseException;
import java.text.ParsePosition;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import javax.measure.quantity.Angle;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.math3.util.FastMath;
import org.jscience.physics.amount.Amount;
import com.google.inject.assistedinject.Assisted;
import com.google.inject.assistedinject.AssistedInject;
import jakarta.inject.Inject;
/**
* Parses angle degree sexagesimal.
*
* @author Erwin Mueller, [email protected]
* @since 1.9
*/
@SuppressWarnings("serial")
public class DegreeSexagesimalFormat extends Format {
private static final String SEC_SUB = "\"";
private static final String MIN_SUB = "'";
private static final String DEGREE_SUB = "°";
/**
* @see #create()
* @return {@link DegreeSexagesimalFormat}
*/
public static DegreeSexagesimalFormat createDegreeSexagesimalFormat() {
return create();
}
/**
* @see #create(int)
* @return {@link DegreeSexagesimalFormat}
*/
public static DegreeSexagesimalFormat create() {
return InjectorInstance.factory.create(4);
}
/**
* Create the sexagesimal degree format.
*
* @param decimal the decimal.
* @return the {@link DegreeSexagesimalFormat}.
*/
public static DegreeSexagesimalFormat create(int decimal) {
return InjectorInstance.factory.create(decimal);
}
/**
* Create the sexagesimal degree format.
*
* @param decimal the decimal.
* @param format the {@link NumberFormat}.
* @return the {@link DegreeSexagesimalFormat}.
*/
public static DegreeSexagesimalFormat create(int decimal, NumberFormat format) {
return InjectorInstance.factory.create(decimal, format);
}
private static class InjectorInstance {
static final DegreeSexagesimalFormatFactory factory = createInjector(new DegreeFormatModule())
.getInstance(DegreeSexagesimalFormatFactory.class);
}
/**
* Tests whether the input string is a valid sexagesimal degree format.
*
* @param string the {@link String} to test.
*
* @return {@code true} if the input string is a valid sexagesimal degree
* format.
*/
public static boolean isValidFormat(String string) {
return PATTERN.matcher(string).matches();
}
private static final Pattern PATTERN = compile(
String.format("^((\\d+)%s)((\\d+)%s)?((\\d+(\\.\\d*)?)%s)?(\\s[NSEW])?$", DEGREE_SUB, MIN_SUB, SEC_SUB));
private static final double MIN = 1d / 60d;
private static final double SEC = 1d / 3600d;
@Inject
private DegreeSexagesimalFormatLogger log;
private final int decimal;
private final double epsilon;
private final NumberFormat format;
/**
* @see DegreeSexagesimalFormatFactory#create()
*/
@AssistedInject
DegreeSexagesimalFormat() {
this(3);
}
/**
* @see DegreeSexagesimalFormatFactory#create(int)
*/
@AssistedInject
DegreeSexagesimalFormat(@Assisted int decimal) {
this(decimal, DecimalFormat.getNumberInstance());
}
/**
* @see DegreeSexagesimalFormatFactory#create(int, NumberFormat)
*/
@AssistedInject
DegreeSexagesimalFormat(@Assisted int decimal, @Assisted NumberFormat format) {
this.decimal = decimal;
this.epsilon = FastMath.pow(10, -(decimal + 1));
this.format = format;
}
/**
* Formats the specified degree sexagesimal.
*
* @param obj the {@link Amount}.
*/
@SuppressWarnings("unchecked")
@Override
public StringBuffer format(Object obj, StringBuffer buff, FieldPosition pos) {
if (obj instanceof Amount) {
formatFormat(buff, ((Amount) obj).doubleValue(DEGREE_ANGLE));
}
if (obj instanceof Number) {
formatFormat(buff, ((Number) obj).doubleValue());
}
return buff;
}
private void formatFormat(StringBuffer buff, double value) {
double degree = (long) value;
double dmin = abs((value - degree) / MIN);
double min = roundToDecimal((long) dmin, decimal);
double sec = roundToDecimal((dmin - min) / MIN, decimal);
buff.append(format.format(degree));
buff.append(DEGREE_SUB);
if (min > epsilon) {
buff.append(format.format(min));
buff.append(MIN_SUB);
}
if (sec > epsilon) {
buff.append(format.format(sec));
buff.append(SEC_SUB);
}
}
/**
* Parses the specified string to degree sexagesimal.
* Format
*
* - {@code "D°M'S.s" [NSEW]"}
*
*
* @return the parsed {@link Amount}.
*/
@Override
public Object parseObject(String source, ParsePosition pos) {
try {
return parse(source, pos);
} catch (ParseException e) {
pos.setErrorIndex(pos.getIndex() + e.getErrorOffset());
return null;
}
}
/**
* @see #parse(String, ParsePosition)
*
* @param source the {@link String} source.
*
* @return the {@link Amount}
*
* @throws ParseException if the string is not in the correct format.
*/
public Amount parse(String source) throws ParseException {
ParsePosition pos = new ParsePosition(0);
Amount result = parse(source, pos);
if (pos.getIndex() == 0) {
throw log.errorParse(source, pos);
}
return result;
}
/**
* @see #parseObject(String)
*
* @param source the {@link String} source.
*
* @param pos the index {@link ParsePosition} position from where to start
* parsing.
*
* @return the {@link Amount}
*
* @throws ParseException if the string is not in the correct format.
*/
public Amount parse(String source, ParsePosition pos) throws ParseException {
try {
source = source.substring(pos.getIndex());
Amount result = decodeAngle(source, pos);
pos.setErrorIndex(-1);
pos.setIndex(pos.getIndex() + source.length());
return result;
} catch (NumberFormatException e) {
log.errorParseNumber(e, source);
pos.setIndex(0);
pos.setErrorIndex(0);
return null;
}
}
private Amount decodeAngle(String source, ParsePosition pos) throws ParseException {
Matcher matcher = PATTERN.matcher(source);
log.checkMatches(matcher.find(), source, pos);
double degree = parseDoubleSave(matcher.group(2));
double min = parseDoubleSave(matcher.group(4)) * MIN;
double sec = parseDoubleSave(matcher.group(6)) * SEC;
Direction direction = parseDirectionSave(matcher.group(8));
double value = (degree + min + sec) * direction.getDirection();
return Amount.valueOf(roundToDecimal(value, decimal), 0, DEGREE_ANGLE);
}
private Direction parseDirectionSave(String string) {
string = StringUtils.trim(string);
return isEmpty(string) ? N : Direction.valueOf(string.toUpperCase());
}
private double parseDoubleSave(String string) {
return isEmpty(string) ? 0 : parseDouble(string);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy