diff --git a/OsmAnd-java/src/main/java/net/osmand/binary/BinaryMapIndexReader.java b/OsmAnd-java/src/main/java/net/osmand/binary/BinaryMapIndexReader.java index 86fce4f208..c8f84796b9 100644 --- a/OsmAnd-java/src/main/java/net/osmand/binary/BinaryMapIndexReader.java +++ b/OsmAnd-java/src/main/java/net/osmand/binary/BinaryMapIndexReader.java @@ -2655,6 +2655,7 @@ public class BinaryMapIndexReader { } } + public TLongObjectHashMap getIncompleteTransportRoutes() throws InvalidProtocolBufferException, IOException { if (incompleteTransportRoutes == null) { incompleteTransportRoutes = new TLongObjectHashMap<>(); @@ -2662,7 +2663,7 @@ public class BinaryMapIndexReader { if (ti.incompleteRoutesLength > 0) { codedIS.seek(ti.incompleteRoutesOffset); int oldLimit = codedIS.pushLimit(ti.incompleteRoutesLength); - transportAdapter.readIncompleteRoutesList(incompleteTransportRoutes); + transportAdapter.readIncompleteRoutesList(incompleteTransportRoutes, ti.filePointer); codedIS.popLimit(oldLimit); } } @@ -2670,5 +2671,4 @@ public class BinaryMapIndexReader { return incompleteTransportRoutes; } - } diff --git a/OsmAnd-java/src/main/java/net/osmand/binary/BinaryMapTransportReaderAdapter.java b/OsmAnd-java/src/main/java/net/osmand/binary/BinaryMapTransportReaderAdapter.java index 358e908dbd..1e733e50f6 100644 --- a/OsmAnd-java/src/main/java/net/osmand/binary/BinaryMapTransportReaderAdapter.java +++ b/OsmAnd-java/src/main/java/net/osmand/binary/BinaryMapTransportReaderAdapter.java @@ -11,6 +11,7 @@ import com.google.protobuf.WireFormat; import gnu.trove.map.hash.TIntObjectHashMap; import gnu.trove.map.hash.TLongObjectHashMap; import net.osmand.binary.BinaryMapIndexReader.SearchRequest; +import net.osmand.data.IncompleteTransportRoute; import net.osmand.data.TransportSchedule; import net.osmand.data.TransportStop; import net.osmand.data.TransportStopExit; @@ -71,7 +72,6 @@ public class BinaryMapTransportReaderAdapter { return bottom; } - IndexStringTable stringTable = null; } @@ -251,7 +251,7 @@ public class BinaryMapTransportReaderAdapter { return ((char) i)+""; } - public void readIncompleteRoutesList(TLongObjectHashMap incompleteRoutes) throws IOException { + public void readIncompleteRoutesList(TLongObjectHashMap incompleteRoutes, int transportIndexStart) throws IOException { boolean end = false; while (!end) { int t = codedIS.readTag(); @@ -263,7 +263,7 @@ public class BinaryMapTransportReaderAdapter { case OsmandOdb.IncompleteTransportRoutes.ROUTES_FIELD_NUMBER: int l = codedIS.readRawVarint32(); int olds = codedIS.pushLimit(l); - net.osmand.data.IncompleteTransportRoute ir = readIncompleteRoute(); + net.osmand.data.IncompleteTransportRoute ir = readIncompleteRoute(transportIndexStart); net.osmand.data.IncompleteTransportRoute itr = incompleteRoutes.get(ir.getRouteId()); if(itr != null) { itr.setNextLinkedRoute(ir); @@ -281,7 +281,7 @@ public class BinaryMapTransportReaderAdapter { } - public net.osmand.data.IncompleteTransportRoute readIncompleteRoute() throws IOException { + public net.osmand.data.IncompleteTransportRoute readIncompleteRoute(int transportIndexStart) throws IOException { net.osmand.data.IncompleteTransportRoute dataObject = new net.osmand.data.IncompleteTransportRoute(); boolean end = false; while(!end){ @@ -295,7 +295,12 @@ public class BinaryMapTransportReaderAdapter { dataObject.setRouteId(codedIS.readUInt64()); break; case OsmandOdb.IncompleteTransportRoute.ROUTEREF_FIELD_NUMBER : - dataObject.setRouteOffset(codedIS.readRawVarint32()); + int delta = codedIS.readRawVarint32(); + if (delta > transportIndexStart) { + dataObject.setRouteOffset(delta); + } else { + dataObject.setRouteOffset(transportIndexStart + delta); + } break; case OsmandOdb.IncompleteTransportRoute.OPERATOR_FIELD_NUMBER : skipUnknownField(t); @@ -318,7 +323,6 @@ public class BinaryMapTransportReaderAdapter { break; } } - return dataObject; } diff --git a/OsmAnd-java/src/main/java/net/osmand/data/IncompleteTransportRoute.java b/OsmAnd-java/src/main/java/net/osmand/data/IncompleteTransportRoute.java index 84be533a5f..9518f1237d 100644 --- a/OsmAnd-java/src/main/java/net/osmand/data/IncompleteTransportRoute.java +++ b/OsmAnd-java/src/main/java/net/osmand/data/IncompleteTransportRoute.java @@ -13,7 +13,11 @@ public class IncompleteTransportRoute { } public void setNextLinkedRoute(IncompleteTransportRoute nextLinkedRoute) { - this.nextLinkedRoute = nextLinkedRoute; + if (this.nextLinkedRoute == null) { + this.nextLinkedRoute = nextLinkedRoute; + } else { + this.nextLinkedRoute.setNextLinkedRoute(nextLinkedRoute); + } } public long getRouteId() { diff --git a/OsmAnd-java/src/main/java/net/osmand/router/BinaryRoutePlanner.java b/OsmAnd-java/src/main/java/net/osmand/router/BinaryRoutePlanner.java index 154d31928e..1b6077cbb1 100644 --- a/OsmAnd-java/src/main/java/net/osmand/router/BinaryRoutePlanner.java +++ b/OsmAnd-java/src/main/java/net/osmand/router/BinaryRoutePlanner.java @@ -370,10 +370,10 @@ public class BinaryRoutePlanner { public void printDebugMemoryInformation(RoutingContext ctx, PriorityQueue graphDirectSegments, PriorityQueue graphReverseSegments, TLongObjectHashMap visitedDirectSegments,TLongObjectHashMap visitedOppositeSegments) { - printInfo(String.format("Time. Total: %.2f, to load: %.2f, to load headers: %.2f, to calc dev: %.2f, to calc rules: %.2f ", + printInfo(String.format("Time. Total: %.2f, to load: %.2f, to load headers: %.2f, to calc dev: %.2f ", (System.nanoTime() - ctx.timeToCalculate) / 1e6, ctx.timeToLoad / 1e6, - ctx.timeToLoadHeaders / 1e6, ctx.timeNanoToCalcDeviation / 1e6, GeneralRouter.TIMER / 1e6)); - GeneralRouter.TIMER = 0; + ctx.timeToLoadHeaders / 1e6, ctx.timeNanoToCalcDeviation / 1e6)); +// GeneralRouter.TIMER = 0; int maxLoadedTiles = Math.max(ctx.maxLoadedTiles, ctx.getCurrentlyLoadedTiles()); printInfo("Current loaded tiles : " + ctx.getCurrentlyLoadedTiles() + ", maximum loaded tiles " + maxLoadedTiles); printInfo("Loaded tiles " + ctx.loadedTiles + " (distinct " + ctx.distinctLoadedTiles + "), unloaded tiles " + ctx.unloadedTiles + @@ -856,6 +856,13 @@ public class BinaryRoutePlanner { super(road, segmentStart); this.distSquare = distSquare; } + + public RouteSegmentPoint(RouteSegmentPoint pnt) { + super(pnt.road, pnt.segStart); + this.distSquare = pnt.distSquare; + this.preciseX = pnt.preciseX; + this.preciseY = pnt.preciseY; + } public double distSquare; public int preciseX; diff --git a/OsmAnd-java/src/main/java/net/osmand/router/GeneralRouter.java b/OsmAnd-java/src/main/java/net/osmand/router/GeneralRouter.java index 0032fb200c..f8de46a783 100644 --- a/OsmAnd-java/src/main/java/net/osmand/router/GeneralRouter.java +++ b/OsmAnd-java/src/main/java/net/osmand/router/GeneralRouter.java @@ -516,6 +516,7 @@ public class GeneralRouter implements VehicleRouter { } private void putCache(RouteDataObjectAttribute attr, RouteRegion reg, int[] types, Float val, boolean extra) { +// TIMER -= System.nanoTime(); Map> ch = evalCache[attr.ordinal()]; if (USE_CACHE) { Map rM = ch.get(reg); @@ -525,7 +526,7 @@ public class GeneralRouter implements VehicleRouter { } rM.put(new IntHolder(types, extra), val); } - TIMER += System.nanoTime(); +// TIMER += System.nanoTime(); } class IntHolder { diff --git a/OsmAnd-java/src/main/java/net/osmand/router/RoutePlannerFrontEnd.java b/OsmAnd-java/src/main/java/net/osmand/router/RoutePlannerFrontEnd.java index 80f2a7382b..acf76552a4 100644 --- a/OsmAnd-java/src/main/java/net/osmand/router/RoutePlannerFrontEnd.java +++ b/OsmAnd-java/src/main/java/net/osmand/router/RoutePlannerFrontEnd.java @@ -32,10 +32,6 @@ public class RoutePlannerFrontEnd { protected static final double GPS_POSSIBLE_ERROR = 7; public boolean useSmartRouteRecalculation = true; - public static double AVERAGE_SPLIT_DISTANCE_GPX = 1500; - public static double MINIMUM_POINT_APPROXIMATION = 50; - public static double MINIMUM_STRAIGHT_STEP_APPROXIMATION = 50; - public RoutePlannerFrontEnd() { } @@ -46,14 +42,29 @@ public class RoutePlannerFrontEnd { COMPLEX } - public static class GpxApproximationResult { + public static class GpxRouteApproximation { + // ! MAIN parameter to approximate (35m good for custom recorded tracks) + public double MINIMUM_POINT_APPROXIMATION = 50; // 35 m good for small deviations + // This parameter could speed up or slow down evaluation (better to make bigger for long routes and smaller for short) + public double MAXIMUM_STEP_APPROXIMATION = 3000; + // don't search subsegments shorter than specified distance (also used to step back for car turns) + public double MINIMUM_STEP_APPROXIMATION = 100; + // Parameter to smoother the track itself (could be 0 if it's not recorded track) + public double SMOOTHEN_POINTS_NO_ROUTE = 2; + + public final RoutingContext ctx; public int routeCalculations = 0; public int routePointsSearched = 0; public int routeDistCalculations = 0; public List res = new ArrayList(); public int routeDistance; + public int routeGapDistance; public int routeDistanceUnmatched; + public GpxRouteApproximation(RoutingContext ctx) { + this.ctx = ctx; + } + @Override public String toString() { return String.format(">> GPX approximation (%d of %d m route calcs, %d route points searched) for %d m: %d m umatched", @@ -192,91 +203,152 @@ public class RoutePlannerFrontEnd { } - // TODO add missing turns for straight lines - // TODO smoothness is not correct for car routing - // TODO native crash - // TODO big gaps when there is no match - // TODO not correct bicycle-> pedestrian - // TODO slow - too many findRouteSegment - // TODO fix progress / timings routing / - // TODO smoothen straight line Douglas-Peucker (remove noise) - public GpxApproximationResult searchGpxRoute(final RoutingContext ctx, List points) throws IOException, InterruptedException { - GpxApproximationResult gctx = new GpxApproximationResult(); + // TODO last segment not correct (cut) before end point and point of straight line + + // TODO add missing turns for straight lines (compare code) + // TODO native matches less roads + + // TODO fix progress - next iteration + // TODO fix timings and remove logging every iteration + public GpxRouteApproximation searchGpxRoute(GpxRouteApproximation gctx, List points) throws IOException, InterruptedException { + gctx.ctx.timeToCalculate = System.nanoTime(); + if (gctx.ctx.calculationProgress == null) { + gctx.ctx.calculationProgress = new RouteCalculationProgress(); + } List gpxPoints = generageGpxPoints(points, gctx); GpxPoint start = gpxPoints.size() > 0 ? gpxPoints.get(0) : null; - + boolean prevRouteFound = false; while (start != null) { - double routeDist = AVERAGE_SPLIT_DISTANCE_GPX; + double routeDist = gctx.MAXIMUM_STEP_APPROXIMATION; GpxPoint next = findNextGpxPointWithin(gctx, gpxPoints, start, routeDist); boolean routeFound = false; - if (next != null && initRoutingPoint(start, gctx, ctx, MINIMUM_POINT_APPROXIMATION)) { - boolean firstAttempt = true; - while ((firstAttempt || next.cumDist - start.cumDist > MINIMUM_POINT_APPROXIMATION) && !routeFound) { - firstAttempt = false; - routeFound = initRoutingPoint(next, gctx, ctx, MINIMUM_POINT_APPROXIMATION); + + if (next != null && initRoutingPoint(start, gctx, gctx.MINIMUM_POINT_APPROXIMATION)) { + while (routeDist >= gctx.MINIMUM_STEP_APPROXIMATION && !routeFound) { + routeFound = initRoutingPoint(next, gctx, gctx.MINIMUM_POINT_APPROXIMATION); if (routeFound) { - routeFound = findGpxRouteSegment(ctx, gctx, gpxPoints, start, next); + routeFound = findGpxRouteSegment(gctx, gpxPoints, start, next, prevRouteFound); + if (routeFound) { + // route is found - cut the end of the route and move to next iteration + boolean stepBack = stepBackAndFindPrevPointInRoute(gctx, gpxPoints, start, next); + if (!stepBack) { + // not supported case (workaround increase MAXIMUM_STEP_APPROXIMATION) + log.info("Consider to increase MAXIMUM_STEP_APPROXIMATION to: " + routeDist*2); + start.routeToTarget = null; + routeFound = false; + break; + } + } } if (!routeFound) { // route is not found move next point closer to start point (distance / 2) routeDist = routeDist / 2; + if (routeDist < gctx.MINIMUM_STEP_APPROXIMATION && routeDist > gctx.MINIMUM_STEP_APPROXIMATION / 2 + 1) { + routeDist = gctx.MINIMUM_STEP_APPROXIMATION; + } next = findNextGpxPointWithin(gctx, gpxPoints, start, routeDist); + if (next != null) { + routeDist = Math.min(next.cumDist - start.cumDist, routeDist); + } } } } - if (routeFound) { - // route is found, cut the end of the route and move to next iteration - start = next; - } else { + // route is not found skip segment and keep it as straight line on display + if (!routeFound) { // route is not found, move start point by - start = findNextGpxPointWithin(gctx, gpxPoints, start, MINIMUM_STRAIGHT_STEP_APPROXIMATION); + next = findNextGpxPointWithin(gctx, gpxPoints, start, gctx.MINIMUM_STEP_APPROXIMATION); + if (prevRouteFound) { + if (next == null) { + // TODO finish + // makeSegmentPointPrecise(prev.routeToTarget.get(prev.routeToTarget.size() - 1), start.loc, false); + } else { + log.warn("NOT found route from: " + start.pnt.getRoad() + " at " + start.pnt.getSegmentStart()); + } + } } - + prevRouteFound = routeFound; + start = next; } - calculateGpxRoute(ctx, gctx, gpxPoints); + + calculateGpxRoute(gctx, gpxPoints); if (!gctx.res.isEmpty()) { - new RouteResultPreparation().printResults(ctx, points.get(0), points.get(points.size() - 1), gctx.res); + new RouteResultPreparation().printResults(gctx.ctx, points.get(0), points.get(points.size() - 1), gctx.res); System.out.println(gctx); } return gctx; } - private void calculateGpxRoute(final RoutingContext ctx, GpxApproximationResult gctx, List gpxPoints) { + private boolean stepBackAndFindPrevPointInRoute(GpxRouteApproximation gctx, + List gpxPoints, GpxPoint start, GpxPoint next) throws IOException { + // step back to find to be sure + // 1) route point is behind GpxPoint - MINIMUM_POINT_APPROXIMATION (end route point could slightly ahead) + // 2) we don't miss correct turn i.e. points could be attached to muliple routes + // 3) to make sure that we perfectly connect to RoadDataObject points + double STEP_BACK_DIST = Math.max(gctx.MINIMUM_POINT_APPROXIMATION, gctx.MINIMUM_STEP_APPROXIMATION); + double d = 0; + int segmendInd = start.routeToTarget.size() - 1; + boolean search = true; + mainLoop: for (; segmendInd >= 0 && search; segmendInd--) { + RouteSegmentResult rr = start.routeToTarget.get(segmendInd); + boolean minus = rr.getStartPointIndex() < rr.getEndPointIndex(); + int nextInd; + for (int j = rr.getEndPointIndex(); j != rr.getStartPointIndex(); j = nextInd) { + nextInd = minus ? j - 1 : j + 1; + d += MapUtils.getDistance(rr.getPoint(j), rr.getPoint(nextInd)); + if (d > STEP_BACK_DIST) { + if (nextInd == rr.getStartPointIndex()) { + segmendInd--; + } else { + rr.setEndPointIndex(nextInd); + } + search = false; + break mainLoop; + } + } + } + if (segmendInd == -1) { + // here all route segments - 1 is longer than needed distance to step back + return false; + } + while (start.routeToTarget.size() > segmendInd + 1) { + start.routeToTarget.remove(segmendInd + 1); + } + RouteSegmentResult res = start.routeToTarget.get(segmendInd); + next.pnt = new RouteSegmentPoint(res.getObject(), res.getEndPointIndex(), 0); + return true; + } + + private void calculateGpxRoute(GpxRouteApproximation gctx, List gpxPoints) { RouteRegion reg = new RouteRegion(); reg.initRouteEncodingRule(0, "highway", "unmatched"); - TIntArrayList lastStraightLine = null; + List lastStraightLine = null; for (int i = 0; i < gpxPoints.size(); ) { GpxPoint pnt = gpxPoints.get(i); if (pnt.routeToTarget != null && !pnt.routeToTarget.isEmpty()) { LatLon startPoint = pnt.routeToTarget.get(0).getStartPoint(); if (lastStraightLine != null) { - lastStraightLine.add(MapUtils.get31TileNumberX(startPoint.getLongitude())); - lastStraightLine.add(MapUtils.get31TileNumberY(startPoint.getLatitude())); + lastStraightLine.add(startPoint); addStraightLine(gctx.res, lastStraightLine, reg, gctx); lastStraightLine = null; } - // TODO - double distLastPnt = gctx.distFromLastPoint(pnt.routeToTarget.get(0).getStartPoint()); - double gpxDistPnt = gctx.distFromLastPoint(pnt.loc); - if (distLastPnt > MINIMUM_POINT_APPROXIMATION / 5 || gpxDistPnt > MINIMUM_POINT_APPROXIMATION / 5) { - System.out.println(String.format("????? routePnt - prevPnt = %f, gpxPoint - prevPnt = %f ", - distLastPnt, gpxDistPnt)); + if (gctx.distFromLastPoint(startPoint) > 1) { + gctx.routeGapDistance += gctx.distFromLastPoint(startPoint); + System.out.println(String.format("????? gap of route point = %f, gap of actual gpxPoint = %f ", + gctx.distFromLastPoint(startPoint), gctx.distFromLastPoint(pnt.loc))); } gctx.res.addAll(pnt.routeToTarget); i = pnt.targetInd; } else { // add straight line from i -> i+1 if (lastStraightLine == null) { - lastStraightLine = new TIntArrayList(); + lastStraightLine = new ArrayList(); // make smooth connection if(gctx.distFromLastPoint(pnt.loc) > 1) { - lastStraightLine.add(MapUtils.get31TileNumberX(gctx.getLastPoint().getLongitude())); - lastStraightLine.add(MapUtils.get31TileNumberY(gctx.getLastPoint().getLatitude())); + lastStraightLine.add(gctx.getLastPoint()); } } - lastStraightLine.add(MapUtils.get31TileNumberX(pnt.loc.getLongitude())); - lastStraightLine.add(MapUtils.get31TileNumberY(pnt.loc.getLatitude())); + lastStraightLine.add(pnt.loc); i++; } } @@ -285,10 +357,10 @@ public class RoutePlannerFrontEnd { lastStraightLine = null; } // clean turns to recaculate them - cleanupResultAndAddTurns(ctx, gctx); + cleanupResultAndAddTurns(gctx); } - private List generageGpxPoints(List points, GpxApproximationResult gctx) { + private List generageGpxPoints(List points, GpxRouteApproximation gctx) { List gpxPoints = new ArrayList<>(points.size()); GpxPoint prev = null; for(int i = 0; i < points.size(); i++) { @@ -305,7 +377,7 @@ public class RoutePlannerFrontEnd { return gpxPoints; } - private void cleanupResultAndAddTurns(final RoutingContext ctx, GpxApproximationResult gctx) { + private void cleanupResultAndAddTurns(GpxRouteApproximation gctx) { // cleanup double joints int LOOK_AHEAD = 4; for(int i = 0; i < gctx.res.size(); i++) { @@ -325,71 +397,119 @@ public class RoutePlannerFrontEnd { r.setTurnType(null); r.setDescription(""); } - preparation.prepareTurnResults(ctx, gctx.res); + preparation.prepareTurnResults(gctx.ctx, gctx.res); } - private void addStraightLine(List res, TIntArrayList lastStraightLine, RouteRegion reg, GpxApproximationResult gctx) { + private void addStraightLine(List res, List lastStraightLine, RouteRegion reg, GpxRouteApproximation gctx) { RouteDataObject rdo = new RouteDataObject(reg); - int l = lastStraightLine.size() / 2; - rdo.pointsX = new int[l]; - rdo.pointsY = new int[l]; - rdo.types = new int[] { 0 } ; - rdo.id = -1; - for (int i = 0; i < l; i++) { - rdo.pointsX[i] = lastStraightLine.get(i * 2); - rdo.pointsY[i] = lastStraightLine.get(i * 2 + 1); - if(i > 0) { - double dist = MapUtils.squareRootDist31(rdo.pointsX[i], rdo.pointsY[i], rdo.pointsX[i-1], rdo.pointsY[i-1]); - gctx.routeDistanceUnmatched += dist; + if(gctx.SMOOTHEN_POINTS_NO_ROUTE > 0) { + simplifyDouglasPeucker(lastStraightLine, gctx.SMOOTHEN_POINTS_NO_ROUTE, 0, lastStraightLine.size() - 1); + } + int s = lastStraightLine.size(); + TIntArrayList x = new TIntArrayList(s); + TIntArrayList y = new TIntArrayList(s); + for (int i = 0; i < s; i++) { + if(lastStraightLine.get(i) != null) { + LatLon l = lastStraightLine.get(i); + int t = x.size() - 1; + x.add(MapUtils.get31TileNumberX(l.getLongitude())); + y.add(MapUtils.get31TileNumberY(l.getLatitude())); + if (t >= 0) { + double dist = MapUtils.squareRootDist31(x.get(t), y.get(t), x.get(t + 1), y.get(t + 1)); + gctx.routeDistanceUnmatched += dist; + } } } + rdo.pointsX = x.toArray(); + rdo.pointsY = y.toArray(); + rdo.types = new int[] { 0 } ; + rdo.id = -1; + // comment to see road without straight connections res.add(new RouteSegmentResult(rdo, 0, rdo.getPointsLength() - 1)); } - private boolean initRoutingPoint(GpxPoint start, GpxApproximationResult gctx, RoutingContext ctx, double distThreshold) throws IOException { + + private void simplifyDouglasPeucker(List l, double eps, int start, int end) { + double dmax = -1; + int index = -1; + LatLon s = l.get(start); + LatLon e = l.get(end); + for (int i = start + 1; i <= end - 1; i++) { + LatLon ip = l.get(i); + double dist = MapUtils.getOrthogonalDistance(ip.getLatitude(), ip.getLongitude(), s.getLatitude(), s.getLongitude(), + e.getLatitude(), e.getLongitude()); + if (dist > dmax) { + dmax = dist; + index = i; + } + } + if (dmax >= eps) { + simplifyDouglasPeucker(l, eps, start, index); + simplifyDouglasPeucker(l, eps, index, end); + } else { + for(int i = start + 1; i < end; i++ ) { + l.set(i, null); + } + } + } + + private boolean initRoutingPoint(GpxPoint start, GpxRouteApproximation gctx, double distThreshold) throws IOException { if (start != null && start.pnt == null) { gctx.routePointsSearched++; - RouteSegmentPoint rsp = findRouteSegment(start.loc.getLatitude(), start.loc.getLongitude(), ctx, null, false); + RouteSegmentPoint rsp = findRouteSegment(start.loc.getLatitude(), start.loc.getLongitude(), gctx.ctx, null, false); if (MapUtils.getDistance(rsp.getPreciseLatLon(), start.loc) < distThreshold) { start.pnt = rsp; } + } + if (start != null && start.pnt != null) { + return true; } - return start != null && start.pnt != null; + return false; } - private GpxPoint findNextGpxPointWithin(GpxApproximationResult gctx, List gpxPoints, + private GpxPoint findNextGpxPointWithin(GpxRouteApproximation gctx, List gpxPoints, GpxPoint start, double dist) { - int targetInd = start.ind + 1; + // returns first point with that has slightly more than dist or last point + int plus = dist > 0 ? 1 : -1; + int targetInd = start.ind + plus; GpxPoint target = null; - while (targetInd < gpxPoints.size()) { + while (targetInd < gpxPoints.size() && targetInd >= 0) { target = gpxPoints.get(targetInd); - if (target.cumDist - start.cumDist > dist) { + if (Math.abs(target.cumDist - start.cumDist) > Math.abs(dist)) { break; } - targetInd++; + targetInd = targetInd + plus; } return target; } - private boolean findGpxRouteSegment(final RoutingContext ctx, GpxApproximationResult gctx, List gpxPoints, - GpxPoint start, GpxPoint target) throws IOException, InterruptedException { + private boolean findGpxRouteSegment(GpxRouteApproximation gctx, List gpxPoints, + GpxPoint start, GpxPoint target, boolean prevRouteCalculated) throws IOException, InterruptedException { List res = null; boolean routeIsCorrect = false; if (start.pnt != null && target.pnt != null) { + start.pnt = new RouteSegmentPoint(start.pnt); + target.pnt = new RouteSegmentPoint(target.pnt); gctx.routeDistCalculations += (target.cumDist - start.cumDist); gctx.routeCalculations++; - res = searchRouteInternalPrepare(ctx, start.pnt, target.pnt, null); + res = searchRouteInternalPrepare(gctx.ctx, start.pnt, target.pnt, null); routeIsCorrect = res != null && !res.isEmpty(); - if (routeIsCorrect) { - makeStartEndPointsPrecise(res, start.pnt.getPreciseLatLon(), target.pnt.getPreciseLatLon(), null); - } for (int k = start.ind + 1; routeIsCorrect && k < target.ind; k++) { GpxPoint ipoint = gpxPoints.get(k); - if (!pointCloseEnough(ipoint, res)) { + if (!pointCloseEnough(gctx, ipoint, res)) { routeIsCorrect = false; } } if (routeIsCorrect) { + // correct start point though don't change end point + if (!prevRouteCalculated) { + // make first position precise + makeSegmentPointPrecise(res.get(0), start.loc, true); + } else { + assert res.get(0).getObject().getId() == start.pnt.getRoad().getId(); + // start point could shift to +-1 due to direction + res.get(0).setStartPointIndex(start.pnt.getSegmentStart()); + } start.routeToTarget = res; start.targetInd = target.ind; } @@ -397,10 +517,11 @@ public class RoutePlannerFrontEnd { return routeIsCorrect; } - private boolean pointCloseEnough(GpxPoint ipoint, List res) { + private boolean pointCloseEnough(GpxRouteApproximation gctx, GpxPoint ipoint, List res) { int px = MapUtils.get31TileNumberX(ipoint.loc.getLongitude()); int py = MapUtils.get31TileNumberY(ipoint.loc.getLatitude()); - double SQR = MINIMUM_POINT_APPROXIMATION * MINIMUM_POINT_APPROXIMATION * 4; + double SQR = gctx.MINIMUM_POINT_APPROXIMATION; + SQR = SQR * SQR; for (RouteSegmentResult sr : res) { int start = sr.getStartPointIndex(); int end = sr.getEndPointIndex(); @@ -525,8 +646,8 @@ public class RoutePlannerFrontEnd { protected void makeStartEndPointsPrecise(List res, LatLon start, LatLon end, List intermediates) { if (res.size() > 0) { - updateResult(res.get(0), start, true); - updateResult(res.get(res.size() - 1), end, false); + makeSegmentPointPrecise(res.get(0), start, true); + makeSegmentPointPrecise(res.get(res.size() - 1), end, false); } } @@ -540,7 +661,7 @@ public class RoutePlannerFrontEnd { return currentsDist; } - private void updateResult(RouteSegmentResult routeSegmentResult, LatLon point, boolean st) { + private void makeSegmentPointPrecise(RouteSegmentResult routeSegmentResult, LatLon point, boolean st) { int px = MapUtils.get31TileNumberX(point.getLongitude()); int py = MapUtils.get31TileNumberY(point.getLatitude()); int pind = st ? routeSegmentResult.getStartPointIndex() : routeSegmentResult.getEndPointIndex(); diff --git a/OsmAnd-java/src/main/java/net/osmand/router/TransportRoutePlanner.java b/OsmAnd-java/src/main/java/net/osmand/router/TransportRoutePlanner.java index c9d471a497..c94831d8f4 100644 --- a/OsmAnd-java/src/main/java/net/osmand/router/TransportRoutePlanner.java +++ b/OsmAnd-java/src/main/java/net/osmand/router/TransportRoutePlanner.java @@ -865,7 +865,6 @@ public class TransportRoutePlanner { lst.add(segment); } } else { - // MapUtils.getDistance(s.getLocation(), route.getForwardStops().get(158).getLocation()); System.err.println(String.format("Routing error: missing stop '%s' in route '%s' id: %d", s.toString(), route.getRef(), route.getId() / 2)); } diff --git a/OsmAnd-java/src/main/java/net/osmand/router/TransportStopsRouteReader.java b/OsmAnd-java/src/main/java/net/osmand/router/TransportStopsRouteReader.java index d077335d0b..7428196332 100644 --- a/OsmAnd-java/src/main/java/net/osmand/router/TransportStopsRouteReader.java +++ b/OsmAnd-java/src/main/java/net/osmand/router/TransportStopsRouteReader.java @@ -8,6 +8,7 @@ import java.util.LinkedHashMap; import java.util.LinkedList; import java.util.List; import java.util.Map; +import java.util.Map.Entry; import gnu.trove.iterator.TIntObjectIterator; import gnu.trove.list.array.TIntArrayList; @@ -15,6 +16,7 @@ import gnu.trove.map.hash.TIntObjectHashMap; import gnu.trove.map.hash.TLongObjectHashMap; import net.osmand.binary.BinaryMapIndexReader; import net.osmand.binary.BinaryMapIndexReader.SearchRequest; +import net.osmand.binary.BinaryMapTransportReaderAdapter.TransportIndex; import net.osmand.data.IncompleteTransportRoute; import net.osmand.data.LatLon; import net.osmand.data.TransportRoute; @@ -43,7 +45,6 @@ public class TransportStopsRouteReader { List stops = r.searchTransportIndex(sr); TIntObjectHashMap routesToLoad = mergeTransportStops(r, loadedTransportStops, stops); loadRoutes(r, routesToLoad); - for (TransportStop stop : stops) { // skip missing stops if (stop.isMissingStop()) { @@ -405,11 +406,4 @@ public class TransportStopsRouteReader { } return allRoutes; } - - - - - - - } diff --git a/OsmAnd/res/values-ar/strings.xml b/OsmAnd/res/values-ar/strings.xml index bf0a05f3ff..064a00bc8d 100644 --- a/OsmAnd/res/values-ar/strings.xml +++ b/OsmAnd/res/values-ar/strings.xml @@ -1898,7 +1898,7 @@ تعطيل 2-مرحلة التوجيه للملاحة بالسيارة. مقياس التنزه في جبال الألب (SAC) تقديم المسارات وفقاً لمقياس SAC. - طبقة رمز التسلق + كيفية اظهار رموز طبقة التسلق (الهايكنق) تقديم المسارات وفقاً لآثار OSMC. انقر فوق أي عنصر لمعرفة المزيد من التفاصيل، اضغط وواصل الضغط للتعطيل أو الحذف. البيانات الحالية على الجهاز (%1$s متاح): إعادة إعلان تعليمات الملاحة في فترات منتظمة. @@ -3797,4 +3797,6 @@ طول الحد هذا الجهاز لا يملك كاميرات السرعة. أحذية تزلج + تمكين التحكم في مستوى التكبير/التصغير بالخريطة باستخدام أزرار مستوى صوت الجهاز. + أزرار مستوى الصوت \ No newline at end of file diff --git a/OsmAnd/res/values-ca/strings.xml b/OsmAnd/res/values-ca/strings.xml index 4c7e5d959a..360d9eea91 100644 --- a/OsmAnd/res/values-ca/strings.xml +++ b/OsmAnd/res/values-ca/strings.xml @@ -3665,4 +3665,38 @@ Abasta l\'àrea: %1$s x %2$s Javanès Gujarati Trieu com desar les tessel·les baixades. + Desinstal·la les càmeres de radar + Informació legal + PDIs de càmeres de radar + En alguns països o regions, l\'ús d\'aplicacions que avisen de la proximitat de càmeres de radar està prohibit. +\n +\nCal que ho configureu segons la llei del vostre país. +\n +\nSeleccioneu %1$s i rebreu alertes i advertiments de les càmeres de radar. +\n +\nSeleccioneu %2$s. Totes les dades relatives a les càmeres de radar (alertes, notificacions, PDIs) desapareixeran fins que es reinstal·li completament OsmAnd . + Desinstal·la + Les alertes de proximitat a càmeres de radar estan prohibides en alguns països. + Mostra o amaga detalls addicionals del mapa + Mapa nocturn + Afegeix orígen en connexió + L\'aplicació d\'aquests canvis esborrarà les dades de la memòria cau per a aquesta font de tessel·les + Indiqueu l\'alçada del vaixell + Podeu indicar l\'alçada del vaixell per evitar ponts baixos. Penseu que si el pont és mòbil, li aplicarem l\'alçada de quan estigui obert. + Indiqueu l\'alçada del vaixell per evitar ponts baixos. Penseu que si el pont és mòbil, li aplicarem l\'alçada de quan estigui obert. + Indiqueu l\'amplada del vaixell per evitar ponts ajustats + Mostra/amaga Mapil·lary + Amaga Mapillary + Mostra Mapillary + Un commutador per mostrar o amagar la capa de Mapillary al mapa. + Especifiqueu la llargada del vehicle que han de permetre les vies. + Llargada màxima + Trajectòria + S\'ha esborrat %1$s + Cal reiniciar per esborrar totalment les dades de les càmeres de radar. + Desinstal·la i Reinicia + Aquest dispositiu no inclou les càmeres de radar. + Patins en línia + Activeu per controlar el nivell d\'ampliació del mapa amb els botons del volum del dispositiu. + Botons de volum pel zoom \ No newline at end of file diff --git a/OsmAnd/res/values-de/phrases.xml b/OsmAnd/res/values-de/phrases.xml index 4add64ed4d..c3fce991db 100644 --- a/OsmAnd/res/values-de/phrases.xml +++ b/OsmAnd/res/values-de/phrases.xml @@ -1648,8 +1648,8 @@ Elektronische Geldbörsen UTA UTA-Karten nicht akzeptiert - Bargeld - Bargeld nicht akzeptiert + Efectivo + Efectivo nicht akzeptiert Girokarte Girokarte nicht akzeptiert Discover-Karte diff --git a/OsmAnd/res/values-de/strings.xml b/OsmAnd/res/values-de/strings.xml index 91e7cb63b1..1cc7fdb88d 100644 --- a/OsmAnd/res/values-de/strings.xml +++ b/OsmAnd/res/values-de/strings.xml @@ -3775,7 +3775,7 @@ Geben Sie Ihre Fahrzeughöhe an, für hohe Fahrzeuge können einige Routenbeschränkungen gelten. Geben Sie Ihr Fahrzeuggewicht an, für schwere Fahrzeuge können einige Routenbeschränkungen gelten. Sie können Schnellaktionen mit Anwendungsprofilen exportieren oder importieren. - Alles Löschen\? + Alles löschen\? Möchten Sie %d Schnellaktionen wirklich unwiderruflich löschen\? Töne Meter @@ -3793,7 +3793,7 @@ Mapillary ausblenden Mapillary anzeigen Eine Umschaltfläche zum ein- oder ausblenden der Mapillary-Layer auf der Karte. - Blitzer Deinstallieren + Blitzer deinstallieren Rechtliches Blitzer-POIs Aktiviert lassen diff --git a/OsmAnd/res/values-eo/phrases.xml b/OsmAnd/res/values-eo/phrases.xml index ffb6b6b3e1..833f1e4da7 100644 --- a/OsmAnd/res/values-eo/phrases.xml +++ b/OsmAnd/res/values-eo/phrases.xml @@ -3823,4 +3823,6 @@ Sageto Vibrado Kesto de senpaga interŝanĝo + Dombloko + Suburbo \ No newline at end of file diff --git a/OsmAnd/res/values-fr/phrases.xml b/OsmAnd/res/values-fr/phrases.xml index 7f290db577..b220d4eaf1 100644 --- a/OsmAnd/res/values-fr/phrases.xml +++ b/OsmAnd/res/values-fr/phrases.xml @@ -3818,7 +3818,7 @@ Remplissage d\'eau potable : non Réseau de remplissage d\'eau potable Uniquement si la marche est autorisée - Signal pour trouver le pôle + Signal pour trouver le poteau Pression Aspiration Pressurisé @@ -3829,4 +3829,5 @@ Vibration : non Flèche Vibration + Boîte à dons \ No newline at end of file diff --git a/OsmAnd/res/values-gl/strings.xml b/OsmAnd/res/values-gl/strings.xml index 80c079348d..6c2608e370 100644 --- a/OsmAnd/res/values-gl/strings.xml +++ b/OsmAnd/res/values-gl/strings.xml @@ -3834,4 +3834,6 @@ Lon %2$s Límite de lonxitude Este dispositivo non ten radares de velocidade. Patíns en liña + Activar para controlar o nivel de achegamento (zoom) do mapa cos botóns de volume do dispositivo. + Os botóns de volume mudan o achegamento \ No newline at end of file diff --git a/OsmAnd/res/values-he/strings.xml b/OsmAnd/res/values-he/strings.xml index 82fd9a4cba..16a5f1030f 100644 --- a/OsmAnd/res/values-he/strings.xml +++ b/OsmAnd/res/values-he/strings.xml @@ -3778,7 +3778,7 @@ נא לציין את רוחב כלי הרכב שלך, יתכן שחלות מגבלות מעבר על כלי רכב רחבים. נא לציין את גובה כלי הרכב שלך, יתכן שחלות מגבלות מעבר על כלי רכב גבוהים. נא לציין את משקל כלי הרכב שלך, יתכן שחלות מגבלות מעבר על כלי רכב כבדים. - גוונים + טונות מטרים הצגה או הסתרה של פירוט נוסף על המפה מפת לילה diff --git a/OsmAnd/res/values-hu/strings.xml b/OsmAnd/res/values-hu/strings.xml index cd5a4a496b..9f026359bb 100644 --- a/OsmAnd/res/values-hu/strings.xml +++ b/OsmAnd/res/values-hu/strings.xml @@ -591,7 +591,7 @@ Régió kijelölése listából Keresztező utca kijelölése Legközelebbi hasznos létesítmények - Alapértelmezett + Térképböngészés Vezetés Kerékpározás Gyaloglás @@ -3755,4 +3755,5 @@ Hosszúság: %2$s Bővítménybeállítások visszaállítása alapértelmezettre Engedélyezésével a térkép nagyítási szintje az eszköz hangerőgombjaival állítható. Hangerőgombok használata nagyításhoz + Görkorcsolya \ No newline at end of file diff --git a/OsmAnd/res/values-nl/phrases.xml b/OsmAnd/res/values-nl/phrases.xml index 2c122f6ba8..2518eb0fe7 100644 --- a/OsmAnd/res/values-nl/phrases.xml +++ b/OsmAnd/res/values-nl/phrases.xml @@ -1239,7 +1239,7 @@ Ja Zonder blindengeleidestrook Ja - Nee + Geluid: nee Alleen als oversteken is toegestaan Reddingstation Verzorgingsplaats @@ -1391,8 +1391,8 @@ IJs IJs Brandstoftype (avia) - Internet verbinding type - Verkoop + Type Internettoegang + Verkooptype Fietsonderhoud Soort Soort @@ -1482,7 +1482,7 @@ Tankstation voor scheepvaart IJshockey Internettoegang - betaald - Internettoegang - gratis + Internettoegang: gratis Vrijmetselaarsloge Zeilclub Padvindersclub @@ -1526,7 +1526,7 @@ Eurowag Eurowag-kaarten niet geaccepteerd E-Z Pass - E-Z Pass niet geaccepteerd + E-ZPass niet geaccepteerd Bankoverschrijving Bankoverschrijving niet geaccepteerd Betaling via SMS @@ -1769,7 +1769,7 @@ Soort brug: enkele hangbrug Brugstructuur: balk Soort brug: enkele brug - "Soort brug: boogbrug" + Soort brug: boogbrug Soort brug: vlotbrug Soort brug: houten Soort brug: viaduct @@ -1874,7 +1874,7 @@ Uitgeverij Avgas UL 91 Avgas 100 LL - Autogas + LPG Jet A-1 fuel AdBlue Brandstof: hout @@ -2308,7 +2308,7 @@ Oppervlak: onverhard Oppervlak: verhard Oppervlak: kasseien - "Paarden " + Paarden Zoogdieren Historische steen Type: moordkruis @@ -2837,7 +2837,7 @@ Richting: tegen de klok in Richting: omhoog Richting: omlaag - "Richting: alle" + Richting: alle Klei Open Gesloten @@ -3198,8 +3198,8 @@ Voedingsmiddelenwinkel Verlichting Dierenvoederplaats - Soort: plank - Soort: telefooncel + Type: plank + Type: telefooncel Openingstijden op kerstdagen Kerstwinkel Watertank @@ -3222,8 +3222,8 @@ Slotenmaker Feestwinkel Sagardotegi - Soort: metaalkast - Soort: leesvak + Type: metalen kast + Type: leesvak Openbare boekenkast Kerstmis: locatie Kerstmis: opmerking @@ -3525,4 +3525,59 @@ Stopcontact: CEE-rood 16A Stopcontact: CEE-blauw: vermogen Stopcontact: CEE-blauw: stroom + Kinder + Academische + Religieuze + Antiquarische + Stripverhalen + Groothandel + Productie van snoepgoed + Veldverzameling + Inspectiemeter + Ventielgroep + Ventiel + Meting + Compressie + Compensatie + Omvormer + Spoorwegen + Overgang + Industrieel + Trafohuisje + Distributie + Transmissie + Alleen maar + Schoenmaker + Rijbewijzen + Rotsformatie + Winkel voor huishoudapparatuur + Waterleidingbedrijf + Notitieboek + Code + Stempelpunt + Wandelcontrolepost + Gemiddelde helling + Laagste punt + Hoogste punt + IJstong + Restant + Schelf + Rots + Kalvend + Hangend + Berg + Vallei + Getijdewater + Monding + Plateau + IJsveld + IJskap + Aantal kabels + Via ferrata + Tokkelbaan + Autoreparatie + Gamingcentrum voor volwassenen + Speelhal + Barbecue: ja + Pick-up punt \ No newline at end of file diff --git a/OsmAnd/res/values-nl/strings.xml b/OsmAnd/res/values-nl/strings.xml index 4417698cf8..f68a059d52 100644 --- a/OsmAnd/res/values-nl/strings.xml +++ b/OsmAnd/res/values-nl/strings.xml @@ -498,7 +498,7 @@ Start route begeleiding Soort vervoer: Kies eerst een bestemming - Routebeschrijving + Route Openingstijden Wijzigingsset openen… Wijzigingsset sluiten… @@ -556,7 +556,7 @@ Kaartbeeld 3D Toon laatst gebruikte POI-kaartlaag. Toon POI-kaartlaag - Kies bron van online of opgeslagen kaartsegmenten + Kies bron van online of gecachte kaartsegmenten. Kaartsegment bron Kaartbron Gebruik internet @@ -707,7 +707,7 @@ Beschikbaar werkgeheugen %1$s MB (Android limiet %2$s MB, Dalvik %3$s MB). Beschikbaar werkgeheugen Werkgeheugen, gereserveerd door app %1$s MB (Dalvik %2$s MB, overig %3$s MB). -Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB). +\n Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB). Totaal werkgeheugen Snelheid van de routesimulatie: Uur @@ -754,7 +754,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).OsmAnd is een open-source navigatiesysteem met online en offline kaarten OsmAnd (OSM Automated Navigation Directions) \n -\n OsmAnd is een open-source navigatie-app die toegang geeft tot een grote variëteit aan wereldwijde OSM-gegevens. Alle kaartgegevens (vector- of rasterkaarten) kunnen in het telefoongeheugen worden opgeslagen voor offline gebruik. OsmAnd biedt ook zowel offline als online route-aanwijzingen, met stembegeleiding. +\n OsmAnd is een open-source navigatie-app die toegang geeft tot een grote variëteit aan wereldwijde OSM-gegevens. Alle kaartgegevens (vector- of rasterkaarten) kunnen in het telefoongeheugen worden opgeslagen voor offline gebruik. OsmAnd biedt ook zowel offline als online routeaanwijzingen, met stembegeleiding. \n \n Enkele van de belangrijkste mogelijkheden: \n - Volledig offline gebruik (gedownloade vector- of rasterkaarten worden opgeslagen op het apparaat) @@ -995,8 +995,8 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Wis Tussenpunten Bewaar Tussenpunten Er zijn al Tussenpunten gemaakt. - Routebeschrijving naar - Routebeschrijving vanaf + Route naar + Route vanaf Kaart: Naar: Via: @@ -1365,8 +1365,8 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Bewerk groep Om deze plugin te installeren is een internetverbinding nodig. Nautische kaartweergave - Om skikaarten te kunnen gebruiken, moet u de speciale offline skikaart downloaden - Om nautische kaarten te kunnen gebruiken, moet u de speciale offline nautische kaart downloaden + Download de speciale offline kaart om skifaciliteiten weer te geven. + Download de speciale offline kaart om nautische details weer te geven. Installeren Bekijk Ski-pistes @@ -1718,7 +1718,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Alles updaten (%1$s MB) Gebruikte gratis downloads Bekijk hoeveel gratis downloads er al gebruikt zijn. - "Kies waar kaarten en andere gegevens opgeslagen worden." + Kies waar kaarten en andere gegevens opgeslagen worden. \'Uit\' toont direct de kaart. Kies het land Nieuwe versie @@ -1961,7 +1961,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Volg ons Richting aangeven met geluid Geef de richting van de bestemming aan met geluid. - Richting aangeven met trilling + Richting aangeven met trillen Geef de richting van de bestemming aan met trillingen. Verplaats de kaart om de positie van de markering te wijzigen @@ -2104,62 +2104,62 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Geldige volledige OLC \nBeslaat gebied: %1$s x %2$s Sneltoets - Actie %d + Sneltoets %d Scherm %d Markeervlaggetje toevoegen POI toevoegen - Kaartstijl wisselen + Kaartstijl wijzigen De kaartstijl is gewijzigd naar \"%s\". - Audio-notitie maken - Video-notitie maken - Maak foto - Voeg OSM-notitie toe - Stembegeleiding Aan/Uit - Stembegeleiding staat uit - Stembegeleiding staat aan - GPX-routepunt maken - Parkeerpositie instellen - Actie toevoegen - Actie bewerken + Audionotitie maken + Videonotitie maken + Fotonotitie maken + OSM-notitie toevoegen + Stembegeleiding aan/uit + Stembegeleiding aan + Stembegeleiding uit + GPX-routepunt toevoegen + Parkeerlocatie instellen + Sneltoets toevoegen + Sneltoets bewerken Favoriet toevoegen Actie toevoegen - Actie verwijderden - "Actie \"%s\" inderdaad verwijderen?" - Toon favorietenscherm + Sneltoets verwijderen + Weet je zeker dat je sneltoets \"%s\" wilt verwijderen\? + Toon Favorieten-scherm Naam voorstellen Een toets om een Markeervlaggetje toe te voegen in het midden van het scherm. Een knop om een GPX-routepunt toe te voegen in het midden van het scherm. - Een knop om een Audio-notitie toe te voegen in het midden van het scherm. - Een knop om een Video-notitie toe te voegen in het midden van het scherm. - Een knop om een Foto-notitie toe te voegen in het midden van het scherm. + Een knop om een audio-notitie toe te voegen in het midden van het scherm. + Een knop om een video-notitie toe te voegen in het midden van het scherm. + Een knop om een foto-notitie toe te voegen in het midden van het scherm. Een knop om een OSM-notitie toe te voegen in het midden van het scherm. Een knop om een POI toe te voegen in het midden van het scherm. Een schakelknop om de stembegeleiding aan of uit te zetten tijdens navigatie. - Een knop om de parkeerpositie toe te voegen in het midden van het scherm. + Een knop om een parkeerlocatie toe te voegen in het midden van het scherm. Bewerkingsdialoog tonen " opgeslagen als " Plaats Sneltoetsnaam gewijzigd in %1$s om duplicaat te voorkomen. Sneltoets-duplicaat - Een schakelknop toont of verbergt de Favoriete punten op de kaart. - Een schakelknop om POI’s op de kaart te tonen of te verbergen. + Een schakelknop om Favorieten al dan niet te tonen op de kaart. + Een schakelknop om POI’s al dan niet op de kaart te tonen. Favorieten tonen /verbergen - Toon favorieten - Verberg favorieten - POI tonen/verbergen - Toon %1$s - Verberg %1$s + Favorieten tonen + Favorieten verbergen + POI’s tonen/verbergen + %1$s tonen + %1$s verbergen Categorie toevoegen Items aanmaken - Kaart-instellingen + Kaartinstellingen Navigatie - Leeg laten om automatisch adres of plaatsnaam te gebruiken. - Dit bericht wordt automatisch aangevuld in het commentaarveld. + Leeg laten om adres of plaatsnaam te gebruiken. + Dit bericht wordt automatisch ingevuld in het commentaarveld. Bericht Categorie om de Favoriet in op te slaan: - Kies een categorie (optie). + Kies een optionele categorie. POI-lijst - Je kunt één of meerdere POI-categorieën kiezen om weer te geven. + Je kunt één of meerdere POI-categorieën kiezen om te tonen op de kaart. Een knop om door de onderstaande lijst te bladeren. Kaartstijl toevoegen Alle parameters invullen @@ -2219,14 +2219,14 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Plaats kiezen Postcode zoeken Autozoom kaart aan/uit - Knop om de automatische kaartzoom volgens de snelheid aan of uit te zetten. + Knop om de snelheidsgebaseerde automatische zoom aan of uit te zetten. Autozoom aan Autozoom uit Bestemming instellen Bestemming vervangen - Eerste Tussenpunt toevoegen - Een druk op de sneltoets voegt het midden van het scherm toe als bestemming. De vorige bestemming wordt het laatste tussenpunt. - Een knop om de bestemming in het schermmidden te plaatsen waardoor de (eventuele) hiervoor geselecteerde bestemming wordt vervangen. + Eerste tussenpunt toevoegen + Een knop om het midden van het scherm in te stellen als bestemming, de vorige bestemming wordt het laatste tussenpunt. + Een knop om het midden van het scherm in te stellen als bestemming, waardoor de (eventueel) eerder ingestelde bestemming wordt vervangen. Een knop om het midden van het scherm als eerste tussenpunt toe te voegen. Geen extra laag Geen achtergrondlaag @@ -2279,9 +2279,9 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Ga ook over privé-terreinen. Zoomniveau weergave: %1$s Voor grote afstanden: voeg tussenpunten toe als binnen de 10 minuten geen route berekend is. - "OsmAnd (OSM, Automated Navigation Directions) is een kaart- en navigatieprogramma dat de gratis, wereldwijde en gedetailleerde OpenStreetMap (OSM) data gebruikt. -\n -\nGebruikt visuele en gesproken navigatie, toont POI\'s (point of interest), maakt en volgt GPX-tracks, toont hoogtelijnen en hoogte informatie (via plug-in), kiest tussen auto, fiets, wandelen of andere, je kan de OSM-kaart bewerken en nog veel meer." + OsmAnd (OSM Automated Navigation Directions) is een kaart- en navigatie-app die gebruik maakt van de gratis, wereldwijde en hoogwaardige OSM-gegevens. +\n +\n Gebruik visuele en gesproken navigatie, toon POI\'s (point of interest), maak en volg GPX-tracks, toon hoogtelijnen en hoogte-informatie (via plug-in), kies tussen autorijden, fietsen of wandelen, bewerk de OSM-kaart en nog veel meer. GPS-navigatie \n • Offline (geen datakosten) of online (snellere) routeberekening \n • Stap-voor-stap navigatie met opgenomen stem of gesproken straatnamen @@ -2311,7 +2311,12 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB). - Wandelen • De kaart toont voetpaden, bergpaden, routes door parken • Wikipedia vertelt u meer over uw omgeving • OV-haltes (tram, bus, trein, metro), inclusief lijnnummers, helpen u te navigeren • Navigatie gebruikt voetpaden • Neem uw weg op als GPX-route of volg een gedownloade route + Lopen, wandelen, stadswandeling +\n • De kaart toont voetpaden en wandelpaden +\n • Wikipedia kan je veel vertellen tijdens een stadswandeling, in je eigen taal +\n • OV-haltes (tram, bus, trein, metro), inclusief lijnnummers, helpen je bij het reizen in een onbekende stad +\n • GPS-navigatie in voetgangermodus gebruikt voor het berekenen van de route voet- en wandelpaden +\n • Upload en volg een GPX-route of neem er zelf één op en deel deze. \n Draag bij aan OpenStreetMap (OSM) \n • Meld fouten @@ -2332,7 +2337,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB). - Indicatie van compleetheid en kwaliteit van de kaart: + Indicatie van de volledigheid en kwaliteit van het kaartmateriaal: \n • West-Europa: **** \n • Oost-Europa: *** \n • Rusland: *** @@ -2342,14 +2347,15 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB). Naam bevat erg veel hoofdletters, toch doorgaan? Zoek in Favorieten - Toon of verberg OSM notities - Toon OSM notities - Verberg OSM notities - Knop om OSM notities te tonen of verbergen. + OSM-notities tonen of verbergen + OSM-notities tonen + OSM-notities verbergen + Knop om OSM-notities al dan niet te tonen. Gesorteerd op afstand Uw bestemming ligt in een gebied dat particulier eigendom is. Wilt u gebruik van particuliere wegen voor deze route toestaan\? Opnieuw zoeken @@ -2373,13 +2379,19 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Rechten Kan bestand niet importeren. Controleer of OsmAnd rechten heeft om het bestand te lezen. Gecorrigeerde afstand - OsmAnd+ (OSM Automated Navigation Directions) is een kaart- en navigatieprogramma dat de gratis, wereldwijde en uitgebreide OpenStreetMap (OSM) gegevens gebruikt. -\nMet akoestische en visuele navigatie, toont POI (points of interest), maakt en beheert GPX-sporen, gebruikt hoogtelijnen en hoogtegegevens, kiest tussen auto, fiets en wandelen, je kan OSM bewerken en veel meer. -\n -\nOsmAnd+ is de betaalde versie van de app. Met de aanschaf ondersteunt u het project, het maken van nieuwe functies, en krijgt u de nieuwste functies. -\n -\nEen paar belangrijke functies: - " Navigatie • werkt online (snel) of offline (zonder roaming-kosten in het buitenalnd) • gesproken stap-voor-stap navigatie (met opgenomen of synthetische stem) • optioneel met rijstrookaanduiding, straatnamen und geschatte aankomsttijd • tussenstops op de route • automatische herberekening bij het verlaten van de route • zoek op adres, soort (bijv.: restaurant, hotel, benzinepomp, museum) of geografische coördinaten " + OsmAnd+ (OSM Automated Navigation Directions) is een kaart- en navigatie-app die de gratis, wereldwijde en hoogwaardige OSM-gegevens gebruikt. +\n Gebruik gesproken en visuele navigatie, toon POI‘S (points of interest), maak en beheer GPX-tracks, toon hoogtelijnen en hoogtegegevens, kies tussen autorijden, fietsen en wandelen, bewerk OSM en nog veel meer. +\n +\n OsmAnd+ is de betaalde versie van de app. Met de aanschaf ondersteun je het project, het ontwikkelen van nieuwe functies, en ontvang je de laatste updates. +\n +\n Een paar van de belangrijkste functies: + " Navigatie +\n • Online (snel) of offline (zonder roamingkosten in het buitenland) +\n • Stap-voor-stap stembegeleiding (met opgenomen of TTS-stemmen) +\n • Optioneel met rijstrookindicatie, weergave van straatnamen en geschatte aankomsttijd +\n • Ondersteuning voor tussenstops op de route +\n • Automatische herberekening bij het afwijken van de route +\n • Zoek op adres, soort (bijv.: restaurant, hotel, benzinepomp, museum) of geografische coördinaten " Kaartweergave \n • Positie en richting tonen \n • Kaart draaien in kompas- of bewegingsrichting @@ -2449,7 +2461,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).GPX bestandsnaam: Toon op kaart na opslaan Afstand meten - Start/stop navigatie + Navigatie starten/stoppen Gemiddeld %1$d van de %2$d Probeer opnieuw @@ -2555,7 +2567,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Maak markeringen aan op de kaart! Twee Een - Kies hoeveel markeervlaggetjes getoond worden: + Kies hoeveel markeervlaggetjes getoond worden. Aantal decimalen Rechts Links @@ -2736,7 +2748,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Zoeken naar het bijbehorende wiki-artikel Artikel niet gevonden Hoe Wikipedia-artikelen te openen? - Wijzig acties + Wijzig sneltoetsen Bladwijzer Punt %1$s verwijderd Punt bewerken @@ -2780,7 +2792,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB). - Druk op een knop en luister naar de bijhorende gesproken melding om ontbrekende of foutieve meldingen te identificeren. + Tik op een knop en luister naar de bijbehorende gesproken prompt om te horen of deze ontbreekt of incorrect is Abonnementen Mogelijk gemaakt door OsmAnd Tariefoverzicht @@ -2825,7 +2837,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Milliradialen Wijzig azimut eenheid. Selecteer de te vermijden openbaar vervoer opties: - %s modus + %smodus Vermijd transport types… Loop Verkort de lengte van \"%s\" tot minder dan 255 karakters. @@ -2884,10 +2896,10 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Wissel Toon meer Getoonde tracks - Toon/Verberg GPX-tracks - Een knop om bepaalde GPX-tracks te tonen of verbergen op de kaart. - Verberg GPX-tracks - Toon GPX-tracks + GPX-tracks tonen/verbergen + Een knop om geselecteerde GPX-tracks al dan niet te tonen op de kaart. + GPX-tracks verbergen + GPX-tracks tonen Voeg a.u.b. eerst de Bestemming in Vorige route Thuisadres toevoegen @@ -2914,10 +2926,10 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Tussenbestemmingen Aankomst om %1$s Open - Tik deze knop om te wisselen tussen de OsmAnd dag en nacht modus. - Dag modus - Nacht modus - Wissel tussen dag/nacht modus + Een schakelknop om OsmAnd om te schakelen tussen dag- en nachtmodus. + Dagmodus + Nachtmodus + Wissel tussen dag-/nachtmodus Oppervlakte Kwaliteit wegdek Steilte @@ -3181,13 +3193,13 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Naast elkaar Kaarten Knop om hoogtelijnen al dan niet te tonen op de kaart. - Toon de hoogtelijnen - Verberg de hoogtelijnen - Toon/verberg de hoogtelijnen + Hoogtelijnen tonen + Hoogtelijnen verbergen + Hoogtelijnen tonen/verbergen Knop om de reliëfschaduw al dan niet te tonen op de kaart. - Toon reliëfschaduw - Verberg reliëfschaduw - Toon/verberg reliëfschaduw + Reliëfschaduw tonen + Reliëfschaduw verbergen + Reliëfschaduw tonen/verbergen Track opgeslagen Toon knooppunt van de fietsroutes Hoogtelijnen en reliëfschaduw @@ -3293,7 +3305,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Sla tracks op in dagelijkse folders Sla tracks op in sub-folders per opname dag (zoals 2018-01-01). Simuleer uw positie door een opgenomen GPX track te gebruiken. - Een knop om het scherm te centreren op het vertrekpunt. Zal dan de bestemming vragen of de routeberekening starten. + Een knop om het midden van het scherm in te stellen als vertrekpunt. Hierna wordt de bestemming gevraagd of de routeberekening gestart. Dialoogvenster kaart downloads Dialogen en meldingen Bepaal popups, dialogen en meldingen. @@ -3462,10 +3474,10 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Kan profiel niet back-uppen. %1$s of %2$s Hellingen - Toon/Verberg de omgeving - Verberg de omgeving - Toon de omgeving - Een knop om een laag op de kaart weer te geven of te verbergen. + Terrein tonen / verbergen + Terrein verbergen + Terrein tonen + Een knop om de terreinlaag al dan niet te tonen op de kaart. Verwijder een beschrijving Voeg een beschrijving toe Selecteer groep @@ -3492,7 +3504,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Stel de hoogte van het vaartuig in om lage bruggen te vermijden. Let op, als de brug beweegbaar is, gebruiken we de hoogte in geopende toestand. Als \"%1$s\" is geactiveerd, is de actieve tijd hiervan afhankelijk. Standaard schermtimeout - tonen + ton meter Toon/verberg extra kaartdetails Nachtkaart @@ -3500,10 +3512,10 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Het toepassen van deze wijzigingen wist de cache van deze tile-bron Stel de hoogte van het vaartuig in Stel de hoogte van het vaartuig in om lage bruggen te vermijden. Let op, als de brug beweegbaar is, gebruiken we de hoogte in geopende toestand. - Toon/verberg Mapillary - Verberg Mapillary - Toon Mapillary - Toon/verberg de Mapillary-laag op de kaart. + Mapillary tonen/verbergen + Mapillary verbergen + Mapillary tonen + Een schakelknop om de Mapillary-laag al dan niet te tonen op de kaart. Geef de toegestane voertuiglengte op voor routes. Maximale lengte Richting @@ -3619,7 +3631,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).OV-informatie verbergen OV-informatie tonen OV-informatie tonen/verbergen - Knop om OV-informatie op de kaart te tonen of verbergen. + Knop om OV-informatie al dan niet te tonen op de kaart. Favoriet toevoegen / bewerken POI toevoegen / bewerken Parkeerlocaties @@ -3632,7 +3644,7 @@ Proportioneel werkgeheugen %4$s MB (Android limiet %5$s MB, Dalvik %6$s MB).Niet ondersteund type Geef de voertuigbreedte op, er zijn mogelijk routebeperkingen voor brede voertuigen. Geef de voertuighoogte op, er zijn mogelijk routebeperkingen voor hoge voertuigen. - Geef het voertuiggewicht op, er zijn mogelijk routbeperkingen voor zwaar verkeer. + Geef het voertuiggewicht op, er zijn mogelijk routebeperkingen voor zwaar verkeer. OsmAnd GPX is corrupt, neem contact op met het support team voor verder onderzoek. Altijd Scherminstellingen diff --git a/OsmAnd/res/values-sr/phrases.xml b/OsmAnd/res/values-sr/phrases.xml index 11e046459a..2fa9363f13 100644 --- a/OsmAnd/res/values-sr/phrases.xml +++ b/OsmAnd/res/values-sr/phrases.xml @@ -1538,7 +1538,7 @@ Истраживачка канцеларија Новинарска канцеларија Канцеларија архитекта - Штампарија + Кладионица Канцеларија за религије Канцеларија удружења Финансијска канцеларија @@ -2765,4 +2765,11 @@ Реф. Реф. тунела Реф. моста + Академске + Дечије + Религијске + Антикварница + Стрип + Свеска + Тип: полица \ No newline at end of file diff --git a/OsmAnd/res/values-sr/strings.xml b/OsmAnd/res/values-sr/strings.xml index 7f9ebdea68..b11bf427f9 100644 --- a/OsmAnd/res/values-sr/strings.xml +++ b/OsmAnd/res/values-sr/strings.xml @@ -1382,7 +1382,7 @@ Радње Ознака Стварајте или мењајте предмете ОСМ-а - Стварајте или мењајте ОСМ тачке од интереса, отворите или коментаришите на ОСМ белешке и доприносите снимањем GPX фајлова. + Стварајте или мењајте ОСМ тачке од интереса, отворите или коментаришите ОСМ белешке и доприносите снимањем GPX фајлова. Избрисано Уређено Додато @@ -3482,8 +3482,8 @@ Доступно Додај произвољну категорију Прикажи само ноћу - Све поставке додатка враћене на подразумевано стање. - Све поставке профила враћене на подразумевано стање. + Све поставке додатка враћене на подразумевано. + Све поставке профила враћене на подразумевано. %1$s/%2$s Сунце залази у %1$s Сунце излази у %1$s @@ -3502,7 +3502,7 @@ Мени %1$s — %2$s — %3$s Укључи додатне податке - Увезени профил садржи додатне податке. Кликните на Увоз да увезете да увезете само профилне податке или одаберите које додатне податке увести. + Увезени профил садржи додатне податке. Кликните на Увоз да увезете само профилне податке или одаберите додатне податке. Поред профила, можете одабрати додатне податке за извоз. Антарктик Можете додати произвољне категорије, сакрити категорије које су Вам сувишне и променити им редослед у списку. Списак се може увести и извести са профилима. @@ -3536,7 +3536,7 @@ \nДодатак ће остати на уређају и после уклањања OsmAnd апликације. Рутирање Подразумевано у апликацију (%s) - Искључи прерачунавање + Без прерачунавања Минимална удаљеност за прерачунавање пута Пут ће бити прерачунат ако је удаљеност до пута већа од одабраног параметра Произвољни профил @@ -3565,7 +3565,7 @@ Поврати све поставке профила\? Чувам нови профил Не могу да направим резервну копију профила. - Прикажи/сакриј терен + Прикажи / сакриј терен Сакриј терен Прикажи терен Дугме да прикаже или сакрите слој терена на карти. @@ -3577,11 +3577,11 @@ Октагон Квадрат Мин - Замени другу тачку са овом + Замени другу тачку са овом. Сноумобил Произвољни OsmAnd додатак - Не могу да читам %1$s. - Не могу да увезем %1$s. + Не могу да читам из „%1$s”. + Не могу да увезем из „%1$s”. Не могу да пишем %1$s. Језици Језик @@ -3718,4 +3718,38 @@ Претрага типова тачака од интереса Комбинујте типове тачака од интереса из различитих категорија. Кликните прекидач да одабере све, кликните на леву страну да одаберете категорију. Маркери на карти + Терен + Провидност + Нивои увећања + Легенда + Увоз завршен + Додате ставке + Увозим + Увозим податке из %1$s + Да ли сте сигурни да желите да очистите снимљене податке\? + Прерачунај пут у случају одступања + Одаберите удаљеност после које се пут прерачунава. + Пут ће бити прерачунат уколико је удаљеност тренутне позиције од пута већа од одабране вредности. + %1$s од %2$s + Прилагођавање корисничког интерфејса + Фиока + Преуреди или сакриј ставке са %1$s. + Разделник + Елементи испод ове тачке су одвојени разделником. + Сакривена + Иако су ове ставке сакривене са главног менија, опције или додаци које они представљају и даље раде. + Сакривање поставки их ресетује на оригинално стање. + Има само четири дугмета. + Главне радње + Можете приступити овим радњама кликтањем на дугме „%1$s”. + Можете померати ставке само унутар ове категорије. + Додатак за програмере + Ски туринг + Ставке + Измене примењене на ’%1$s’ профил. + За приказ Википедијиних тачака од интереса су потребне додатне карте. + Одаберите језик за Википедијине чланке на карти. Пребаците на било који други доступни језик док читате чланак. + Неки чланци са Википедија су можда доступни на Вашем језику. + Кантонски + Јужномински \ No newline at end of file diff --git a/OsmAnd/res/values-uk/strings.xml b/OsmAnd/res/values-uk/strings.xml index b126f56677..8a5184b325 100644 --- a/OsmAnd/res/values-uk/strings.xml +++ b/OsmAnd/res/values-uk/strings.xml @@ -1533,7 +1533,7 @@ Не вдалося створити мапи у вказаній теці Не вдалося перемістити файли Зовнішнє сховище - Толочне сховище + Багатокористувацьке сховище Внутрішня пам’ять застосунку Вказано вручну Внутрішня пам’ять diff --git a/OsmAnd/res/values-zh-rTW/phrases.xml b/OsmAnd/res/values-zh-rTW/phrases.xml index 8f0f380d7d..d3dc318221 100644 --- a/OsmAnd/res/values-zh-rTW/phrases.xml +++ b/OsmAnd/res/values-zh-rTW/phrases.xml @@ -3829,4 +3829,6 @@ 箭頭 震動 禮物箱 + 街區 + 自治市鎮 \ No newline at end of file diff --git a/OsmAnd/res/values/phrases.xml b/OsmAnd/res/values/phrases.xml index d4dffcaa9b..1f9d2ee344 100644 --- a/OsmAnd/res/values/phrases.xml +++ b/OsmAnd/res/values/phrases.xml @@ -4237,4 +4237,8 @@ Give box + Borough + City block + + \ No newline at end of file diff --git a/OsmAnd/src/net/osmand/plus/poi/PoiUIFilter.java b/OsmAnd/src/net/osmand/plus/poi/PoiUIFilter.java index e23e0c3485..4b1471a396 100644 --- a/OsmAnd/src/net/osmand/plus/poi/PoiUIFilter.java +++ b/OsmAnd/src/net/osmand/plus/poi/PoiUIFilter.java @@ -834,12 +834,19 @@ public class PoiUIFilter implements SearchPoiTypeFilter, Comparable if (!poiTypes.isRegisteredType(type)) { type = poiTypes.getOtherPoiCategory(); } - LinkedHashSet acceptedTypesSet = acceptedTypes.get(type); - if (acceptedTypesSet != null && acceptedTypesSet.contains(subtype)) { - return true; + if (acceptedTypes.containsKey(type)) { + LinkedHashSet acceptedTypesSet = acceptedTypes.get(type); + if (acceptedTypesSet == null || acceptedTypesSet.contains(subtype)) { + return true; + } } - acceptedTypesSet = acceptedTypesOrigin.get(type); - return acceptedTypesSet != null && acceptedTypesSet.contains(subtype); + if (acceptedTypesOrigin.containsKey(type)) { + LinkedHashSet acceptedTypesSet = acceptedTypesOrigin.get(type); + if (acceptedTypesSet == null || acceptedTypesSet.contains(subtype)) { + return true; + } + } + return false; } @Override