2018-08-03 12:50:51 +02:00
|
|
|
package net.osmand.router;
|
|
|
|
|
2019-07-10 14:52:20 +02:00
|
|
|
import java.io.IOException;
|
|
|
|
import java.util.ArrayList;
|
2019-07-29 16:17:27 +02:00
|
|
|
import java.util.Collection;
|
2019-07-10 14:52:20 +02:00
|
|
|
import java.util.Collections;
|
|
|
|
import java.util.Comparator;
|
2019-07-29 16:13:55 +02:00
|
|
|
import java.util.Iterator;
|
2019-07-10 14:52:20 +02:00
|
|
|
import java.util.LinkedHashMap;
|
|
|
|
import java.util.List;
|
|
|
|
import java.util.Map;
|
|
|
|
import java.util.PriorityQueue;
|
|
|
|
|
|
|
|
import gnu.trove.iterator.TIntIterator;
|
|
|
|
import gnu.trove.list.array.TIntArrayList;
|
|
|
|
import gnu.trove.map.hash.TIntObjectHashMap;
|
|
|
|
import gnu.trove.map.hash.TLongObjectHashMap;
|
2019-07-29 16:17:27 +02:00
|
|
|
import net.osmand.binary.BinaryMapIndexReader;
|
|
|
|
import net.osmand.binary.BinaryMapIndexReader.SearchRequest;
|
|
|
|
import net.osmand.data.LatLon;
|
|
|
|
import net.osmand.data.QuadRect;
|
|
|
|
import net.osmand.data.TransportRoute;
|
|
|
|
import net.osmand.data.TransportSchedule;
|
|
|
|
import net.osmand.data.TransportStop;
|
|
|
|
import net.osmand.osm.edit.Node;
|
|
|
|
import net.osmand.osm.edit.Way;
|
|
|
|
import net.osmand.util.MapUtils;
|
2019-07-10 14:52:20 +02:00
|
|
|
|
2018-08-03 12:50:51 +02:00
|
|
|
public class TransportRoutePlanner {
|
|
|
|
|
2019-03-05 23:49:48 +01:00
|
|
|
private static final boolean MEASURE_TIME = false;
|
2019-04-05 19:59:49 +02:00
|
|
|
public static final long GEOMETRY_WAY_ID = -1;
|
|
|
|
public static final long STOPS_WAY_ID = -2;
|
2018-08-03 12:50:51 +02:00
|
|
|
|
2019-01-09 17:13:49 +01:00
|
|
|
public List<TransportRouteResult> buildRoute(TransportRoutingContext ctx, LatLon start, LatLon end) throws IOException, InterruptedException {
|
2018-08-04 13:03:06 +02:00
|
|
|
ctx.startCalcTime = System.currentTimeMillis();
|
2018-08-03 12:50:51 +02:00
|
|
|
List<TransportRouteSegment> startStops = ctx.getTransportStops(start);
|
|
|
|
List<TransportRouteSegment> endStops = ctx.getTransportStops(end);
|
|
|
|
|
|
|
|
TLongObjectHashMap<TransportRouteSegment> endSegments = new TLongObjectHashMap<TransportRouteSegment>();
|
|
|
|
for(TransportRouteSegment s : endStops) {
|
|
|
|
endSegments.put(s.getId(), s);
|
|
|
|
}
|
2019-03-15 14:02:24 +01:00
|
|
|
if(startStops.size() == 0) {
|
|
|
|
return Collections.emptyList();
|
|
|
|
}
|
2019-02-13 11:42:26 +01:00
|
|
|
PriorityQueue<TransportRouteSegment> queue = new PriorityQueue<TransportRouteSegment>(startStops.size(), new SegmentsComparator(ctx));
|
2018-08-04 13:03:06 +02:00
|
|
|
for(TransportRouteSegment r : startStops){
|
|
|
|
r.walkDist = (float) MapUtils.getDistance(r.getLocation(), start);
|
|
|
|
r.distFromStart = r.walkDist / ctx.cfg.walkSpeed;
|
|
|
|
queue.add(r);
|
|
|
|
}
|
|
|
|
double finishTime = ctx.cfg.maxRouteTime;
|
2019-03-14 16:34:20 +01:00
|
|
|
double maxTravelTimeCmpToWalk = MapUtils.getDistance(start, end) / ctx.cfg.walkSpeed - ctx.cfg.changeTime / 2;
|
2018-08-04 13:03:06 +02:00
|
|
|
List<TransportRouteSegment> results = new ArrayList<TransportRouteSegment>();
|
2019-01-09 17:13:49 +01:00
|
|
|
initProgressBar(ctx, start, end);
|
2018-08-04 13:03:06 +02:00
|
|
|
while (!queue.isEmpty()) {
|
2019-03-05 23:49:48 +01:00
|
|
|
long beginMs = MEASURE_TIME ? System.currentTimeMillis() : 0;
|
2018-12-19 17:53:38 +01:00
|
|
|
if (ctx.calculationProgress != null && ctx.calculationProgress.isCancelled) {
|
|
|
|
return null;
|
|
|
|
}
|
2018-08-03 12:50:51 +02:00
|
|
|
TransportRouteSegment segment = queue.poll();
|
2018-08-04 13:03:06 +02:00
|
|
|
TransportRouteSegment ex = ctx.visitedSegments.get(segment.getId());
|
|
|
|
if(ex != null) {
|
|
|
|
if(ex.distFromStart > segment.distFromStart) {
|
|
|
|
System.err.println(String.format("%.1f (%s) > %.1f (%s)", ex.distFromStart, ex, segment.distFromStart, segment));
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2018-08-06 00:01:25 +02:00
|
|
|
ctx.visitedRoutesCount++;
|
2018-08-04 13:03:06 +02:00
|
|
|
ctx.visitedSegments.put(segment.getId(), segment);
|
2019-03-05 23:49:48 +01:00
|
|
|
|
2019-03-17 17:06:30 +01:00
|
|
|
if (segment.getDepth() > ctx.cfg.maxNumberOfChanges + 1) {
|
2018-08-04 13:03:06 +02:00
|
|
|
continue;
|
|
|
|
}
|
2019-03-14 16:34:20 +01:00
|
|
|
if (segment.distFromStart > finishTime + ctx.cfg.finishTimeSeconds ||
|
|
|
|
segment.distFromStart > maxTravelTimeCmpToWalk) {
|
2018-08-04 13:03:06 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
long segmentId = segment.getId();
|
|
|
|
TransportRouteSegment finish = null;
|
|
|
|
double minDist = 0;
|
|
|
|
double travelDist = 0;
|
|
|
|
double travelTime = 0;
|
2019-03-17 14:09:41 +01:00
|
|
|
final float routeTravelSpeed = ctx.cfg.getSpeedByRouteType(segment.road.getType());
|
|
|
|
if(routeTravelSpeed == 0) {
|
|
|
|
continue;
|
|
|
|
}
|
2018-08-04 13:03:06 +02:00
|
|
|
TransportStop prevStop = segment.getStop(segment.segStart);
|
2018-08-05 23:46:01 +02:00
|
|
|
List<TransportRouteSegment> sgms = new ArrayList<TransportRouteSegment>();
|
2018-08-04 13:03:06 +02:00
|
|
|
for (int ind = 1 + segment.segStart; ind < segment.getLength(); ind++) {
|
2018-12-19 17:53:38 +01:00
|
|
|
if (ctx.calculationProgress != null && ctx.calculationProgress.isCancelled) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
segmentId ++;
|
2018-08-04 13:03:06 +02:00
|
|
|
ctx.visitedSegments.put(segmentId, segment);
|
|
|
|
TransportStop stop = segment.getStop(ind);
|
|
|
|
// could be geometry size
|
|
|
|
double segmentDist = MapUtils.getDistance(prevStop.getLocation(), stop.getLocation());
|
|
|
|
travelDist += segmentDist;
|
2018-08-15 14:43:47 +02:00
|
|
|
if(ctx.cfg.useSchedule) {
|
|
|
|
TransportSchedule sc = segment.road.getSchedule();
|
|
|
|
int interval = sc.avgStopIntervals.get(ind - 1);
|
|
|
|
travelTime += interval * 10;
|
|
|
|
} else {
|
2019-03-17 14:09:41 +01:00
|
|
|
travelTime += ctx.cfg.stopTime + segmentDist / routeTravelSpeed;
|
2018-08-15 14:43:47 +02:00
|
|
|
}
|
2019-03-15 13:44:49 +01:00
|
|
|
if(segment.distFromStart + travelTime > finishTime + ctx.cfg.finishTimeSeconds) {
|
2019-03-05 23:49:48 +01:00
|
|
|
break;
|
|
|
|
}
|
2018-08-05 23:46:01 +02:00
|
|
|
sgms.clear();
|
|
|
|
sgms = ctx.getTransportStops(stop.x31, stop.y31, true, sgms);
|
2019-03-15 14:41:05 +01:00
|
|
|
ctx.visitedStops++;
|
2018-08-04 13:03:06 +02:00
|
|
|
for (TransportRouteSegment sgm : sgms) {
|
2018-12-19 17:53:38 +01:00
|
|
|
if (ctx.calculationProgress != null && ctx.calculationProgress.isCancelled) {
|
|
|
|
return null;
|
|
|
|
}
|
2018-08-04 13:03:06 +02:00
|
|
|
if (segment.wasVisited(sgm)) {
|
|
|
|
continue;
|
|
|
|
}
|
2019-03-15 13:44:49 +01:00
|
|
|
TransportRouteSegment nextSegment = new TransportRouteSegment(sgm);
|
|
|
|
nextSegment.parentRoute = segment;
|
|
|
|
nextSegment.parentStop = ind;
|
|
|
|
nextSegment.walkDist = MapUtils.getDistance(nextSegment.getLocation(), stop.getLocation());
|
|
|
|
nextSegment.parentTravelTime = travelTime;
|
|
|
|
nextSegment.parentTravelDist = travelDist;
|
|
|
|
double walkTime = nextSegment.walkDist / ctx.cfg.walkSpeed
|
2019-03-17 14:09:41 +01:00
|
|
|
+ ctx.cfg.getChangeTime() + ctx.cfg.getBoardingTime();
|
2019-03-15 13:44:49 +01:00
|
|
|
nextSegment.distFromStart = segment.distFromStart + travelTime + walkTime;
|
2018-08-20 21:00:22 +02:00
|
|
|
if(ctx.cfg.useSchedule) {
|
|
|
|
int tm = (sgm.departureTime - ctx.cfg.scheduleTimeOfDay) * 10;
|
2019-03-15 13:44:49 +01:00
|
|
|
if(tm >= nextSegment.distFromStart) {
|
|
|
|
nextSegment.distFromStart = tm;
|
|
|
|
queue.add(nextSegment);
|
2018-08-20 21:00:22 +02:00
|
|
|
}
|
|
|
|
} else {
|
2019-03-15 13:44:49 +01:00
|
|
|
queue.add(nextSegment);
|
2018-08-20 21:00:22 +02:00
|
|
|
}
|
2018-08-04 13:03:06 +02:00
|
|
|
}
|
2019-03-15 13:44:49 +01:00
|
|
|
TransportRouteSegment finalSegment = endSegments.get(segmentId);
|
2018-08-04 13:03:06 +02:00
|
|
|
double distToEnd = MapUtils.getDistance(stop.getLocation(), end);
|
2019-03-15 13:44:49 +01:00
|
|
|
if (finalSegment != null && distToEnd < ctx.cfg.walkRadius) {
|
2018-08-04 13:03:06 +02:00
|
|
|
if (finish == null || minDist > distToEnd) {
|
|
|
|
minDist = distToEnd;
|
2019-03-15 13:44:49 +01:00
|
|
|
finish = new TransportRouteSegment(finalSegment);
|
2018-08-04 13:03:06 +02:00
|
|
|
finish.parentRoute = segment;
|
|
|
|
finish.parentStop = ind;
|
|
|
|
finish.walkDist = distToEnd;
|
|
|
|
finish.parentTravelTime = travelTime;
|
|
|
|
finish.parentTravelDist = travelDist;
|
|
|
|
|
|
|
|
double walkTime = distToEnd / ctx.cfg.walkSpeed;
|
|
|
|
finish.distFromStart = segment.distFromStart + travelTime + walkTime;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
prevStop = stop;
|
|
|
|
}
|
|
|
|
if (finish != null) {
|
|
|
|
if (finishTime > finish.distFromStart) {
|
|
|
|
finishTime = finish.distFromStart;
|
|
|
|
}
|
2019-03-14 16:34:20 +01:00
|
|
|
if(finish.distFromStart < finishTime + ctx.cfg.finishTimeSeconds &&
|
|
|
|
(finish.distFromStart < maxTravelTimeCmpToWalk || results.size() == 0)) {
|
2018-08-04 13:03:06 +02:00
|
|
|
results.add(finish);
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
}
|
2019-01-09 17:13:49 +01:00
|
|
|
|
|
|
|
if (ctx.calculationProgress != null && ctx.calculationProgress.isCancelled) {
|
|
|
|
throw new InterruptedException("Route calculation interrupted");
|
|
|
|
}
|
2019-03-05 23:49:48 +01:00
|
|
|
if (MEASURE_TIME) {
|
|
|
|
long time = System.currentTimeMillis() - beginMs;
|
|
|
|
if (time > 10) {
|
|
|
|
System.out.println(String.format("%d ms ref - %s id - %d", time, segment.road.getRef(),
|
|
|
|
segment.road.getId()));
|
|
|
|
}
|
|
|
|
}
|
2019-01-09 17:13:49 +01:00
|
|
|
updateCalculationProgress(ctx, queue);
|
|
|
|
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
2018-08-04 13:03:06 +02:00
|
|
|
|
|
|
|
return prepareResults(ctx, results);
|
|
|
|
}
|
2019-01-09 17:13:49 +01:00
|
|
|
|
|
|
|
private void initProgressBar(TransportRoutingContext ctx, LatLon start, LatLon end) {
|
2019-02-25 16:01:05 +01:00
|
|
|
if (ctx.calculationProgress != null) {
|
|
|
|
ctx.calculationProgress.distanceFromEnd = 0;
|
|
|
|
ctx.calculationProgress.reverseSegmentQueueSize = 0;
|
|
|
|
ctx.calculationProgress.directSegmentQueueSize = 0;
|
2019-03-17 14:09:41 +01:00
|
|
|
float speed = (float) ctx.cfg.defaultTravelSpeed + 1; // assume
|
2019-02-25 16:01:05 +01:00
|
|
|
ctx.calculationProgress.totalEstimatedDistance = (float) (MapUtils.getDistance(start, end) / speed);
|
|
|
|
}
|
2019-01-09 17:13:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private void updateCalculationProgress(TransportRoutingContext ctx, PriorityQueue<TransportRouteSegment> queue) {
|
|
|
|
if (ctx.calculationProgress != null) {
|
|
|
|
ctx.calculationProgress.directSegmentQueueSize = queue.size();
|
|
|
|
if (queue.size() > 0) {
|
|
|
|
TransportRouteSegment peek = queue.peek();
|
|
|
|
ctx.calculationProgress.distanceFromBegin = (float) Math.max(peek.distFromStart,
|
|
|
|
ctx.calculationProgress.distanceFromBegin);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
|
|
|
|
private List<TransportRouteResult> prepareResults(TransportRoutingContext ctx, List<TransportRouteSegment> results) {
|
|
|
|
Collections.sort(results, new SegmentsComparator(ctx));
|
|
|
|
List<TransportRouteResult> lst = new ArrayList<TransportRouteResult>();
|
2019-03-15 14:41:05 +01:00
|
|
|
System.out.println(String.format("Calculated %.1f seconds, found %d results, visited %d routes / %d stops, loaded %d tiles (%d ms read, %d ms total), loaded ways %d (%d wrong)",
|
|
|
|
(System.currentTimeMillis() - ctx.startCalcTime) / 1000.0, results.size(),
|
|
|
|
ctx.visitedRoutesCount, ctx.visitedStops,
|
|
|
|
ctx.quadTree.size(), ctx.readTime / (1000 * 1000), ctx.loadTime / (1000 * 1000),
|
|
|
|
ctx.loadedWays, ctx.wrongLoadedWays));
|
2018-08-04 13:03:06 +02:00
|
|
|
for(TransportRouteSegment res : results) {
|
2018-12-19 17:53:38 +01:00
|
|
|
if (ctx.calculationProgress != null && ctx.calculationProgress.isCancelled) {
|
|
|
|
return null;
|
|
|
|
}
|
2018-08-04 13:03:06 +02:00
|
|
|
TransportRouteResult route = new TransportRouteResult(ctx);
|
|
|
|
route.routeTime = res.distFromStart;
|
|
|
|
route.finishWalkDist = res.walkDist;
|
|
|
|
TransportRouteSegment p = res;
|
|
|
|
while (p != null) {
|
2018-12-19 17:53:38 +01:00
|
|
|
if (ctx.calculationProgress != null && ctx.calculationProgress.isCancelled) {
|
|
|
|
return null;
|
|
|
|
}
|
2018-08-04 13:03:06 +02:00
|
|
|
if (p.parentRoute != null) {
|
2019-03-15 13:44:49 +01:00
|
|
|
TransportRouteResultSegment sg = new TransportRouteResultSegment();
|
|
|
|
sg.route = p.parentRoute.road;
|
|
|
|
sg.start = p.parentRoute.segStart;
|
|
|
|
sg.end = p.parentStop;
|
|
|
|
sg.walkDist = p.parentRoute.walkDist;
|
|
|
|
sg.walkTime = sg.walkDist / ctx.cfg.walkSpeed;
|
|
|
|
sg.depTime = p.departureTime;
|
2019-03-17 14:09:41 +01:00
|
|
|
sg.travelDistApproximate = p.parentTravelDist;
|
2019-03-15 13:44:49 +01:00
|
|
|
sg.travelTime = p.parentTravelTime;
|
2018-08-04 13:03:06 +02:00
|
|
|
route.segments.add(0, sg);
|
|
|
|
}
|
|
|
|
p = p.parentRoute;
|
|
|
|
}
|
|
|
|
// test if faster routes fully included
|
|
|
|
boolean include = false;
|
|
|
|
for(TransportRouteResult s : lst) {
|
2018-12-19 17:53:38 +01:00
|
|
|
if (ctx.calculationProgress != null && ctx.calculationProgress.isCancelled) {
|
|
|
|
return null;
|
|
|
|
}
|
2018-08-04 13:03:06 +02:00
|
|
|
if(includeRoute(s, route)) {
|
|
|
|
include = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(!include) {
|
|
|
|
lst.add(route);
|
|
|
|
System.out.println(route.toString());
|
|
|
|
} else {
|
|
|
|
// System.err.println(route.toString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return lst;
|
|
|
|
}
|
|
|
|
|
|
|
|
private boolean includeRoute(TransportRouteResult fastRoute, TransportRouteResult testRoute) {
|
|
|
|
if(testRoute.segments.size() < fastRoute.segments.size()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
int j = 0;
|
|
|
|
for(int i = 0; i < fastRoute.segments.size(); i++, j++) {
|
|
|
|
TransportRouteResultSegment fs = fastRoute.segments.get(i);
|
|
|
|
while(j < testRoute.segments.size()) {
|
|
|
|
TransportRouteResultSegment ts = testRoute.segments.get(j);
|
|
|
|
if(fs.route.getId().longValue() != ts.route.getId().longValue()) {
|
|
|
|
j++;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(j >= testRoute.segments.size()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private static class SegmentsComparator implements Comparator<TransportRouteSegment> {
|
|
|
|
|
|
|
|
public SegmentsComparator(TransportRoutingContext ctx) {
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public int compare(TransportRouteSegment o1, TransportRouteSegment o2) {
|
2018-08-04 13:03:06 +02:00
|
|
|
return Double.compare(o1.distFromStart, o2.distFromStart);
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
|
|
|
|
public static class TransportRouteResultSegment {
|
2019-03-15 13:44:49 +01:00
|
|
|
|
2019-03-13 15:58:40 +01:00
|
|
|
private static final boolean DISPLAY_FULL_SEGMENT_ROUTE = false;
|
|
|
|
private static final int DISPLAY_SEGMENT_IND = 0;
|
2019-03-15 13:44:49 +01:00
|
|
|
public TransportRoute route;
|
|
|
|
public double walkTime;
|
2019-03-17 14:09:41 +01:00
|
|
|
public double travelDistApproximate;
|
2019-03-15 13:44:49 +01:00
|
|
|
public double travelTime;
|
|
|
|
public int start;
|
|
|
|
public int end;
|
|
|
|
public double walkDist ;
|
|
|
|
public int depTime;
|
2018-08-04 13:03:06 +02:00
|
|
|
|
2019-03-15 13:44:49 +01:00
|
|
|
public TransportRouteResultSegment() {
|
2018-08-20 21:00:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public int getArrivalTime() {
|
|
|
|
if(route.getSchedule() != null && depTime != -1) {
|
|
|
|
int tm = depTime;
|
|
|
|
TIntArrayList intervals = route.getSchedule().avgStopIntervals;
|
|
|
|
for(int i = start; i <= end; i++) {
|
|
|
|
if(i == end) {
|
|
|
|
return tm;
|
|
|
|
}
|
|
|
|
if(intervals.size() > i) {
|
|
|
|
tm += intervals.get(i);
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1;
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
2018-08-04 13:03:06 +02:00
|
|
|
|
2019-03-17 14:09:41 +01:00
|
|
|
public double getTravelTime() {
|
|
|
|
return travelTime;
|
|
|
|
}
|
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
public TransportStop getStart() {
|
|
|
|
return route.getForwardStops().get(start);
|
|
|
|
}
|
|
|
|
|
|
|
|
public TransportStop getEnd() {
|
|
|
|
return route.getForwardStops().get(end);
|
|
|
|
}
|
2018-12-20 12:28:09 +01:00
|
|
|
|
2019-03-01 12:35:59 +01:00
|
|
|
public List<TransportStop> getTravelStops() {
|
2019-03-11 17:51:46 +01:00
|
|
|
return route.getForwardStops().subList(start, end + 1);
|
2019-03-01 12:35:59 +01:00
|
|
|
}
|
|
|
|
|
2019-03-14 20:13:04 +01:00
|
|
|
public QuadRect getSegmentRect() {
|
|
|
|
double left = 0, right = 0;
|
|
|
|
double top = 0, bottom = 0;
|
|
|
|
for (Node n : getNodes()) {
|
|
|
|
if (left == 0 && right == 0) {
|
|
|
|
left = n.getLongitude();
|
|
|
|
right = n.getLongitude();
|
|
|
|
top = n.getLatitude();
|
|
|
|
bottom = n.getLatitude();
|
|
|
|
} else {
|
|
|
|
left = Math.min(left, n.getLongitude());
|
|
|
|
right = Math.max(right, n.getLongitude());
|
|
|
|
top = Math.max(top, n.getLatitude());
|
|
|
|
bottom = Math.min(bottom, n.getLatitude());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return left == 0 && right == 0 ? null : new QuadRect(left, top, right, bottom);
|
|
|
|
}
|
|
|
|
|
2018-12-20 12:28:09 +01:00
|
|
|
public List<Node> getNodes() {
|
|
|
|
List<Node> nodes = new ArrayList<>();
|
|
|
|
List<Way> ways = getGeometry();
|
|
|
|
for (Way way : ways) {
|
|
|
|
nodes.addAll(way.getNodes());
|
|
|
|
}
|
|
|
|
return nodes;
|
|
|
|
}
|
|
|
|
|
2018-08-05 22:35:28 +02:00
|
|
|
public List<Way> getGeometry() {
|
2019-04-05 18:49:25 +02:00
|
|
|
List<Way> list = new ArrayList<>();
|
2018-08-05 22:35:28 +02:00
|
|
|
route.mergeForwardWays();
|
2019-04-05 18:49:25 +02:00
|
|
|
if (DISPLAY_FULL_SEGMENT_ROUTE) {
|
2019-03-13 15:58:40 +01:00
|
|
|
System.out.println("TOTAL SEGMENTS: " + route.getForwardWays().size());
|
2019-04-05 18:49:25 +02:00
|
|
|
if (route.getForwardWays().size() > DISPLAY_SEGMENT_IND) {
|
2019-03-13 15:58:40 +01:00
|
|
|
return Collections.singletonList(route.getForwardWays().get(DISPLAY_SEGMENT_IND));
|
|
|
|
}
|
|
|
|
return route.getForwardWays();
|
|
|
|
}
|
2018-08-05 22:35:28 +02:00
|
|
|
List<Way> fw = route.getForwardWays();
|
|
|
|
double minStart = 150;
|
|
|
|
double minEnd = 150;
|
|
|
|
LatLon str = getStart().getLocation();
|
|
|
|
LatLon en = getEnd().getLocation();
|
|
|
|
int endInd = -1;
|
2019-04-05 18:49:25 +02:00
|
|
|
List<Node> res = new ArrayList<>();
|
|
|
|
for (int i = 0; i < fw.size() ; i++) {
|
2018-08-05 22:35:28 +02:00
|
|
|
List<Node> nodes = fw.get(i).getNodes();
|
2019-04-05 18:49:25 +02:00
|
|
|
for (int j = 0; j < nodes.size(); j++) {
|
2018-08-05 22:35:28 +02:00
|
|
|
Node n = nodes.get(j);
|
2019-04-05 18:49:25 +02:00
|
|
|
if (MapUtils.getDistance(str, n.getLatitude(), n.getLongitude()) < minStart) {
|
2018-08-05 22:35:28 +02:00
|
|
|
minStart = MapUtils.getDistance(str, n.getLatitude(), n.getLongitude());
|
|
|
|
res.clear();
|
|
|
|
}
|
|
|
|
res.add(n);
|
2019-04-05 18:49:25 +02:00
|
|
|
if (MapUtils.getDistance(en, n.getLatitude(), n.getLongitude()) < minEnd) {
|
2018-08-05 22:35:28 +02:00
|
|
|
endInd = res.size();
|
|
|
|
minEnd = MapUtils.getDistance(en, n.getLatitude(), n.getLongitude());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-04-05 19:59:49 +02:00
|
|
|
Way way;
|
2019-04-05 18:49:25 +02:00
|
|
|
if (res.isEmpty() || endInd == -1) {
|
2019-04-05 19:59:49 +02:00
|
|
|
way = new Way(STOPS_WAY_ID);
|
2018-08-05 22:35:28 +02:00
|
|
|
for (int i = start; i <= end; i++) {
|
|
|
|
LatLon l = getStop(i).getLocation();
|
|
|
|
Node n = new Node(l.getLatitude(), l.getLongitude(), -1);
|
|
|
|
way.addNode(n);
|
|
|
|
}
|
|
|
|
} else {
|
2019-04-05 19:59:49 +02:00
|
|
|
way = new Way(GEOMETRY_WAY_ID);
|
2019-04-05 18:49:25 +02:00
|
|
|
for(int k = 0; k < res.size() && k < endInd; k++) {
|
2018-08-05 22:35:28 +02:00
|
|
|
way.addNode(res.get(k));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
list.add(way);
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
public double getTravelDist() {
|
|
|
|
double d = 0;
|
|
|
|
for (int k = start; k < end; k++) {
|
|
|
|
d += MapUtils.getDistance(route.getForwardStops().get(k).getLocation(),
|
|
|
|
route.getForwardStops().get(k + 1).getLocation());
|
|
|
|
}
|
|
|
|
return d;
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
2018-08-05 10:55:08 +02:00
|
|
|
|
|
|
|
public TransportStop getStop(int i) {
|
|
|
|
return route.getForwardStops().get(i);
|
|
|
|
}
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
public static class TransportRouteResult {
|
|
|
|
|
|
|
|
List<TransportRouteResultSegment> segments = new ArrayList<TransportRouteResultSegment>(4);
|
|
|
|
double finishWalkDist;
|
|
|
|
double routeTime;
|
2018-08-05 22:35:28 +02:00
|
|
|
private final TransportRoutingConfiguration cfg;
|
2018-08-04 13:03:06 +02:00
|
|
|
|
|
|
|
public TransportRouteResult(TransportRoutingContext ctx) {
|
2018-08-05 22:35:28 +02:00
|
|
|
cfg = ctx.cfg;
|
2018-08-04 13:03:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public List<TransportRouteResultSegment> getSegments() {
|
|
|
|
return segments;
|
|
|
|
}
|
2018-12-14 12:06:59 +01:00
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
public double getWalkDist() {
|
|
|
|
double d = finishWalkDist;
|
|
|
|
for (TransportRouteResultSegment s : segments) {
|
|
|
|
d += s.walkDist;
|
|
|
|
}
|
|
|
|
return d;
|
|
|
|
}
|
2018-12-14 10:16:23 +01:00
|
|
|
|
|
|
|
public double getFinishWalkDist() {
|
|
|
|
return finishWalkDist;
|
|
|
|
}
|
|
|
|
|
|
|
|
public double getWalkSpeed() {
|
|
|
|
return cfg.walkSpeed;
|
|
|
|
}
|
|
|
|
|
2018-08-05 22:35:28 +02:00
|
|
|
public double getRouteTime() {
|
|
|
|
return routeTime;
|
|
|
|
}
|
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
public int getStops() {
|
|
|
|
int stops = 0;
|
|
|
|
for(TransportRouteResultSegment s : segments) {
|
|
|
|
stops += (s.end - s.start);
|
|
|
|
}
|
|
|
|
return stops;
|
|
|
|
}
|
2019-07-10 14:52:20 +02:00
|
|
|
|
|
|
|
public boolean isRouteStop(TransportStop stop) {
|
|
|
|
for(TransportRouteResultSegment s : segments) {
|
|
|
|
if (s.getTravelStops().contains(stop)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public TransportRouteResultSegment getRouteStopSegment(TransportStop stop) {
|
|
|
|
for(TransportRouteResultSegment s : segments) {
|
|
|
|
if (s.getTravelStops().contains(stop)) {
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
public double getTravelDist() {
|
|
|
|
double d = 0;
|
|
|
|
for (TransportRouteResultSegment s : segments) {
|
|
|
|
d += s.getTravelDist();
|
|
|
|
}
|
|
|
|
return d;
|
|
|
|
}
|
|
|
|
|
2018-08-05 22:35:28 +02:00
|
|
|
public double getTravelTime() {
|
2019-03-17 14:09:41 +01:00
|
|
|
double t = 0;
|
|
|
|
for (TransportRouteResultSegment s : segments) {
|
|
|
|
if (cfg.useSchedule) {
|
2018-08-15 14:43:47 +02:00
|
|
|
TransportSchedule sts = s.route.getSchedule();
|
|
|
|
for (int k = s.start; k < s.end; k++) {
|
|
|
|
t += sts.getAvgStopIntervals()[k] * 10;
|
|
|
|
}
|
2019-03-17 14:09:41 +01:00
|
|
|
} else {
|
2019-03-17 14:19:26 +01:00
|
|
|
t += cfg.getBoardingTime();
|
2019-03-17 14:09:41 +01:00
|
|
|
t += s.getTravelTime();
|
2018-08-15 14:43:47 +02:00
|
|
|
}
|
|
|
|
}
|
2019-03-17 14:09:41 +01:00
|
|
|
return t;
|
2018-08-05 22:35:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public double getWalkTime() {
|
|
|
|
return getWalkDist() / cfg.walkSpeed;
|
|
|
|
}
|
2019-03-15 17:06:51 +01:00
|
|
|
|
|
|
|
public double getChangeTime() {
|
|
|
|
return cfg.getChangeTime();
|
|
|
|
}
|
|
|
|
|
2019-03-23 09:33:27 +01:00
|
|
|
public double getBoardingTime() {
|
|
|
|
return cfg.getBoardingTime();
|
|
|
|
}
|
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
public int getChanges() {
|
|
|
|
return segments.size() - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public String toString() {
|
|
|
|
StringBuilder bld = new StringBuilder();
|
|
|
|
bld.append(String.format("Route %d stops, %d changes, %.2f min: %.2f m (%.1f min) to walk, %.2f m (%.1f min) to travel\n",
|
2018-08-05 22:35:28 +02:00
|
|
|
getStops(), getChanges(), routeTime / 60, getWalkDist(), getWalkTime() / 60.0,
|
|
|
|
getTravelDist(), getTravelTime() / 60.0));
|
2018-08-04 13:03:06 +02:00
|
|
|
for(int i = 0; i < segments.size(); i++) {
|
|
|
|
TransportRouteResultSegment s = segments.get(i);
|
2018-08-20 21:00:22 +02:00
|
|
|
String time = "";
|
|
|
|
String arriveTime = "";
|
|
|
|
if(s.depTime != -1) {
|
|
|
|
time = String.format("at %s", formatTransporTime(s.depTime));
|
|
|
|
}
|
|
|
|
int aTime = s.getArrivalTime();
|
|
|
|
if(aTime != -1) {
|
|
|
|
arriveTime = String.format("and arrive at %s", formatTransporTime(aTime));
|
|
|
|
}
|
|
|
|
bld.append(String.format(" %d. %s: walk %.1f m to '%s' and travel %s to '%s' by %s %d stops %s\n",
|
|
|
|
i + 1, s.route.getRef(), s.walkDist, s.getStart().getName(),
|
|
|
|
time, s.getEnd().getName(),s.route.getName(), (s.end - s.start), arriveTime));
|
2018-08-04 13:03:06 +02:00
|
|
|
}
|
|
|
|
bld.append(String.format(" F. Walk %.1f m to reach your destination", finishWalkDist));
|
|
|
|
return bld.toString();
|
|
|
|
}
|
2018-08-20 21:00:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public static String formatTransporTime(int i) {
|
|
|
|
int h = i / 60 / 6;
|
|
|
|
int mh = i - h * 60 * 6;
|
|
|
|
int m = mh / 6;
|
|
|
|
int s = (mh - m * 6) * 10;
|
|
|
|
String tm = String.format("%02d:%02d:%02d ", h, m, s);
|
|
|
|
return tm;
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
public static class TransportRouteSegment {
|
2018-08-04 13:03:06 +02:00
|
|
|
|
|
|
|
final int segStart;
|
2018-08-03 12:50:51 +02:00
|
|
|
final TransportRoute road;
|
2018-08-15 14:43:47 +02:00
|
|
|
final int departureTime;
|
2018-08-03 12:50:51 +02:00
|
|
|
private static final int SHIFT = 10; // assume less than 1024 stops
|
2018-08-20 21:00:22 +02:00
|
|
|
private static final int SHIFT_DEPTIME = 14; // assume less than 1024 stops
|
2018-08-03 12:50:51 +02:00
|
|
|
|
|
|
|
TransportRouteSegment parentRoute = null;
|
2019-03-15 13:44:49 +01:00
|
|
|
int parentStop; // last stop to exit for parent route
|
|
|
|
double parentTravelTime; // travel time for parent route
|
|
|
|
double parentTravelDist; // travel distance for parent route (inaccurate)
|
2018-08-04 13:03:06 +02:00
|
|
|
// walk distance to start route location (or finish in case last segment)
|
|
|
|
double walkDist = 0;
|
|
|
|
|
|
|
|
// main field accumulated all time spent from beginning of journey
|
|
|
|
double distFromStart = 0;
|
|
|
|
|
2018-08-03 12:50:51 +02:00
|
|
|
|
2018-08-15 14:43:47 +02:00
|
|
|
|
|
|
|
|
2018-08-03 12:50:51 +02:00
|
|
|
public TransportRouteSegment(TransportRoute road, int stopIndex) {
|
|
|
|
this.road = road;
|
|
|
|
this.segStart = (short) stopIndex;
|
2018-08-15 14:43:47 +02:00
|
|
|
this.departureTime = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
public TransportRouteSegment(TransportRoute road, int stopIndex, int depTime) {
|
|
|
|
this.road = road;
|
|
|
|
this.segStart = (short) stopIndex;
|
|
|
|
this.departureTime = depTime;
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
public TransportRouteSegment(TransportRouteSegment c) {
|
|
|
|
this.road = c.road;
|
|
|
|
this.segStart = c.segStart;
|
2018-08-15 14:43:47 +02:00
|
|
|
this.departureTime = c.departureTime;
|
2018-08-04 13:03:06 +02:00
|
|
|
}
|
2018-08-03 12:50:51 +02:00
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
|
|
|
|
public boolean wasVisited(TransportRouteSegment rrs) {
|
2018-08-20 21:00:22 +02:00
|
|
|
if (rrs.road.getId().longValue() == road.getId().longValue() &&
|
|
|
|
rrs.departureTime == departureTime) {
|
2018-08-04 13:03:06 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if(parentRoute != null) {
|
|
|
|
return parentRoute.wasVisited(rrs);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public TransportStop getStop(int i) {
|
|
|
|
return road.getForwardStops().get(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public LatLon getLocation() {
|
|
|
|
return road.getForwardStops().get(segStart).getLocation();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-08-03 12:50:51 +02:00
|
|
|
public int getLength() {
|
|
|
|
return road.getForwardStops().size();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public long getId() {
|
2018-08-20 21:00:22 +02:00
|
|
|
long l = road.getId();
|
|
|
|
|
|
|
|
l = l << SHIFT_DEPTIME;
|
|
|
|
if(departureTime >= (1 << SHIFT_DEPTIME)) {
|
|
|
|
throw new IllegalStateException("too long dep time" + departureTime);
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
2018-08-20 21:00:22 +02:00
|
|
|
l += (departureTime + 1);
|
|
|
|
|
|
|
|
l = l << SHIFT;
|
|
|
|
if (segStart >= (1 << SHIFT)) {
|
2018-08-03 12:50:51 +02:00
|
|
|
throw new IllegalStateException("too many stops " + road.getId() + " " + segStart);
|
|
|
|
}
|
2018-08-20 21:00:22 +02:00
|
|
|
l += segStart;
|
|
|
|
|
|
|
|
if(l < 0 ) {
|
|
|
|
throw new IllegalStateException("too long id " + road.getId());
|
|
|
|
}
|
|
|
|
return l ;
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
public int getDepth() {
|
|
|
|
if(parentRoute != null) {
|
|
|
|
return parentRoute.getDepth() + 1;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
@Override
|
|
|
|
public String toString() {
|
2018-08-20 21:00:22 +02:00
|
|
|
return String.format("Route: %s, stop: %s %s", road.getName(), road.getForwardStops().get(segStart).getName(),
|
|
|
|
departureTime == -1 ? "" : formatTransporTime(departureTime) );
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
public static class TransportRoutingContext {
|
|
|
|
|
|
|
|
public RouteCalculationProgress calculationProgress;
|
2018-08-04 13:03:06 +02:00
|
|
|
public TLongObjectHashMap<TransportRouteSegment> visitedSegments = new TLongObjectHashMap<TransportRouteSegment>();
|
2018-08-05 23:46:01 +02:00
|
|
|
public TransportRoutingConfiguration cfg;
|
2018-08-04 13:03:06 +02:00
|
|
|
|
2018-08-03 12:50:51 +02:00
|
|
|
|
2018-08-05 23:46:01 +02:00
|
|
|
public TLongObjectHashMap<List<TransportRouteSegment>> quadTree;
|
2018-08-15 14:43:47 +02:00
|
|
|
public final Map<BinaryMapIndexReader, TIntObjectHashMap<TransportRoute>> routeMap =
|
2018-08-03 12:50:51 +02:00
|
|
|
new LinkedHashMap<BinaryMapIndexReader, TIntObjectHashMap<TransportRoute>>();
|
|
|
|
|
2018-08-06 00:01:25 +02:00
|
|
|
// stats
|
|
|
|
public long startCalcTime;
|
|
|
|
public int visitedRoutesCount;
|
2019-03-15 14:41:05 +01:00
|
|
|
public int visitedStops;
|
2018-08-06 00:01:25 +02:00
|
|
|
public int wrongLoadedWays;
|
|
|
|
public int loadedWays;
|
|
|
|
public long loadTime;
|
|
|
|
public long readTime;
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
private final int walkRadiusIn31;
|
|
|
|
private final int walkChangeRadiusIn31;
|
2018-08-03 12:50:51 +02:00
|
|
|
|
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
|
|
|
|
|
|
|
|
public TransportRoutingContext(TransportRoutingConfiguration cfg, BinaryMapIndexReader... readers) {
|
|
|
|
this.cfg = cfg;
|
|
|
|
walkRadiusIn31 = (int) (cfg.walkRadius / MapUtils.getTileDistanceWidth(31));
|
|
|
|
walkChangeRadiusIn31 = (int) (cfg.walkChangeRadius / MapUtils.getTileDistanceWidth(31));
|
2018-08-05 23:46:01 +02:00
|
|
|
quadTree = new TLongObjectHashMap<List<TransportRouteSegment>>();
|
2018-08-03 12:50:51 +02:00
|
|
|
for (BinaryMapIndexReader r : readers) {
|
2018-08-15 14:43:47 +02:00
|
|
|
routeMap.put(r, new TIntObjectHashMap<TransportRoute>());
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-04 13:03:06 +02:00
|
|
|
public List<TransportRouteSegment> getTransportStops(LatLon loc) throws IOException {
|
|
|
|
int y = MapUtils.get31TileNumberY(loc.getLatitude());
|
|
|
|
int x = MapUtils.get31TileNumberX(loc.getLongitude());
|
2018-08-05 23:46:01 +02:00
|
|
|
return getTransportStops(x, y, false, new ArrayList<TransportRouteSegment>());
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
|
2018-08-05 23:46:01 +02:00
|
|
|
public List<TransportRouteSegment> getTransportStops(int x, int y, boolean change, List<TransportRouteSegment> res) throws IOException {
|
|
|
|
return loadNativeTransportStops(x, y, change, res);
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
|
2018-08-05 23:46:01 +02:00
|
|
|
private List<TransportRouteSegment> loadNativeTransportStops(int sx, int sy, boolean change, List<TransportRouteSegment> res) throws IOException {
|
2018-08-04 13:03:06 +02:00
|
|
|
long nanoTime = System.nanoTime();
|
|
|
|
int d = change ? walkChangeRadiusIn31 : walkRadiusIn31;
|
|
|
|
int lx = (sx - d ) >> (31 - cfg.ZOOM_TO_LOAD_TILES);
|
|
|
|
int rx = (sx + d ) >> (31 - cfg.ZOOM_TO_LOAD_TILES);
|
|
|
|
int ty = (sy - d ) >> (31 - cfg.ZOOM_TO_LOAD_TILES);
|
|
|
|
int by = (sy + d ) >> (31 - cfg.ZOOM_TO_LOAD_TILES);
|
2018-08-03 12:50:51 +02:00
|
|
|
for(int x = lx; x <= rx; x++) {
|
|
|
|
for(int y = ty; y <= by; y++) {
|
2019-03-15 17:38:55 +01:00
|
|
|
long tileId = (((long)x) << (cfg.ZOOM_TO_LOAD_TILES + 1)) + y;
|
2018-08-05 23:46:01 +02:00
|
|
|
List<TransportRouteSegment> list = quadTree.get(tileId);
|
|
|
|
if(list == null) {
|
|
|
|
list = loadTile(x, y);
|
|
|
|
quadTree.put(tileId, list);
|
|
|
|
}
|
|
|
|
for(TransportRouteSegment r : list) {
|
|
|
|
TransportStop st = r.getStop(r.segStart);
|
|
|
|
if (Math.abs(st.x31 - sx) > walkRadiusIn31 || Math.abs(st.y31 - sy) > walkRadiusIn31) {
|
|
|
|
wrongLoadedWays++;
|
|
|
|
} else {
|
|
|
|
loadedWays++;
|
|
|
|
res.add(r);
|
|
|
|
}
|
|
|
|
}
|
2018-08-04 13:03:06 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
loadTime += System.nanoTime() - nanoTime;
|
2018-08-05 23:46:01 +02:00
|
|
|
return res;
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-08-05 23:46:01 +02:00
|
|
|
private List<TransportRouteSegment> loadTile(int x, int y) throws IOException {
|
2018-08-06 00:01:25 +02:00
|
|
|
long nanoTime = System.nanoTime();
|
2018-08-05 23:46:01 +02:00
|
|
|
List<TransportRouteSegment> lst = new ArrayList<TransportRouteSegment>();
|
2018-08-04 13:03:06 +02:00
|
|
|
int pz = (31 - cfg.ZOOM_TO_LOAD_TILES);
|
2019-07-22 15:36:07 +02:00
|
|
|
SearchRequest<TransportStop> sr = BinaryMapIndexReader.buildSearchTransportRequest(x << pz, (x + 1) << pz,
|
2018-08-03 12:50:51 +02:00
|
|
|
y << pz, (y + 1) << pz, -1, null);
|
2019-07-29 15:36:19 +02:00
|
|
|
|
2019-07-29 16:13:55 +02:00
|
|
|
// could be global ?
|
|
|
|
TLongObjectHashMap<TransportStop> loadedTransportStops = new TLongObjectHashMap<TransportStop>();
|
2019-07-29 15:36:19 +02:00
|
|
|
|
2019-07-29 16:13:55 +02:00
|
|
|
TIntArrayList localFileRoutesToLoad = new TIntArrayList();
|
2019-07-22 15:36:07 +02:00
|
|
|
for (BinaryMapIndexReader r : routeMap.keySet()) {
|
2018-08-03 12:50:51 +02:00
|
|
|
sr.clearSearchResults();
|
|
|
|
List<TransportStop> stops = r.searchTransportIndex(sr);
|
2019-07-29 16:13:55 +02:00
|
|
|
|
|
|
|
localFileRoutesToLoad.clear();
|
|
|
|
prepareRoutesToLoad(loadedTransportStops, stops, localFileRoutesToLoad);
|
|
|
|
|
|
|
|
|
2019-07-29 15:36:19 +02:00
|
|
|
// load routes and create transport segments
|
2019-07-29 16:13:55 +02:00
|
|
|
TIntObjectHashMap<TransportRoute> localFileRoutes = new TIntObjectHashMap<TransportRoute>();
|
|
|
|
if (localFileRoutesToLoad.size() > 0) {
|
|
|
|
localFileRoutesToLoad.sort();
|
2019-07-29 15:36:19 +02:00
|
|
|
TIntArrayList referencesToLoad = new TIntArrayList();
|
2018-08-15 14:43:47 +02:00
|
|
|
TIntObjectHashMap<TransportRoute> loadedRoutes = routeMap.get(r);
|
2019-07-29 16:13:55 +02:00
|
|
|
TIntIterator it = localFileRoutesToLoad.iterator();
|
|
|
|
int p = localFileRoutesToLoad.get(0) + 1; // different
|
2019-07-22 15:36:07 +02:00
|
|
|
while (it.hasNext()) {
|
2018-08-15 14:43:47 +02:00
|
|
|
int nxt = it.next();
|
|
|
|
if (p != nxt) {
|
|
|
|
if (loadedRoutes.contains(nxt)) {
|
2019-07-29 16:13:55 +02:00
|
|
|
localFileRoutes.put(nxt, loadedRoutes.get(nxt));
|
2018-08-15 14:43:47 +02:00
|
|
|
} else {
|
2019-07-22 15:36:07 +02:00
|
|
|
referencesToLoad.add(nxt);
|
2018-08-15 14:43:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-07-29 16:13:55 +02:00
|
|
|
r.loadTransportRoutes(referencesToLoad.toArray(), localFileRoutes);
|
|
|
|
loadedRoutes.putAll(localFileRoutes);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (TransportStop stop : stops) {
|
|
|
|
long stopId = stop.getId();
|
|
|
|
TransportStop multifileStop = loadedTransportStops.get(stopId);
|
|
|
|
int[] rrs = stop.getReferencesToRoutes();
|
|
|
|
if (multifileStop == stop) {
|
|
|
|
// clear up so it won't be used as it is multi file stop
|
|
|
|
stop.setReferencesToRoutes(null);
|
|
|
|
}
|
|
|
|
if(rrs != null && !multifileStop.isDeleted()) {
|
|
|
|
for(int i = 0; i < rrs.length; i++) {
|
|
|
|
TransportRoute route = localFileRoutes.get(rrs[i]);
|
|
|
|
if(route == null) {
|
|
|
|
System.err.println(String.format("Something went wrong by loading route %d for stop %s", rrs[i], stop));
|
|
|
|
} else {
|
|
|
|
// duplicates won't be added
|
|
|
|
multifileStop.addRouteId(route.getId());
|
|
|
|
multifileStop.addRoute(route);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
2019-07-22 15:36:07 +02:00
|
|
|
}
|
2019-07-29 15:36:19 +02:00
|
|
|
|
2019-07-29 16:17:27 +02:00
|
|
|
loadTransportSegments(loadedTransportStops.valueCollection(), lst);
|
2019-07-29 15:36:19 +02:00
|
|
|
|
2018-08-06 00:01:25 +02:00
|
|
|
readTime += System.nanoTime() - nanoTime;
|
2018-08-05 23:46:01 +02:00
|
|
|
return lst;
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
|
2019-07-29 16:13:55 +02:00
|
|
|
private List<TransportStop> prepareRoutesToLoad(TLongObjectHashMap<TransportStop> loadedTransportStops,
|
|
|
|
List<TransportStop> stops, TIntArrayList resRoutesToLoad) {
|
|
|
|
Iterator<TransportStop> it = stops.iterator();
|
|
|
|
while (it.hasNext()) {
|
|
|
|
TransportStop stop = it.next();
|
|
|
|
long stopId = stop.getId();
|
|
|
|
TransportStop multifileStop = loadedTransportStops.get(stopId);
|
|
|
|
long[] routesIds = stop.getRoutesIds();
|
|
|
|
long[] delRIds = stop.getDeletedRoutesIds();
|
|
|
|
if (multifileStop == null) {
|
|
|
|
loadedTransportStops.put(stopId, stop);
|
|
|
|
if(!stop.isDeleted()) {
|
|
|
|
resRoutesToLoad.addAll(stop.getReferencesToRoutes());
|
|
|
|
}
|
|
|
|
} else if(multifileStop.isDeleted()){
|
|
|
|
// stop has noting to load, so not needed
|
|
|
|
it.remove();
|
|
|
|
} else {
|
|
|
|
if (delRIds != null) {
|
|
|
|
for (long deletedRouteId : delRIds) {
|
|
|
|
multifileStop.addDeletedRouteId(deletedRouteId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (routesIds != null && routesIds.length > 0) {
|
|
|
|
int[] refs = stop.getReferencesToRoutes();
|
|
|
|
for (int i = 0; i < routesIds.length; i++) {
|
|
|
|
long routeId = routesIds[i];
|
|
|
|
if (!multifileStop.hasRoute(routeId) && !multifileStop.isRouteDeleted(routeId)) {
|
|
|
|
resRoutesToLoad.add(refs[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (stop.hasReferencesToRoutes()) {
|
|
|
|
// old format
|
|
|
|
resRoutesToLoad.addAll(stop.getReferencesToRoutes());
|
|
|
|
} else {
|
|
|
|
// stop has noting to load, so not needed
|
|
|
|
it.remove();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return stops;
|
|
|
|
}
|
|
|
|
|
2019-07-29 16:17:27 +02:00
|
|
|
private void loadTransportSegments(Collection<TransportStop> stops, List<TransportRouteSegment> lst) throws IOException {
|
2018-08-03 12:50:51 +02:00
|
|
|
for(TransportStop s : stops) {
|
2019-03-04 12:29:26 +01:00
|
|
|
if (s.isDeleted()) {
|
|
|
|
continue;
|
|
|
|
}
|
2019-07-29 15:36:19 +02:00
|
|
|
for (TransportRoute route : s.getRoutes()) {
|
|
|
|
int stopIndex = -1;
|
|
|
|
double dist = TransportRoute.SAME_STOP;
|
|
|
|
for (int k = 0; k < route.getForwardStops().size(); k++) {
|
|
|
|
TransportStop st = route.getForwardStops().get(k);
|
|
|
|
if(st.getId().longValue() == s.getId().longValue() ) {
|
|
|
|
stopIndex = k;
|
|
|
|
break;
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
2019-07-29 15:36:19 +02:00
|
|
|
double d = MapUtils.getDistance(st.getLocation(), s.getLocation());
|
|
|
|
if (d < dist) {
|
|
|
|
stopIndex = k;
|
|
|
|
dist = d;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (stopIndex != -1) {
|
|
|
|
if (cfg.useSchedule) {
|
|
|
|
loadScheduleRouteSegment(lst, route, stopIndex);
|
2018-08-05 23:46:01 +02:00
|
|
|
} else {
|
2019-07-29 15:36:19 +02:00
|
|
|
TransportRouteSegment segment = new TransportRouteSegment(route, stopIndex);
|
|
|
|
lst.add(segment);
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
2019-07-29 15:36:19 +02:00
|
|
|
} 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));
|
2018-08-03 12:50:51 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-20 21:00:22 +02:00
|
|
|
private void loadScheduleRouteSegment(List<TransportRouteSegment> lst, TransportRoute route, int stopIndex) {
|
|
|
|
if(route.getSchedule() != null) {
|
|
|
|
TIntArrayList ti = route.getSchedule().tripIntervals;
|
|
|
|
int cnt = ti.size();
|
|
|
|
int t = 0;
|
|
|
|
// improve by using exact data
|
|
|
|
int stopTravelTime = 0;
|
|
|
|
TIntArrayList avgStopIntervals = route.getSchedule().avgStopIntervals;
|
|
|
|
for (int i = 0; i < stopIndex; i++) {
|
|
|
|
if (avgStopIntervals.size() > i) {
|
|
|
|
stopTravelTime += avgStopIntervals.getQuick(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for(int i = 0; i < cnt; i++) {
|
|
|
|
t += ti.getQuick(i);
|
|
|
|
int startTime = t + stopTravelTime;
|
|
|
|
if(startTime >= cfg.scheduleTimeOfDay && startTime <= cfg.scheduleTimeOfDay + cfg.scheduleMaxTime ) {
|
|
|
|
TransportRouteSegment segment = new TransportRouteSegment(route, stopIndex, startTime);
|
|
|
|
lst.add(segment);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-15 14:43:47 +02:00
|
|
|
|
2018-08-03 12:50:51 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|