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

ezvcard.util.GeoUri Maven / Gradle / Ivy

Go to download

A library that reads and writes vCards, supporting all versions of the vCard standard (2.1, 3.0, and 4.0) as well as xCard (XML-encoded vCards), hCard (HTML-encoded vCards), and jCard (JSON-encoded vCards).

There is a newer version: 0.9.6
Show newest version
package ezvcard.util;

import java.net.URI;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import ezvcard.Messages;

/*
 Copyright (c) 2012-2023, Michael Angstadt
 All rights reserved.

 Redistribution and use in source and binary forms, with or without
 modification, are permitted provided that the following conditions are met: 

 1. Redistributions of source code must retain the above copyright notice, this
 list of conditions and the following disclaimer. 
 2. Redistributions in binary form must reproduce the above copyright notice,
 this list of conditions and the following disclaimer in the documentation
 and/or other materials provided with the distribution. 

 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
 ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
 WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
 ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.

 The views and conclusions contained in the software and documentation are those
 of the authors and should not be interpreted as representing official policies, 
 either expressed or implied, of the FreeBSD Project.
 */

/**
 * 

* Represents a URI for encoding a geographical position. *

*

* Example geo URI: {@code geo:40.714623,-74.006605} *

*

* This class is immutable. Use the {@link Builder} object to construct a new * instance, or the {@link #parse} method to parse a geo URI string. *

*

* Examples: *

* *
 * GeoUri uri = new GeoUri.Builder(40.714623, -74.006605).coordC(1.1).build();
 * GeoUri uri = GeoUri.parse("geo:40.714623,-74.006605,1.1");
 * GeoUri copy = new GeoUri.Builder(original).coordC(2.1).build();
 * 
* @author Michael Angstadt * @see RFC 5870 */ public final class GeoUri { /** * The coordinate reference system used by GPS (the default). */ public static final String CRS_WGS84 = "wgs84"; /** * The characters which are allowed to exist un-encoded inside of a * parameter value. */ private static final boolean validParameterValueCharacters[] = new boolean[128]; static { for (int i = '0'; i <= '9'; i++) { validParameterValueCharacters[i] = true; } for (int i = 'A'; i <= 'Z'; i++) { validParameterValueCharacters[i] = true; } for (int i = 'a'; i <= 'z'; i++) { validParameterValueCharacters[i] = true; } String s = "!$&'()*+-.:[]_~"; for (int i = 0; i < s.length(); i++) { char c = s.charAt(i); validParameterValueCharacters[c] = true; } } /** * Finds hex values in a parameter value. */ private static final Pattern hexPattern = Pattern.compile("(?i)%([0-9a-f]{2})"); private static final String PARAM_CRS = "crs"; private static final String PARAM_UNCERTAINTY = "u"; private final Double coordA; private final Double coordB; private final Double coordC; private final String crs; private final Double uncertainty; private final Map parameters; private GeoUri(Builder builder) { this.coordA = (builder.coordA == null) ? Double.valueOf(0.0) : builder.coordA; this.coordB = (builder.coordB == null) ? Double.valueOf(0.0) : builder.coordB; this.coordC = builder.coordC; this.crs = builder.crs; this.uncertainty = builder.uncertainty; this.parameters = Collections.unmodifiableMap(builder.parameters); } /** * Parses a geo URI string. * @param uri the URI string (e.g. "geo:40.714623,-74.006605") * @return the parsed geo URI * @throws IllegalArgumentException if the string is not a valid geo URI */ public static GeoUri parse(String uri) { //URI format: geo:LAT,LONG;prop1=value1;prop2=value2 String scheme = "geo:"; if (uri.length() < scheme.length() || !uri.substring(0, scheme.length()).equalsIgnoreCase(scheme)) { //not a geo URI throw Messages.INSTANCE.getIllegalArgumentException(18, scheme); } Builder builder = new Builder(null, null); ClearableStringBuilder buffer = new ClearableStringBuilder(); String paramName = null; boolean coordinatesDone = false; for (int i = scheme.length(); i < uri.length(); i++) { char c = uri.charAt(i); if (c == ',' && !coordinatesDone) { handleEndOfCoordinate(buffer, builder); continue; } if (c == ';') { if (coordinatesDone) { handleEndOfParameter(buffer, paramName, builder); paramName = null; } else { handleEndOfCoordinate(buffer, builder); if (builder.coordB == null) { throw Messages.INSTANCE.getIllegalArgumentException(21); } coordinatesDone = true; } continue; } if (c == '=' && coordinatesDone && paramName == null) { paramName = buffer.getAndClear(); continue; } buffer.append(c); } if (coordinatesDone) { handleEndOfParameter(buffer, paramName, builder); } else { handleEndOfCoordinate(buffer, builder); if (builder.coordB == null) { throw Messages.INSTANCE.getIllegalArgumentException(21); } } return builder.build(); } private static void handleEndOfCoordinate(ClearableStringBuilder buffer, Builder builder) { String s = buffer.getAndClear(); if (builder.coordA == null) { try { builder.coordA = Double.parseDouble(s); } catch (NumberFormatException e) { throw new IllegalArgumentException(Messages.INSTANCE.getExceptionMessage(22, "A"), e); } return; } if (builder.coordB == null) { try { builder.coordB = Double.parseDouble(s); } catch (NumberFormatException e) { throw new IllegalArgumentException(Messages.INSTANCE.getExceptionMessage(22, "B"), e); } return; } if (builder.coordC == null) { try { builder.coordC = Double.parseDouble(s); } catch (NumberFormatException e) { throw new IllegalArgumentException(Messages.INSTANCE.getExceptionMessage(22, "C"), e); } return; } } private static void addParameter(String name, String value, Builder builder) { value = decodeParameterValue(value); if (PARAM_CRS.equalsIgnoreCase(name)) { builder.crs = value; return; } if (PARAM_UNCERTAINTY.equalsIgnoreCase(name)) { try { builder.uncertainty = Double.valueOf(value); return; } catch (NumberFormatException e) { //if it can't be parsed, then treat it as an ordinary parameter } } builder.parameters.put(name, value); } private static void handleEndOfParameter(ClearableStringBuilder buffer, String paramName, Builder builder) { String s = buffer.getAndClear(); if (paramName == null) { if (s.length() > 0) { addParameter(s, "", builder); } return; } addParameter(paramName, s, builder); } /** * Gets the first coordinate (latitude). * @return the first coordinate or null if there is none */ public Double getCoordA() { return coordA; } /** * Gets the second coordinate (longitude). * @return the second coordinate or null if there is none */ public Double getCoordB() { return coordB; } /** * Gets the third coordinate (altitude). * @return the third coordinate or null if there is none */ public Double getCoordC() { return coordC; } /** * Gets the coordinate reference system. * @return the coordinate reference system or null if using the default * (WGS-84) */ public String getCrs() { return crs; } /** * Gets the uncertainty (how accurate the coordinates are). * @return the uncertainty (in meters) or null if not set */ public Double getUncertainty() { return uncertainty; } /** * Gets a parameter value. * @param name the parameter name * @return the parameter value or null if not found */ public String getParameter(String name) { return parameters.get(name); } /** * Gets all the parameters. * @return all the parameters */ public Map getParameters() { return parameters; } /** * Creates a {@link URI} object from this geo URI. * @return the {@link URI} object */ public URI toUri() { return URI.create(toString()); } /** * Converts this geo URI to its string representation. * @return the geo URI's string representation */ @Override public String toString() { return toString(6); } /** * Converts this geo URI to its string representation. * @param decimals the number of decimals to display for floating point * values * @return the geo URI's string representation */ public String toString(int decimals) { VCardFloatFormatter formatter = new VCardFloatFormatter(decimals); StringBuilder sb = new StringBuilder("geo:"); sb.append(formatter.format(coordA)); sb.append(','); sb.append(formatter.format(coordB)); if (coordC != null) { sb.append(','); sb.append(coordC); } //if the CRS is WGS-84, then it doesn't have to be displayed if (crs != null && !crs.equalsIgnoreCase(CRS_WGS84)) { writeParameter(PARAM_CRS, crs, sb); } if (uncertainty != null) { writeParameter(PARAM_UNCERTAINTY, formatter.format(uncertainty), sb); } for (Map.Entry entry : parameters.entrySet()) { String name = entry.getKey(); String value = entry.getValue(); writeParameter(name, value, sb); } return sb.toString(); } /** * Writes a parameter to a string. * @param name the parameter name * @param value the parameter value * @param sb the string to write to */ private void writeParameter(String name, String value, StringBuilder sb) { sb.append(';').append(name).append('=').append(encodeParameterValue(value)); } /** * Encodes a string for safe inclusion in a parameter value. * @param value the string to encode * @return the encoded value */ private static String encodeParameterValue(String value) { StringBuilder sb = null; for (int i = 0; i < value.length(); i++) { char c = value.charAt(i); if (c < validParameterValueCharacters.length && validParameterValueCharacters[c]) { if (sb != null) { sb.append(c); } } else { if (sb == null) { sb = new StringBuilder(value.length() * 2); sb.append(value, 0, i); } String hex = Integer.toString(c, 16); sb.append('%').append(hex); } } return (sb == null) ? value : sb.toString(); } /** * Decodes escaped characters in a parameter value. * @param value the parameter value * @return the decoded value */ private static String decodeParameterValue(String value) { Matcher m = hexPattern.matcher(value); StringBuffer sb = null; while (m.find()) { if (sb == null) { sb = new StringBuffer(value.length()); } int hex = Integer.parseInt(m.group(1), 16); m.appendReplacement(sb, Character.toString((char) hex)); } if (sb == null) { return value; } m.appendTail(sb); return sb.toString(); } @Override public int hashCode() { final int prime = 31; int result = 1; result = prime * result + ((coordA == null) ? 0 : coordA.hashCode()); result = prime * result + ((coordB == null) ? 0 : coordB.hashCode()); result = prime * result + ((coordC == null) ? 0 : coordC.hashCode()); result = prime * result + ((crs == null) ? 0 : crs.toLowerCase().hashCode()); result = prime * result + ((parameters == null) ? 0 : StringUtils.toLowerCase(parameters).hashCode()); result = prime * result + ((uncertainty == null) ? 0 : uncertainty.hashCode()); return result; } @Override public boolean equals(Object obj) { if (this == obj) return true; if (obj == null) return false; if (getClass() != obj.getClass()) return false; GeoUri other = (GeoUri) obj; if (coordA == null) { if (other.coordA != null) return false; } else if (!coordA.equals(other.coordA)) return false; if (coordB == null) { if (other.coordB != null) return false; } else if (!coordB.equals(other.coordB)) return false; if (coordC == null) { if (other.coordC != null) return false; } else if (!coordC.equals(other.coordC)) return false; if (crs == null) { if (other.crs != null) return false; } else if (!crs.equalsIgnoreCase(other.crs)) return false; if (uncertainty == null) { if (other.uncertainty != null) return false; } else if (!uncertainty.equals(other.uncertainty)) return false; if (parameters == null) { if (other.parameters != null) return false; } else { if (other.parameters == null) return false; if (parameters.size() != other.parameters.size()) return false; Map parametersLower = StringUtils.toLowerCase(parameters); Map otherParametersLower = StringUtils.toLowerCase(other.parameters); if (!parametersLower.equals(otherParametersLower)) return false; } return true; } /** * Builder class for {@link GeoUri}. * @author Michael Angstadt */ public static class Builder { private Double coordA; private Double coordB; private Double coordC; private String crs; private Double uncertainty; private Map parameters; private CharacterBitSet validParamChars = new CharacterBitSet("a-zA-Z0-9-"); /** * Creates a new {@link GeoUri} builder. * @param coordA the first coordinate (i.e. latitude) * @param coordB the second coordinate (i.e. longitude) */ public Builder(Double coordA, Double coordB) { parameters = new LinkedHashMap<>(0); //set initial size to 0 because parameters are rarely used coordA(coordA); coordB(coordB); } /** * Creates a new {@link GeoUri} builder. * @param original the {@link GeoUri} object to copy from */ public Builder(GeoUri original) { coordA(original.coordA); coordB(original.coordB); this.coordC = original.coordC; this.crs = original.crs; this.uncertainty = original.uncertainty; this.parameters = new LinkedHashMap<>(original.parameters); } /** * Sets the first coordinate (latitude). * @param coordA the first coordinate * @return this */ public Builder coordA(Double coordA) { this.coordA = coordA; return this; } /** * Sets the second coordinate (longitude). * @param coordB the second coordinate * @return this */ public Builder coordB(Double coordB) { this.coordB = coordB; return this; } /** * Sets the third coordinate (altitude). * @param coordC the third coordinate or null to remove * @return this */ public Builder coordC(Double coordC) { this.coordC = coordC; return this; } /** * Sets the coordinate reference system. * @param crs the coordinate reference system (can only contain letters, * numbers, and hyphens) or null to use the default (WGS-84) * @throws IllegalArgumentException if the CRS name contains invalid * characters * @return this */ public Builder crs(String crs) { if (crs != null && !validParamChars.containsOnly(crs)) { throw Messages.INSTANCE.getIllegalArgumentException(24); } this.crs = crs; return this; } /** * Sets the uncertainty (how accurate the coordinates are). * @param uncertainty the uncertainty (in meters) or null to remove * @return this */ public Builder uncertainty(Double uncertainty) { this.uncertainty = uncertainty; return this; } /** * Adds a parameter. * @param name the parameter name (can only contain letters, numbers, * and hyphens) * @param value the parameter value or null to remove the parameter * @throws IllegalArgumentException if the parameter name contains * invalid characters * @return this */ public Builder parameter(String name, String value) { if (!validParamChars.containsOnly(name)) { throw Messages.INSTANCE.getIllegalArgumentException(23); } if (value == null) { parameters.remove(name); } else { parameters.put(name, value); } return this; } /** * Builds the final {@link GeoUri} object. * @return the object */ public GeoUri build() { return new GeoUri(this); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy