2011-07-03 15:08:34 +02:00
|
|
|
package net.osmand.router;
|
|
|
|
|
|
|
|
import java.util.LinkedHashMap;
|
|
|
|
import java.util.Map;
|
|
|
|
|
|
|
|
import net.osmand.binary.BinaryMapDataObject;
|
|
|
|
import net.osmand.binary.BinaryMapIndexReader.TagValuePair;
|
|
|
|
import net.osmand.osm.MapRenderingTypes;
|
|
|
|
import net.osmand.router.BinaryRoutePlanner.RouteSegment;
|
|
|
|
|
2011-07-03 16:29:23 +02:00
|
|
|
public class CarRouter extends VehicleRouter {
|
2011-07-03 17:32:24 +02:00
|
|
|
// no distinguish for speed in city/outside city (for now)
|
|
|
|
private Map<String, Double> autoNotDefinedValues = new LinkedHashMap<String, Double>();
|
|
|
|
private Map<String, Double> autoPriorityValues = new LinkedHashMap<String, Double>();
|
|
|
|
{
|
|
|
|
autoNotDefinedValues.put("motorway", 110d);
|
|
|
|
autoNotDefinedValues.put("motorway_link", 80d);
|
|
|
|
autoNotDefinedValues.put("trunk", 100d);
|
|
|
|
autoNotDefinedValues.put("trunk_link", 80d);
|
|
|
|
autoNotDefinedValues.put("primary", 65d);
|
|
|
|
autoNotDefinedValues.put("primary_link", 45d);
|
|
|
|
autoNotDefinedValues.put("secondary", 50d);
|
|
|
|
autoNotDefinedValues.put("secondary_link", 40d);
|
|
|
|
autoNotDefinedValues.put("tertiary", 35d);
|
|
|
|
autoNotDefinedValues.put("tertiary_link", 30d);
|
|
|
|
autoNotDefinedValues.put("residential", 30d);
|
|
|
|
autoNotDefinedValues.put("road", 30d);
|
|
|
|
autoNotDefinedValues.put("service", 20d);
|
|
|
|
autoNotDefinedValues.put("unclassified", 20d);
|
|
|
|
autoNotDefinedValues.put("track", 20d);
|
|
|
|
autoNotDefinedValues.put("path", 20d);
|
|
|
|
autoNotDefinedValues.put("living_street", 20d);
|
|
|
|
|
|
|
|
autoPriorityValues.put("motorway", 1.5);
|
|
|
|
autoPriorityValues.put("motorway_link", 1.0);
|
|
|
|
autoPriorityValues.put("trunk", 1.5);
|
|
|
|
autoPriorityValues.put("trunk_link", 1d);
|
|
|
|
autoPriorityValues.put("primary", 1.3d);
|
|
|
|
autoPriorityValues.put("primary_link", 1d);
|
|
|
|
autoPriorityValues.put("secondary", 1.0d);
|
|
|
|
autoPriorityValues.put("secondary_link", 1.0d);
|
2011-07-03 21:24:13 +02:00
|
|
|
autoPriorityValues.put("tertiary", 0.85d);
|
|
|
|
autoPriorityValues.put("tertiary_link", 0.85d);
|
|
|
|
autoPriorityValues.put("residential", 0.7d);
|
2011-07-03 17:32:24 +02:00
|
|
|
autoPriorityValues.put("service", 0.6d);
|
|
|
|
autoPriorityValues.put("unclassified", 0.4d);
|
|
|
|
autoPriorityValues.put("road", 0.4d);
|
2011-07-07 00:53:03 +02:00
|
|
|
autoPriorityValues.put("track", 0.2d);
|
|
|
|
autoPriorityValues.put("path", 0.2d);
|
2011-07-03 17:32:24 +02:00
|
|
|
autoPriorityValues.put("living_street", 0.5d);
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean acceptLine(TagValuePair pair) {
|
|
|
|
if (pair.tag.equals("highway")) {
|
|
|
|
return autoNotDefinedValues.containsKey(pair.value);
|
2011-07-03 15:08:34 +02:00
|
|
|
}
|
2011-07-03 17:32:24 +02:00
|
|
|
return false;
|
|
|
|
}
|
2011-07-03 15:08:34 +02:00
|
|
|
|
2011-07-03 17:32:24 +02:00
|
|
|
public boolean acceptPoint(TagValuePair pair) {
|
|
|
|
if (pair.tag.equals("traffic_calming")) {
|
|
|
|
return true;
|
|
|
|
} else if (pair.tag.equals("highway") && pair.value.equals("traffic_signals")) {
|
|
|
|
return true;
|
|
|
|
} else if (pair.tag.equals("highway") && pair.value.equals("speed_camera")) {
|
|
|
|
return true;
|
|
|
|
} else if (pair.tag.equals("railway") && pair.value.equals("crossing")) {
|
|
|
|
return true;
|
|
|
|
} else if (pair.tag.equals("railway") && pair.value.equals("level_crossing")) {
|
|
|
|
return true;
|
2011-07-03 15:08:34 +02:00
|
|
|
}
|
2011-07-03 17:32:24 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isOneWay(int highwayAttributes) {
|
|
|
|
return MapRenderingTypes.isOneWayWay(highwayAttributes) || MapRenderingTypes.isRoundabout(highwayAttributes);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* return delay in seconds
|
|
|
|
*/
|
|
|
|
public double defineObstacle(BinaryMapDataObject road, int point) {
|
|
|
|
if ((road.getTypes()[0] & 3) == MapRenderingTypes.POINT_TYPE) {
|
|
|
|
// possibly not only first type needed ?
|
2011-07-03 15:08:34 +02:00
|
|
|
TagValuePair pair = road.getTagValue(0);
|
2011-07-03 17:32:24 +02:00
|
|
|
if (pair != null) {
|
|
|
|
if (pair.tag.equals("highway") && pair.value.equals("traffic_signals")) {
|
|
|
|
return 20;
|
|
|
|
} else if (pair.tag.equals("railway") && pair.value.equals("crossing")) {
|
|
|
|
return 25;
|
|
|
|
} else if (pair.tag.equals("railway") && pair.value.equals("level_crossing")) {
|
|
|
|
return 25;
|
2011-07-03 15:08:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-07-03 17:32:24 +02:00
|
|
|
return 0;
|
|
|
|
}
|
2011-07-05 10:22:33 +02:00
|
|
|
|
|
|
|
@Override
|
2011-07-07 00:53:03 +02:00
|
|
|
public double getRoadPriorityHeuristicToIncrease(BinaryMapDataObject road) {
|
2011-07-05 10:22:33 +02:00
|
|
|
TagValuePair pair = road.getTagValue(0);
|
|
|
|
boolean highway = "highway".equals(pair.tag);
|
|
|
|
double priority = highway && autoPriorityValues.containsKey(pair.value) ? autoPriorityValues.get(pair.value) : 0.5d;
|
2011-07-07 00:53:03 +02:00
|
|
|
// allow to get out from motorway to primary roads
|
|
|
|
// if("motorway_link".equals(pair.value) || "trunk".equals(pair.value) ||
|
|
|
|
// "trunk_link".equals(pair.value) || "motorway".equals(pair.value)) {
|
|
|
|
// return 1.3d;
|
|
|
|
// } else
|
|
|
|
if(priority >= 1){
|
2011-07-05 10:22:33 +02:00
|
|
|
return 1;
|
|
|
|
} else if(priority >= 0.7){
|
|
|
|
return 0.7;
|
|
|
|
} else if(priority >= 0.5){
|
|
|
|
return 0.5;
|
|
|
|
} else {
|
|
|
|
return 0.3;
|
|
|
|
}
|
|
|
|
}
|
2011-07-07 00:53:03 +02:00
|
|
|
|
|
|
|
public double getRoadPriorityToCalculateRoute(BinaryMapDataObject road) {
|
|
|
|
TagValuePair pair = road.getTagValue(0);
|
|
|
|
boolean highway = "highway".equals(pair.tag);
|
|
|
|
double priority = highway && autoPriorityValues.containsKey(pair.value) ? autoPriorityValues.get(pair.value) : 0.5d;
|
|
|
|
return priority;
|
|
|
|
};
|
2011-07-03 15:08:34 +02:00
|
|
|
|
2011-07-03 17:32:24 +02:00
|
|
|
/**
|
|
|
|
* return speed in m/s
|
|
|
|
*/
|
|
|
|
public double defineSpeed(BinaryMapDataObject road) {
|
|
|
|
TagValuePair pair = road.getTagValue(0);
|
|
|
|
double speed = MapRenderingTypes.getMaxSpeedIfDefined(road.getHighwayAttributes()) / 3.6d;
|
|
|
|
boolean highway = "highway".equals(pair.tag);
|
2011-07-05 10:22:33 +02:00
|
|
|
double priority = highway && autoPriorityValues.containsKey(pair.value) ? autoPriorityValues.get(pair.value) : 0.5d;
|
2011-07-03 17:32:24 +02:00
|
|
|
if (speed == 0 && highway) {
|
|
|
|
Double value = autoNotDefinedValues.get(pair.value);
|
|
|
|
if (value == null) {
|
|
|
|
value = 50d;
|
|
|
|
}
|
|
|
|
speed = value / 3.6d;
|
2011-07-03 15:08:34 +02:00
|
|
|
}
|
2011-07-03 17:32:24 +02:00
|
|
|
return speed * priority;
|
|
|
|
}
|
2011-07-03 15:08:34 +02:00
|
|
|
|
2011-07-03 17:32:24 +02:00
|
|
|
/**
|
|
|
|
* Used for A* routing to calculate g(x)
|
|
|
|
*
|
|
|
|
* @return minimal speed at road
|
|
|
|
*/
|
|
|
|
public double getMinDefaultSpeed() {
|
|
|
|
return 9;
|
|
|
|
}
|
2011-07-03 15:08:34 +02:00
|
|
|
|
2011-07-03 17:32:24 +02:00
|
|
|
/**
|
|
|
|
* Used for A* routing to predict h(x) : it should be < (!) any g(x)
|
|
|
|
*
|
|
|
|
* @return maximum speed to calculate shortest distance
|
|
|
|
*/
|
|
|
|
public double getMaxDefaultSpeed() {
|
|
|
|
return 30;
|
|
|
|
}
|
|
|
|
|
|
|
|
public double calculateTurnTime(RouteSegment segment, RouteSegment next, int segmentEnd) {
|
|
|
|
boolean end = (segmentEnd == segment.road.getPointsLength() - 1 || segmentEnd == 0);
|
|
|
|
boolean start = next.segmentStart == 0;
|
2011-07-03 19:02:21 +02:00
|
|
|
// that addition highly affects to trunk roads !(prefer trunk/motorway)
|
|
|
|
if (end && start) {
|
|
|
|
if (next.road.getPointsLength() > 1) {
|
|
|
|
int x = segment.road.getPoint31XTile(segmentEnd);
|
|
|
|
int y = segment.road.getPoint31XTile(segmentEnd);
|
|
|
|
int prevSegmentEnd = segmentEnd - 1;
|
|
|
|
if(prevSegmentEnd < 0){
|
|
|
|
prevSegmentEnd = segmentEnd + 1;
|
|
|
|
}
|
|
|
|
int px = segment.road.getPoint31XTile(prevSegmentEnd);
|
|
|
|
int py = segment.road.getPoint31XTile(prevSegmentEnd);
|
|
|
|
double a1 = Math.atan2(y - py, x - px);
|
|
|
|
double a2 = Math.atan2(y - next.road.getPoint31YTile(1), x - next.road.getPoint31XTile(1));
|
|
|
|
double diff = Math.abs(a1 - a2);
|
|
|
|
if (diff > Math.PI / 2 && diff < 3 * Math.PI / 2) {
|
|
|
|
return 25;
|
|
|
|
}
|
2011-07-03 15:08:34 +02:00
|
|
|
}
|
|
|
|
return 0;
|
2011-07-03 17:32:24 +02:00
|
|
|
} else {
|
2011-07-03 19:02:21 +02:00
|
|
|
return 15;
|
2011-07-03 15:08:34 +02:00
|
|
|
}
|
2011-07-03 17:32:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|