/* * GeoTools - The Open Source Java GIS Toolkit * http://geotools.org * * (C) 2002-2008, Open Source Geospatial Foundation (OSGeo) * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; * version 2.1 of the License. * * 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. */ package org.geotools.gml2.bindings; import java.util.StringTokenizer; import javax.xml.namespace.QName; import org.geotools.gml2.GML; import org.geotools.xml.AbstractComplexBinding; import org.geotools.xml.ElementInstance; import org.geotools.xml.Node; import org.w3c.dom.Document; import org.w3c.dom.Element; import com.vividsolutions.jts.geom.Coordinate; import com.vividsolutions.jts.geom.CoordinateSequence; import com.vividsolutions.jts.geom.CoordinateSequenceFactory; /** * Binding object for the type http://www.opengis.net/gml:CoordinatesType. * * <p> * <pre> * <code> * <complexType name="CoordinatesType"> * <annotation> * <documentation> Coordinates can be included in a single * string, but there is no facility for validating * string content. The value of the 'cs' attribute * is the separator for coordinate values, and the value of the * 'ts' attribute gives the tuple separator * (a single space by default); the default values may * be changed to reflect local usage. </documentation> * </annotation> * <simpleContent> * <extension base="string"> * <attribute name="decimal" type="string" use="optional" default="."/> * <attribute name="cs" type="string" use="optional" default=","/> * <attribute name="ts" type="string" use="optional" default=" "/> * </extension> * </simpleContent> * </complexType> * * </code> * </pre> * </p> * * @generated * * * @source $URL$ */ public class GMLCoordinatesTypeBinding extends AbstractComplexBinding { CoordinateSequenceFactory csFactory; public GMLCoordinatesTypeBinding(CoordinateSequenceFactory csFactory) { this.csFactory = csFactory; } /** * @generated */ public QName getTarget() { return GML.CoordinatesType; } /** * <!-- begin-user-doc --> * <!-- end-user-doc --> * * @generated modifiable */ public Class getType() { return CoordinateSequence.class; } /** * <!-- begin-user-doc --> * Returns an object of type {@see com.vividsolutions.jts.geom.CoordinateSequence} * TODO: this method should do more validation of the string * <!-- end-user-doc --> * * @generated modifiable */ public Object parse(ElementInstance instance, Node node, Object value) throws Exception { //get the coordinate and tuple seperators String decimal = "."; String cs = ","; String ts = " "; if (node.getAttribute("decimal") != null) { decimal = (String) node.getAttribute("decimal").getValue(); } if (node.getAttribute("cs") != null) { cs = (String) node.getAttribute("cs").getValue(); } if (node.getAttribute("ts") != null) { ts = (String) node.getAttribute("ts").getValue(); } //do the parsing String text = instance.getText(); //eliminate newlines, repeated spaces, etc final String anyBlankSeq = "\\s+"; final String singleSpace = " "; text = text.replaceAll(anyBlankSeq, singleSpace); //first tokenize by tuple seperators StringTokenizer tuples = new StringTokenizer(text, ts); CoordinateSequence seq = null; int i = 0; int ncoords = tuples.countTokens(); //number of coordinates while (tuples.hasMoreTokens()) { String tuple = tuples.nextToken(); //next tokenize by coordinate seperator String[] oords = tuple.split(cs); //next tokenize by decimal String x = null; //next tokenize by decimal String y = null; //next tokenize by decimal String z = null; //must be at least 1D x = ".".equals(decimal) ? oords[0] : oords[0].replaceAll(decimal, "."); //check for 2 and 3 D if (oords.length > 1) { y = ".".equals(decimal) ? oords[1] : oords[1].replaceAll(decimal, "."); } if (oords.length > 2) { z = ".".equals(decimal) ? oords[2] : oords[2].replaceAll(decimal, "."); } if (seq == null) { seq = csFactory.create(ncoords, oords.length); } seq.setOrdinate(i, CoordinateSequence.X, Double.parseDouble(x)); if (y != null) { seq.setOrdinate(i, CoordinateSequence.Y, Double.parseDouble(y)); } if (z != null) { seq.setOrdinate(i, CoordinateSequence.Z, Double.parseDouble(z)); } i++; } return seq; } public Element encode(Object object, Document document, Element value) throws Exception { CoordinateSequence coordinates = (CoordinateSequence) object; StringBuffer buf = new StringBuffer(); for (int i = 0; i < coordinates.size(); i++) { Coordinate c = coordinates.getCoordinate(i); buf.append(c.x); boolean y = (coordinates.getDimension() > 1) && !new Double(c.y).isNaN(); if (y) { buf.append("," + c.y); } boolean z = y && (coordinates.getDimension() > 2) && !new Double(c.z).isNaN(); if (z) { buf.append("," + c.z); } if (i < (coordinates.size() - 1)) { buf.append(" "); } } value.appendChild(document.createTextNode(buf.toString())); return value; } }