com.graphhopper.reader.osm.OSMTurnRelation Maven / Gradle / Ivy
/*
* Licensed to GraphHopper GmbH under one or more contributor
* license agreements. See the NOTICE file distributed with this work for
* additional information regarding copyright ownership.
*
* GraphHopper GmbH licenses this file to you under the Apache License,
* Version 2.0 (the "License"); you may not use this file except in
* compliance with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.graphhopper.reader.osm;
import com.graphhopper.routing.util.TurnCostEncoder;
import com.graphhopper.util.EdgeExplorer;
import com.graphhopper.util.EdgeIterator;
import java.util.*;
/**
* Helper object which gives node cost entries for a given OSM-relation of type "restriction"
*
*
* @author Karl Hübner
*/
public class OSMTurnRelation {
private final long fromOsmWayId;
private final long viaOsmNodeId;
private final long toOsmWayId;
private final Type restriction;
OSMTurnRelation(long fromWayID, long viaNodeID, long toWayID, Type restrictionType) {
this.fromOsmWayId = fromWayID;
this.viaOsmNodeId = viaNodeID;
this.toOsmWayId = toWayID;
this.restriction = restrictionType;
}
long getOsmIdFrom() {
return fromOsmWayId;
}
long getOsmIdTo() {
return toOsmWayId;
}
/**
* Transforms this relation into a collection of turn cost entries
*
*
* @param edgeOutExplorer an edge filter which only allows outgoing edges
* @param edgeInExplorer an edge filter which only allows incoming edges
* @return a collection of node cost entries which can be added to the graph later
*/
public Collection getRestrictionAsEntries(TurnCostEncoder encoder,
EdgeExplorer edgeOutExplorer, EdgeExplorer edgeInExplorer, OSMReader osmReader) {
int nodeVia = osmReader.getInternalNodeIdOfOsmNode(this.viaOsmNodeId);
try {
// street with restriction was not included (access or tag limits etc)
if (nodeVia == OSMReader.EMPTY_NODE)
return Collections.emptyList();
int edgeIdFrom = EdgeIterator.NO_EDGE;
// get all incoming edges and receive the edge which is defined by fromOsm
EdgeIterator iter = edgeInExplorer.setBaseNode(nodeVia);
while (iter.next()) {
if (osmReader.getOsmIdOfInternalEdge(iter.getEdge()) == this.fromOsmWayId) {
edgeIdFrom = iter.getEdge();
break;
}
}
if (edgeIdFrom == EdgeIterator.NO_EDGE)
return Collections.emptyList();
final Collection entries = new ArrayList<>();
// get all outgoing edges of the via node
iter = edgeOutExplorer.setBaseNode(nodeVia);
// for TYPE_ONLY_* we add ALL restrictions (from, via, * ) EXCEPT the given turn
// for TYPE_NOT_* we add ONE restriction (from, via, to)
while (iter.next()) {
int edgeId = iter.getEdge();
long wayId = osmReader.getOsmIdOfInternalEdge(edgeId);
if (edgeId != edgeIdFrom && this.restriction == Type.ONLY && wayId != this.toOsmWayId
|| this.restriction == Type.NOT && wayId == this.toOsmWayId && wayId >= 0) {
final TurnCostTableEntry entry = new TurnCostTableEntry();
entry.nodeVia = nodeVia;
entry.edgeFrom = edgeIdFrom;
entry.edgeTo = iter.getEdge();
entry.flags = encoder.getTurnFlags(true, 0);
entries.add(entry);
if (this.restriction == Type.NOT)
break;
}
}
return entries;
} catch (Exception e) {
throw new IllegalStateException("Could not built turn table entry for relation of node with osmId:" + this.viaOsmNodeId, e);
}
}
@Override
public String toString() {
return "*-(" + fromOsmWayId + ")->" + viaOsmNodeId + "-(" + toOsmWayId + ")->*";
}
enum Type {
UNSUPPORTED, NOT, ONLY;
private static final Map tags = new HashMap<>();
static {
tags.put("no_left_turn", NOT);
tags.put("no_right_turn", NOT);
tags.put("no_straight_on", NOT);
tags.put("no_u_turn", NOT);
tags.put("only_right_turn", ONLY);
tags.put("only_left_turn", ONLY);
tags.put("only_straight_on", ONLY);
}
public static Type getRestrictionType(String tag) {
Type result = null;
if (tag != null) {
result = tags.get(tag);
}
return (result != null) ? result : UNSUPPORTED;
}
}
/**
* Helper class to processing purposes only
*/
public static class TurnCostTableEntry {
public int edgeFrom;
public int nodeVia;
public int edgeTo;
public long flags;
/**
* @return an unique id (edgeFrom, edgeTo) to avoid duplicate entries if multiple encoders
* are involved.
*/
public long getItemId() {
return ((long) edgeFrom) << 32 | ((long) edgeTo);
}
@Override
public String toString() {
return "*-(" + edgeFrom + ")->" + nodeVia + "-(" + edgeTo + ")->*";
}
}
}