find border sequence from a drawn line
This commit is contained in:
parent
e15a2655b5
commit
1e6d308293
4 changed files with 91 additions and 31 deletions
|
@ -437,6 +437,51 @@ def make_osm():
|
|||
xml = xml + '</osm>'
|
||||
return Response(xml, mimetype='application/x-osm+xml')
|
||||
|
||||
@app.route('/josmbord')
|
||||
def josm_borders_along():
|
||||
name = request.args.get('name')
|
||||
line = request.args.get('line')
|
||||
cur = g.conn.cursor()
|
||||
# select all outer osm borders inside a buffer of the given line
|
||||
cur.execute("""
|
||||
with linestr as (
|
||||
select ST_Intersection(geom, ST_Buffer(ST_GeomFromText(%s, 4326), 0.1)) as line
|
||||
from {table} where name = %s
|
||||
), osmborders as (
|
||||
select (ST_Dump(way)).geom as g from {osm}, linestr where ST_Intersects(line, way)
|
||||
)
|
||||
select ST_AsGeoJSON((ST_Dump(ST_LineMerge(ST_Intersection(ST_Collect(ST_ExteriorRing(g)), line)))).geom) from osmborders, linestr group by line
|
||||
""".format(table=config.TABLE, osm=config.OSM_TABLE), (line, name))
|
||||
|
||||
node_pool = { 'id': 1 } # 'lat_lon': id
|
||||
lines = []
|
||||
for rec in cur:
|
||||
geometry = json.loads(rec[0])
|
||||
if geometry['type'] == 'LineString':
|
||||
nodes = parse_linestring(node_pool, geometry['coordinates'])
|
||||
elif geometry['type'] == 'MultiLineString':
|
||||
nodes = []
|
||||
for line in geometry['coordinates']:
|
||||
nodes.extend(parse_linestring(node_pool, line))
|
||||
if len(nodes) > 0:
|
||||
lines.append(nodes)
|
||||
|
||||
xml = '<?xml version="1.0" encoding="UTF-8"?><osm version="0.6" upload="false">'
|
||||
for latlon, node_id in node_pool.items():
|
||||
if latlon != 'id':
|
||||
(lat, lon) = latlon.split()
|
||||
xml = xml + '<node id="{id}" visible="true" version="1" lat="{lat}" lon="{lon}" />'.format(id=node_id, lat=lat, lon=lon)
|
||||
|
||||
wrid = 1
|
||||
for line in lines:
|
||||
xml = xml + '<way id="{id}" visible="true" version="1">'.format(id=wrid)
|
||||
for nd in line:
|
||||
xml = xml + '<nd ref="{ref}" />'.format(ref=nd)
|
||||
xml = xml + '</way>'
|
||||
wrid = wrid + 1
|
||||
xml = xml + '</osm>'
|
||||
return Response(xml, mimetype='application/x-osm+xml')
|
||||
|
||||
def quoteattr(value):
|
||||
value = value.replace('&', '&').replace('>', '>').replace('<', '<')
|
||||
value = value.replace('\n', ' ').replace('\r', ' ').replace('\t', '	')
|
||||
|
@ -450,19 +495,22 @@ def ring_hash(refs):
|
|||
def parse_polygon(node_pool, rings, polygon):
|
||||
role = 'outer'
|
||||
for ring in polygon:
|
||||
nodes = []
|
||||
for lonlat in ring:
|
||||
ref = '{} {}'.format(lonlat[1], lonlat[0])
|
||||
if ref in node_pool:
|
||||
node_id = node_pool[ref]
|
||||
else:
|
||||
node_id = node_pool['id']
|
||||
node_pool[ref] = node_id
|
||||
node_pool['id'] = node_id + 1
|
||||
nodes.append(node_id)
|
||||
rings.append([role, nodes])
|
||||
rings.append([role, parse_linestring(node_pool, ring)])
|
||||
role = 'inner'
|
||||
|
||||
def parse_linestring(node_pool, linestring):
|
||||
nodes = []
|
||||
for lonlat in linestring:
|
||||
ref = '{} {}'.format(lonlat[1], lonlat[0])
|
||||
if ref in node_pool:
|
||||
node_id = node_pool[ref]
|
||||
else:
|
||||
node_id = node_pool['id']
|
||||
node_pool[ref] = node_id
|
||||
node_pool['id'] = node_id + 1
|
||||
nodes.append(node_id)
|
||||
return nodes
|
||||
|
||||
def append_way(way, way2):
|
||||
another = list(way2) # make copy to not modify original list
|
||||
if way[0] == way[-1] or another[0] == another[-1]:
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
# postgresql connection string
|
||||
CONNECTION = 'dbname=borders'
|
||||
# passed to flask.Debug
|
||||
DEBUG = True
|
||||
DEBUG = False
|
||||
# if the main table is read-only
|
||||
READONLY = False
|
||||
# main table name
|
||||
|
|
|
@ -15,6 +15,8 @@ function init() {
|
|||
map = L.map('map', { editable: true }).setView([30, 0], 3);
|
||||
var hash = new L.Hash(map);
|
||||
L.tileLayer('http://tile.openstreetmap.org/{z}/{x}/{y}.png', { attribution: '© OpenStreetMap' }).addTo(map);
|
||||
L.tileLayer('http://korona.geog.uni-heidelberg.de/tiles/adminb/x={x}&y={y}&z={z}',
|
||||
{ attribution: '© GIScience Heidelberg' }).addTo(map);
|
||||
bordersLayer = L.layerGroup();
|
||||
map.addLayer(bordersLayer);
|
||||
|
||||
|
@ -302,14 +304,8 @@ function bOldBorders() {
|
|||
}
|
||||
}
|
||||
|
||||
function bJOSM() {
|
||||
var b = map.getBounds();
|
||||
var url = getServer('josm') + '?' + $.param({
|
||||
'xmin': b.getWest(),
|
||||
'xmax': b.getEast(),
|
||||
'ymin': b.getSouth(),
|
||||
'ymax': b.getNorth()
|
||||
});
|
||||
function importInJOSM(method, data ) {
|
||||
var url = getServer(method) + '?' + $.param(data);
|
||||
$.ajax({
|
||||
url: 'http://127.0.0.1:8111/import',
|
||||
data: { url: url, new_layer: 'true' },
|
||||
|
@ -320,22 +316,24 @@ function bJOSM() {
|
|||
});
|
||||
}
|
||||
|
||||
function bJosmOld() {
|
||||
function bJOSM() {
|
||||
var b = map.getBounds();
|
||||
var url = getServer('josm') + '?' + $.param({
|
||||
'table': OLD_BORDERS_NAME,
|
||||
importInJOSM('josm', {
|
||||
'xmin': b.getWest(),
|
||||
'xmax': b.getEast(),
|
||||
'ymin': b.getSouth(),
|
||||
'ymax': b.getNorth()
|
||||
});
|
||||
$.ajax({
|
||||
url: 'http://127.0.0.1:8111/import',
|
||||
data: { url: url, new_layer: 'true' },
|
||||
complete: function(t) {
|
||||
if( t.status != 200 )
|
||||
window.alert('Please enable remote_control in JOSM');
|
||||
}
|
||||
}
|
||||
|
||||
function bJosmOld() {
|
||||
var b = map.getBounds();
|
||||
importInJOSM('josm', {
|
||||
'table': OLD_BORDERS_NAME,
|
||||
'xmin': b.getWest(),
|
||||
'xmax': b.getEast(),
|
||||
'ymin': b.getSouth(),
|
||||
'ymax': b.getNorth()
|
||||
});
|
||||
}
|
||||
|
||||
|
@ -459,6 +457,19 @@ function bSplitDo() {
|
|||
bSplitCancel();
|
||||
}
|
||||
|
||||
function bSplitJosm() {
|
||||
var wkt = '', lls = splitLayer.getLatLngs();
|
||||
for( i = 0; i < lls.length; i++ ) {
|
||||
if( i > 0 )
|
||||
wkt += ',';
|
||||
wkt += L.Util.formatNum(lls[i].lng, 6) + ' ' + L.Util.formatNum(lls[i].lat, 6);
|
||||
}
|
||||
importInJOSM('josmbord', {
|
||||
'name': splitSelected,
|
||||
'line': 'LINESTRING(' + wkt + ')'
|
||||
});
|
||||
}
|
||||
|
||||
function bSplitCancel() {
|
||||
map.editTools.stopDrawing();
|
||||
if( splitLayer != null )
|
||||
|
|
|
@ -32,7 +32,7 @@
|
|||
<div id="panel">
|
||||
<div id="header">
|
||||
<div id="filter">
|
||||
Раскраска по <select size="1" id="f_type" value="size" onchange="filterSelect(this.value)">
|
||||
Раскраска по <select size="1" id="f_type" value="size" onchange="filterSelect()">
|
||||
<option value="size">размеру</option>
|
||||
<option value="topo">топологии</option>
|
||||
<option value="chars">буквам в назв.</option>
|
||||
|
@ -111,6 +111,7 @@
|
|||
Нарисуйте линию через выбранную область, затем нажмите кнопку<br>
|
||||
<div id="s_do">
|
||||
<button onclick="bSplitDo()">Разрезать область</button><br>
|
||||
<button onclick="bSplitJosm()">Границы вдоль — в JOSM</button><br>
|
||||
<button onclick="bSplitAgain()">Нарисовать по-другому</button>
|
||||
</div>
|
||||
<br>
|
||||
|
|
Loading…
Add table
Reference in a new issue