tmp
This commit is contained in:
parent
a4628db80f
commit
9f532914f2
2 changed files with 79 additions and 12 deletions
|
@ -10,6 +10,7 @@ import urllib.request
|
|||
|
||||
from subway_structure import (
|
||||
distance,
|
||||
distance_on_line,
|
||||
download_cities,
|
||||
find_transfers,
|
||||
get_unused_entrances_geojson,
|
||||
|
@ -306,15 +307,21 @@ def prepare_mapsme_data(transfers, cities):
|
|||
'route_id': uid(route.id, 'r'),
|
||||
'itineraries': []
|
||||
}
|
||||
for variant in route:
|
||||
for i, variant in enumerate(route):
|
||||
logging.warn('%s[%s]', route.name, i)
|
||||
itin = []
|
||||
time = 0
|
||||
vertex = 0
|
||||
for i, stop in enumerate(variant):
|
||||
stops[stop.stoparea.id] = stop.stoparea
|
||||
itin.append([uid(stop.stoparea.id), time])
|
||||
if i+1 < len(variant):
|
||||
# TODO: estimate segment length and calculate proper time
|
||||
d = distance(stop.stop, variant[i+1].stop)
|
||||
d = distance_on_line(stop.stop, variant[i+1].stop, variant.tracks, vertex)
|
||||
if not d:
|
||||
d = distance(stop.stop, variant[i+1].stop)
|
||||
else:
|
||||
vertex = d[1]
|
||||
d = d[0]
|
||||
time += round(d*3.6/SPEED_ON_LINE) + 20
|
||||
routes['itineraries'].append({'stops': itin, 'interval': DEFAULT_INTERVAL})
|
||||
network['routes'].append(routes)
|
||||
|
|
|
@ -61,20 +61,28 @@ def project_on_segment(p, p1, p2):
|
|||
res = (p1[0] + u*dp[0], p1[1] + u*dp[1])
|
||||
if res[0] < min(p1[0], p2[0]) or res[0] > max(p1[0], p2[0]):
|
||||
return None
|
||||
return res
|
||||
return res, u
|
||||
|
||||
|
||||
def project_on_line(p, line):
|
||||
def project_on_line_relative(p, line, start_vertex=0):
|
||||
"""Projects p on a list of coordinates and returns a tuple of (i, u, p):
|
||||
index of vertex, a relative position [0; 1] to vertex i+1
|
||||
and a projected point itself."""
|
||||
result = None
|
||||
if len(line) < 2:
|
||||
return result
|
||||
d_min = MAX_DISTANCE_STOP_TO_LINE * 5
|
||||
# First, check vertices in the line
|
||||
for vertex in line:
|
||||
d = distance(p, vertex)
|
||||
for i in range(start_vertex, len(line)):
|
||||
d = distance(p, line[i])
|
||||
if d < d_min:
|
||||
result = vertex
|
||||
if i == len(line) - 1:
|
||||
result = i-1, 1, line[i]
|
||||
else:
|
||||
result = i, 0, line[i]
|
||||
d_min = d
|
||||
# And then calculate distances to each segment
|
||||
for seg in range(len(line)-1):
|
||||
for seg in range(start_vertex, len(line)-1):
|
||||
# Check bbox for speed
|
||||
if not ((min(line[seg][0], line[seg+1][0]) - MAX_DISTANCE_STOP_TO_LINE <= p[0] <=
|
||||
max(line[seg][0], line[seg+1][0]) + MAX_DISTANCE_STOP_TO_LINE) and
|
||||
|
@ -83,11 +91,57 @@ def project_on_line(p, line):
|
|||
continue
|
||||
proj = project_on_segment(p, line[seg], line[seg+1])
|
||||
if proj:
|
||||
d = distance(p, proj)
|
||||
d = distance(p, proj[0])
|
||||
if d < d_min:
|
||||
result = proj
|
||||
result = seg, proj[1], proj[0]
|
||||
d_min = d
|
||||
return NOWHERE_STOP if not result else result
|
||||
return result
|
||||
|
||||
|
||||
def project_on_line(p, line, start_vertex=0):
|
||||
pos = project_on_line_relative(p, line, start_vertex)
|
||||
if not pos:
|
||||
return NOWHERE_STOP
|
||||
return pos[2]
|
||||
|
||||
|
||||
def distance_on_line(p1, p2, line, start_vertex=0):
|
||||
"""Calculates distance via line between projections
|
||||
of points p1 and p2. Returns a TUPLE of (d, vertex):
|
||||
d is the distance and vertex is the number of the second
|
||||
vertex, to continue calculations for the next point."""
|
||||
pos1 = project_on_line_relative(p1, line, start_vertex)
|
||||
if not pos1:
|
||||
logging.warn('p1 %s is not projected', p1)
|
||||
return None
|
||||
pos2 = project_on_line_relative(p2, line, pos1[0])
|
||||
if not pos2:
|
||||
if line[0] == line[-1]:
|
||||
logging.warn('Trying circular')
|
||||
pos2 = project_on_line_relative(p2, line, 0)
|
||||
if not pos2:
|
||||
logging.warn('p2 %s is not projected, line %s to %s', p2, line[0], line[-1])
|
||||
return None
|
||||
if pos2[0] < pos1[0] or (pos2[0] == pos1[0] and pos2[1] < pos1[1]):
|
||||
logging.warn('Pos1 %s is after pos2 %s', pos1, pos2)
|
||||
if line[0] == line[-1]:
|
||||
# If a line is circular, calculate distance properly
|
||||
pos2 += len(line)
|
||||
line = line + line
|
||||
else:
|
||||
logging.warn('Line is not circular')
|
||||
pos1, pos2 = pos2, pos1
|
||||
if pos1[0] == pos2[0]:
|
||||
return distance(line[pos1[0]], line[pos1[0]+1]) * (
|
||||
pos2[1]-pos1[1]), pos1[0]
|
||||
d = 0
|
||||
if pos1[1] < 1:
|
||||
d += distance(line[pos1[0]], line[pos1[0]+1]) * (1-pos1[1])
|
||||
for i in range(pos1[0]+1, pos2[0]):
|
||||
d += distance(line[i], line[i+1])
|
||||
if pos2[1] > 0:
|
||||
d += distance(line[pos2[0]], line[pos2[0]+1]) * pos2[1]
|
||||
return d, pos2[0]
|
||||
|
||||
|
||||
def angle_between(p1, c, p2):
|
||||
|
@ -601,6 +655,12 @@ class Route:
|
|||
msg = 'Angle between stops around "{}" is too narrow, {} degrees'.format(
|
||||
self.stops[si+1].stoparea.name, angle)
|
||||
city.error_if(angle < 20, msg, relation)
|
||||
proj1 = project_on_line_relative(self.stops[1].stop, self.tracks)
|
||||
proj2 = project_on_line_relative(self.stops[min(len(self.stops)-1, 3)].stop,
|
||||
self.tracks)
|
||||
if proj1[0] > proj2[0] or (proj1[0] == proj2[0] and proj1[1] > proj2[1]):
|
||||
city.warn('Tracks seem to go in the opposite direction to stops', relation)
|
||||
self.tracks.reverse()
|
||||
|
||||
def __len__(self):
|
||||
return len(self.stops)
|
||||
|
|
Loading…
Add table
Reference in a new issue