Java tutorial
/* Copyright 2010 - 2014 by Brian Uri! This file is part of DDMSence. This library is free software; you can redistribute it and/or modify it under the terms of version 3.0 of the GNU Lesser General Public License as published by the Free Software Foundation. This library is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with DDMSence. If not, see <http://www.gnu.org/licenses/>. You can contact the author at ddmsence@urizone.net. The DDMSence home page is located at http://ddmsence.urizone.net/ */ package buri.ddmsence.ddms.summary.gml; import java.io.Serializable; import java.util.ArrayList; import java.util.Collections; import java.util.List; import nu.xom.Element; import buri.ddmsence.AbstractAttributeGroup; import buri.ddmsence.ddms.IBuilder; import buri.ddmsence.ddms.InvalidDDMSException; import buri.ddmsence.ddms.OutputFormat; import buri.ddmsence.ddms.Resource; import buri.ddmsence.util.DDMSVersion; import buri.ddmsence.util.LazyList; import buri.ddmsence.util.Util; import com.google.gson.JsonObject; /** * Attribute group for the four SRS attributes used on GML and TSPI-based shapes. * <br /><br /> * {@ddms.versions 11111} * * <p> * Because the GML-Profile defines these attributes locally inside of attribute groups, they are not in any namespace. * Some older examples on the DDMS website inaccurately display the attributes with the gml: prefix. * </p> * * {@table.header History} * The GML profile was removed in favour of TSPI in DDMS 5.0. However, these SRS attributes continue to appear on TSPI shapes. * {@table.footer} * {@table.header Nested Elements} * None. * {@table.footer} * {@table.header Attributes} * {@child.info srsName|0..1|11111} * {@child.info srsDimension|0..1|11111} * {@child.info axisLabels|0..*|11111} * {@child.info uomLabels|0..*|11111} * {@table.footer} * {@table.header Validation Rules} * {@ddms.rule If set, srsName must be a valid URI.|Error|11111} * {@ddms.rule If set, srsDimension must be positive.|Error|11111} * {@ddms.rule If srsName is not set, axisLabels must not be set.|Error|11111} * {@ddms.rule If axisLabels is not set, uomLabels must not be set.|Error|11111} * {@ddms.rule If set, each axisLabels value must be a NCName.|Error|11111} * {@ddms.rule If set, each uomLabel value must be a NCName.|Error|11111} * <p>When validating this attribute group, the required/optional nature of the srsName attribute is not checked. * Because that limitation depends on the parent element (for example, gml:Point and gml:Polygon require an srsName, but gml:pos * does not), the parent element should be responsible for checking.</p> * {@table.footer} * * @author Brian Uri! * @since 0.9.b */ public final class SRSAttributes extends AbstractAttributeGroup { private String _srsName = null; private Integer _srsDimension = null; private List<String> _axisLabels = null; private List<String> _uomLabels = null; /** The prefix of the shared attributes */ public static final String NO_PREFIX = ""; /** The namespace of the shared attributes */ public static final String NO_NAMESPACE = ""; private static final String SRS_NAME_NAME = "srsName"; private static final String SRS_DIMENSION_NAME = "srsDimension"; private static final String AXIS_LABELS_NAME = "axisLabels"; private static final String UOM_LABELS_NAME = "uomLabels"; /** * Returns a non-null instance of SRS attributes. If the instance passed in is not null, it will be returned. * * @param srsAttributes the attributes to return by default * @return a non-null attributes instance * @throws InvalidDDMSException if there are problems creating the empty attributes instance */ public static SRSAttributes getNonNullInstance(SRSAttributes srsAttributes) throws InvalidDDMSException { return (srsAttributes == null ? new SRSAttributes(null, null, null, null) : srsAttributes); } /** * Base constructor * * @param element the XOM element which is decorated with these attributes. */ public SRSAttributes(Element element) throws InvalidDDMSException { setNamespace(NO_NAMESPACE); _srsName = element.getAttributeValue(SRS_NAME_NAME, NO_NAMESPACE); String srsDimension = element.getAttributeValue(SRS_DIMENSION_NAME, NO_NAMESPACE); if (!Util.isEmpty(srsDimension)) { _srsDimension = Integer.valueOf(srsDimension); } String axisLabels = element.getAttributeValue(AXIS_LABELS_NAME, NO_NAMESPACE); _axisLabels = new ArrayList<String>(); if (!Util.isEmpty(axisLabels)) { _axisLabels.addAll(Util.getXsListAsList(axisLabels)); } String uomLabels = element.getAttributeValue(UOM_LABELS_NAME, NO_NAMESPACE); _uomLabels = new ArrayList<String>(); if (!Util.isEmpty(uomLabels)) { _uomLabels.addAll(Util.getXsListAsList(uomLabels)); } validate(DDMSVersion.getVersionForNamespace(element.getNamespaceURI())); } /** * Constructor which builds from raw data. * * @param srsName the srsName * @param srsDimension the srsDimension * @param axisLabels the axis labels * @param uomLabels the labels for UOM * @throws InvalidDDMSException if any required information is missing or malformed */ public SRSAttributes(String srsName, Integer srsDimension, List<String> axisLabels, List<String> uomLabels) throws InvalidDDMSException { setNamespace(NO_NAMESPACE); if (axisLabels == null) axisLabels = Collections.emptyList(); if (uomLabels == null) uomLabels = Collections.emptyList(); _srsName = srsName; _srsDimension = srsDimension; _axisLabels = axisLabels; _uomLabels = uomLabels; validate(DDMSVersion.getCurrentVersion()); } /** * Convenience method to add these attributes onto an existing XOM Element * * @param element the element to decorate * @throws InvalidDDMSException if the DDMS version of the element is different */ protected void addTo(Element element) throws InvalidDDMSException { // No compatible version check, since these attributes are in no namespace. Util.addAttribute(element, NO_PREFIX, SRS_NAME_NAME, NO_NAMESPACE, getSrsName()); if (getSrsDimension() != null) Util.addAttribute(element, NO_PREFIX, SRS_DIMENSION_NAME, NO_NAMESPACE, String.valueOf(getSrsDimension())); Util.addAttribute(element, NO_PREFIX, AXIS_LABELS_NAME, NO_NAMESPACE, getAxisLabelsAsXsList()); Util.addAttribute(element, NO_PREFIX, UOM_LABELS_NAME, NO_NAMESPACE, getUomLabelsAsXsList()); } /** * @param version the DDMS version to validate against. This cannot be stored in the attribute group because some * DDMSVersions have the same attribute XML namespace (e.g. XLink, ISM, NTK, GML after DDMS 2.0). * @throws InvalidDDMSException if any required information is missing or malformed */ protected void validate(DDMSVersion version) throws InvalidDDMSException { if (!Util.isEmpty(getSrsName())) Util.requireDDMSValidURI(getSrsName()); if (getSrsDimension() != null && getSrsDimension().intValue() < 0) throw new InvalidDDMSException("The srsDimension must be a positive integer."); if (Util.isEmpty(getSrsName()) && !getAxisLabels().isEmpty()) throw new InvalidDDMSException("The axisLabels attribute must only be used in tandem with an srsName."); if (getAxisLabels().isEmpty() && !getUomLabels().isEmpty()) throw new InvalidDDMSException("The uomLabels attribute must only be used in tandem with axisLabels."); Util.requireValidNCNames(getAxisLabels()); Util.requireValidNCNames(getUomLabels()); super.validate(version); } /** * @see AbstractAttributeGroup#isEmpty() */ public boolean isEmpty() { return (Util.isEmpty(getSrsName()) && getSrsDimension() == null && getAxisLabels().isEmpty() && getUomLabels().isEmpty()); } /** * @see AbstractAttributeGroup#getJSONObject() */ public JsonObject getJSONObject() { JsonObject object = new JsonObject(); addJson(object, "srsName", getSrsName()); addJson(object, "srsDimension", getSrsDimension()); addJson(object, "axisLabels", getAxisLabels()); addJson(object, "uomLabels", getUomLabels()); return (object); } /** * @see AbstractAttributeGroup#getName() */ public String getName() { return ("srsAttributes"); } /** * @see AbstractAttributeGroup#getHTMLTextOutput(OutputFormat, String) */ public String getHTMLTextOutput(OutputFormat format, String prefix) { Util.requireHTMLText(format); String localPrefix = Util.getNonNullString(prefix); StringBuffer text = new StringBuffer(); text.append(Resource.buildHTMLTextOutput(format, localPrefix + "srsName", getSrsName())); if (getSrsDimension() != null) { text.append(Resource.buildHTMLTextOutput(format, localPrefix + "srsDimension", String.valueOf(getSrsDimension()))); } text.append(Resource.buildHTMLTextOutput(format, localPrefix + "axisLabels", getAxisLabelsAsXsList())); text.append(Resource.buildHTMLTextOutput(format, localPrefix + "uomLabels", getUomLabelsAsXsList())); return (text.toString()); } /** * @see Object#equals(Object) */ public boolean equals(Object obj) { if (!(obj instanceof SRSAttributes)) return (false); SRSAttributes test = (SRSAttributes) obj; return (getSrsName().equals(test.getSrsName()) && Util.nullEquals(getSrsDimension(), test.getSrsDimension()) && Util.listEquals(getAxisLabels(), test.getAxisLabels()) && Util.listEquals(getUomLabels(), test.getUomLabels())); } /** * @see Object#hashCode() */ public int hashCode() { int result = 0; result = 7 * result + getSrsName().hashCode(); if (getSrsDimension() != null) result = 7 * result + getSrsDimension().hashCode(); result = 7 * result + getAxisLabels().hashCode(); result = 7 * result + getUomLabels().hashCode(); return (result); } /** * Accessor for the srsName. */ public String getSrsName() { return (Util.getNonNullString(_srsName)); } /** * Accessor for the srsDimension. May return null if not set. */ public Integer getSrsDimension() { return (_srsDimension); } /** * Accessor for the axisLabels. Will return an empty list if not set. */ public List<String> getAxisLabels() { return (Collections.unmodifiableList(_axisLabels)); } /** * Accessor for the String representation of the axisLabels */ public String getAxisLabelsAsXsList() { return (Util.getXsList(getAxisLabels())); } /** * Accessor for the uomLabels. Will return an empty list if not set. */ public List<String> getUomLabels() { return (Collections.unmodifiableList(_uomLabels)); } /** * Accessor for the String representation of the uomLabels */ public String getUomLabelsAsXsList() { return (Util.getXsList(getUomLabels())); } /** * Builder for these attributes. * * <p>This class does not implement the IBuilder interface, because the behavior of commit() is at odds with the * standard commit() method. As an attribute group, an empty attribute group will always be returned instead of * null. * * @see IBuilder * @author Brian Uri! * @since 1.8.0 */ public static class Builder implements Serializable { private static final long serialVersionUID = 6071979027185230870L; private String _srsName; private Integer _srsDimension; private List<String> _axisLabels; private List<String> _uomLabels; /** * Empty constructor */ public Builder() { } /** * Constructor which starts from an existing component. */ public Builder(SRSAttributes attributes) { setSrsName(attributes.getSrsName()); setSrsDimension(attributes.getSrsDimension()); setAxisLabels(attributes.getAxisLabels()); setUomLabels(attributes.getUomLabels()); } /** * Finalizes the data gathered for this builder instance. Will always return an empty instance instead of * a null one. * * @throws InvalidDDMSException if any required information is missing or malformed */ public SRSAttributes commit() throws InvalidDDMSException { return (new SRSAttributes(getSrsName(), getSrsDimension(), getAxisLabels(), getUomLabels())); } /** * Checks if any values have been provided for this Builder. * * @return true if every field is empty */ public boolean isEmpty() { return (Util.isEmpty(getSrsName()) && getSrsDimension() == null && Util.containsOnlyEmptyValues(getAxisLabels()) && Util.containsOnlyEmptyValues(getUomLabels())); } /** * Builder accessor for the srsName */ public String getSrsName() { return _srsName; } /** * Builder accessor for the srsName */ public void setSrsName(String srsName) { _srsName = srsName; } /** * Builder accessor for the srsDimension */ public Integer getSrsDimension() { return _srsDimension; } /** * Builder accessor for the srsDimension */ public void setSrsDimension(Integer srsDimension) { _srsDimension = srsDimension; } /** * Builder accessor for the axisLabels */ public List<String> getAxisLabels() { if (_axisLabels == null) _axisLabels = new LazyList(String.class); return _axisLabels; } /** * Builder accessor for the axisLabels */ public void setAxisLabels(List<String> axisLabels) { _axisLabels = new LazyList(axisLabels, String.class); } /** * Builder accessor for the uomLabels */ public List<String> getUomLabels() { if (_uomLabels == null) _uomLabels = new LazyList(String.class); return _uomLabels; } /** * Builder accessor for the uomLabels */ public void setUomLabels(List<String> uomLabels) { _uomLabels = new LazyList(uomLabels, String.class); } } }