Rev 4373 |
View as "text/plain" |
Blame |
Compare with Previous |
Last modification |
View Log
| RSS feed
/*
* Copyright (C) 2012.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 3 or
* version 2 as published by the Free Software Foundation.
*
* This program 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
* General Public License for more details.
*/
package uk.me.parabola.mkgmap.reader.osm;
import java.awt.Rectangle;
import java.util.ArrayList;
import java.util.List;
import java.util.Map.Entry;
import uk.me.parabola.imgfmt.app.Area;
import uk.me.parabola.imgfmt.app.Coord;
import uk.me.parabola.log.Logger;
import uk.me.parabola.util.EnhancedProperties;
/**
* The hook removes all elements that will not be included in the map and can therefore
* be safely removed. This improves the performance because the elements do not have
* to go through the style system.
*
* @author WanMil
*/
public class UnusedElementsRemoverHook
implements OsmReadingHooks
{
private static final Logger log =
Logger.
getLogger(UnusedElementsRemoverHook.
class);
private ElementSaver saver
;
/** node with tags of this list must not be removed */
private List<Entry
<String,
String>> areasToPoiNodeTags
;
@
Override
public boolean init
(ElementSaver saver, EnhancedProperties props
) {
this.
saver = saver
;
// Get the tags from the POIGeneratorHook which are used to define the point
// where the POI is placed in polygons. They must not be removed if the polygon
// is not removed. Checking if the polygon is not removed is too costly therefore
// all nodes with these tags are kept.
if (props.
containsKey("add-pois-to-areas"))
areasToPoiNodeTags = POIGeneratorHook.
getPoiPlacementTags(props
);
else
areasToPoiNodeTags =
new ArrayList<>();
return true;
}
@
Override
public void end
() {
long t1 =
System.
currentTimeMillis();
log.
info("Removing unused elements");
final Area bbox = saver.
getBoundingBox();
long nodes = saver.
getNodes().
size();
// go through all nodes
for (Node node :
new ArrayList<Node>(saver.
getNodes().
values())) {
// nodes without tags can be removed
if (node.
getTagCount() ==
0) {
saver.
getNodes().
remove(node.
getId());
continue;
}
// check if the node is within the tile bounding box
if (!bbox.
contains(node.
getLocation())) {
boolean removeNode =
true;
for (Entry
<String,
String> tag : areasToPoiNodeTags
) {
// check if the node has a tag used by the POIGeneratorHook
String val = node.
getTag(tag.
getKey());
if (val
!=
null && (tag.
getValue() ==
null || val.
equals(tag.
getValue()))) {
// the node contains one tag that might be
// interesting for the POIGeneratorHook
// do not remove it
removeNode =
false;
break;
}
}
if (removeNode
) {
saver.
getNodes().
remove(node.
getId());
} else {
log.
debug("Keep node", node,
"because it contains a tag which might be required for the area-to-poi function.");
}
}
}
Rectangle bboxRect =
new Rectangle(bbox.
getMinLong(), bbox.
getMinLat(), bbox.
getWidth(), bbox.
getHeight());
long ways = saver.
getWays().
size();
for (Way way :
new ArrayList<Way
>(saver.
getWays().
values())) {
if (way.
isViaWay())
continue;
if (way.
getPoints().
isEmpty()) {
// empty way will not appear in the map => remove it
saver.
getWays().
remove(way.
getId());
continue;
}
// check if a way has no tags
// it is presumed that the RelationStyleHook was already executed
if (way.
getTagCount() ==
0) {
saver.
getWays().
remove(way.
getId());
continue;
}
// check if the way is completely outside the tile bounding box
boolean coordInBbox =
false;
Coord prevC =
null;
// It is possible that the way is larger than the bounding box and therefore
// contains the bbox completely. Especially this is true for the sea polygon
// when using --generate-sea=polygon
for (Coord c : way.
getPoints()) {
if (bbox.
contains(c
)) {
coordInBbox =
true;
break;
} else if (prevC
!=
null &&
bboxRect.
intersectsLine(prevC.
getLongitude(), prevC.
getLatitude(), c.
getLongitude(), c.
getLatitude())) {
// the line intersects the bounding box
if (log.
isDebugEnabled()) {
log.
debug("Intersection!");
log.
debug("Bbox:", bbox
);
log.
debug("Way coords:", prevC, c
);
}
coordInBbox =
true;
break;
}
prevC = c
;
}
if (!coordInBbox
) {
// no coord of the way is within the bounding box
// check if the way possibly covers the bounding box completely
Area wayBbox =
Area.
getBBox(way.
getPoints());
if (wayBbox.
contains(saver.
getBoundingBox())) {
log.
debug(way,
"possibly covers the bbox completely. Keep it.");
} else {
saver.
getWays().
remove(way.
getId());
}
}
}
log.
info("Nodes: before:", nodes,
"after:", saver.
getNodes().
size());
log.
info("Ways: before:", ways,
"after:", saver.
getWays().
size());
log.
info("Removing unused elements took",
(System.
currentTimeMillis()-t1
),
"ms");
}
}