2010-04-28 00:23:24 +02:00
|
|
|
package com.osmand.osm.io;
|
|
|
|
|
2010-06-07 20:07:04 +02:00
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_CHANGESET;
|
2010-05-14 23:05:18 +02:00
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_ID;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_K;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_LAT;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_LON;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_REF;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_ROLE;
|
2010-06-07 20:07:04 +02:00
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_TIMESTAMP;
|
2010-05-14 23:05:18 +02:00
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_TYPE;
|
2010-06-07 20:07:04 +02:00
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_UID;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_USER;
|
2010-05-14 23:05:18 +02:00
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_V;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_VERSION;
|
2010-06-07 20:07:04 +02:00
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ATTR_VISIBLE;
|
2010-05-14 23:05:18 +02:00
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ELEM_MEMBER;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ELEM_ND;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ELEM_NODE;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ELEM_OSM;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ELEM_RELATION;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ELEM_TAG;
|
|
|
|
import static com.osmand.osm.io.OsmBaseStorage.ELEM_WAY;
|
|
|
|
|
2010-04-28 00:23:24 +02:00
|
|
|
import java.io.IOException;
|
|
|
|
import java.io.OutputStream;
|
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.Collection;
|
|
|
|
import java.util.List;
|
|
|
|
import java.util.Map;
|
|
|
|
import java.util.Stack;
|
|
|
|
import java.util.Map.Entry;
|
|
|
|
|
|
|
|
import javax.xml.stream.XMLStreamException;
|
|
|
|
import javax.xml.stream.XMLStreamWriter;
|
|
|
|
|
2010-05-16 23:49:11 +02:00
|
|
|
import com.osmand.Algoritms;
|
|
|
|
import com.osmand.data.MapObject;
|
2010-04-28 00:23:24 +02:00
|
|
|
import com.osmand.osm.Entity;
|
2010-06-07 20:07:04 +02:00
|
|
|
import com.osmand.osm.EntityInfo;
|
2010-04-28 00:23:24 +02:00
|
|
|
import com.osmand.osm.Node;
|
|
|
|
import com.osmand.osm.Relation;
|
|
|
|
import com.osmand.osm.Way;
|
|
|
|
import com.sun.org.apache.xerces.internal.impl.PropertyManager;
|
|
|
|
import com.sun.xml.internal.stream.writers.XMLStreamWriterImpl;
|
|
|
|
|
2010-05-17 16:38:56 +02:00
|
|
|
public class OsmStorageWriter {
|
2010-04-28 00:23:24 +02:00
|
|
|
|
|
|
|
private final String INDENT = " ";
|
|
|
|
private final String INDENT2 = INDENT + INDENT;
|
|
|
|
|
|
|
|
|
2010-05-17 16:38:56 +02:00
|
|
|
public OsmStorageWriter(){
|
2010-04-28 00:23:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-05-16 23:49:11 +02:00
|
|
|
public void saveStorage(OutputStream output, OsmBaseStorage storage, Collection<Long> interestedObjects, boolean includeLinks) throws XMLStreamException, IOException {
|
|
|
|
Map<Long, Entity> entities = storage.getRegisteredEntities();
|
2010-06-07 20:07:04 +02:00
|
|
|
Map<Long, EntityInfo> entityInfo = storage.getRegisteredEntityInfo();
|
2010-04-28 00:23:24 +02:00
|
|
|
PropertyManager propertyManager = new PropertyManager(PropertyManager.CONTEXT_WRITER);
|
|
|
|
// transformer.setOutputProperty(OutputKeys.INDENT, "yes");
|
|
|
|
// String indent = "{http://xml.apache.org/xslt}indent-amount";
|
|
|
|
// transformer.setOutputProperty(indent, "4");
|
|
|
|
|
|
|
|
XMLStreamWriter streamWriter = new XMLStreamWriterImpl(output, propertyManager);
|
|
|
|
List<Node> nodes = new ArrayList<Node>();
|
|
|
|
List<Way> ways = new ArrayList<Way>();
|
|
|
|
List<Relation> relations = new ArrayList<Relation>();
|
|
|
|
if(interestedObjects == null){
|
|
|
|
interestedObjects = entities.keySet();
|
|
|
|
}
|
|
|
|
Stack<Long> toResolve = new Stack<Long>();
|
|
|
|
toResolve.addAll(interestedObjects);
|
|
|
|
while(!toResolve.isEmpty()){
|
|
|
|
Long l = toResolve.pop();
|
|
|
|
if(entities.get(l) instanceof Node){
|
|
|
|
nodes.add((Node) entities.get(l));
|
|
|
|
} else if(entities.get(l) instanceof Way){
|
|
|
|
ways.add((Way) entities.get(l));
|
2010-05-06 16:55:49 +02:00
|
|
|
if(includeLinks){
|
|
|
|
toResolve.addAll(((Way)entities.get(l)).getNodeIds());
|
|
|
|
}
|
2010-04-28 00:23:24 +02:00
|
|
|
} else if(entities.get(l) instanceof Relation){
|
|
|
|
relations.add((Relation) entities.get(l));
|
2010-05-06 16:55:49 +02:00
|
|
|
if(includeLinks){
|
|
|
|
toResolve.addAll(((Relation)entities.get(l)).getMemberIds());
|
|
|
|
}
|
2010-04-28 00:23:24 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
streamWriter.writeStartDocument();
|
|
|
|
|
|
|
|
writeStartElement(streamWriter, ELEM_OSM, "");
|
|
|
|
streamWriter.writeAttribute(ATTR_VERSION, "0.6");
|
|
|
|
for(Node n : nodes){
|
|
|
|
writeStartElement(streamWriter, ELEM_NODE, INDENT);
|
|
|
|
streamWriter.writeAttribute(ATTR_LAT, n.getLatitude()+"");
|
|
|
|
streamWriter.writeAttribute(ATTR_LON, n.getLongitude()+"");
|
|
|
|
streamWriter.writeAttribute(ATTR_ID, n.getId()+"");
|
2010-06-08 20:29:38 +02:00
|
|
|
writeEntityAttributes(streamWriter, n, entityInfo.get(n.getId()));
|
2010-04-28 00:23:24 +02:00
|
|
|
writeTags(streamWriter, n);
|
2010-05-16 23:49:11 +02:00
|
|
|
writeEndElement(streamWriter, INDENT);
|
2010-04-28 00:23:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for(Way w : ways){
|
|
|
|
writeStartElement(streamWriter, ELEM_WAY, INDENT);
|
|
|
|
streamWriter.writeAttribute(ATTR_ID, w.getId()+"");
|
2010-06-08 20:29:38 +02:00
|
|
|
writeEntityAttributes(streamWriter, w, entityInfo.get(w.getId()));
|
2010-04-28 00:23:24 +02:00
|
|
|
for(Long r : w.getNodeIds()){
|
|
|
|
writeStartElement(streamWriter, ELEM_ND, INDENT2);
|
|
|
|
streamWriter.writeAttribute(ATTR_REF, r+"");
|
2010-05-16 23:49:11 +02:00
|
|
|
writeEndElement(streamWriter, INDENT2);
|
2010-04-28 00:23:24 +02:00
|
|
|
}
|
|
|
|
writeTags(streamWriter, w);
|
2010-05-16 23:49:11 +02:00
|
|
|
writeEndElement(streamWriter, INDENT);
|
2010-04-28 00:23:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for(Relation r : relations){
|
|
|
|
writeStartElement(streamWriter, ELEM_RELATION, INDENT);
|
|
|
|
streamWriter.writeAttribute(ATTR_ID, r.getId()+"");
|
2010-06-08 20:29:38 +02:00
|
|
|
writeEntityAttributes(streamWriter, r, entityInfo.get(r.getId()));
|
2010-04-28 00:23:24 +02:00
|
|
|
for(Entry<Long, String> e : r.getMembersMap().entrySet()){
|
|
|
|
writeStartElement(streamWriter, ELEM_MEMBER, INDENT2);
|
|
|
|
streamWriter.writeAttribute(ATTR_REF, e.getKey()+"");
|
|
|
|
String s = e.getValue();
|
|
|
|
if(s == null){
|
|
|
|
s = "";
|
|
|
|
}
|
|
|
|
streamWriter.writeAttribute(ATTR_ROLE, s);
|
2010-05-16 23:49:11 +02:00
|
|
|
streamWriter.writeAttribute(ATTR_TYPE, getEntityType(entities, e.getKey()));
|
|
|
|
writeEndElement(streamWriter, INDENT2);
|
2010-04-28 00:23:24 +02:00
|
|
|
}
|
|
|
|
writeTags(streamWriter, r);
|
2010-05-16 23:49:11 +02:00
|
|
|
writeEndElement(streamWriter, INDENT);
|
2010-04-28 00:23:24 +02:00
|
|
|
}
|
|
|
|
|
2010-05-16 23:49:11 +02:00
|
|
|
writeEndElement(streamWriter, ""); // osm
|
2010-04-28 00:23:24 +02:00
|
|
|
streamWriter.writeEndDocument();
|
|
|
|
streamWriter.flush();
|
|
|
|
}
|
|
|
|
|
2010-06-08 20:29:38 +02:00
|
|
|
private void writeEntityAttributes(XMLStreamWriter writer, Entity i, EntityInfo info) throws XMLStreamException{
|
|
|
|
if(i.getId() < 0 && (info == null || info.getAction() == null)){
|
|
|
|
writer.writeAttribute("action", "modify");
|
|
|
|
}
|
2010-06-07 20:07:04 +02:00
|
|
|
if(info != null){
|
2010-06-08 20:29:38 +02:00
|
|
|
// for josm editor
|
|
|
|
if(info.getAction() != null){
|
|
|
|
writer.writeAttribute("action", info.getAction());
|
|
|
|
}
|
2010-06-07 20:07:04 +02:00
|
|
|
if(info.getChangeset() != null){
|
|
|
|
writer.writeAttribute(ATTR_CHANGESET, info.getChangeset());
|
|
|
|
}
|
|
|
|
if(info.getTimestamp() != null){
|
|
|
|
writer.writeAttribute(ATTR_TIMESTAMP, info.getTimestamp());
|
|
|
|
}
|
|
|
|
if(info.getUid() != null){
|
|
|
|
writer.writeAttribute(ATTR_UID, info.getUid());
|
|
|
|
}
|
|
|
|
if(info.getUser() != null){
|
|
|
|
writer.writeAttribute(ATTR_USER, info.getUser());
|
|
|
|
}
|
|
|
|
if(info.getVisible() != null){
|
|
|
|
writer.writeAttribute(ATTR_VISIBLE, info.getVisible());
|
|
|
|
}
|
|
|
|
if(info.getVersion() != null){
|
|
|
|
writer.writeAttribute(ATTR_VERSION, info.getVersion());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-05-16 23:49:11 +02:00
|
|
|
private String getEntityType(Map<Long, Entity> entities , Long id){
|
2010-04-28 00:23:24 +02:00
|
|
|
Entity e = entities.get(id);
|
|
|
|
if(e instanceof Way){
|
|
|
|
return "way";
|
|
|
|
} else if(e instanceof Relation){
|
|
|
|
return "relation";
|
|
|
|
}
|
|
|
|
return "node";
|
|
|
|
}
|
|
|
|
|
2010-05-25 18:53:52 +02:00
|
|
|
|
2010-05-27 13:02:02 +02:00
|
|
|
|
|
|
|
public boolean couldBeWrited(MapObject e){
|
2010-05-16 23:49:11 +02:00
|
|
|
if(!Algoritms.isEmpty(e.getName()) && e.getLocation() != null){
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-04-28 00:23:24 +02:00
|
|
|
private void writeStartElement(XMLStreamWriter writer, String name, String indent) throws XMLStreamException{
|
|
|
|
writer.writeCharacters("\n"+indent);
|
|
|
|
writer.writeStartElement(name);
|
|
|
|
}
|
|
|
|
|
2010-05-16 23:49:11 +02:00
|
|
|
private void writeEndElement(XMLStreamWriter writer, String indent) throws XMLStreamException{
|
|
|
|
writer.writeCharacters("\n"+indent);
|
|
|
|
writer.writeEndElement();
|
|
|
|
}
|
2010-04-28 00:23:24 +02:00
|
|
|
|
|
|
|
private void writeTags(XMLStreamWriter writer, Entity e) throws XMLStreamException{
|
|
|
|
for(Entry<String, String> en : e.getTags().entrySet()){
|
|
|
|
writeStartElement(writer, ELEM_TAG, INDENT2);
|
|
|
|
writer.writeAttribute(ATTR_K, en.getKey());
|
|
|
|
writer.writeAttribute(ATTR_V, en.getValue());
|
|
|
|
writer.writeEndElement();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-06-02 15:27:50 +02:00
|
|
|
|