OsmAnd/OsmAnd-java/src/main/java/net/osmand/router/RouteStatisticsHelper.java

487 lines
16 KiB
Java
Raw Normal View History

package net.osmand.router;
2019-03-07 18:02:41 +01:00
import net.osmand.render.RenderingRuleSearchRequest;
import net.osmand.render.RenderingRulesStorage;
2019-03-01 12:35:59 +01:00
import java.util.ArrayList;
import java.util.List;
2019-07-07 18:30:01 +02:00
import java.util.Locale;
2019-03-01 12:35:59 +01:00
import java.util.Map;
import java.util.TreeMap;
2019-07-07 18:30:01 +02:00
public class RouteStatisticsHelper {
public static final String UNDEFINED_ATTR = "undefined";
2019-03-07 18:02:41 +01:00
2019-07-07 18:30:01 +02:00
public static List<RouteStatistics> calculateRouteStatistic(List<RouteSegmentResult> route, RenderingRulesStorage currentRenderer,
RenderingRulesStorage defaultRenderer, RenderingRuleSearchRequest currentSearchRequest,
RenderingRuleSearchRequest defaultSearchRequest) {
2019-03-07 18:02:41 +01:00
2019-07-07 18:30:01 +02:00
List<RouteStatistics> result = new ArrayList<>();
2019-03-07 18:02:41 +01:00
2019-07-07 18:30:01 +02:00
String[] attrNames = { "surface", "highway", "smoothness" };
String[] colorAttrNames = { "surfaceColor", "roadClassColor", "smoothnessColor" };
2019-07-07 18:30:01 +02:00
for (int i = 0; i < attrNames.length; i++) {
String attrName = attrNames[i];
String colorAttrName = colorAttrNames[i];
RouteStatisticComputer statisticComputer = new RoutePlainStatisticComputer(route, attrName, colorAttrName, currentRenderer, defaultRenderer, currentSearchRequest, defaultSearchRequest);
result.add(statisticComputer.computeStatistic());
}
2019-03-07 18:02:41 +01:00
2019-07-07 18:30:01 +02:00
return result;
2019-03-07 18:02:41 +01:00
}
private abstract static class RouteStatisticComputer<E extends Comparable<E>> {
2019-07-07 18:30:01 +02:00
final List<RouteSegmentResult> route;
final String attrName;
final String colorAttrName;
final RenderingRulesStorage currentRenderer;
final RenderingRulesStorage defaultRenderer;
final RenderingRuleSearchRequest currentRenderingRuleSearchRequest;
final RenderingRuleSearchRequest defaultRenderingRuleSearchRequest;
2019-03-07 18:02:41 +01:00
2019-07-07 18:30:01 +02:00
RouteStatisticComputer(List<RouteSegmentResult> route, String attrName, String colorAttrName,
RenderingRulesStorage currentRenderer, RenderingRulesStorage defaultRenderer,
RenderingRuleSearchRequest currentRenderingRuleSearchRequest, RenderingRuleSearchRequest defaultRenderingRuleSearchRequest) {
2019-03-07 18:02:41 +01:00
this.route = route;
2019-07-07 18:30:01 +02:00
this.attrName = attrName;
this.colorAttrName = colorAttrName;
this.currentRenderer = currentRenderer;
this.defaultRenderer = defaultRenderer;
this.currentRenderingRuleSearchRequest = currentRenderingRuleSearchRequest;
this.defaultRenderingRuleSearchRequest = defaultRenderingRuleSearchRequest;
2019-03-07 18:02:41 +01:00
}
2019-07-07 18:30:01 +02:00
Map<E, RouteSegmentAttribute<E>> makePartition(List<RouteSegmentAttribute<E>> routeAttributes) {
2019-03-07 18:02:41 +01:00
Map<E, RouteSegmentAttribute<E>> partition = new TreeMap<>();
for (RouteSegmentAttribute<E> attribute : routeAttributes) {
E key = attribute.getAttribute();
RouteSegmentAttribute<E> pattr = partition.get(key);
if (pattr == null) {
pattr = new RouteSegmentAttribute<>(attribute);
partition.put(key, pattr);
}
pattr.incrementDistanceBy(attribute.getDistance());
}
return partition;
}
private float computeTotalDistance(List<RouteSegmentAttribute<E>> attributes) {
float distance = 0f;
for (RouteSegmentAttribute attribute : attributes) {
distance += attribute.getDistance();
}
return distance;
}
protected List<RouteSegmentAttribute<E>> processRoute() {
int index = 0;
List<RouteSegmentAttribute<E>> routes = new ArrayList<>();
E prev = null;
2019-07-07 18:30:01 +02:00
for (RouteSegmentResult segment : route) {
2019-03-07 18:02:41 +01:00
E current = getAttribute(segment);
if (prev != null && !prev.equals(current)) {
index++;
}
if (index >= routes.size()) {
int color = getColor(current);
String propertyName = getPropertyName(current);
routes.add(new RouteSegmentAttribute<>(index, current, propertyName, color));
}
RouteSegmentAttribute surface = routes.get(index);
surface.incrementDistanceBy(segment.getDistance());
prev = current;
}
return routes;
}
2019-07-07 18:30:01 +02:00
RouteStatistics<E> computeStatistic() {
2019-03-07 18:02:41 +01:00
List<RouteSegmentAttribute<E>> routeAttributes = processRoute();
Map<E, RouteSegmentAttribute<E>> partition = makePartition(routeAttributes);
float totalDistance = computeTotalDistance(routeAttributes);
2019-07-07 18:30:01 +02:00
return new RouteStatistics<>(routeAttributes, partition, totalDistance);
2019-03-07 18:02:41 +01:00
}
RenderingRuleSearchRequest getSearchRequest(boolean useCurrentRenderer) {
return new RenderingRuleSearchRequest(useCurrentRenderer ? currentRenderingRuleSearchRequest : defaultRenderingRuleSearchRequest);
}
public int getColor(E attribute) {
int color = 0;
RenderingRuleSearchRequest currentRequest = getSearchRequest(true);
if (searchRenderingAttribute(currentRenderer, currentRequest, attribute)) {
color = currentRequest.getIntPropertyValue(currentRenderer.PROPS.R_ATTR_COLOR_VALUE);
} else {
RenderingRuleSearchRequest defaultRequest = getSearchRequest(false);
if (searchRenderingAttribute(defaultRenderer, defaultRequest, attribute)) {
color = defaultRequest.getIntPropertyValue(defaultRenderer.PROPS.R_ATTR_COLOR_VALUE);
}
}
return color;
}
2019-03-07 18:02:41 +01:00
public abstract E getAttribute(RouteSegmentResult segment);
2019-03-07 18:02:41 +01:00
public abstract String getPropertyName(E attribute);
protected abstract boolean searchRenderingAttribute(RenderingRulesStorage rrs, RenderingRuleSearchRequest req, E attribute);
2019-03-07 18:02:41 +01:00
}
2019-07-07 18:30:01 +02:00
private static class RoutePlainStatisticComputer extends RouteStatisticComputer<String> {
2019-03-07 18:02:41 +01:00
2019-07-07 18:30:01 +02:00
public RoutePlainStatisticComputer(List<RouteSegmentResult> route, String attrName, String colorAttrName,
RenderingRulesStorage currentRenderer, RenderingRulesStorage defaultRenderer,
RenderingRuleSearchRequest currentSearchRequest, RenderingRuleSearchRequest defaultSearchRequest) {
super(route, attrName, colorAttrName, currentRenderer, defaultRenderer, currentSearchRequest, defaultSearchRequest);
2019-03-07 18:02:41 +01:00
}
@Override
public String getAttribute(RouteSegmentResult segment) {
2019-07-07 18:30:01 +02:00
String attribute = segment.getObjectAttribute(attrName);
if (attribute == null) {
2019-03-07 18:02:41 +01:00
return UNDEFINED_ATTR;
}
String type = getAttributeType(attribute);
2019-07-07 18:30:01 +02:00
return type != null ? type : UNDEFINED_ATTR;
2019-03-07 18:02:41 +01:00
}
private String getAttributeType(String attribute) {
String type = null;
RenderingRuleSearchRequest currentRequest = getSearchRequest(true);
if (searchRenderingAttribute(currentRenderer, currentRequest, attribute)) {
type = currentRequest.getStringPropertyValue(currentRenderer.PROPS.R_ATTR_STRING_VALUE);
if (currentRequest.searchRenderingAttribute(type)) {
type = currentRequest.getStringPropertyValue(currentRenderer.PROPS.R_ATTR_STRING_VALUE);
}
} else {
RenderingRuleSearchRequest defaultRequest = getSearchRequest(false);
if (searchRenderingAttribute(defaultRenderer, defaultRequest, attribute)) {
type = defaultRequest.getStringPropertyValue(currentRenderer.PROPS.R_ATTR_STRING_VALUE);
if (defaultRequest.searchRenderingAttribute(type)) {
type = defaultRequest.getStringPropertyValue(currentRenderer.PROPS.R_ATTR_STRING_VALUE);
}
}
}
2019-07-07 18:30:01 +02:00
return type == null ? UNDEFINED_ATTR : type;
}
2019-03-07 18:02:41 +01:00
2019-07-07 18:30:01 +02:00
@Override
public String getPropertyName(String attribute) {
String type = getAttributeType(attribute);
return type != null ? type : attribute;
}
2019-03-07 18:02:41 +01:00
2019-07-07 18:30:01 +02:00
@Override
protected boolean searchRenderingAttribute(RenderingRulesStorage rrs, RenderingRuleSearchRequest req, String attribute) {
String additional = attrName + "=" + attribute;
req.setStringFilter(rrs.PROPS.R_ATTR_STRING_VALUE, attrName + "_" + attribute);
req.setStringFilter(rrs.PROPS.R_ADDITIONAL, additional);
boolean result = req.searchRenderingAttribute(colorAttrName);
/*
if (!result) {
req.clearState();
req.setStringFilter(rrs.PROPS.R_TAG, attrName);
req.setStringFilter(rrs.PROPS.R_VALUE, attribute);
result = req.searchRenderingAttribute(colorAttrName);
}
*/
return result;
}
}
private static class RouteBoundariesStatisticComputer extends RouteStatisticComputer<Boundaries> {
2019-03-07 18:02:41 +01:00
private final List<Incline> inclines;
2019-07-07 18:30:01 +02:00
public RouteBoundariesStatisticComputer(List<Incline> inclines, String attrName, String colorAttrName,
RenderingRulesStorage currentRenderer, RenderingRulesStorage defaultRenderer,
RenderingRuleSearchRequest currentSearchRequest, RenderingRuleSearchRequest defaultSearchRequest) {
super(null, attrName, colorAttrName, currentRenderer, defaultRenderer, currentSearchRequest, defaultSearchRequest);
2019-03-07 18:02:41 +01:00
this.inclines = inclines;
}
@Override
public List<RouteSegmentAttribute<Boundaries>> processRoute() {
List<RouteSegmentAttribute<Boundaries>> routeInclines = new ArrayList<>();
int index = 0;
Boundaries prev = null;
Incline prevIncline = null;
for (Incline incline : inclines) {
Boundaries current = incline.getBoundaries();
if (prev != null && !prev.equals(current)) {
index++;
}
if (index >= routeInclines.size()) {
String propertyName = getPropertyName(current);
int color = getColor(current);
RouteSegmentAttribute<Boundaries> attribute = new RouteSegmentAttribute<>(index, current, propertyName, color);
if (prevIncline != null) {
attribute.setInitDistance(prevIncline.getDistance());
}
routeInclines.add(attribute);
}
RouteSegmentAttribute routeIncline = routeInclines.get(index);
routeIncline.relativeSum(incline.getDistance());
prev = current;
prevIncline = incline;
}
return routeInclines;
}
@Override
public Boundaries getAttribute(RouteSegmentResult segment) {
return null;
}
@Override
public String getPropertyName(Boundaries attribute) {
int lowerBoundary = Math.round(attribute.getLowerBoundary());
int upperBoundary = Math.round(attribute.getUpperBoundary());
if (lowerBoundary >= Boundaries.MIN_DIVIDED_INCLINE) {
lowerBoundary++;
2019-03-07 18:02:41 +01:00
}
2019-07-07 18:30:01 +02:00
return String.format(Locale.US, "%d%% ... %d%%", lowerBoundary, upperBoundary);
2019-03-07 18:02:41 +01:00
}
@Override
public boolean searchRenderingAttribute(RenderingRulesStorage rrs, RenderingRuleSearchRequest req, Boundaries attribute) {
int lowerBoundary = Math.round(attribute.getLowerBoundary());
int upperBoundary = Math.round(attribute.getUpperBoundary());
StringBuilder range = new StringBuilder();
if (lowerBoundary >= Boundaries.MIN_DIVIDED_INCLINE) {
lowerBoundary++;
2019-03-15 23:18:36 +01:00
} else {
lowerBoundary = Boundaries.MIN_INCLINE;
}
if (upperBoundary > Boundaries.MAX_DIVIDED_INCLINE) {
upperBoundary = Boundaries.MAX_INCLINE;
}
range.append(lowerBoundary);
range.append(upperBoundary < 0 ? "_" : "-");
range.append(upperBoundary);
2019-07-07 18:30:01 +02:00
String additional = attrName + "=" + range;
req.setStringFilter(rrs.PROPS.R_ADDITIONAL, additional);
2019-07-07 18:30:01 +02:00
return req.searchRenderingAttribute(colorAttrName);
2019-03-07 18:02:41 +01:00
}
}
public static class RouteSegmentAttribute<E> {
private final int index;
private final E attribute;
private final int color;
private final String propertyName;
private float distance;
private float initDistance;
2019-07-07 18:30:01 +02:00
RouteSegmentAttribute(int index, E attribute, String propertyName, int color) {
2019-03-07 18:02:41 +01:00
this.index = index;
this.attribute = attribute;
this.propertyName = propertyName;
this.color = color;
}
2019-07-07 18:30:01 +02:00
RouteSegmentAttribute(RouteSegmentAttribute<E> segmentAttribute) {
2019-03-07 18:02:41 +01:00
this.index = segmentAttribute.getIndex();
this.attribute = segmentAttribute.getAttribute();
this.propertyName = segmentAttribute.getPropertyName();
this.color = segmentAttribute.getColor();
}
public int getIndex() {
return index;
}
public E getAttribute() {
return attribute;
}
public float getDistance() {
return distance;
}
public void setInitDistance(float initDistance) {
this.initDistance = initDistance;
}
public void incrementDistanceBy(float distance) {
this.distance += distance;
}
public void relativeSum(float distance) {
this.distance = this.distance + ((distance - this.initDistance) - this.distance);
}
public String getPropertyName() {
return propertyName;
}
public int getColor() {
return color;
}
@Override
public String toString() {
return "RouteSegmentAttribute{" +
"index=" + index +
", attribute='" + attribute + '\'' +
", color='" + color + '\'' +
", distance=" + distance +
'}';
2019-03-01 12:35:59 +01:00
}
}
2019-03-07 18:02:41 +01:00
public static class Incline {
private final float inclineValue;
2019-03-07 18:02:41 +01:00
private final float distance;
private Boundaries boundaries;
2019-03-07 18:02:41 +01:00
public Incline(float inclineValue, float distance) {
this.inclineValue = inclineValue;
this.distance = distance;
}
public void computeBoundaries(float minIncline, float maxIncline) {
this.boundaries = Boundaries.newBoundariesFor(inclineValue, minIncline, maxIncline);
2019-03-07 18:02:41 +01:00
}
public float getValue() {
return inclineValue;
}
public float getDistance() {
return distance;
}
public Boundaries getBoundaries() {
return this.boundaries;
}
@Override
public String toString() {
return "Incline{" +
", incline=" + inclineValue +
", distance=" + distance +
'}';
2019-03-01 12:35:59 +01:00
}
}
2019-03-07 18:02:41 +01:00
public static class Boundaries implements Comparable<Boundaries> {
private static final int MIN_INCLINE = -100;
private static final int MIN_DIVIDED_INCLINE = -20;
private static final int MAX_INCLINE = 100;
private static final int MAX_DIVIDED_INCLINE = 20;
private static final int STEP = 4;
private static final int NUM;
private static final int[] BOUNDARIES_ARRAY;
static {
NUM = ((MAX_DIVIDED_INCLINE - MIN_DIVIDED_INCLINE) / STEP) + 3;
BOUNDARIES_ARRAY = new int[NUM];
BOUNDARIES_ARRAY[0] = MIN_INCLINE;
for (int i = 1; i < NUM - 1; i++) {
BOUNDARIES_ARRAY[i] = MIN_DIVIDED_INCLINE + (i - 1) * STEP;
}
BOUNDARIES_ARRAY[NUM - 1] = MAX_INCLINE;
}
private final float upperBoundary;
private final float lowerBoundary;
private Boundaries(float upperBoundary, float lowerBoundary) {
this.upperBoundary = upperBoundary;
this.lowerBoundary = lowerBoundary;
}
public static Boundaries newBoundariesFor(float incline, float minIncline, float maxIncline) {
int maxRoundedIncline = Math.round(maxIncline);
int minRoundedIncline = Math.round(minIncline);
2019-03-07 18:02:41 +01:00
if (incline > MAX_INCLINE) {
return new Boundaries(MAX_INCLINE, MAX_DIVIDED_INCLINE);
2019-03-07 18:02:41 +01:00
}
if (incline < MIN_INCLINE) {
return new Boundaries(MIN_DIVIDED_INCLINE, MIN_INCLINE);
2019-03-07 18:02:41 +01:00
}
for (int i = 1; i < NUM; i++) {
if (incline >= BOUNDARIES_ARRAY[i - 1] && incline < BOUNDARIES_ARRAY[i]) {
if (i == 1) {
return new Boundaries(BOUNDARIES_ARRAY[i], minRoundedIncline);
} else if (i == NUM - 1) {
return new Boundaries(maxRoundedIncline, BOUNDARIES_ARRAY[i - 1]);
} else {
return new Boundaries(BOUNDARIES_ARRAY[i], BOUNDARIES_ARRAY[i - 1]);
}
2019-03-07 18:02:41 +01:00
}
}
return null;
}
2019-03-01 12:35:59 +01:00
2019-03-07 18:02:41 +01:00
public float getUpperBoundary() {
return upperBoundary;
}
public float getLowerBoundary() {
return lowerBoundary;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
Boundaries that = (Boundaries) o;
if (Float.compare(that.upperBoundary, upperBoundary) != 0) return false;
return Float.compare(that.lowerBoundary, lowerBoundary) == 0;
}
@Override
public int hashCode() {
int result = (upperBoundary != +0.0f ? Float.floatToIntBits(upperBoundary) : 0);
result = 31 * result + (lowerBoundary != +0.0f ? Float.floatToIntBits(lowerBoundary) : 0);
return result;
}
@Override
public int compareTo(Boundaries boundaries) {
return (int) (getLowerBoundary() - boundaries.getLowerBoundary());
}
@Override
public String toString() {
2019-07-07 18:30:01 +02:00
return String.format(Locale.US, "%d%% ... %d%%", Math.round(getLowerBoundary()), Math.round(getUpperBoundary()));
2019-03-07 18:02:41 +01:00
}
}
2019-07-07 18:30:01 +02:00
public static class RouteStatistics<E> {
2019-03-07 18:02:41 +01:00
private final List<RouteSegmentAttribute<E>> elements;
private final Map<E, RouteSegmentAttribute<E>> partition;
private final float totalDistance;
2019-07-07 18:30:01 +02:00
private RouteStatistics(List<RouteSegmentAttribute<E>> elements,
Map<E, RouteSegmentAttribute<E>> partition,
float totalDistance) {
2019-03-07 18:02:41 +01:00
this.elements = elements;
this.partition = partition;
this.totalDistance = totalDistance;
}
2019-03-01 12:35:59 +01:00
2019-03-07 18:02:41 +01:00
public float getTotalDistance() {
return totalDistance;
2019-03-01 12:35:59 +01:00
}
2019-03-07 18:02:41 +01:00
public List<RouteSegmentAttribute<E>> getElements() {
return elements;
2019-03-01 12:35:59 +01:00
}
2019-03-07 18:02:41 +01:00
public Map<E, RouteSegmentAttribute<E>> getPartition() {
return partition;
2019-03-01 12:35:59 +01:00
}
}
2019-03-07 18:02:41 +01:00
}