2013-04-18 23:35:02 +02:00
package net.osmand.router ;
2017-02-10 16:55:56 +01:00
import java.io.FileWriter ;
2013-04-18 23:35:02 +02:00
import java.io.IOException ;
2019-07-08 00:37:20 +02:00
import java.io.InputStream ;
2013-04-18 23:35:02 +02:00
import java.text.MessageFormat ;
import java.util.ArrayList ;
import java.util.Arrays ;
import java.util.Collections ;
import java.util.Iterator ;
2019-07-08 00:37:20 +02:00
import java.util.LinkedHashMap ;
2013-04-18 23:35:02 +02:00
import java.util.List ;
2019-07-08 00:37:20 +02:00
import java.util.Map ;
2013-04-18 23:35:02 +02:00
2019-07-08 00:37:20 +02:00
import org.apache.commons.logging.Log ;
import org.xmlpull.v1.XmlPullParser ;
import org.xmlpull.v1.XmlPullParserException ;
import gnu.trove.iterator.TIntIterator ;
import gnu.trove.list.array.TIntArrayList ;
import gnu.trove.set.hash.TIntHashSet ;
2014-09-24 22:32:37 +02:00
import net.osmand.PlatformUtil ;
2013-04-18 23:35:02 +02:00
import net.osmand.binary.BinaryMapIndexReader ;
2014-03-01 12:30:25 +01:00
import net.osmand.binary.BinaryMapRouteReaderAdapter.RouteTypeRule ;
2013-04-18 23:35:02 +02:00
import net.osmand.binary.RouteDataObject ;
import net.osmand.data.LatLon ;
2016-11-07 00:59:59 +01:00
import net.osmand.osm.MapRenderingTypes ;
2019-07-08 00:37:20 +02:00
import net.osmand.render.RenderingRuleSearchRequest ;
import net.osmand.render.RenderingRulesStorage ;
import net.osmand.render.RenderingRulesStorage.RenderingRulesStorageResolver ;
2013-04-18 23:35:02 +02:00
import net.osmand.router.BinaryRoutePlanner.FinalRouteSegment ;
import net.osmand.router.BinaryRoutePlanner.RouteSegment ;
2017-11-27 21:04:17 +01:00
import net.osmand.router.GeneralRouter.GeneralRouterProfile ;
2018-06-16 11:19:21 +02:00
import net.osmand.router.RoutePlannerFrontEnd.RouteCalculationMode ;
2019-07-08 00:37:20 +02:00
import net.osmand.router.RouteStatisticsHelper.RouteStatistics ;
2016-04-27 01:05:42 +02:00
import net.osmand.util.Algorithms ;
2018-07-31 21:41:38 +02:00
import net.osmand.util.MapAlgorithms ;
2013-04-18 23:35:02 +02:00
import net.osmand.util.MapUtils ;
public class RouteResultPreparation {
2014-02-01 02:06:36 +01:00
public static boolean PRINT_TO_CONSOLE_ROUTE_INFORMATION_TO_TEST = false ;
2017-02-10 16:55:56 +01:00
public static String PRINT_TO_GPX_FILE = null ;
2015-09-13 10:58:27 +02:00
private static final float TURN_DEGREE_MIN = 45 ;
2018-06-16 11:19:21 +02:00
public static final int SHIFT_ID = 6 ;
2014-09-24 22:32:37 +02:00
private Log log = PlatformUtil . getLog ( RouteResultPreparation . class ) ;
2013-04-18 23:35:02 +02:00
/ * *
* Helper method to prepare final result
* /
2014-01-19 21:29:11 +01:00
List < RouteSegmentResult > prepareResult ( RoutingContext ctx , FinalRouteSegment finalSegment ) throws IOException {
2013-04-18 23:35:02 +02:00
List < RouteSegmentResult > result = convertFinalSegmentToResults ( ctx , finalSegment ) ;
2014-01-19 21:29:11 +01:00
prepareResult ( ctx , result ) ;
2013-04-18 23:35:02 +02:00
return result ;
}
2018-07-31 21:41:38 +02:00
private static class CombineAreaRoutePoint {
int x31 ;
int y31 ;
int originalIndex ;
}
private void combineWayPointsForAreaRouting ( RoutingContext ctx , List < RouteSegmentResult > result ) {
for ( int i = 0 ; i < result . size ( ) ; i + + ) {
RouteSegmentResult rsr = result . get ( i ) ;
RouteDataObject obj = rsr . getObject ( ) ;
boolean area = false ;
if ( obj . getPoint31XTile ( 0 ) = = obj . getPoint31XTile ( obj . getPointsLength ( ) - 1 ) & &
obj . getPoint31YTile ( 0 ) = = obj . getPoint31YTile ( obj . getPointsLength ( ) - 1 ) ) {
area = true ;
}
if ( ! area | | ! ctx . getRouter ( ) . isArea ( obj ) ) {
continue ;
}
List < CombineAreaRoutePoint > originalWay = new ArrayList < CombineAreaRoutePoint > ( ) ;
List < CombineAreaRoutePoint > routeWay = new ArrayList < CombineAreaRoutePoint > ( ) ;
for ( int j = 0 ; j < obj . getPointsLength ( ) ; j + + ) {
CombineAreaRoutePoint pnt = new CombineAreaRoutePoint ( ) ;
pnt . x31 = obj . getPoint31XTile ( j ) ;
pnt . y31 = obj . getPoint31YTile ( j ) ;
pnt . originalIndex = j ;
originalWay . add ( pnt ) ;
if ( j > = rsr . getStartPointIndex ( ) & & j < = rsr . getEndPointIndex ( ) ) {
routeWay . add ( pnt ) ;
} else if ( j < = rsr . getStartPointIndex ( ) & & j > = rsr . getEndPointIndex ( ) ) {
routeWay . add ( 0 , pnt ) ;
}
}
int originalSize = routeWay . size ( ) ;
simplifyAreaRouteWay ( routeWay , originalWay ) ;
int newsize = routeWay . size ( ) ;
if ( routeWay . size ( ) ! = originalSize ) {
RouteDataObject nobj = new RouteDataObject ( obj ) ;
nobj . pointsX = new int [ newsize ] ;
nobj . pointsY = new int [ newsize ] ;
for ( int k = 0 ; k < newsize ; k + + ) {
nobj . pointsX [ k ] = routeWay . get ( k ) . x31 ;
nobj . pointsY [ k ] = routeWay . get ( k ) . y31 ;
}
// in future point names might be used
nobj . restrictions = null ;
nobj . restrictionsVia = null ;
nobj . pointTypes = null ;
nobj . pointNames = null ;
nobj . pointNameTypes = null ;
RouteSegmentResult nrsr = new RouteSegmentResult ( nobj , 0 , newsize - 1 ) ;
result . set ( i , nrsr ) ;
}
}
}
private void simplifyAreaRouteWay ( List < CombineAreaRoutePoint > routeWay , List < CombineAreaRoutePoint > originalWay ) {
boolean changed = true ;
while ( changed ) {
changed = false ;
int connectStart = - 1 ;
int connectLen = 0 ;
double dist = 0 ;
int length = routeWay . size ( ) - 1 ;
while ( length > 0 & & connectLen = = 0 ) {
for ( int i = 0 ; i < routeWay . size ( ) - length ; i + + ) {
CombineAreaRoutePoint p = routeWay . get ( i ) ;
CombineAreaRoutePoint n = routeWay . get ( i + length ) ;
if ( segmentLineBelongsToPolygon ( p , n , originalWay ) ) {
double ndist = BinaryRoutePlanner . squareRootDist ( p . x31 , p . y31 , n . x31 , n . y31 ) ;
if ( ndist > dist ) {
ndist = dist ;
connectStart = i ;
connectLen = length ;
}
}
}
length - - ;
}
while ( connectLen > 1 ) {
routeWay . remove ( connectStart + 1 ) ;
connectLen - - ;
changed = true ;
}
}
}
private boolean segmentLineBelongsToPolygon ( CombineAreaRoutePoint p , CombineAreaRoutePoint n ,
List < CombineAreaRoutePoint > originalWay ) {
int intersections = 0 ;
int mx = p . x31 / 2 + n . x31 / 2 ;
int my = p . y31 / 2 + n . y31 / 2 ;
for ( int i = 1 ; i < originalWay . size ( ) ; i + + ) {
CombineAreaRoutePoint p2 = originalWay . get ( i - 1 ) ;
CombineAreaRoutePoint n2 = originalWay . get ( i ) ;
if ( p . originalIndex ! = i & & p . originalIndex ! = i - 1 ) {
if ( n . originalIndex ! = i & & n . originalIndex ! = i - 1 ) {
if ( MapAlgorithms . linesIntersect ( p . x31 , p . y31 , n . x31 , n . y31 , p2 . x31 , p2 . y31 , n2 . x31 , n2 . y31 ) ) {
return false ;
}
}
}
int fx = MapAlgorithms . ray_intersect_x ( p2 . x31 , p2 . y31 , n2 . x31 , n2 . y31 , my ) ;
if ( Integer . MIN_VALUE ! = fx & & mx > = fx ) {
intersections + + ;
}
}
return intersections % 2 = = 1 ;
}
2013-04-18 23:35:02 +02:00
2014-01-19 21:29:11 +01:00
List < RouteSegmentResult > prepareResult ( RoutingContext ctx , List < RouteSegmentResult > result ) throws IOException {
2019-04-01 17:11:01 +02:00
for ( int i = 0 ; i < result . size ( ) ; i + + ) {
checkAndInitRouteRegion ( ctx , result . get ( i ) . getObject ( ) ) ;
}
2018-12-14 12:06:59 +01:00
combineWayPointsForAreaRouting ( ctx , result ) ;
2013-04-18 23:35:02 +02:00
validateAllPointsConnected ( result ) ;
splitRoadsAndAttachRoadSegments ( ctx , result ) ;
calculateTimeSpeed ( ctx , result ) ;
2015-01-13 10:25:56 +01:00
2015-01-15 02:42:33 +01:00
for ( int i = 0 ; i < result . size ( ) ; i + + ) {
TurnType turnType = getTurnInfo ( result , i , ctx . leftSideNavigation ) ;
result . get ( i ) . setTurnType ( turnType ) ;
}
2015-01-13 10:25:56 +01:00
determineTurnsToMerge ( ctx . leftSideNavigation , result ) ;
2016-06-23 19:44:36 +02:00
ignorePrecedingStraightsOnSameIntersection ( ctx . leftSideNavigation , result ) ;
2015-01-15 02:42:33 +01:00
justifyUTurns ( ctx . leftSideNavigation , result ) ;
2015-01-13 10:25:56 +01:00
addTurnInfoDescriptions ( result ) ;
2013-04-18 23:35:02 +02:00
return result ;
}
2016-06-23 19:44:36 +02:00
protected void ignorePrecedingStraightsOnSameIntersection ( boolean leftside , List < RouteSegmentResult > result ) {
//Issue 2571: Ignore TurnType.C if immediately followed by another turn in non-motorway cases, as these likely belong to the very same intersection
RouteSegmentResult nextSegment = null ;
double distanceToNextTurn = 999999 ;
for ( int i = result . size ( ) - 1 ; i > = 0 ; i - - ) {
// Mark next "real" turn
if ( nextSegment ! = null & & nextSegment . getTurnType ( ) ! = null & &
nextSegment . getTurnType ( ) . getValue ( ) ! = TurnType . C & &
! isMotorway ( nextSegment ) ) {
if ( distanceToNextTurn = = 999999 ) {
distanceToNextTurn = 0 ;
}
}
RouteSegmentResult currentSegment = result . get ( i ) ;
// Identify preceding goStraights within distance limit and suppress
if ( currentSegment ! = null ) {
distanceToNextTurn + = currentSegment . getDistance ( ) ;
if ( currentSegment . getTurnType ( ) ! = null & &
currentSegment . getTurnType ( ) . getValue ( ) = = TurnType . C & &
distanceToNextTurn < = 100 ) {
result . get ( i ) . getTurnType ( ) . setSkipToSpeak ( true ) ;
} else {
nextSegment = currentSegment ;
distanceToNextTurn = 999999 ;
}
}
}
}
2015-01-15 02:42:33 +01:00
private void justifyUTurns ( boolean leftSide , List < RouteSegmentResult > result ) {
int next ;
for ( int i = 0 ; i < result . size ( ) - 1 ; i = next ) {
next = i + 1 ;
TurnType t = result . get ( i ) . getTurnType ( ) ;
// justify turn
if ( t ! = null ) {
TurnType jt = justifyUTurn ( leftSide , result , i , t ) ;
if ( jt ! = null ) {
result . get ( i ) . setTurnType ( jt ) ;
next = i + 2 ;
}
}
}
}
2019-06-10 11:58:14 +02:00
// decrease speed proportionally from 15ms (50kmh)
private static final double SLOW_DOWN_SPEED_THRESHOLD = 15 ;
// reference speed 30ms (108kmh) - 2ms (7kmh)
private static final double SLOW_DOWN_SPEED = 2 ;
2013-04-18 23:35:02 +02:00
private void calculateTimeSpeed ( RoutingContext ctx , List < RouteSegmentResult > result ) throws IOException {
2017-11-27 21:04:17 +01:00
//for Naismith
boolean usePedestrianHeight = ( ( ( ( GeneralRouter ) ctx . getRouter ( ) ) . getProfile ( ) = = GeneralRouterProfile . PEDESTRIAN ) & & ( ( GeneralRouter ) ctx . getRouter ( ) ) . getHeightObstacles ( ) ) ;
2013-04-18 23:35:02 +02:00
for ( int i = 0 ; i < result . size ( ) ; i + + ) {
RouteSegmentResult rr = result . get ( i ) ;
RouteDataObject road = rr . getObject ( ) ;
double distOnRoadToPass = 0 ;
2014-02-02 14:06:09 +01:00
double speed = ctx . getRouter ( ) . defineVehicleSpeed ( road ) ;
2013-04-18 23:35:02 +02:00
if ( speed = = 0 ) {
2019-06-24 14:03:53 +02:00
speed = ctx . getRouter ( ) . getDefaultSpeed ( ) ;
2013-11-06 23:04:55 +01:00
} else {
2019-06-10 11:58:14 +02:00
if ( speed > SLOW_DOWN_SPEED_THRESHOLD ) {
speed = speed - ( speed / SLOW_DOWN_SPEED_THRESHOLD - 1 ) * SLOW_DOWN_SPEED ;
2013-11-06 23:04:55 +01:00
}
2013-04-18 23:35:02 +02:00
}
boolean plus = rr . getStartPointIndex ( ) < rr . getEndPointIndex ( ) ;
int next ;
double distance = 0 ;
2017-11-27 21:04:17 +01:00
//for Naismith
float prevHeight = - 99999 . 0f ;
float [ ] heightDistanceArray = null ;
if ( usePedestrianHeight ) {
road . calculateHeightArray ( ) ;
heightDistanceArray = road . heightDistanceArray ;
}
2013-04-18 23:35:02 +02:00
for ( int j = rr . getStartPointIndex ( ) ; j ! = rr . getEndPointIndex ( ) ; j = next ) {
next = plus ? j + 1 : j - 1 ;
double d = measuredDist ( road . getPoint31XTile ( j ) , road . getPoint31YTile ( j ) , road . getPoint31XTile ( next ) ,
road . getPoint31YTile ( next ) ) ;
distance + = d ;
double obstacle = ctx . getRouter ( ) . defineObstacle ( road , j ) ;
if ( obstacle < 0 ) {
obstacle = 0 ;
}
2017-11-27 21:04:17 +01:00
distOnRoadToPass + = d / speed + obstacle ; //this is time in seconds
2013-04-18 23:35:02 +02:00
2017-11-27 21:04:17 +01:00
//for Naismith
if ( usePedestrianHeight ) {
int heightIndex = 2 * j + 1 ;
if ( heightDistanceArray ! = null & & heightIndex < heightDistanceArray . length ) {
float height = heightDistanceArray [ heightIndex ] ;
if ( prevHeight ! = - 99999 . 0f ) {
float heightDiff = height - prevHeight ;
if ( heightDiff > 0 ) { //ascent only
distOnRoadToPass + = heightDiff * 6 . 0f ; //Naismith's rule: add 1 hour per every 600m of ascent
}
}
prevHeight = height ;
}
}
2013-04-18 23:35:02 +02:00
}
2017-11-27 21:04:17 +01:00
2013-04-18 23:35:02 +02:00
// last point turn time can be added
// if(i + 1 < result.size()) { distOnRoadToPass += ctx.getRouter().calculateTurnTime(); }
rr . setSegmentTime ( ( float ) distOnRoadToPass ) ;
rr . setSegmentSpeed ( ( float ) speed ) ;
rr . setDistance ( ( float ) distance ) ;
}
}
private void splitRoadsAndAttachRoadSegments ( RoutingContext ctx , List < RouteSegmentResult > result ) throws IOException {
for ( int i = 0 ; i < result . size ( ) ; i + + ) {
if ( ctx . checkIfMemoryLimitCritical ( ctx . config . memoryLimitation ) ) {
ctx . unloadUnusedTiles ( ctx . config . memoryLimitation ) ;
}
RouteSegmentResult rr = result . get ( i ) ;
boolean plus = rr . getStartPointIndex ( ) < rr . getEndPointIndex ( ) ;
int next ;
for ( int j = rr . getStartPointIndex ( ) ; j ! = rr . getEndPointIndex ( ) ; j = next ) {
next = plus ? j + 1 : j - 1 ;
if ( j = = rr . getStartPointIndex ( ) ) {
attachRoadSegments ( ctx , result , i , j , plus ) ;
}
if ( next ! = rr . getEndPointIndex ( ) ) {
attachRoadSegments ( ctx , result , i , next , plus ) ;
}
List < RouteSegmentResult > attachedRoutes = rr . getAttachedRoutes ( next ) ;
2013-11-26 16:23:05 +01:00
boolean tryToSplit = next ! = rr . getEndPointIndex ( ) & & ! rr . getObject ( ) . roundabout ( ) & & attachedRoutes ! = null ;
if ( rr . getDistance ( next , plus ) = = 0 ) {
// same point will be processed next step
tryToSplit = false ;
}
if ( tryToSplit ) {
// avoid small zigzags
2013-04-18 23:35:02 +02:00
float before = rr . getBearing ( next , ! plus ) ;
float after = rr . getBearing ( next , plus ) ;
2013-11-26 16:23:05 +01:00
if ( rr . getDistance ( next , plus ) < 5 ) {
after = before + 180 ;
} else if ( rr . getDistance ( next , ! plus ) < 5 ) {
before = after - 180 ;
}
2013-04-18 23:35:02 +02:00
boolean straight = Math . abs ( MapUtils . degreesDiff ( before + 180 , after ) ) < TURN_DEGREE_MIN ;
boolean isSplit = false ;
// split if needed
for ( RouteSegmentResult rs : attachedRoutes ) {
double diff = MapUtils . degreesDiff ( before + 180 , rs . getBearingBegin ( ) ) ;
if ( Math . abs ( diff ) < = TURN_DEGREE_MIN ) {
isSplit = true ;
} else if ( ! straight & & Math . abs ( diff ) < 100 ) {
isSplit = true ;
}
}
if ( isSplit ) {
int endPointIndex = rr . getEndPointIndex ( ) ;
RouteSegmentResult split = new RouteSegmentResult ( rr . getObject ( ) , next , endPointIndex ) ;
split . copyPreattachedRoutes ( rr , Math . abs ( next - rr . getStartPointIndex ( ) ) ) ;
rr . setEndPointIndex ( next ) ;
result . add ( i + 1 , split ) ;
i + + ;
// switch current segment to the splitted
rr = split ;
}
}
}
}
}
private void checkAndInitRouteRegion ( RoutingContext ctx , RouteDataObject road ) throws IOException {
BinaryMapIndexReader reader = ctx . reverseMap . get ( road . region ) ;
if ( reader ! = null ) {
reader . initRouteRegion ( road . region ) ;
}
}
private void validateAllPointsConnected ( List < RouteSegmentResult > result ) {
for ( int i = 1 ; i < result . size ( ) ; i + + ) {
RouteSegmentResult rr = result . get ( i ) ;
RouteSegmentResult pr = result . get ( i - 1 ) ;
double d = MapUtils . getDistance ( pr . getPoint ( pr . getEndPointIndex ( ) ) , rr . getPoint ( rr . getStartPointIndex ( ) ) ) ;
if ( d > 0 ) {
System . err . println ( " Points are not connected : " + pr . getObject ( ) + " ( " + pr . getEndPointIndex ( ) + " ) -> " + rr . getObject ( )
+ " ( " + rr . getStartPointIndex ( ) + " ) " + d + " meters " ) ;
}
}
}
private List < RouteSegmentResult > convertFinalSegmentToResults ( RoutingContext ctx , FinalRouteSegment finalSegment ) {
List < RouteSegmentResult > result = new ArrayList < RouteSegmentResult > ( ) ;
if ( finalSegment ! = null ) {
ctx . routingTime = finalSegment . distanceFromStart ;
println ( " Routing calculated time distance " + finalSegment . distanceFromStart ) ;
// Get results from opposite direction roads
2014-01-31 01:43:52 +01:00
RouteSegment segment = finalSegment . reverseWaySearch ? finalSegment :
finalSegment . opposite . getParentRoute ( ) ;
int parentSegmentStart = finalSegment . reverseWaySearch ? finalSegment . opposite . getSegmentStart ( ) :
finalSegment . opposite . getParentSegmentEnd ( ) ;
2014-01-28 21:03:25 +01:00
float parentRoutingTime = - 1 ;
2013-04-18 23:35:02 +02:00
while ( segment ! = null ) {
RouteSegmentResult res = new RouteSegmentResult ( segment . road , parentSegmentStart , segment . getSegmentStart ( ) ) ;
2014-01-28 21:03:25 +01:00
parentRoutingTime = calcRoutingTime ( parentRoutingTime , finalSegment , segment , res ) ;
2013-04-18 23:35:02 +02:00
parentSegmentStart = segment . getParentSegmentEnd ( ) ;
segment = segment . getParentRoute ( ) ;
2014-01-19 23:58:17 +01:00
addRouteSegmentToResult ( ctx , result , res , false ) ;
2013-04-18 23:35:02 +02:00
}
// reverse it just to attach good direction roads
Collections . reverse ( result ) ;
segment = finalSegment . reverseWaySearch ? finalSegment . opposite . getParentRoute ( ) : finalSegment ;
int parentSegmentEnd = finalSegment . reverseWaySearch ? finalSegment . opposite . getParentSegmentEnd ( ) : finalSegment . opposite . getSegmentStart ( ) ;
2014-01-28 21:03:25 +01:00
parentRoutingTime = - 1 ;
2013-04-18 23:35:02 +02:00
while ( segment ! = null ) {
RouteSegmentResult res = new RouteSegmentResult ( segment . road , segment . getSegmentStart ( ) , parentSegmentEnd ) ;
2014-01-28 21:03:25 +01:00
parentRoutingTime = calcRoutingTime ( parentRoutingTime , finalSegment , segment , res ) ;
2013-04-18 23:35:02 +02:00
parentSegmentEnd = segment . getParentSegmentEnd ( ) ;
segment = segment . getParentRoute ( ) ;
// happens in smart recalculation
2014-01-19 23:58:17 +01:00
addRouteSegmentToResult ( ctx , result , res , true ) ;
2013-04-18 23:35:02 +02:00
}
Collections . reverse ( result ) ;
2014-01-28 21:03:25 +01:00
// checkTotalRoutingTime(result);
2013-04-18 23:35:02 +02:00
}
return result ;
}
2014-01-28 21:03:25 +01:00
protected void checkTotalRoutingTime ( List < RouteSegmentResult > result ) {
float totalRoutingTime = 0 ;
for ( RouteSegmentResult r : result ) {
totalRoutingTime + = r . getRoutingTime ( ) ;
}
println ( " Total routing time ! " + totalRoutingTime ) ;
}
private float calcRoutingTime ( float parentRoutingTime , RouteSegment finalSegment , RouteSegment segment ,
RouteSegmentResult res ) {
if ( segment ! = finalSegment ) {
if ( parentRoutingTime ! = - 1 ) {
res . setRoutingTime ( parentRoutingTime - segment . distanceFromStart ) ;
}
parentRoutingTime = segment . distanceFromStart ;
}
return parentRoutingTime ;
}
2013-04-18 23:35:02 +02:00
2014-01-19 23:58:17 +01:00
private void addRouteSegmentToResult ( RoutingContext ctx , List < RouteSegmentResult > result , RouteSegmentResult res , boolean reverse ) {
2013-04-18 23:35:02 +02:00
if ( res . getStartPointIndex ( ) ! = res . getEndPointIndex ( ) ) {
if ( result . size ( ) > 0 ) {
RouteSegmentResult last = result . get ( result . size ( ) - 1 ) ;
2014-01-19 23:58:17 +01:00
if ( last . getObject ( ) . id = = res . getObject ( ) . id & & ctx . calculationMode ! = RouteCalculationMode . BASE ) {
2013-04-18 23:35:02 +02:00
if ( combineTwoSegmentResult ( res , last , reverse ) ) {
return ;
}
}
}
result . add ( res ) ;
}
}
private boolean combineTwoSegmentResult ( RouteSegmentResult toAdd , RouteSegmentResult previous ,
boolean reverse ) {
boolean ld = previous . getEndPointIndex ( ) > previous . getStartPointIndex ( ) ;
boolean rd = toAdd . getEndPointIndex ( ) > toAdd . getStartPointIndex ( ) ;
if ( rd = = ld ) {
if ( toAdd . getStartPointIndex ( ) = = previous . getEndPointIndex ( ) & & ! reverse ) {
previous . setEndPointIndex ( toAdd . getEndPointIndex ( ) ) ;
2014-01-28 21:03:25 +01:00
previous . setRoutingTime ( previous . getRoutingTime ( ) + toAdd . getRoutingTime ( ) ) ;
2013-04-18 23:35:02 +02:00
return true ;
} else if ( toAdd . getEndPointIndex ( ) = = previous . getStartPointIndex ( ) & & reverse ) {
previous . setStartPointIndex ( toAdd . getStartPointIndex ( ) ) ;
2014-01-28 21:03:25 +01:00
previous . setRoutingTime ( previous . getRoutingTime ( ) + toAdd . getRoutingTime ( ) ) ;
2013-04-18 23:35:02 +02:00
return true ;
}
}
return false ;
}
void printResults ( RoutingContext ctx , LatLon start , LatLon end , List < RouteSegmentResult > result ) {
float completeTime = 0 ;
float completeDistance = 0 ;
for ( RouteSegmentResult r : result ) {
completeTime + = r . getSegmentTime ( ) ;
completeDistance + = r . getDistance ( ) ;
}
println ( " ROUTE : " ) ;
double startLat = start . getLatitude ( ) ;
double startLon = start . getLongitude ( ) ;
double endLat = end . getLatitude ( ) ;
double endLon = end . getLongitude ( ) ;
2018-10-17 02:03:44 +02:00
String msg = String . format ( " <test regions= \" \" description= \" \" best_percent= \" \" vehicle= \" %s \" \ n "
+ " start_lat= \" %.5f \" start_lon= \" %.5f \" target_lat= \" %.5f \" target_lon= \" %.5f \" "
+ " routing_time= \" %.2f \" loadedTiles= \" %d \" visitedSegments= \" %d \" complete_distance= \" %.2f \" complete_time= \" %.2f \" > " ,
ctx . config . routerName , startLat , startLon , endLat , endLon , ctx . routingTime , ctx . loadedTiles ,
ctx . visitedSegments , completeDistance , completeTime ) ;
// String msg = MessageFormat.format("<test regions=\"\" description=\"\" best_percent=\"\" vehicle=\"{4}\" \n"
// + " start_lat=\"{0}\" start_lon=\"{1}\" target_lat=\"{2}\" target_lon=\"{3}\" {5} >",
// startLat + "", startLon + "", endLat + "", endLon + "", ctx.config.routerName,
// "loadedTiles = \"" + ctx.loadedTiles + "\" " + "visitedSegments = \"" + ctx.visitedSegments + "\" " +
// "complete_distance = \"" + completeDistance + "\" " + "complete_time = \"" + completeTime + "\" " +
// "routing_time = \"" + ctx.routingTime + "\" ");
2014-09-24 22:32:37 +02:00
log . info ( msg ) ;
println ( msg ) ;
2013-04-18 23:35:02 +02:00
if ( PRINT_TO_CONSOLE_ROUTE_INFORMATION_TO_TEST ) {
2017-02-10 16:55:56 +01:00
org . xmlpull . v1 . XmlSerializer serializer = null ;
if ( PRINT_TO_GPX_FILE ! = null ) {
serializer = PlatformUtil . newSerializer ( ) ;
try {
serializer . setOutput ( new FileWriter ( PRINT_TO_GPX_FILE ) ) ;
2017-02-13 19:13:42 +01:00
serializer . setFeature ( " http://xmlpull.org/v1/doc/features.html#indent-output " , true ) ;
// indentation as 3 spaces
// serializer.setProperty("http://xmlpull.org/v1/doc/properties.html#serializer-indentation", " ");
// // also set the line separator
// serializer.setProperty("http://xmlpull.org/v1/doc/properties.html#serializer-line-separator", "\n");
2017-02-10 16:55:56 +01:00
serializer . startDocument ( " UTF-8 " , true ) ;
serializer . startTag ( " " , " gpx " ) ;
serializer . attribute ( " " , " version " , " 1.1 " ) ;
serializer . attribute ( " " , " xmlns " , " http://www.topografix.com/GPX/1/1 " ) ;
serializer . attribute ( " " , " xmlns:xsi " , " http://www.w3.org/2001/XMLSchema-instance " ) ;
serializer . attribute ( " " , " xmlns:schemaLocation " , " http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd " ) ;
serializer . startTag ( " " , " trk " ) ;
serializer . startTag ( " " , " trkseg " ) ;
} catch ( IOException e ) {
e . printStackTrace ( ) ;
serializer = null ;
}
}
2017-02-23 21:43:30 +01:00
double lastHeight = - 180 ;
2013-04-18 23:35:02 +02:00
for ( RouteSegmentResult res : result ) {
String name = res . getObject ( ) . getName ( ) ;
2016-12-04 11:42:33 +01:00
String ref = res . getObject ( ) . getRef ( " " , false , res . isForwardDirection ( ) ) ;
2013-04-18 23:35:02 +02:00
if ( name = = null ) {
name = " " ;
}
if ( ref ! = null ) {
name + = " ( " + ref + " ) " ;
}
StringBuilder additional = new StringBuilder ( ) ;
additional . append ( " time = \" " ) . append ( res . getSegmentTime ( ) ) . append ( " \" " ) ;
2019-07-10 15:24:23 +02:00
if ( res . getRoutingTime ( ) > 0 ) {
additional . append ( " rspeed = \" " )
. append ( ( int ) Math . round ( res . getDistance ( ) / res . getRoutingTime ( ) * 3 . 6 ) ) . append ( " \" " ) ;
}
// additional.append("rtime = \"").append(res.getRoutingTime()).append("\" ");
2013-04-18 23:35:02 +02:00
additional . append ( " name = \" " ) . append ( name ) . append ( " \" " ) ;
// float ms = res.getSegmentSpeed();
2015-08-31 00:11:48 +02:00
float ms = res . getObject ( ) . getMaximumSpeed ( res . isForwardDirection ( ) ) ;
2013-04-18 23:35:02 +02:00
if ( ms > 0 ) {
2019-07-10 15:24:23 +02:00
additional . append ( " maxspeed = \" " ) . append ( ( int ) Math . round ( ms * 3 . 6f ) ) . append ( " \" " ) . append ( res . getObject ( ) . getHighway ( ) ) . append ( " " ) ;
2013-04-18 23:35:02 +02:00
}
additional . append ( " distance = \" " ) . append ( res . getDistance ( ) ) . append ( " \" " ) ;
if ( res . getTurnType ( ) ! = null ) {
additional . append ( " turn = \" " ) . append ( res . getTurnType ( ) ) . append ( " \" " ) ;
additional . append ( " turn_angle = \" " ) . append ( res . getTurnType ( ) . getTurnAngle ( ) ) . append ( " \" " ) ;
if ( res . getTurnType ( ) . getLanes ( ) ! = null ) {
additional . append ( " lanes = \" " ) . append ( Arrays . toString ( res . getTurnType ( ) . getLanes ( ) ) ) . append ( " \" " ) ;
}
}
additional . append ( " start_bearing = \" " ) . append ( res . getBearingBegin ( ) ) . append ( " \" " ) ;
additional . append ( " end_bearing = \" " ) . append ( res . getBearingEnd ( ) ) . append ( " \" " ) ;
2016-12-22 01:34:22 +01:00
additional . append ( " height = \" " ) . append ( Arrays . toString ( res . getHeightValues ( ) ) ) . append ( " \" " ) ;
2013-04-18 23:35:02 +02:00
additional . append ( " description = \" " ) . append ( res . getDescription ( ) ) . append ( " \" " ) ;
2016-04-26 22:59:47 +02:00
println ( MessageFormat . format ( " \ t<segment id= \" {0} \" oid= \" {1} \" start= \" {2} \" end= \" {3} \" {4}/> " ,
2018-06-16 11:19:21 +02:00
( res . getObject ( ) . getId ( ) > > ( SHIFT_ID ) ) + " " , res . getObject ( ) . getId ( ) + " " ,
2013-04-18 23:35:02 +02:00
res . getStartPointIndex ( ) + " " , res . getEndPointIndex ( ) + " " , additional . toString ( ) ) ) ;
2017-02-10 16:55:56 +01:00
int inc = res . getStartPointIndex ( ) < res . getEndPointIndex ( ) ? 1 : - 1 ;
2017-02-13 19:13:42 +01:00
int indexnext = res . getStartPointIndex ( ) ;
2017-02-23 21:43:30 +01:00
LatLon prev = null ;
2017-02-13 19:13:42 +01:00
for ( int index = res . getStartPointIndex ( ) ; index ! = res . getEndPointIndex ( ) ; ) {
index = indexnext ;
indexnext + = inc ;
2017-02-10 16:55:56 +01:00
if ( serializer ! = null ) {
try {
LatLon l = res . getPoint ( index ) ;
serializer . startTag ( " " , " trkpt " ) ;
serializer . attribute ( " " , " lat " , l . getLatitude ( ) + " " ) ;
serializer . attribute ( " " , " lon " , l . getLongitude ( ) + " " ) ;
2017-02-13 19:13:42 +01:00
float [ ] vls = res . getObject ( ) . heightDistanceArray ;
2017-02-23 21:43:30 +01:00
double dist = prev = = null ? 0 : MapUtils . getDistance ( prev , l ) ;
2017-02-13 19:13:42 +01:00
if ( index * 2 + 1 < vls . length ) {
2017-02-23 21:43:30 +01:00
double h = vls [ 2 * index + 1 ] ;
2017-02-10 16:55:56 +01:00
serializer . startTag ( " " , " ele " ) ;
2017-02-23 21:43:30 +01:00
serializer . text ( h + " " ) ;
2017-02-10 16:55:56 +01:00
serializer . endTag ( " " , " ele " ) ;
2017-02-23 21:43:30 +01:00
if ( lastHeight ! = - 180 & & dist > 0 ) {
serializer . startTag ( " " , " cmt " ) ;
2017-02-23 21:45:34 +01:00
serializer . text ( ( float ) ( ( h - lastHeight ) / dist * 100 ) + " % " +
2017-02-23 21:48:56 +01:00
" degree " + ( float ) Math . atan ( ( ( h - lastHeight ) / dist ) ) / Math . PI * 180 +
2017-02-23 21:45:34 +01:00
" asc " + ( float ) ( h - lastHeight ) + " dist "
2017-02-23 21:43:30 +01:00
+ ( float ) dist ) ;
serializer . endTag ( " " , " cmt " ) ;
2017-02-26 16:52:56 +01:00
serializer . startTag ( " " , " slope " ) ;
serializer . text ( ( h - lastHeight ) / dist * 100 + " " ) ;
serializer . endTag ( " " , " slope " ) ;
2017-02-23 21:43:30 +01:00
}
2017-02-13 19:13:42 +01:00
serializer . startTag ( " " , " desc " ) ;
2018-06-16 11:19:21 +02:00
serializer . text ( ( res . getObject ( ) . getId ( ) > > ( SHIFT_ID ) ) + " " + index ) ;
2017-02-13 19:13:42 +01:00
serializer . endTag ( " " , " desc " ) ;
2017-02-23 21:43:30 +01:00
lastHeight = h ;
} else if ( lastHeight ! = - 180 ) {
2017-03-08 21:23:14 +01:00
// serializer.startTag("","ele");
// serializer.text(lastHeight +"");
// serializer.endTag("","ele");
2017-02-10 16:55:56 +01:00
}
serializer . endTag ( " " , " trkpt " ) ;
2017-02-23 21:43:30 +01:00
prev = l ;
2017-02-10 16:55:56 +01:00
} catch ( IOException e ) {
e . printStackTrace ( ) ;
}
}
}
2014-03-01 12:30:25 +01:00
printAdditionalPointInfo ( res ) ;
2013-04-18 23:35:02 +02:00
}
2017-02-10 16:55:56 +01:00
if ( serializer ! = null ) {
try {
serializer . endTag ( " " , " trkseg " ) ;
serializer . endTag ( " " , " trk " ) ;
serializer . endTag ( " " , " gpx " ) ;
serializer . endDocument ( ) ;
serializer . flush ( ) ;
} catch ( IOException e ) {
e . printStackTrace ( ) ;
}
}
2013-04-18 23:35:02 +02:00
}
println ( " </test> " ) ;
2017-02-10 19:29:57 +01:00
println ( msg ) ;
2019-07-08 00:37:20 +02:00
2019-07-08 01:30:33 +02:00
// calculateStatistics(result);
2019-07-08 00:37:20 +02:00
}
private void calculateStatistics ( List < RouteSegmentResult > result ) {
InputStream is = RenderingRulesStorage . class . getResourceAsStream ( " default.render.xml " ) ;
final Map < String , String > renderingConstants = new LinkedHashMap < String , String > ( ) ;
try {
InputStream pis = RenderingRulesStorage . class . getResourceAsStream ( " default.render.xml " ) ;
try {
XmlPullParser parser = PlatformUtil . newXMLPullParser ( ) ;
parser . setInput ( pis , " UTF-8 " ) ;
int tok ;
while ( ( tok = parser . next ( ) ) ! = XmlPullParser . END_DOCUMENT ) {
if ( tok = = XmlPullParser . START_TAG ) {
String tagName = parser . getName ( ) ;
if ( tagName . equals ( " renderingConstant " ) ) {
if ( ! renderingConstants . containsKey ( parser . getAttributeValue ( " " , " name " ) ) ) {
renderingConstants . put ( parser . getAttributeValue ( " " , " name " ) ,
parser . getAttributeValue ( " " , " value " ) ) ;
}
}
}
}
} finally {
pis . close ( ) ;
}
RenderingRulesStorage rrs = new RenderingRulesStorage ( " default " , renderingConstants ) ;
rrs . parseRulesFromXmlInputStream ( is , new RenderingRulesStorageResolver ( ) {
@Override
public RenderingRulesStorage resolve ( String name , RenderingRulesStorageResolver ref )
throws XmlPullParserException , IOException {
throw new UnsupportedOperationException ( ) ;
}
} ) ;
RenderingRuleSearchRequest req = new RenderingRuleSearchRequest ( rrs ) ;
List < RouteStatistics > rsr = RouteStatisticsHelper . calculateRouteStatistic ( result , null , rrs , null , req ) ;
for ( RouteStatistics r : rsr ) {
System . out . println ( r ) ;
}
} catch ( Exception e ) {
throw new IllegalStateException ( e . getMessage ( ) , e ) ;
}
2013-04-18 23:35:02 +02:00
}
2014-03-01 12:30:25 +01:00
private void printAdditionalPointInfo ( RouteSegmentResult res ) {
boolean plus = res . getStartPointIndex ( ) < res . getEndPointIndex ( ) ;
for ( int k = res . getStartPointIndex ( ) ; k ! = res . getEndPointIndex ( ) ; ) {
int [ ] tp = res . getObject ( ) . getPointTypes ( k ) ;
2015-07-06 07:08:10 +02:00
String [ ] pointNames = res . getObject ( ) . getPointNames ( k ) ;
int [ ] pointNameTypes = res . getObject ( ) . getPointNameTypes ( k ) ;
if ( tp ! = null | | pointNameTypes ! = null ) {
StringBuilder bld = new StringBuilder ( ) ;
2016-12-22 01:34:22 +01:00
bld . append ( " <point " + ( k ) ) ;
2015-07-06 07:08:10 +02:00
if ( tp ! = null ) {
for ( int t = 0 ; t < tp . length ; t + + ) {
RouteTypeRule rr = res . getObject ( ) . region . quickGetEncodingRule ( tp [ t ] ) ;
bld . append ( " " + rr . getTag ( ) + " = \" " + rr . getValue ( ) + " \" " ) ;
}
}
if ( pointNameTypes ! = null ) {
for ( int t = 0 ; t < pointNameTypes . length ; t + + ) {
RouteTypeRule rr = res . getObject ( ) . region . quickGetEncodingRule ( pointNameTypes [ t ] ) ;
bld . append ( " " + rr . getTag ( ) + " = \" " + pointNames [ t ] + " \" " ) ;
}
2014-03-01 12:30:25 +01:00
}
2015-07-06 07:08:10 +02:00
bld . append ( " /> " ) ;
println ( " \ t " + bld . toString ( ) ) ;
2014-03-01 12:30:25 +01:00
}
if ( plus ) {
k + + ;
} else {
k - - ;
}
}
}
2013-04-18 23:35:02 +02:00
2015-01-13 10:25:56 +01:00
protected void addTurnInfoDescriptions ( List < RouteSegmentResult > result ) {
int prevSegment = - 1 ;
float dist = 0 ;
for ( int i = 0 ; i < = result . size ( ) ; i + + ) {
if ( i = = result . size ( ) | | result . get ( i ) . getTurnType ( ) ! = null ) {
2013-04-18 23:35:02 +02:00
if ( prevSegment > = 0 ) {
String turn = result . get ( prevSegment ) . getTurnType ( ) . toString ( ) ;
2015-01-13 10:25:56 +01:00
result . get ( prevSegment ) . setDescription (
turn + MessageFormat . format ( " and go {0,number,#.##} meters " , dist ) ) ;
if ( result . get ( prevSegment ) . getTurnType ( ) . isSkipToSpeak ( ) ) {
result . get ( prevSegment ) . setDescription ( " -* " + result . get ( prevSegment ) . getDescription ( ) ) ;
2013-04-18 23:35:02 +02:00
}
}
prevSegment = i ;
dist = 0 ;
}
if ( i < result . size ( ) ) {
dist + = result . get ( i ) . getDistance ( ) ;
}
}
2015-01-13 10:25:56 +01:00
}
2014-11-16 16:04:54 +01:00
2015-01-13 10:25:56 +01:00
protected TurnType justifyUTurn ( boolean leftside , List < RouteSegmentResult > result , int i , TurnType t ) {
2016-04-09 14:11:09 +02:00
boolean tl = TurnType . isLeftTurnNoUTurn ( t . getValue ( ) ) ;
boolean tr = TurnType . isRightTurnNoUTurn ( t . getValue ( ) ) ;
2015-01-13 10:25:56 +01:00
if ( tl | | tr ) {
2015-01-15 02:42:33 +01:00
TurnType tnext = result . get ( i + 1 ) . getTurnType ( ) ;
2016-04-09 14:11:09 +02:00
if ( tnext ! = null & & result . get ( i ) . getDistance ( ) < 50 ) { //
2015-01-13 10:25:56 +01:00
boolean ut = true ;
if ( i > 0 ) {
double uTurn = MapUtils . degreesDiff ( result . get ( i - 1 ) . getBearingEnd ( ) , result
. get ( i + 1 ) . getBearingBegin ( ) ) ;
if ( Math . abs ( uTurn ) < 120 ) {
ut = false ;
}
}
2016-07-25 01:15:13 +02:00
// String highway = result.get(i).getObject().getHighway();
// if(highway == null || highway.endsWith("track") || highway.endsWith("services") || highway.endsWith("service")
// || highway.endsWith("path")) {
// ut = false;
// }
if ( result . get ( i - 1 ) . getObject ( ) . getOneway ( ) = = 0 | | result . get ( i + 1 ) . getObject ( ) . getOneway ( ) = = 0 ) {
2015-01-13 10:25:56 +01:00
ut = false ;
}
2016-07-25 01:15:13 +02:00
if ( ! Algorithms . objectEquals ( getStreetName ( result , i - 1 , false ) ,
getStreetName ( result , i + 1 , true ) ) ) {
2016-04-27 01:05:42 +02:00
ut = false ;
}
2015-01-13 10:25:56 +01:00
if ( ut ) {
2015-01-15 02:42:33 +01:00
tnext . setSkipToSpeak ( true ) ;
2016-04-12 22:20:01 +02:00
if ( tl & & TurnType . isLeftTurnNoUTurn ( tnext . getValue ( ) ) ) {
2016-04-09 14:11:09 +02:00
TurnType tt = TurnType . valueOf ( TurnType . TU , false ) ;
tt . setLanes ( t . getLanes ( ) ) ;
return tt ;
2016-04-12 22:20:01 +02:00
} else if ( tr & & TurnType . isRightTurnNoUTurn ( tnext . getValue ( ) ) ) {
2016-04-09 14:11:09 +02:00
TurnType tt = TurnType . valueOf ( TurnType . TU , true ) ;
tt . setLanes ( t . getLanes ( ) ) ;
return tt ;
2015-01-13 10:25:56 +01:00
}
}
}
}
return null ;
2014-11-23 19:28:21 +01:00
}
2016-07-25 01:15:13 +02:00
private String getStreetName ( List < RouteSegmentResult > result , int i , boolean dir ) {
String nm = result . get ( i ) . getObject ( ) . getName ( ) ;
if ( Algorithms . isEmpty ( nm ) ) {
if ( ! dir ) {
if ( i > 0 ) {
nm = result . get ( i - 1 ) . getObject ( ) . getName ( ) ;
}
} else {
if ( i < result . size ( ) - 1 ) {
nm = result . get ( i + 1 ) . getObject ( ) . getName ( ) ;
}
}
}
return nm ;
}
2014-11-23 19:28:21 +01:00
private void determineTurnsToMerge ( boolean leftside , List < RouteSegmentResult > result ) {
2015-01-15 02:42:33 +01:00
RouteSegmentResult nextSegment = null ;
double dist = 0 ;
for ( int i = result . size ( ) - 1 ; i > = 0 ; i - - ) {
2014-11-16 16:04:54 +01:00
RouteSegmentResult currentSegment = result . get ( i ) ;
2014-11-28 15:39:18 +01:00
TurnType currentTurn = currentSegment . getTurnType ( ) ;
2015-01-15 02:42:33 +01:00
dist + = currentSegment . getDistance ( ) ;
if ( currentTurn = = null | | currentTurn . getLanes ( ) = = null ) {
// skip
} else {
2016-07-03 01:12:59 +02:00
boolean merged = false ;
2015-01-15 02:42:33 +01:00
if ( nextSegment ! = null ) {
String hw = currentSegment . getObject ( ) . getHighway ( ) ;
double mergeDistance = 200 ;
if ( hw ! = null & & ( hw . startsWith ( " trunk " ) | | hw . startsWith ( " motorway " ) ) ) {
mergeDistance = 400 ;
}
if ( dist < mergeDistance ) {
2016-07-03 01:12:59 +02:00
mergeTurnLanes ( leftside , currentSegment , nextSegment ) ;
2016-08-29 00:19:19 +02:00
inferCommonActiveLane ( currentSegment . getTurnType ( ) , nextSegment . getTurnType ( ) ) ;
2016-07-03 01:12:59 +02:00
merged = true ;
2016-06-01 14:44:05 +02:00
}
}
2016-07-03 01:12:59 +02:00
if ( ! merged ) {
2016-06-01 14:44:05 +02:00
TurnType tt = currentSegment . getTurnType ( ) ;
2016-06-01 19:46:04 +02:00
inferActiveTurnLanesFromTurn ( tt , TurnType . C ) ;
2015-01-15 02:42:33 +01:00
}
nextSegment = currentSegment ;
dist = 0 ;
2014-11-16 16:04:54 +01:00
}
}
}
2016-06-01 19:46:04 +02:00
private void inferActiveTurnLanesFromTurn ( TurnType tt , int type ) {
boolean found = false ;
if ( tt . getValue ( ) = = type & & tt . getLanes ( ) ! = null ) {
for ( int it = 0 ; it < tt . getLanes ( ) . length ; it + + ) {
int turn = tt . getLanes ( ) [ it ] ;
if ( TurnType . getPrimaryTurn ( turn ) = = type | |
TurnType . getSecondaryTurn ( turn ) = = type | |
TurnType . getTertiaryTurn ( turn ) = = type ) {
found = true ;
break ;
}
}
}
if ( found ) {
for ( int it = 0 ; it < tt . getLanes ( ) . length ; it + + ) {
int turn = tt . getLanes ( ) [ it ] ;
if ( TurnType . getPrimaryTurn ( turn ) ! = type ) {
if ( TurnType . getSecondaryTurn ( turn ) = = type ) {
int st = TurnType . getSecondaryTurn ( turn ) ;
TurnType . setSecondaryTurn ( tt . getLanes ( ) , it , TurnType . getPrimaryTurn ( turn ) ) ;
TurnType . setPrimaryTurn ( tt . getLanes ( ) , it , st ) ;
} else if ( TurnType . getTertiaryTurn ( turn ) = = type ) {
int st = TurnType . getTertiaryTurn ( turn ) ;
TurnType . setTertiaryTurn ( tt . getLanes ( ) , it , TurnType . getPrimaryTurn ( turn ) ) ;
TurnType . setPrimaryTurn ( tt . getLanes ( ) , it , st ) ;
} else {
tt . getLanes ( ) [ it ] = turn & ( ~ 1 ) ;
}
}
}
}
}
2015-01-13 10:25:56 +01:00
2015-01-15 02:42:33 +01:00
private class MergeTurnLaneTurn {
TurnType turn ;
int [ ] originalLanes ;
int [ ] disabledLanes ;
int activeStartIndex = - 1 ;
int activeEndIndex = - 1 ;
int activeLen = 0 ;
public MergeTurnLaneTurn ( RouteSegmentResult segment ) {
this . turn = segment . getTurnType ( ) ;
if ( turn ! = null ) {
originalLanes = turn . getLanes ( ) ;
2015-01-13 10:25:56 +01:00
}
2015-01-15 02:42:33 +01:00
if ( originalLanes ! = null ) {
disabledLanes = new int [ originalLanes . length ] ;
for ( int i = 0 ; i < originalLanes . length ; i + + ) {
int ln = originalLanes [ i ] ;
disabledLanes [ i ] = ln & ~ 1 ;
if ( ( ln & 1 ) > 0 ) {
if ( activeStartIndex = = - 1 ) {
activeStartIndex = i ;
}
activeEndIndex = i ;
activeLen + + ;
2014-11-16 16:04:54 +01:00
}
}
2015-01-13 10:25:56 +01:00
}
2015-01-15 02:42:33 +01:00
}
public boolean isActiveTurnMostLeft ( ) {
return activeStartIndex = = 0 ;
}
public boolean isActiveTurnMostRight ( ) {
return activeEndIndex = = originalLanes . length - 1 ;
}
}
2016-06-01 14:44:05 +02:00
private boolean mergeTurnLanes ( boolean leftSide , RouteSegmentResult currentSegment , RouteSegmentResult nextSegment ) {
2015-01-15 02:42:33 +01:00
MergeTurnLaneTurn active = new MergeTurnLaneTurn ( currentSegment ) ;
MergeTurnLaneTurn target = new MergeTurnLaneTurn ( nextSegment ) ;
if ( active . activeLen < 2 ) {
2016-06-01 14:44:05 +02:00
return false ;
2015-01-15 02:42:33 +01:00
}
if ( target . activeStartIndex = = - 1 ) {
2016-06-01 14:44:05 +02:00
return false ;
2015-01-15 02:42:33 +01:00
}
boolean changed = false ;
if ( target . isActiveTurnMostLeft ( ) ) {
// let only the most left lanes be enabled
2016-08-29 00:19:19 +02:00
if ( target . activeLen < active . activeLen ) {
2015-01-15 02:42:33 +01:00
active . activeEndIndex - = ( active . activeLen - target . activeLen ) ;
changed = true ;
2014-11-16 16:04:54 +01:00
}
2015-01-15 02:42:33 +01:00
} else if ( target . isActiveTurnMostRight ( ) ) {
// next turn is right
// let only the most right lanes be enabled
2016-10-18 10:31:39 +02:00
if ( target . activeLen < active . activeLen ) {
2015-01-15 02:42:33 +01:00
active . activeStartIndex + = ( active . activeLen - target . activeLen ) ;
changed = true ;
2014-11-16 16:04:54 +01:00
}
2015-01-15 02:42:33 +01:00
} else {
// next turn is get through (take out the left and the right turn)
2015-01-15 03:17:26 +01:00
if ( target . activeLen < active . activeLen ) {
2016-04-13 00:47:05 +02:00
if ( target . originalLanes . length = = active . activeLen ) {
active . activeEndIndex = active . activeStartIndex + target . activeEndIndex ;
active . activeStartIndex = active . activeStartIndex + target . activeStartIndex ;
changed = true ;
} else {
2017-04-03 17:36:21 +02:00
int straightActiveLen = 0 ;
int straightActiveBegin = - 1 ;
for ( int i = active . activeStartIndex ; i < = active . activeEndIndex ; i + + ) {
if ( TurnType . hasAnyTurnLane ( active . originalLanes [ i ] , TurnType . C ) ) {
straightActiveLen + + ;
if ( straightActiveBegin = = - 1 ) {
straightActiveBegin = i ;
}
}
2016-06-15 00:59:00 +02:00
}
2018-05-26 00:02:57 +02:00
if ( straightActiveBegin ! = - 1 & & straightActiveLen < = target . activeLen ) {
2017-04-03 17:36:21 +02:00
active . activeStartIndex = straightActiveBegin ;
2018-05-26 00:02:57 +02:00
active . activeEndIndex = straightActiveBegin + straightActiveLen - 1 ;
2017-04-03 17:36:21 +02:00
changed = true ;
} else {
// cause the next-turn goes forward exclude left most and right most lane
if ( active . activeStartIndex = = 0 ) {
active . activeStartIndex + + ;
active . activeLen - - ;
}
if ( active . activeEndIndex = = active . originalLanes . length - 1 ) {
active . activeEndIndex - - ;
active . activeLen - - ;
}
float ratio = ( active . activeLen - target . activeLen ) / 2f ;
if ( ratio > 0 ) {
active . activeEndIndex = ( int ) Math . ceil ( active . activeEndIndex - ratio ) ;
active . activeStartIndex = ( int ) Math . floor ( active . activeStartIndex + ratio ) ;
}
changed = true ;
2016-06-15 00:59:00 +02:00
}
2016-04-13 00:47:05 +02:00
}
2014-11-16 16:04:54 +01:00
}
2015-01-15 02:42:33 +01:00
}
if ( ! changed ) {
2016-06-01 14:44:05 +02:00
return false ;
2015-01-15 02:42:33 +01:00
}
// set the allowed lane bit
for ( int i = 0 ; i < active . disabledLanes . length ; i + + ) {
2016-04-13 00:47:05 +02:00
if ( i > = active . activeStartIndex & & i < = active . activeEndIndex & &
active . originalLanes [ i ] % 2 = = 1 ) {
2015-01-15 02:42:33 +01:00
active . disabledLanes [ i ] | = 1 ;
2014-11-25 02:24:47 +01:00
}
2014-11-16 16:04:54 +01:00
}
2015-01-15 02:42:33 +01:00
TurnType currentTurn = currentSegment . getTurnType ( ) ;
currentTurn . setLanes ( active . disabledLanes ) ;
2016-06-01 14:44:05 +02:00
return true ;
2013-04-18 23:35:02 +02:00
}
2016-08-29 00:19:19 +02:00
private void inferCommonActiveLane ( TurnType currentTurn , TurnType nextTurn ) {
2016-06-01 19:46:04 +02:00
int [ ] lanes = currentTurn . getLanes ( ) ;
2016-08-29 00:19:19 +02:00
TIntHashSet turnSet = new TIntHashSet ( ) ;
2016-06-01 19:46:04 +02:00
for ( int i = 0 ; i < lanes . length ; i + + ) {
2016-08-29 00:19:19 +02:00
if ( lanes [ i ] % 2 = = 1 ) {
int singleTurn = TurnType . getPrimaryTurn ( lanes [ i ] ) ;
turnSet . add ( singleTurn ) ;
if ( TurnType . getSecondaryTurn ( lanes [ i ] ) ! = 0 ) {
turnSet . add ( TurnType . getSecondaryTurn ( lanes [ i ] ) ) ;
}
if ( TurnType . getTertiaryTurn ( lanes [ i ] ) ! = 0 ) {
turnSet . add ( TurnType . getTertiaryTurn ( lanes [ i ] ) ) ;
}
2016-06-01 19:46:04 +02:00
}
}
2016-08-29 00:45:21 +02:00
int singleTurn = 0 ;
2016-08-29 00:19:19 +02:00
if ( turnSet . size ( ) = = 1 ) {
singleTurn = turnSet . iterator ( ) . next ( ) ;
2016-08-29 00:45:21 +02:00
} else if ( currentTurn . goAhead ( ) & & turnSet . contains ( nextTurn . getValue ( ) ) ) {
if ( currentTurn . isPossibleLeftTurn ( ) & &
TurnType . isLeftTurn ( nextTurn . getValue ( ) ) ) {
singleTurn = nextTurn . getValue ( ) ;
} else if ( currentTurn . isPossibleLeftTurn ( ) & &
TurnType . isLeftTurn ( nextTurn . getActiveCommonLaneTurn ( ) ) ) {
singleTurn = nextTurn . getActiveCommonLaneTurn ( ) ;
} else if ( currentTurn . isPossibleRightTurn ( ) & &
TurnType . isRightTurn ( nextTurn . getValue ( ) ) ) {
singleTurn = nextTurn . getValue ( ) ;
} else if ( currentTurn . isPossibleRightTurn ( ) & &
TurnType . isRightTurn ( nextTurn . getActiveCommonLaneTurn ( ) ) ) {
singleTurn = nextTurn . getActiveCommonLaneTurn ( ) ;
}
}
if ( singleTurn = = 0 ) {
2016-06-01 19:46:04 +02:00
singleTurn = currentTurn . getValue ( ) ;
2016-10-18 10:31:39 +02:00
if ( singleTurn = = TurnType . KL | | singleTurn = = TurnType . KR ) {
return ;
}
2016-06-01 19:46:04 +02:00
}
for ( int i = 0 ; i < lanes . length ; i + + ) {
if ( lanes [ i ] % 2 = = 1 & & TurnType . getPrimaryTurn ( lanes [ i ] ) ! = singleTurn ) {
if ( TurnType . getSecondaryTurn ( lanes [ i ] ) = = singleTurn ) {
TurnType . setSecondaryTurn ( lanes , i , TurnType . getPrimaryTurn ( lanes [ i ] ) ) ;
TurnType . setPrimaryTurn ( lanes , i , singleTurn ) ;
} else if ( TurnType . getTertiaryTurn ( lanes [ i ] ) = = singleTurn ) {
TurnType . setTertiaryTurn ( lanes , i , TurnType . getPrimaryTurn ( lanes [ i ] ) ) ;
TurnType . setPrimaryTurn ( lanes , i , singleTurn ) ;
2016-08-29 00:19:19 +02:00
} else {
// disable lane
lanes [ i ] = lanes [ i ] - 1 ;
2016-06-01 19:46:04 +02:00
}
}
}
}
2013-04-18 23:35:02 +02:00
private static final int MAX_SPEAK_PRIORITY = 5 ;
private int highwaySpeakPriority ( String highway ) {
if ( highway = = null | | highway . endsWith ( " track " ) | | highway . endsWith ( " services " ) | | highway . endsWith ( " service " )
| | highway . endsWith ( " path " ) ) {
return MAX_SPEAK_PRIORITY ;
}
if ( highway . endsWith ( " _link " ) | | highway . endsWith ( " unclassified " ) | | highway . endsWith ( " road " )
| | highway . endsWith ( " living_street " ) | | highway . endsWith ( " residential " ) ) {
return 1 ;
}
return 0 ;
}
private TurnType getTurnInfo ( List < RouteSegmentResult > result , int i , boolean leftSide ) {
if ( i = = 0 ) {
return TurnType . valueOf ( TurnType . C , false ) ;
}
RouteSegmentResult prev = result . get ( i - 1 ) ;
if ( prev . getObject ( ) . roundabout ( ) ) {
// already analyzed!
return null ;
}
RouteSegmentResult rr = result . get ( i ) ;
if ( rr . getObject ( ) . roundabout ( ) ) {
return processRoundaboutTurn ( result , i , leftSide , prev , rr ) ;
}
TurnType t = null ;
if ( prev ! = null ) {
boolean noAttachedRoads = rr . getAttachedRoutes ( rr . getStartPointIndex ( ) ) . size ( ) = = 0 ;
// add description about turn
double mpi = MapUtils . degreesDiff ( prev . getBearingEnd ( ) , rr . getBearingBegin ( ) ) ;
if ( noAttachedRoads ) {
// TODO VICTOR : look at the comment inside direction route
2014-01-09 01:33:46 +01:00
// ? avoid small zigzags is covered at (search for "zigzags")
2013-04-18 23:35:02 +02:00
// double begin = rr.getObject().directionRoute(rr.getStartPointIndex(), rr.getStartPointIndex() <
// rr.getEndPointIndex(), 25);
// mpi = MapUtils.degreesDiff(prev.getBearingEnd(), begin);
}
if ( mpi > = TURN_DEGREE_MIN ) {
2015-09-11 19:43:31 +02:00
if ( mpi < 45 ) {
2015-09-13 10:58:27 +02:00
// Slight turn detection here causes many false positives where drivers would expect a "normal" TL. Best use limit-angle=TURN_DEGREE_MIN, this reduces TSL to the turn-lanes cases.
2013-04-18 23:35:02 +02:00
t = TurnType . valueOf ( TurnType . TSLL , leftSide ) ;
} else if ( mpi < 120 ) {
t = TurnType . valueOf ( TurnType . TL , leftSide ) ;
2015-09-11 19:43:31 +02:00
} else if ( mpi < 150 | | leftSide ) {
2013-04-18 23:35:02 +02:00
t = TurnType . valueOf ( TurnType . TSHL , leftSide ) ;
} else {
t = TurnType . valueOf ( TurnType . TU , leftSide ) ;
}
2015-01-15 02:42:33 +01:00
int [ ] lanes = getTurnLanesInfo ( prev , t . getValue ( ) ) ;
t . setLanes ( lanes ) ;
2013-04-18 23:35:02 +02:00
} else if ( mpi < - TURN_DEGREE_MIN ) {
2015-09-11 19:43:31 +02:00
if ( mpi > - 45 ) {
2013-04-18 23:35:02 +02:00
t = TurnType . valueOf ( TurnType . TSLR , leftSide ) ;
} else if ( mpi > - 120 ) {
t = TurnType . valueOf ( TurnType . TR , leftSide ) ;
2015-09-11 19:43:31 +02:00
} else if ( mpi > - 150 | | ! leftSide ) {
2013-04-18 23:35:02 +02:00
t = TurnType . valueOf ( TurnType . TSHR , leftSide ) ;
} else {
2015-01-15 02:42:33 +01:00
t = TurnType . valueOf ( TurnType . TRU , leftSide ) ;
2014-11-23 19:45:31 +01:00
}
2015-01-15 02:42:33 +01:00
int [ ] lanes = getTurnLanesInfo ( prev , t . getValue ( ) ) ;
t . setLanes ( lanes ) ;
2013-04-18 23:35:02 +02:00
} else {
2015-01-15 02:42:33 +01:00
t = attachKeepLeftInfoAndLanes ( leftSide , prev , rr ) ;
2013-04-18 23:35:02 +02:00
}
if ( t ! = null ) {
t . setTurnAngle ( ( float ) - mpi ) ;
}
}
return t ;
}
2016-06-15 01:20:36 +02:00
private int [ ] getTurnLanesInfo ( RouteSegmentResult prevSegm , int mainTurnType ) { String turnLanes = getTurnLanesString ( prevSegm ) ;
2016-04-12 23:44:17 +02:00
int [ ] lanesArray ;
2014-10-13 15:32:35 +02:00
if ( turnLanes = = null ) {
2016-04-13 00:14:31 +02:00
if ( prevSegm . getTurnType ( ) ! = null & & prevSegm . getTurnType ( ) . getLanes ( ) ! = null
& & prevSegm . getDistance ( ) < 100 ) {
2016-04-12 23:44:17 +02:00
int [ ] lns = prevSegm . getTurnType ( ) . getLanes ( ) ;
TIntArrayList lst = new TIntArrayList ( ) ;
for ( int i = 0 ; i < lns . length ; i + + ) {
if ( lns [ i ] % 2 = = 1 ) {
lst . add ( ( lns [ i ] > > 1 ) < < 1 ) ;
}
}
if ( lst . isEmpty ( ) ) {
return null ;
}
lanesArray = lst . toArray ( ) ;
} else {
return null ;
}
} else {
lanesArray = calculateRawTurnLanes ( turnLanes , mainTurnType ) ;
2014-10-13 15:32:35 +02:00
}
2014-11-23 19:45:31 +01:00
// Manually set the allowed lanes.
boolean isSet = setAllowedLanes ( mainTurnType , lanesArray ) ;
if ( ! isSet & & lanesArray . length > 0 ) {
// In some cases (at least in the US), the rightmost lane might not have a right turn indicated as per turn:lanes,
// but is allowed and being used here. This section adds in that indicator. The same applies for where leftSide is true.
2015-01-15 02:42:33 +01:00
boolean leftTurn = TurnType . isLeftTurn ( mainTurnType ) ;
2014-11-23 19:45:31 +01:00
int ind = leftTurn ? 0 : lanesArray . length - 1 ;
2016-06-15 00:59:00 +02:00
int primaryTurn = TurnType . getPrimaryTurn ( lanesArray [ ind ] ) ;
2016-03-28 15:51:52 +02:00
final int st = TurnType . getSecondaryTurn ( lanesArray [ ind ] ) ;
2014-11-23 19:45:31 +01:00
if ( leftTurn ) {
2016-06-15 00:59:00 +02:00
if ( ! TurnType . isLeftTurn ( primaryTurn ) ) {
2014-10-13 16:53:52 +02:00
// This was just to make sure that there's no bad data.
2015-01-15 02:42:33 +01:00
TurnType . setPrimaryTurnAndReset ( lanesArray , ind , TurnType . TL ) ;
2016-06-15 00:59:00 +02:00
TurnType . setSecondaryTurn ( lanesArray , ind , primaryTurn ) ;
2016-03-28 15:51:52 +02:00
TurnType . setTertiaryTurn ( lanesArray , ind , st ) ;
2016-06-15 00:59:00 +02:00
primaryTurn = TurnType . TL ;
2015-05-27 04:56:31 +02:00
lanesArray [ ind ] | = 1 ;
2014-10-13 16:53:52 +02:00
}
2014-11-23 19:45:31 +01:00
} else {
2016-06-15 00:59:00 +02:00
if ( ! TurnType . isRightTurn ( primaryTurn ) ) {
2014-10-13 16:53:52 +02:00
// This was just to make sure that there's no bad data.
2015-01-15 02:42:33 +01:00
TurnType . setPrimaryTurnAndReset ( lanesArray , ind , TurnType . TR ) ;
2016-06-15 00:59:00 +02:00
TurnType . setSecondaryTurn ( lanesArray , ind , primaryTurn ) ;
2016-03-28 15:51:52 +02:00
TurnType . setTertiaryTurn ( lanesArray , ind , st ) ;
2016-06-15 00:59:00 +02:00
primaryTurn = TurnType . TR ;
2015-05-27 04:56:31 +02:00
lanesArray [ ind ] | = 1 ;
2014-10-13 16:53:52 +02:00
}
}
2016-06-15 00:59:00 +02:00
setAllowedLanes ( primaryTurn , lanesArray ) ;
2014-10-13 16:53:52 +02:00
}
2014-11-23 19:45:31 +01:00
return lanesArray ;
}
2014-10-13 16:53:52 +02:00
2014-11-23 19:45:31 +01:00
protected boolean setAllowedLanes ( int mainTurnType , int [ ] lanesArray ) {
boolean turnSet = false ;
2014-10-13 15:32:35 +02:00
for ( int i = 0 ; i < lanesArray . length ; i + + ) {
2014-11-23 19:45:31 +01:00
if ( TurnType . getPrimaryTurn ( lanesArray [ i ] ) = = mainTurnType ) {
2014-10-13 15:32:35 +02:00
lanesArray [ i ] | = 1 ;
2014-11-23 19:45:31 +01:00
turnSet = true ;
2014-10-13 15:32:35 +02:00
}
}
2014-11-23 19:45:31 +01:00
return turnSet ;
2014-10-13 15:32:35 +02:00
}
2013-04-18 23:35:02 +02:00
private TurnType processRoundaboutTurn ( List < RouteSegmentResult > result , int i , boolean leftSide , RouteSegmentResult prev ,
RouteSegmentResult rr ) {
int exit = 1 ;
RouteSegmentResult last = rr ;
2016-08-11 02:18:34 +02:00
RouteSegmentResult firstRoundabout = rr ;
RouteSegmentResult lastRoundabout = rr ;
2013-04-18 23:35:02 +02:00
for ( int j = i ; j < result . size ( ) ; j + + ) {
RouteSegmentResult rnext = result . get ( j ) ;
last = rnext ;
if ( rnext . getObject ( ) . roundabout ( ) ) {
2016-08-11 02:18:34 +02:00
lastRoundabout = rnext ;
2013-04-18 23:35:02 +02:00
boolean plus = rnext . getStartPointIndex ( ) < rnext . getEndPointIndex ( ) ;
int k = rnext . getStartPointIndex ( ) ;
if ( j = = i ) {
// first exit could be immediately after roundabout enter
// k = plus ? k + 1 : k - 1;
}
while ( k ! = rnext . getEndPointIndex ( ) ) {
int attachedRoads = rnext . getAttachedRoutes ( k ) . size ( ) ;
if ( attachedRoads > 0 ) {
exit + + ;
}
k = plus ? k + 1 : k - 1 ;
}
} else {
break ;
}
}
// combine all roundabouts
2014-10-02 13:47:34 +02:00
TurnType t = TurnType . getExitTurn ( exit , 0 , leftSide ) ;
2016-08-11 02:18:34 +02:00
// usually covers more than expected
float turnAngleBasedOnOutRoads = ( float ) MapUtils . degreesDiff ( last . getBearingBegin ( ) , prev . getBearingEnd ( ) ) ;
// usually covers less than expected
float turnAngleBasedOnCircle = ( float ) - MapUtils . degreesDiff ( firstRoundabout . getBearingBegin ( ) , lastRoundabout . getBearingEnd ( ) + 180 ) ;
if ( Math . abs ( turnAngleBasedOnOutRoads - turnAngleBasedOnCircle ) > 180 ) {
t . setTurnAngle ( turnAngleBasedOnCircle ) ;
} else {
t . setTurnAngle ( ( turnAngleBasedOnCircle + turnAngleBasedOnOutRoads ) / 2 ) ;
}
2013-04-18 23:35:02 +02:00
return t ;
}
2015-01-15 02:42:33 +01:00
private class RoadSplitStructure {
boolean keepLeft = false ;
boolean keepRight = false ;
boolean speak = false ;
2016-06-01 19:46:04 +02:00
List < int [ ] > leftLanesInfo = new ArrayList < int [ ] > ( ) ;
2015-01-15 02:42:33 +01:00
int leftLanes = 0 ;
2016-06-01 19:46:04 +02:00
List < int [ ] > rightLanesInfo = new ArrayList < int [ ] > ( ) ;
2015-01-15 02:42:33 +01:00
int rightLanes = 0 ;
int roadsOnLeft = 0 ;
int addRoadsOnLeft = 0 ;
int roadsOnRight = 0 ;
int addRoadsOnRight = 0 ;
}
2013-04-18 23:35:02 +02:00
2015-01-15 02:42:33 +01:00
private TurnType attachKeepLeftInfoAndLanes ( boolean leftSide , RouteSegmentResult prevSegm , RouteSegmentResult currentSegm ) {
2013-04-18 23:35:02 +02:00
List < RouteSegmentResult > attachedRoutes = currentSegm . getAttachedRoutes ( currentSegm . getStartPointIndex ( ) ) ;
2015-01-15 02:42:33 +01:00
if ( attachedRoutes = = null | | attachedRoutes . size ( ) = = 0 ) {
return null ;
2013-04-18 23:35:02 +02:00
}
2015-01-15 02:42:33 +01:00
// keep left/right
RoadSplitStructure rs = calculateRoadSplitStructure ( prevSegm , currentSegm , attachedRoutes ) ;
if ( rs . roadsOnLeft + rs . roadsOnRight = = 0 ) {
return null ;
}
// turn lanes exist
String turnLanes = getTurnLanesString ( prevSegm ) ;
if ( turnLanes ! = null ) {
return createKeepLeftRightTurnBasedOnTurnTypes ( rs , prevSegm , currentSegm , turnLanes , leftSide ) ;
}
// turn lanes don't exist
if ( rs . keepLeft | | rs . keepRight ) {
return createSimpleKeepLeftRightTurn ( leftSide , prevSegm , currentSegm , rs ) ;
}
return null ;
}
protected TurnType createKeepLeftRightTurnBasedOnTurnTypes ( RoadSplitStructure rs , RouteSegmentResult prevSegm ,
2016-05-24 10:42:36 +02:00
RouteSegmentResult currentSegm , String turnLanes , boolean leftSide ) {
2015-01-15 02:42:33 +01:00
// Maybe going straight at a 90-degree intersection
TurnType t = TurnType . valueOf ( TurnType . C , leftSide ) ;
int [ ] rawLanes = calculateRawTurnLanes ( turnLanes , TurnType . C ) ;
2016-08-29 00:45:21 +02:00
boolean possiblyLeftTurn = rs . roadsOnLeft = = 0 ;
boolean possiblyRightTurn = rs . roadsOnRight = = 0 ;
for ( int k = 0 ; k < rawLanes . length ; k + + ) {
int turn = TurnType . getPrimaryTurn ( rawLanes [ k ] ) ;
int sturn = TurnType . getSecondaryTurn ( rawLanes [ k ] ) ;
int tturn = TurnType . getTertiaryTurn ( rawLanes [ k ] ) ;
if ( turn = = TurnType . TU | | sturn = = TurnType . TU | | tturn = = TurnType . TU ) {
possiblyLeftTurn = true ;
}
if ( turn = = TurnType . TRU | | sturn = = TurnType . TRU | | sturn = = TurnType . TRU ) {
possiblyRightTurn = true ;
}
}
t . setPossibleLeftTurn ( possiblyLeftTurn ) ;
t . setPossibleRightTurn ( possiblyRightTurn ) ;
2015-01-15 02:42:33 +01:00
if ( rs . keepLeft | | rs . keepRight ) {
String [ ] splitLaneOptions = turnLanes . split ( " \\ | " , - 1 ) ;
int activeBeginIndex = findActiveIndex ( rawLanes , splitLaneOptions , rs . leftLanes , true ,
2016-06-01 19:46:04 +02:00
rs . leftLanesInfo , rs . roadsOnLeft , rs . addRoadsOnLeft ) ;
2017-04-03 17:36:21 +02:00
if ( ! rs . keepLeft & & activeBeginIndex ! = - 1 & &
splitLaneOptions . length > 0 & & ! splitLaneOptions [ splitLaneOptions . length - 1 ] . contains ( " ; " ) ) {
activeBeginIndex = Math . max ( activeBeginIndex , 1 ) ;
}
2015-01-15 02:42:33 +01:00
int activeEndIndex = findActiveIndex ( rawLanes , splitLaneOptions , rs . rightLanes , false ,
2016-06-01 19:46:04 +02:00
rs . rightLanesInfo , rs . roadsOnRight , rs . addRoadsOnRight ) ;
2017-04-03 17:36:21 +02:00
if ( ! rs . keepRight & & activeEndIndex ! = - 1 & &
splitLaneOptions . length > 0 & & ! splitLaneOptions [ 0 ] . contains ( " ; " ) ) {
activeEndIndex = Math . min ( activeEndIndex , rawLanes . length - 1 ) ;
}
2015-01-15 02:42:33 +01:00
if ( activeBeginIndex = = - 1 | | activeEndIndex = = - 1 | | activeBeginIndex > activeEndIndex ) {
// something went wrong
return createSimpleKeepLeftRightTurn ( leftSide , prevSegm , currentSegm , rs ) ;
}
for ( int k = 0 ; k < rawLanes . length ; k + + ) {
if ( k > = activeBeginIndex & & k < = activeEndIndex ) {
rawLanes [ k ] | = 1 ;
}
}
2016-08-11 23:35:04 +02:00
int tp = inferSlightTurnFromLanes ( rawLanes , rs ) ;
2015-01-15 02:42:33 +01:00
if ( tp ! = t . getValue ( ) & & tp ! = 0 ) {
t = TurnType . valueOf ( tp , leftSide ) ;
}
} else {
for ( int k = 0 ; k < rawLanes . length ; k + + ) {
2016-04-12 23:05:18 +02:00
int turn = TurnType . getPrimaryTurn ( rawLanes [ k ] ) ;
2016-05-30 11:08:31 +02:00
int sturn = TurnType . getSecondaryTurn ( rawLanes [ k ] ) ;
2016-08-29 00:45:21 +02:00
int tturn = TurnType . getTertiaryTurn ( rawLanes [ k ] ) ;
2015-01-15 02:42:33 +01:00
boolean active = false ;
2016-08-11 23:35:04 +02:00
// some turns go through many segments (to turn right or left)
// so on one first segment the lane could be available and may be only 1 possible
// all undesired lanes will be disabled through the 2nd pass
if ( ( TurnType . isRightTurn ( sturn ) & & possiblyRightTurn ) | |
( TurnType . isLeftTurn ( sturn ) & & possiblyLeftTurn ) ) {
2016-08-29 00:19:19 +02:00
// we can't predict here whether it will be a left turn or straight on,
// it could be done during 2nd pass
2016-08-29 00:45:21 +02:00
TurnType . setPrimaryTurn ( rawLanes , k , sturn ) ;
TurnType . setSecondaryTurn ( rawLanes , k , turn ) ;
active = true ;
} else if ( ( TurnType . isRightTurn ( tturn ) & & possiblyRightTurn ) | |
( TurnType . isLeftTurn ( tturn ) & & possiblyLeftTurn ) ) {
TurnType . setPrimaryTurn ( rawLanes , k , tturn ) ;
TurnType . setTertiaryTurn ( rawLanes , k , turn ) ;
2016-06-01 19:46:04 +02:00
active = true ;
2016-08-11 23:35:04 +02:00
} else if ( ( TurnType . isRightTurn ( turn ) & & possiblyRightTurn ) | |
( TurnType . isLeftTurn ( turn ) & & possiblyLeftTurn ) ) {
2015-01-15 02:42:33 +01:00
active = true ;
2016-08-11 23:35:04 +02:00
} else if ( turn = = TurnType . C ) {
2015-01-15 02:42:33 +01:00
active = true ;
}
if ( active ) {
rawLanes [ k ] | = 1 ;
}
}
}
t . setSkipToSpeak ( ! rs . speak ) ;
t . setLanes ( rawLanes ) ;
return t ;
}
protected int findActiveIndex ( int [ ] rawLanes , String [ ] splitLaneOptions , int lanes , boolean left ,
2016-06-01 19:46:04 +02:00
List < int [ ] > lanesInfo , int roads , int addRoads ) {
2015-01-15 02:42:33 +01:00
int activeStartIndex = - 1 ;
boolean lookupSlightTurn = addRoads > 0 ;
2017-04-11 23:41:52 +02:00
TIntHashSet addedTurns = new TIntHashSet ( ) ;
2016-06-01 19:46:04 +02:00
// if we have information increase number of roads per each turn direction
int diffTurnRoads = roads ;
2016-08-11 01:17:25 +02:00
int increaseTurnRoads = 0 ;
2016-06-01 19:46:04 +02:00
for ( int [ ] li : lanesInfo ) {
TIntHashSet set = new TIntHashSet ( ) ;
if ( li ! = null ) {
for ( int k = 0 ; k < li . length ; k + + ) {
TurnType . collectTurnTypes ( li [ k ] , set ) ;
}
}
2016-08-11 01:17:25 +02:00
increaseTurnRoads = Math . max ( set . size ( ) - 1 , 0 ) ;
2016-06-01 19:46:04 +02:00
}
2015-01-15 02:42:33 +01:00
for ( int i = 0 ; i < rawLanes . length ; i + + ) {
int ind = left ? i : ( rawLanes . length - i - 1 ) ;
if ( ! lookupSlightTurn | |
2017-04-03 17:36:21 +02:00
TurnType . hasAnySlightTurnLane ( rawLanes [ ind ] ) ) {
2016-06-01 19:46:04 +02:00
String [ ] laneTurns = splitLaneOptions [ ind ] . split ( " ; " ) ;
int cnt = 0 ;
for ( String lTurn : laneTurns ) {
2017-04-11 23:41:52 +02:00
boolean added = addedTurns . add ( TurnType . convertType ( lTurn ) ) ;
2016-06-01 19:46:04 +02:00
if ( added ) {
cnt + + ;
diffTurnRoads - - ;
}
2015-01-15 02:42:33 +01:00
}
lanes - = cnt ;
2017-04-03 17:36:21 +02:00
//lanes--;
2015-01-15 02:42:33 +01:00
// we already found slight turn others are turn in different direction
lookupSlightTurn = false ;
}
2016-08-11 01:17:25 +02:00
if ( lanes < 0 | | diffTurnRoads + increaseTurnRoads < 0 ) {
2015-01-15 02:42:33 +01:00
activeStartIndex = ind ;
break ;
2016-08-11 01:17:25 +02:00
} else if ( diffTurnRoads < 0 & & activeStartIndex < 0 ) {
activeStartIndex = ind ;
2015-01-15 02:42:33 +01:00
}
}
return activeStartIndex ;
}
protected RoadSplitStructure calculateRoadSplitStructure ( RouteSegmentResult prevSegm , RouteSegmentResult currentSegm ,
List < RouteSegmentResult > attachedRoutes ) {
RoadSplitStructure rs = new RoadSplitStructure ( ) ;
2013-04-18 23:35:02 +02:00
int speakPriority = Math . max ( highwaySpeakPriority ( prevSegm . getObject ( ) . getHighway ( ) ) , highwaySpeakPriority ( currentSegm . getObject ( ) . getHighway ( ) ) ) ;
2015-01-15 02:42:33 +01:00
for ( RouteSegmentResult attached : attachedRoutes ) {
2016-11-07 00:59:59 +01:00
boolean restricted = false ;
for ( int k = 0 ; k < prevSegm . getObject ( ) . getRestrictionLength ( ) ; k + + ) {
if ( prevSegm . getObject ( ) . getRestrictionId ( k ) = = attached . getObject ( ) . getId ( ) & &
prevSegm . getObject ( ) . getRestrictionType ( k ) < = MapRenderingTypes . RESTRICTION_NO_STRAIGHT_ON ) {
restricted = true ;
break ;
}
}
if ( restricted ) {
continue ;
}
2015-01-15 02:42:33 +01:00
double ex = MapUtils . degreesDiff ( attached . getBearingBegin ( ) , currentSegm . getBearingBegin ( ) ) ;
double mpi = Math . abs ( MapUtils . degreesDiff ( prevSegm . getBearingEnd ( ) , attached . getBearingBegin ( ) ) ) ;
int rsSpeakPriority = highwaySpeakPriority ( attached . getObject ( ) . getHighway ( ) ) ;
2016-04-13 23:00:27 +02:00
int lanes = countLanesMinOne ( attached ) ;
2018-01-24 23:27:37 +01:00
int [ ] turnLanes = parseTurnLanes ( attached . getObject ( ) , attached . getBearingBegin ( ) * Math . PI / 180 ) ;
2016-04-13 23:00:27 +02:00
boolean smallStraightVariation = mpi < TURN_DEGREE_MIN ;
boolean smallTargetVariation = Math . abs ( ex ) < TURN_DEGREE_MIN ;
boolean attachedOnTheRight = ex > = 0 ;
if ( attachedOnTheRight ) {
rs . roadsOnRight + + ;
} else {
rs . roadsOnLeft + + ;
}
2015-01-15 02:42:33 +01:00
if ( rsSpeakPriority ! = MAX_SPEAK_PRIORITY | | speakPriority = = MAX_SPEAK_PRIORITY ) {
if ( smallTargetVariation | | smallStraightVariation ) {
if ( attachedOnTheRight ) {
rs . keepLeft = true ;
rs . rightLanes + = lanes ;
2016-06-01 19:46:04 +02:00
if ( turnLanes ! = null ) {
rs . rightLanesInfo . add ( turnLanes ) ;
}
2015-01-15 02:42:33 +01:00
} else {
rs . keepRight = true ;
rs . leftLanes + = lanes ;
2016-06-01 19:46:04 +02:00
if ( turnLanes ! = null ) {
rs . leftLanesInfo . add ( turnLanes ) ;
}
2015-01-15 02:42:33 +01:00
}
2016-05-24 10:31:43 +02:00
rs . speak = rs . speak | | rsSpeakPriority < = speakPriority ;
2015-01-15 02:42:33 +01:00
} else {
if ( attachedOnTheRight ) {
rs . addRoadsOnRight + + ;
} else {
rs . addRoadsOnLeft + + ;
2013-04-18 23:35:02 +02:00
}
}
}
}
2015-01-15 02:42:33 +01:00
return rs ;
}
protected TurnType createSimpleKeepLeftRightTurn ( boolean leftSide , RouteSegmentResult prevSegm ,
RouteSegmentResult currentSegm , RoadSplitStructure rs ) {
2014-11-23 19:45:31 +01:00
int current = countLanesMinOne ( currentSegm ) ;
2015-01-15 02:42:33 +01:00
int ls = current + rs . leftLanes + rs . rightLanes ;
int [ ] lanes = new int [ ls ] ;
2014-11-23 19:45:31 +01:00
for ( int it = 0 ; it < ls ; it + + ) {
2015-01-15 02:42:33 +01:00
if ( it < rs . leftLanes | | it > = rs . leftLanes + current ) {
2014-11-23 19:45:31 +01:00
lanes [ it ] = 0 ;
} else {
lanes [ it ] = 1 ;
}
2013-04-18 23:35:02 +02:00
}
2014-11-23 19:45:31 +01:00
// sometimes links are
2016-05-23 21:22:33 +02:00
if ( ( current < = rs . leftLanes + rs . rightLanes ) & & ( rs . leftLanes > 1 | | rs . rightLanes > 1 ) ) {
rs . speak = true ;
}
2013-04-18 23:35:02 +02:00
double devation = Math . abs ( MapUtils . degreesDiff ( prevSegm . getBearingEnd ( ) , currentSegm . getBearingBegin ( ) ) ) ;
2013-04-21 13:08:21 +02:00
boolean makeSlightTurn = devation > 5 & & ( ! isMotorway ( prevSegm ) | | ! isMotorway ( currentSegm ) ) ;
2015-01-15 02:42:33 +01:00
TurnType t = null ;
if ( rs . keepLeft & & rs . keepRight ) {
2014-11-16 16:04:54 +01:00
t = TurnType . valueOf ( TurnType . C , leftSide ) ;
2015-01-15 02:42:33 +01:00
} else if ( rs . keepLeft ) {
2013-11-26 16:23:05 +01:00
t = TurnType . valueOf ( makeSlightTurn ? TurnType . TSLL : TurnType . KL , leftSide ) ;
2015-01-15 02:42:33 +01:00
} else if ( rs . keepRight ) {
2013-11-26 16:23:05 +01:00
t = TurnType . valueOf ( makeSlightTurn ? TurnType . TSLR : TurnType . KR , leftSide ) ;
2015-01-15 02:42:33 +01:00
} else {
2014-11-16 16:04:54 +01:00
return t ;
2013-04-18 23:35:02 +02:00
}
2015-01-15 02:42:33 +01:00
t . setSkipToSpeak ( ! rs . speak ) ;
t . setLanes ( lanes ) ;
2013-04-18 23:35:02 +02:00
return t ;
}
2014-10-02 17:50:02 +02:00
2014-11-23 19:45:31 +01:00
protected int countLanesMinOne ( RouteSegmentResult attached ) {
final boolean oneway = attached . getObject ( ) . getOneway ( ) ! = 0 ;
int lns = attached . getObject ( ) . getLanes ( ) ;
if ( lns = = 0 ) {
String tls = getTurnLanesString ( attached ) ;
if ( tls ! = null ) {
2014-11-23 19:51:47 +01:00
return Math . max ( 1 , countOccurrences ( tls , '|' ) ) ;
2014-11-23 19:45:31 +01:00
}
}
if ( oneway ) {
return Math . max ( 1 , lns ) ;
}
2014-10-02 17:50:02 +02:00
try {
if ( attached . isForwardDirection ( ) & & attached . getObject ( ) . getValue ( " lanes:forward " ) ! = null ) {
return Integer . parseInt ( attached . getObject ( ) . getValue ( " lanes:forward " ) ) ;
} else if ( ! attached . isForwardDirection ( ) & & attached . getObject ( ) . getValue ( " lanes:backward " ) ! = null ) {
return Integer . parseInt ( attached . getObject ( ) . getValue ( " lanes:backward " ) ) ;
}
2014-11-23 19:51:47 +01:00
} catch ( NumberFormatException e ) {
2014-10-02 17:50:02 +02:00
e . printStackTrace ( ) ;
}
2014-11-23 19:45:31 +01:00
return Math . max ( 1 , ( lns + 1 ) / 2 ) ;
2014-10-02 17:50:02 +02:00
}
2014-10-04 16:39:25 +02:00
2014-11-25 10:48:15 +01:00
protected static String getTurnLanesString ( RouteSegmentResult segment ) {
2014-10-04 16:39:25 +02:00
if ( segment . getObject ( ) . getOneway ( ) = = 0 ) {
if ( segment . isForwardDirection ( ) ) {
return segment . getObject ( ) . getValue ( " turn:lanes:forward " ) ;
} else {
return segment . getObject ( ) . getValue ( " turn:lanes:backward " ) ;
}
} else {
return segment . getObject ( ) . getValue ( " turn:lanes " ) ;
}
}
2015-01-15 02:42:33 +01:00
2014-10-04 16:39:25 +02:00
2014-10-08 04:38:48 +02:00
private int countOccurrences ( String haystack , char needle ) {
int count = 0 ;
for ( int i = 0 ; i < haystack . length ( ) ; i + + ) {
if ( haystack . charAt ( i ) = = needle ) {
count + + ;
}
}
return count ;
}
2014-11-25 10:48:15 +01:00
public static int [ ] parseTurnLanes ( RouteDataObject ro , double dirToNorthEastPi ) {
String turnLanes = null ;
if ( ro . getOneway ( ) = = 0 ) {
// we should get direction to detect forward or backward
double cmp = ro . directionRoute ( 0 , true ) ;
if ( Math . abs ( MapUtils . alignAngleDifference ( dirToNorthEastPi - cmp ) ) < Math . PI / 2 ) {
turnLanes = ro . getValue ( " turn:lanes:forward " ) ;
} else {
turnLanes = ro . getValue ( " turn:lanes:backward " ) ;
}
} else {
turnLanes = ro . getValue ( " turn:lanes " ) ;
}
if ( turnLanes = = null ) {
return null ;
}
2015-01-15 02:42:33 +01:00
return calculateRawTurnLanes ( turnLanes , 0 ) ;
2014-11-25 10:48:15 +01:00
}
2014-12-09 22:34:33 +01:00
public static int [ ] parseLanes ( RouteDataObject ro , double dirToNorthEastPi ) {
int lns = 0 ;
try {
if ( ro . getOneway ( ) = = 0 ) {
// we should get direction to detect forward or backward
double cmp = ro . directionRoute ( 0 , true ) ;
if ( Math . abs ( MapUtils . alignAngleDifference ( dirToNorthEastPi - cmp ) ) < Math . PI / 2 ) {
if ( ro . getValue ( " lanes:forward " ) ! = null ) {
lns = Integer . parseInt ( ro . getValue ( " lanes:forward " ) ) ;
}
} else {
if ( ro . getValue ( " lanes:backward " ) ! = null ) {
lns = Integer . parseInt ( ro . getValue ( " lanes:backward " ) ) ;
}
}
if ( lns = = 0 & & ro . getValue ( " lanes " ) ! = null ) {
lns = Integer . parseInt ( ro . getValue ( " lanes " ) ) / 2 ;
}
} else {
lns = Integer . parseInt ( ro . getValue ( " lanes " ) ) ;
}
if ( lns > 0 ) {
return new int [ lns ] ;
}
} catch ( NumberFormatException e ) {
}
return null ;
}
2015-01-15 02:42:33 +01:00
private static int [ ] calculateRawTurnLanes ( String turnLanes , int calcTurnType ) {
String [ ] splitLaneOptions = turnLanes . split ( " \\ | " , - 1 ) ;
2014-11-23 19:45:31 +01:00
int [ ] lanes = new int [ splitLaneOptions . length ] ;
2014-10-04 16:39:25 +02:00
for ( int i = 0 ; i < splitLaneOptions . length ; i + + ) {
String [ ] laneOptions = splitLaneOptions [ i ] . split ( " ; " ) ;
2016-04-05 11:58:05 +02:00
boolean isTertiaryTurn = false ;
2014-10-04 16:39:25 +02:00
for ( int j = 0 ; j < laneOptions . length ; j + + ) {
2017-04-11 23:41:52 +02:00
int turn = TurnType . convertType ( laneOptions [ j ] ) ;
2014-10-04 16:39:25 +02:00
2015-01-15 02:42:33 +01:00
final int primary = TurnType . getPrimaryTurn ( lanes [ i ] ) ;
if ( primary = = 0 ) {
TurnType . setPrimaryTurnAndReset ( lanes , i , turn ) ;
2014-10-04 16:39:25 +02:00
} else {
2015-01-15 02:42:33 +01:00
if ( turn = = calcTurnType | |
( TurnType . isRightTurn ( calcTurnType ) & & TurnType . isRightTurn ( turn ) ) | |
( TurnType . isLeftTurn ( calcTurnType ) & & TurnType . isLeftTurn ( turn ) )
) {
2016-03-28 15:51:52 +02:00
TurnType . setPrimaryTurnShiftOthers ( lanes , i , turn ) ;
2016-04-05 11:58:05 +02:00
} else if ( ! isTertiaryTurn ) {
2016-03-28 15:51:52 +02:00
TurnType . setSecondaryTurnShiftOthers ( lanes , i , turn ) ;
2016-04-05 11:58:05 +02:00
isTertiaryTurn = true ;
} else {
TurnType . setTertiaryTurn ( lanes , i , turn ) ;
break ;
2014-10-06 20:10:57 +02:00
}
2014-10-04 16:39:25 +02:00
}
}
}
2014-11-23 19:45:31 +01:00
return lanes ;
2014-10-04 16:39:25 +02:00
}
2016-08-11 23:35:04 +02:00
private int inferSlightTurnFromLanes ( int [ ] oLanes , RoadSplitStructure rs ) {
2014-11-23 19:45:31 +01:00
TIntHashSet possibleTurns = new TIntHashSet ( ) ;
for ( int i = 0 ; i < oLanes . length ; i + + ) {
if ( ( oLanes [ i ] & 1 ) = = 0 ) {
2014-10-21 14:50:29 +02:00
continue ;
}
if ( possibleTurns . isEmpty ( ) ) {
// Nothing is in the list to compare to, so add the first elements
2014-11-23 19:45:31 +01:00
possibleTurns . add ( TurnType . getPrimaryTurn ( oLanes [ i ] ) ) ;
if ( TurnType . getSecondaryTurn ( oLanes [ i ] ) ! = 0 ) {
possibleTurns . add ( TurnType . getSecondaryTurn ( oLanes [ i ] ) ) ;
2014-10-21 14:50:29 +02:00
}
2016-03-28 15:51:52 +02:00
if ( TurnType . getTertiaryTurn ( oLanes [ i ] ) ! = 0 ) {
possibleTurns . add ( TurnType . getTertiaryTurn ( oLanes [ i ] ) ) ;
}
2014-10-21 14:50:29 +02:00
} else {
2014-11-23 19:45:31 +01:00
TIntArrayList laneTurns = new TIntArrayList ( ) ;
laneTurns . add ( TurnType . getPrimaryTurn ( oLanes [ i ] ) ) ;
if ( TurnType . getSecondaryTurn ( oLanes [ i ] ) ! = 0 ) {
laneTurns . add ( TurnType . getSecondaryTurn ( oLanes [ i ] ) ) ;
2014-10-21 14:50:29 +02:00
}
2016-03-28 15:51:52 +02:00
if ( TurnType . getTertiaryTurn ( oLanes [ i ] ) ! = 0 ) {
laneTurns . add ( TurnType . getTertiaryTurn ( oLanes [ i ] ) ) ;
}
2014-10-21 14:50:29 +02:00
possibleTurns . retainAll ( laneTurns ) ;
if ( possibleTurns . isEmpty ( ) ) {
// No common turns, so can't determine anything.
2014-11-23 19:45:31 +01:00
return 0 ;
2014-10-21 14:50:29 +02:00
}
}
}
// Remove all turns from lanes not selected...because those aren't it
2014-11-23 19:45:31 +01:00
for ( int i = 0 ; i < oLanes . length ; i + + ) {
if ( ( oLanes [ i ] & 1 ) = = 0 & & ! possibleTurns . isEmpty ( ) ) {
possibleTurns . remove ( ( Integer ) TurnType . getPrimaryTurn ( oLanes [ i ] ) ) ;
if ( TurnType . getSecondaryTurn ( oLanes [ i ] ) ! = 0 ) {
possibleTurns . remove ( ( Integer ) TurnType . getSecondaryTurn ( oLanes [ i ] ) ) ;
2014-10-21 14:50:29 +02:00
}
2016-03-28 15:51:52 +02:00
if ( TurnType . getTertiaryTurn ( oLanes [ i ] ) ! = 0 ) {
possibleTurns . remove ( ( Integer ) TurnType . getTertiaryTurn ( oLanes [ i ] ) ) ;
}
2014-10-21 14:50:29 +02:00
}
}
2017-10-18 13:59:10 +02:00
// remove all non-slight turns // TEST don't pass
// if(possibleTurns.size() > 1) {
// TIntIterator it = possibleTurns.iterator();
// while(it.hasNext()) {
// int nxt = it.next();
// if(!TurnType.isSlightTurn(nxt)) {
// it.remove();
// }
// }
// }
2016-08-11 23:35:04 +02:00
int infer = 0 ;
if ( possibleTurns . size ( ) = = 1 ) {
infer = possibleTurns . iterator ( ) . next ( ) ;
} else if ( possibleTurns . size ( ) > 1 ) {
if ( rs . keepLeft & & rs . keepRight & & possibleTurns . contains ( TurnType . C ) ) {
infer = TurnType . C ;
} else if ( rs . keepLeft | | rs . keepRight ) {
TIntIterator it = possibleTurns . iterator ( ) ;
infer = it . next ( ) ;
while ( it . hasNext ( ) ) {
int next = it . next ( ) ;
int orderInfer = TurnType . orderFromLeftToRight ( infer ) ;
int orderNext = TurnType . orderFromLeftToRight ( next ) ;
if ( rs . keepLeft & & orderNext < orderInfer ) {
infer = next ;
} else if ( rs . keepRight & & orderNext > orderInfer ) {
infer = next ;
}
}
}
}
2014-10-21 14:50:29 +02:00
// Checking to see that there is only one unique turn
2016-08-11 23:35:04 +02:00
if ( infer ! = 0 ) {
2015-01-15 03:17:26 +01:00
for ( int i = 0 ; i < oLanes . length ; i + + ) {
if ( TurnType . getSecondaryTurn ( oLanes [ i ] ) = = infer ) {
int pt = TurnType . getPrimaryTurn ( oLanes [ i ] ) ;
int en = oLanes [ i ] & 1 ;
TurnType . setPrimaryTurnAndReset ( oLanes , i , infer ) ;
oLanes [ i ] | = en ;
TurnType . setSecondaryTurn ( oLanes , i , pt ) ;
}
}
2014-10-21 14:50:29 +02:00
}
2016-08-11 23:35:04 +02:00
return infer ;
2014-10-21 14:50:29 +02:00
}
2013-04-21 13:08:21 +02:00
private boolean isMotorway ( RouteSegmentResult s ) {
String h = s . getObject ( ) . getHighway ( ) ;
return " motorway " . equals ( h ) | | " motorway_link " . equals ( h ) | |
" trunk " . equals ( h ) | | " trunk_link " . equals ( h ) ;
}
2013-04-18 23:35:02 +02:00
private void attachRoadSegments ( RoutingContext ctx , List < RouteSegmentResult > result , int routeInd , int pointInd , boolean plus ) throws IOException {
RouteSegmentResult rr = result . get ( routeInd ) ;
RouteDataObject road = rr . getObject ( ) ;
long nextL = pointInd < road . getPointsLength ( ) - 1 ? getPoint ( road , pointInd + 1 ) : 0 ;
long prevL = pointInd > 0 ? getPoint ( road , pointInd - 1 ) : 0 ;
// attach additional roads to represent more information about the route
RouteSegmentResult previousResult = null ;
// by default make same as this road id
long previousRoadId = road . getId ( ) ;
if ( pointInd = = rr . getStartPointIndex ( ) & & routeInd > 0 ) {
previousResult = result . get ( routeInd - 1 ) ;
previousRoadId = previousResult . getObject ( ) . getId ( ) ;
if ( previousRoadId ! = road . getId ( ) ) {
if ( previousResult . getStartPointIndex ( ) < previousResult . getEndPointIndex ( )
& & previousResult . getEndPointIndex ( ) < previousResult . getObject ( ) . getPointsLength ( ) - 1 ) {
rr . attachRoute ( pointInd , new RouteSegmentResult ( previousResult . getObject ( ) , previousResult . getEndPointIndex ( ) ,
previousResult . getObject ( ) . getPointsLength ( ) - 1 ) ) ;
} else if ( previousResult . getStartPointIndex ( ) > previousResult . getEndPointIndex ( )
& & previousResult . getEndPointIndex ( ) > 0 ) {
rr . attachRoute ( pointInd , new RouteSegmentResult ( previousResult . getObject ( ) , previousResult . getEndPointIndex ( ) , 0 ) ) ;
}
}
}
Iterator < RouteSegment > it ;
if ( rr . getPreAttachedRoutes ( pointInd ) ! = null ) {
final RouteSegmentResult [ ] list = rr . getPreAttachedRoutes ( pointInd ) ;
it = new Iterator < BinaryRoutePlanner . RouteSegment > ( ) {
int i = 0 ;
@Override
public boolean hasNext ( ) {
return i < list . length ;
}
@Override
public RouteSegment next ( ) {
RouteSegmentResult r = list [ i + + ] ;
return new RouteSegment ( r . getObject ( ) , r . getStartPointIndex ( ) ) ;
}
@Override
public void remove ( ) {
}
} ;
} else {
RouteSegment rt = ctx . loadRouteSegment ( road . getPoint31XTile ( pointInd ) , road . getPoint31YTile ( pointInd ) , ctx . config . memoryLimitation ) ;
it = rt = = null ? null : rt . getIterator ( ) ;
}
// try to attach all segments except with current id
while ( it ! = null & & it . hasNext ( ) ) {
RouteSegment routeSegment = it . next ( ) ;
if ( routeSegment . road . getId ( ) ! = road . getId ( ) & & routeSegment . road . getId ( ) ! = previousRoadId ) {
RouteDataObject addRoad = routeSegment . road ;
checkAndInitRouteRegion ( ctx , addRoad ) ;
// TODO restrictions can be considered as well
int oneWay = ctx . getRouter ( ) . isOneWay ( addRoad ) ;
if ( oneWay > = 0 & & routeSegment . getSegmentStart ( ) < addRoad . getPointsLength ( ) - 1 ) {
long pointL = getPoint ( addRoad , routeSegment . getSegmentStart ( ) + 1 ) ;
if ( pointL ! = nextL & & pointL ! = prevL ) {
// if way contains same segment (nodes) as different way (do not attach it)
rr . attachRoute ( pointInd , new RouteSegmentResult ( addRoad , routeSegment . getSegmentStart ( ) , addRoad . getPointsLength ( ) - 1 ) ) ;
}
}
if ( oneWay < = 0 & & routeSegment . getSegmentStart ( ) > 0 ) {
long pointL = getPoint ( addRoad , routeSegment . getSegmentStart ( ) - 1 ) ;
// if way contains same segment (nodes) as different way (do not attach it)
if ( pointL ! = nextL & & pointL ! = prevL ) {
rr . attachRoute ( pointInd , new RouteSegmentResult ( addRoad , routeSegment . getSegmentStart ( ) , 0 ) ) ;
}
}
}
}
}
private static void println ( String logMsg ) {
// log.info(logMsg);
System . out . println ( logMsg ) ;
}
private long getPoint ( RouteDataObject road , int pointInd ) {
return ( ( ( long ) road . getPoint31XTile ( pointInd ) ) < < 31 ) + ( long ) road . getPoint31YTile ( pointInd ) ;
}
private static double measuredDist ( int x1 , int y1 , int x2 , int y2 ) {
return MapUtils . getDistance ( MapUtils . get31LatitudeY ( y1 ) , MapUtils . get31LongitudeX ( x1 ) ,
MapUtils . get31LatitudeY ( y2 ) , MapUtils . get31LongitudeX ( x2 ) ) ;
}
2018-10-26 15:33:56 +02:00
2018-10-26 18:07:14 +02:00
2013-04-18 23:35:02 +02:00
}