-
Notifications
You must be signed in to change notification settings - Fork 410
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
refactor(isochrones): introduce DRYness to isochrone map builders (#1699
- Loading branch information
Showing
5 changed files
with
355 additions
and
675 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
280 changes: 280 additions & 0 deletions
280
ors-engine/src/main/java/org/heigit/ors/isochrones/builders/AbstractIsochroneMapBuilder.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,280 @@ | ||
package org.heigit.ors.isochrones.builders; | ||
|
||
import com.graphhopper.routing.SPTEntry; | ||
import com.graphhopper.routing.util.FlagEncoder; | ||
import com.graphhopper.routing.util.HikeFlagEncoder; | ||
import com.graphhopper.util.*; | ||
import org.apache.log4j.Logger; | ||
import org.heigit.ors.isochrones.Isochrone; | ||
import org.heigit.ors.isochrones.IsochroneMap; | ||
import org.heigit.ors.routing.RouteSearchContext; | ||
import org.heigit.ors.routing.graphhopper.extensions.flagencoders.FootFlagEncoder; | ||
import org.heigit.ors.routing.graphhopper.extensions.flagencoders.ORSAbstractFlagEncoder; | ||
import org.heigit.ors.routing.graphhopper.extensions.flagencoders.WheelchairFlagEncoder; | ||
import org.heigit.ors.routing.graphhopper.extensions.flagencoders.bike.CommonBikeFlagEncoder; | ||
import org.heigit.ors.util.GeomUtility; | ||
import org.locationtech.jts.geom.*; | ||
|
||
import java.util.ArrayList; | ||
import java.util.Arrays; | ||
import java.util.List; | ||
|
||
import static org.locationtech.jts.algorithm.hull.ConcaveHull.concaveHullByLength; | ||
|
||
public abstract class AbstractIsochroneMapBuilder implements IsochroneMapBuilder { | ||
private static final double MAX_SPLIT_LENGTH = 20000.0; | ||
protected static final DistanceCalc dcFast = new DistancePlaneProjection(); | ||
protected GeometryFactory geometryFactory; | ||
protected RouteSearchContext searchContext; | ||
protected double defaultSmoothingDistance = 0.012;// Use a default length of ~1333m | ||
protected Polygon previousIsochronePolygon = null; | ||
|
||
public abstract Logger getLogger(); | ||
|
||
public void initialize(RouteSearchContext searchContext) { | ||
this.searchContext = searchContext; | ||
this.geometryFactory = new GeometryFactory(); | ||
} | ||
|
||
/** | ||
* Converts the smoothing factor into a distance (which can be used in algorithms for generating isochrone polygons). | ||
* The distance value returned is dependent on the radius and smoothing factor. | ||
* | ||
* @param smoothingFactor A factor that should be used in the smoothing process. Lower numbers produce a smaller | ||
* distance (and so likely a more detailed polygon) | ||
* @param maxRadius The maximum radius of the isochrone (in metres) | ||
* @return | ||
*/ | ||
protected double convertSmoothingFactorToDistance(float smoothingFactor, double maxRadius) { | ||
final double MINIMUM_DISTANCE = 0.006; | ||
|
||
if (smoothingFactor == -1) { | ||
// No user defined smoothing factor, so use defaults | ||
|
||
// For shorter isochrones, we want to use a smaller minimum distance else we get inaccurate results | ||
if (maxRadius < 5000) | ||
return MINIMUM_DISTANCE; | ||
|
||
return defaultSmoothingDistance; | ||
} | ||
|
||
double intervalDegrees = GeomUtility.metresToDegrees(maxRadius); | ||
double maxLength = (intervalDegrees / 100f) * smoothingFactor; | ||
|
||
if (maxLength < MINIMUM_DISTANCE) | ||
maxLength = MINIMUM_DISTANCE; | ||
return maxLength; | ||
} | ||
|
||
protected void addPoint(List<Coordinate> points, double lon, double lat) { | ||
Coordinate p = new Coordinate(lon, lat); | ||
points.add(p); | ||
} | ||
|
||
protected void addBufferPoints(List<Coordinate> points, double lon0, double lat0, double lon1, | ||
double lat1, boolean addLast, double bufferSize) { | ||
double dx = (lon0 - lon1); | ||
double dy = (lat0 - lat1); | ||
double normLength = Math.sqrt((dx * dx) + (dy * dy)); | ||
double scale = bufferSize / normLength; | ||
|
||
double dx2 = -dy * scale; | ||
double dy2 = dx * scale; | ||
|
||
addPoint(points, lon0 + dx2, lat0 + dy2); | ||
addPoint(points, lon0 - dx2, lat0 - dy2); | ||
|
||
// add a middle point if two points are too far from each other | ||
if (normLength > 2 * bufferSize) { | ||
addPoint(points, (lon0 + lon1) / 2.0 + dx2, (lat0 + lat1) / 2.0 + dy2); | ||
addPoint(points, (lon0 + lon1) / 2.0 - dx2, (lat0 + lat1) / 2.0 - dy2); | ||
} | ||
|
||
if (addLast) { | ||
addPoint(points, lon1 + dx2, lat1 + dy2); | ||
addPoint(points, lon1 - dx2, lat1 - dy2); | ||
} | ||
} | ||
|
||
protected void addPointsFromEdge(List<Coordinate> points, double isolineCost, float maxCost, float minCost, double bufferSize, double edgeDist, List<Coordinate> pl) { | ||
double edgeCost = maxCost - minCost; | ||
double costPerMeter = edgeCost / edgeDist; | ||
double distPolyline = 0.0; | ||
|
||
double lat0 = pl.get(0).y; | ||
double lon0 = pl.get(0).x; | ||
double lat1; | ||
double lon1; | ||
|
||
for (int i = 1; i < pl.size(); ++i) { | ||
lat1 = pl.get(i).y; | ||
lon1 = pl.get(i).x; | ||
|
||
distPolyline += dcFast.calcDist(lat0, lon0, lat1, lon1); | ||
|
||
double distCost = minCost + distPolyline * costPerMeter; | ||
if (distCost >= isolineCost) { | ||
double segLength = (1 - (distCost - isolineCost) / edgeCost); | ||
double lon2 = lon0 + segLength * (lon1 - lon0); | ||
double lat2 = lat0 + segLength * (lat1 - lat0); | ||
|
||
addBufferPoints(points, lon0, lat0, lon2, lat2, true, bufferSize); | ||
|
||
break; | ||
} else { | ||
addBufferPoints(points, lon0, lat0, lon1, lat1, false, bufferSize); | ||
} | ||
|
||
lat0 = lat1; | ||
lon0 = lon1; | ||
} | ||
} | ||
|
||
/** | ||
* Splits a line between two points of the distance is longer than a limit | ||
* | ||
* @param lat0 latitude of first point | ||
* @param lon0 longitude of first point | ||
* @param lat1 latitude of second point | ||
* @param lon1 longitude of second point | ||
* @param minlim limit above which the edge will be split (in meters) | ||
* @param maxlim limit above which the edge will NOT be split anymore (in meters) | ||
*/ | ||
protected void splitLineSegment(double lat0, double lon0, double lat1, double lon1, List<Coordinate> points, double minlim, double maxlim) { | ||
double dist = dcFast.calcDist(lat0, lon0, lat1, lon1); | ||
|
||
if (dist > minlim && dist < maxlim) { | ||
int n = (int) Math.ceil(dist / minlim); | ||
for (int i = 1; i < n; i++) { | ||
addPoint(points, lon0 + i * (lon1 - lon0) / n, lat0 + i * (lat1 - lat0) / n); | ||
} | ||
} | ||
} | ||
|
||
protected List<Coordinate> edgeToPoints(EdgeIteratorState iter, double minSplitLength) { | ||
// always use mode=3, since other ones do not provide correct results | ||
PointList pl = iter.fetchWayGeometry(FetchMode.ALL); | ||
|
||
List<Coordinate> points = new ArrayList<>(2 * pl.size()); | ||
|
||
if (!pl.isEmpty()) { | ||
double lat0 = pl.getLat(0); | ||
double lon0 = pl.getLon(0); | ||
double lat1; | ||
double lon1; | ||
for (int i = 1; i < pl.size(); i++) { | ||
lat1 = pl.getLat(i); | ||
lon1 = pl.getLon(i); | ||
addPoint(points, lon0, lat0); | ||
splitLineSegment(lat0, lon0, lat1, lon1, points, minSplitLength, MAX_SPLIT_LENGTH); | ||
lon0 = lon1; | ||
lat0 = lat1; | ||
} | ||
addPoint(points, lon0, lat0); | ||
} | ||
|
||
return points; | ||
} | ||
|
||
protected void addBufferedEdgeGeometry(List<Coordinate> points, double minSplitLength, EdgeIteratorState iter, boolean detailedShape, SPTEntry goalEdge, double bufferSize) { | ||
List<Coordinate> pl = edgeToPoints(iter, minSplitLength); | ||
if (pl.isEmpty()) { | ||
return; | ||
} | ||
|
||
int size = pl.size(); | ||
|
||
double lat0 = pl.get(0).y; | ||
double lon0 = pl.get(0).x; | ||
double lat1; | ||
double lon1; | ||
|
||
if (detailedShape) { | ||
for (int i = 1; i < size; ++i) { | ||
lat1 = pl.get(i).y; | ||
lon1 = pl.get(i).x; | ||
|
||
addBufferPoints(points, lon0, lat0, lon1, lat1, goalEdge.edge < 0 && i == size - 1, bufferSize); | ||
|
||
lon0 = lon1; | ||
lat0 = lat1; | ||
} | ||
} else { | ||
points.addAll(pl); | ||
} | ||
} | ||
|
||
protected void addBorderEdgeGeometry(List<Coordinate> points, double isolineCost, double minSplitLength, EdgeIteratorState iter, float maxCost, float minCost, double bufferSize) { | ||
List<Coordinate> pl = edgeToPoints(iter, minSplitLength); | ||
if (pl.isEmpty()) { | ||
return; | ||
} | ||
addPointsFromEdge(points, isolineCost, maxCost, minCost, bufferSize, iter.getDistance(), pl); | ||
} | ||
|
||
protected double determineMaxSpeed() { | ||
FlagEncoder encoder = searchContext.getEncoder(); | ||
double maxSpeed = encoder.getMaxSpeed(); | ||
|
||
if (encoder instanceof FootFlagEncoder || encoder instanceof HikeFlagEncoder) { | ||
// in the GH FootFlagEncoder, the maximum speed is set to 15km/h which is way too high | ||
maxSpeed = 4; | ||
} | ||
|
||
if (encoder instanceof WheelchairFlagEncoder) { | ||
maxSpeed = WheelchairFlagEncoder.MEAN_SPEED; | ||
} | ||
|
||
return maxSpeed; | ||
} | ||
|
||
protected double determineMeanSpeed(double maxSpeed) { | ||
double meanSpeed = maxSpeed; | ||
FlagEncoder encoder = searchContext.getEncoder(); | ||
|
||
if (encoder instanceof ORSAbstractFlagEncoder flagEncoder) { | ||
meanSpeed = flagEncoder.getMeanSpeed(); | ||
} | ||
|
||
if (encoder instanceof CommonBikeFlagEncoder flagEncoder) { | ||
meanSpeed = flagEncoder.getMeanSpeed(); | ||
} | ||
|
||
return meanSpeed; | ||
} | ||
|
||
protected List<Coordinate> createCoordinateListFromPolygon(Polygon poly) { | ||
List<Coordinate> ringCoordinates = new ArrayList<>(Arrays.asList(poly.getExteriorRing().getCoordinates())); | ||
// remove the last point as for a closed ring it equals the first one | ||
ringCoordinates.remove(ringCoordinates.size() - 1); | ||
|
||
return ringCoordinates; | ||
} | ||
|
||
protected void addIsochrone(IsochroneMap isochroneMap, GeometryCollection points, double isoValue, double meanRadius, double smoothingDistance) { | ||
if (points.isEmpty()) | ||
return; | ||
|
||
StopWatch sw = new StopWatch(); | ||
if (getLogger().isDebugEnabled()) { | ||
sw = new StopWatch(); | ||
sw.start(); | ||
} | ||
|
||
Geometry shellGeometry = concaveHullByLength(points, smoothingDistance); | ||
if (shellGeometry instanceof GeometryCollection geomColl) { | ||
if (geomColl.isEmpty()) | ||
return; | ||
} | ||
|
||
Polygon poly = (Polygon) shellGeometry; | ||
previousIsochronePolygon = poly; | ||
|
||
if (getLogger().isDebugEnabled()) { | ||
sw.stop(); | ||
getLogger().debug("Build shell concave hull " + sw.getSeconds()); | ||
} | ||
|
||
isochroneMap.addIsochrone(new Isochrone(poly, isoValue, meanRadius)); | ||
} | ||
} |
Oops, something went wrong.