1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215
|
package tim.prune.load.xml;
import java.util.ArrayList;
import org.xml.sax.Attributes;
import org.xml.sax.SAXException;
import tim.prune.data.Field;
import tim.prune.load.TrackNameList;
/**
* Class for handling specifics of parsing Gpx files
*/
public class GpxHandler extends XmlHandler
{
private boolean _insidePoint = false;
private boolean _insideWaypoint = false;
private boolean _startSegment = true;
private boolean _isTrackPoint = false;
private int _trackNum = -1;
private GpxTag _name = new GpxTag(), _trackName = new GpxTag();
private String _latitude = null, _longitude = null;
private GpxTag _elevation = new GpxTag(), _time = new GpxTag();
private GpxTag _type = new GpxTag(), _description = new GpxTag();
private GpxTag _link = new GpxTag();
private GpxTag _currentTag = null;
private ArrayList<String[]> _pointList = new ArrayList<String[]>();
private ArrayList<String> _linkList = new ArrayList<String>();
private TrackNameList _trackNameList = new TrackNameList();
/**
* Receive the start of a tag
* @see org.xml.sax.ContentHandler#startElement(java.lang.String, java.lang.String, java.lang.String, org.xml.sax.Attributes)
*/
public void startElement(String uri, String localName, String qName,
Attributes attributes) throws SAXException
{
// Read the parameters for waypoints and track points
String tag = qName.toLowerCase();
if (tag.equals("wpt") || tag.equals("trkpt") || tag.equals("rtept"))
{
_insidePoint = true;
_insideWaypoint = tag.equals("wpt");
_isTrackPoint = tag.equals("trkpt");
final int numAttributes = attributes.getLength();
for (int i=0; i<numAttributes; i++)
{
String att = attributes.getQName(i).toLowerCase();
if (att.equals("lat")) {_latitude = attributes.getValue(i);}
else if (att.equals("lon")) {_longitude = attributes.getValue(i);}
}
_elevation.setValue(null);
_name.setValue(null);
_time.setValue(null);
_type.setValue(null);
_link.setValue(null);
_description.setValue(null);
}
else if (tag.equals("ele")) {
_currentTag = _elevation;
}
else if (tag.equals("name")) {
_currentTag = (_insidePoint?_name:_trackName);
}
else if (tag.equals("time")) {
_currentTag = _time;
}
else if (tag.equals("type")) {
_currentTag = _type;
}
else if (tag.equals("description")) {
_currentTag = _description;
}
else if (tag.equals("link")) {
_link.setValue(attributes.getValue("href"));
}
else if (tag.equals("trkseg")) {
_startSegment = true;
}
else if (tag.equals("trk"))
{
_trackNum++;
_trackName.setValue(null);
}
super.startElement(uri, localName, qName, attributes);
}
/**
* Process end tag
* @see org.xml.sax.ContentHandler#endElement(java.lang.String, java.lang.String, java.lang.String)
*/
public void endElement(String uri, String localName, String qName)
throws SAXException
{
String tag = qName.toLowerCase();
if (tag.equals("wpt") || tag.equals("trkpt") || tag.equals("rtept"))
{
processPoint();
_insidePoint = false;
}
else {
_currentTag = null;
}
super.endElement(uri, localName, qName);
}
/**
* Process character text (inside tags or between them)
* @see org.xml.sax.ContentHandler#characters(char[], int, int)
*/
public void characters(char[] ch, int start, int length)
throws SAXException
{
String value = new String(ch, start, length);
if (_currentTag != null) {
_currentTag.setValue(checkCharacters(_currentTag.getValue(), value));
}
super.characters(ch, start, length);
}
/**
* Check to concatenate partially-received values, if necessary
* @param inVariable variable containing characters received until now
* @param inValue new value received
* @return concatenation
*/
private static String checkCharacters(String inVariable, String inValue)
{
if (inVariable == null) {return inValue;}
return inVariable + inValue;
}
/**
* Process a point, either a waypoint or track point
*/
private void processPoint()
{
// Put the values into a String array matching the order in getFieldArray()
String[] values = new String[8];
values[0] = _latitude;
values[1] = _longitude;
values[2] = _elevation.getValue();
if (_insideWaypoint) {values[3] = _name.getValue();}
values[4] = _time.getValue();
if (_startSegment && !_insideWaypoint)
{
values[5] = "1";
_startSegment = false;
}
values[6] = _type.getValue();
values[7] = _description.getValue();
_pointList.add(values);
_trackNameList.addPoint(_trackNum, _trackName.getValue(), _isTrackPoint);
_linkList.add(_link.getValue());
}
/**
* @see tim.prune.load.xml.XmlHandler#getFieldArray()
*/
public Field[] getFieldArray()
{
final Field[] fields = {Field.LATITUDE, Field.LONGITUDE, Field.ALTITUDE,
Field.WAYPT_NAME, Field.TIMESTAMP, Field.NEW_SEGMENT,
Field.WAYPT_TYPE, Field.DESCRIPTION};
return fields;
}
/**
* Return the parsed information as a 2d array
* @see tim.prune.load.xml.XmlHandler#getDataArray()
*/
public String[][] getDataArray()
{
int numPoints = _pointList.size();
// construct data array
String[][] result = new String[numPoints][];
for (int i=0; i<numPoints; i++)
{
result[i] = _pointList.get(i);
}
return result;
}
/**
* @return array of links, or null if none
*/
public String[] getLinkArray()
{
int numPoints = _linkList.size();
boolean hasLink = false;
String[] result = new String[numPoints];
for (int i=0; i<numPoints; i++)
{
result[i] = _linkList.get(i);
if (result[i] != null) {hasLink = true;}
}
if (!hasLink) {result = null;}
return result;
}
/**
* @return track name list
*/
public TrackNameList getTrackNameList() {
return _trackNameList;
}
}
|