From 84b480676bc330c7e145ffe6ff8fef1b1929f9b2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Kom=D1=8Fpa?= Date: Fri, 30 Apr 2010 23:42:00 +0300 Subject: [PATCH] Initial support for quadtiles - broken: dragging - broken: beautiful maps - added: python/lxml converter to quadtiles - added: zoomlevel - changed: messaging now uses container class, not tuples - changed: default place ;) --- src/kothic.py | 169 ++++++++++++++------------ src/python-osm-converter/osm2tiles.py | 162 ++++++++++++++++++++++++ 2 files changed, 252 insertions(+), 79 deletions(-) create mode 100644 src/python-osm-converter/osm2tiles.py diff --git a/src/kothic.py b/src/kothic.py index e03beea..a908e26 100755 --- a/src/kothic.py +++ b/src/kothic.py @@ -1,7 +1,19 @@ #!/usr/bin/env python -#use Cairo; -#import cairo -#use Gtk2 '-init'; +# -*- coding: utf-8 -*- +# This file is part of kothic, the realtime map renderer. + +# kothic is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. + +# kothic is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. + +# You should have received a copy of the GNU General Public License +# along with kothic. If not, see . import pygtk pygtk.require('2.0') import gtk @@ -11,27 +23,9 @@ import string import threading import time import Queue +from twms import projections -#use Glib qw(TRUE FALSE); -#use Time::HiRes qw(gettimeofday tv_interval); -#use strict; -#use POSIX qw(ceil floor); -#use style qw(@style); -#use navigator_lib; -#my $dbname = "moscow.db"; -#my $dbargs = {AutoCommit => 0, -# PrintError => 1, -#}; -# - -#my $latmin; -#my $latmax; -#my $lonmin; -#my $lonmax; - -#my $latcenter = 55.6304; -#my $loncenter = 37.49305; class Renderer(threading.Thread): def __init__(self, comm): @@ -45,21 +39,21 @@ class Renderer(threading.Thread): request = self.comm[0].get() if(self.comm[0].empty): break - print (" got request:", request) - res = RasterTile(request[2][0], request[2][1]) - res.update_surface(request[0][0], request[0][1], request[1], self.tc, request[3]) + #print (" got request:", request) + res = RasterTile(request.size[0], request.size[1], request.zoomlevel, request.data_projection) + res.update_surface(request.center_lonlat[0], request.center_lonlat[1], request.zoom, self.tc, request.style) print (" render complete") comm[1].put(res) -# comm[2].get_window().invalidate_rect(None, True) - comm[2].queue_draw() class Navigator: def __init__(self, comm): self.comm = comm - self.lat_c = 55.6304 - self.lon_c = 37.49305 + self.lon_c = 27.6749791 + self.lat_c = 53.8621394 self.width, self.height = 640, 480 - self.zoom = self.width/0.02; + self.zoomlevel = 17 + self.data_projection = "EPSG:4326" + self.zoom = self.width/0.09; self.window = gtk.Window(gtk.WINDOW_TOPLEVEL) self.dx = 0 self.dy = 0 @@ -102,16 +96,29 @@ class Navigator: self.window.set_size_request(self.width, self.height) self.window.add(da) self.window.connect("delete_event", self.delete_ev) - self.comm.append(da) def motion_ev(self, widget, event): # print("Motion") if self.drag: self.dx = event.x - self.drag_x self.dy = event.y - self.drag_y if((abs(self.dx) > 100 or abs(self.dy) > 100) and self.f): - self.comm[0].put((self.rastertile.screen2latlon(self.rastertile.w/2 - self.dx, self.rastertile.h/2 - self.dy), self.zoom, (self.width + self.border*2, self.height + self.border*2), self.style)) + com = MessageContainer() + com.center_lonlat = self.rastertile.screen2lonlat(self.rastertile.w/2 - self.dx, self.rastertile.h/2 - self.dy) + com.data_projection = self.data_projection + com.zoomlevel = self.zoomlevel + com.zoom = self.zoom + com.size = (self.width + self.border*2, self.height + self.border*2) + com.style = self.style + self.comm[0].put(com) self.request_d = (self.dx, self.dy) self.f = False + if not self.comm[1].empty(): + self.rastertile = self.comm[1].get() + self.f = True + self.drag_x += self.request_d[0] + self.drag_y += self.request_d[1] + self.dx = event.x - self.drag_x + self.dy = event.y - self.drag_y widget.queue_draw() def delete_ev(self, widget, event): gtk.main_quit() @@ -126,12 +133,21 @@ class Navigator: # print("ll:", self.latcenter, self.loncenter) print("LL before: ",self.lat_c, self.lon_c) print("dd: ",self.dx, self.dy) - self.lat_c, self.lon_c = self.rastertile.screen2latlon(self.rastertile.w/2 - self.dx, self.rastertile.h/2 - self.dy); -# self.dx = self.dy = 0 + self.lon_c, self.lat_c = self.rastertile.screen2lonlat(self.rastertile.w/2 - self.dx, self.rastertile.h/2 - self.dy); + self.dx = self.dy = 0 self.f = True print("LL after: ",self.lat_c, self.lon_c) + + com = MessageContainer() + com.center_lonlat = (self.lon_c,self.lat_c) + com.data_projection = self.data_projection + com.zoomlevel = self.zoomlevel + com.zoom = self.zoom + com.size = (self.width + self.border*2, self.height + self.border*2) + com.style = self.style + self.comm[0].put(com) # self.rastertile.update_surface( self.lat_c, self.lon_c, self.zoom, self.tilecache, self.style) - self.comm[0].put(((self.lat_c, self.lon_c), self.zoom, (self.width + self.border*2, self.height + self.border*2), self.style)) + widget.queue_draw() def expose_ev(self, widget, event): # print("Expose") @@ -141,29 +157,24 @@ class Navigator: self.height = widget.allocation.height self.rastertile = None if self.rastertile is None: - self.rastertile = RasterTile(self.width + self.border*2, self.height + self.border*2) - self.rastertile.update_surface(self.lat_c, self.lon_c, self.zoom, self.tilecache, self.style) - if not self.comm[1].empty(): - ort = self.rastertile - nrt = self.comm[1].get() - lat, lon = ort.screen2latlon(ort.w/2 - self.dx, ort.h/2 - self.dy) - ox, oy = nrt.latlon2screen(lat, lon, nrt.zoom) - ox -= nrt.w/2 - oy -= nrt.h/2 - print (ox, oy) - self.rastertile.offset_x = ox - self.rastertile.offset_y = oy - self.f = True - self.rastertile = nrt - + self.rastertile = RasterTile(self.width + self.border*2, self.height + self.border*2, self.zoomlevel, self.data_projection) + self.rastertile.update_surface(self.lon_c, self.lat_c, self.zoom, self.tilecache, self.style) cr = widget.window.cairo_create() - cr.set_source_surface(self.rastertile.surface, self.dx-self.border + self.rastertile.offset_x, self.dy - self.border + self.rastertile.offset_y) + cr.set_source_surface(self.rastertile.surface, self.dx-self.border, self.dy - self.border) cr.paint() +# cr. def main(self): self.window.show_all() gtk.main() +class MessageContainer: + """ + A class to keep messages to render-threads in. + """ + pass + + def line(cr, c): cr.move_to(c[0], c[1]) for k in range(2, len(c), 2): @@ -185,34 +196,36 @@ def ways(t): def load_tile(k): print("loading tile: ", k) - f = open(key_to_filename(k)) + try: + f = open(key_to_filename(k)) + except IOError: + print "Failed open: %s" % key_to_filename(k) + return {} t = {} - while True: - str = f.readline() - if str is None or str == "": - break - str = str.rstrip("\n") - a = str.split(" ") + for line in f: + a = line.split(" ") w = Way(a[0], int(a[1]), int(a[2]), map(lambda x: float(x), a[3:])) t[w.id] = w f.close() return t class RasterTile: - def __init__(self, width, height): + def __init__(self, width, height, zoom, data_projection): self.w = width self.h = height self.surface = cairo.ImageSurface(cairo.FORMAT_RGB24, self.w, self.h) - self.offset_x = 0 - self.offset_y = 0 + self.x_offset = 0 + self.y_offset = 0 self.lat_c = None self.lon_c = None - self.zoom = None - def screen2latlon(self, x, y): - return -(y - self.h/2)/self.zoom + self.lat_c, (x - self.w/2)/(math.cos(self.lat_c*math.pi/180)*self.zoom) + self.lon_c - def latlon2screen(self, lat, lon, lcc): + self.zoomlevel = zoom + self.zoom = None + self.data_projection = data_projection + def screen2lonlat(self, x, y): + return (x - self.w/2)/(math.cos(self.lat_c*math.pi/180)*self.zoom) + self.lon_c, -(y - self.h/2)/self.zoom + self.lat_c + def lonlat2screen(self, lon, lat, lcc): return (lon - self.lon_c)*lcc*self.zoom + self.w/2, -(lat - self.lat_c)*self.zoom + self.h/2 - def update_surface(self, lat, lon, zoom, tilecache, style): + def update_surface(self, lon, lat, zoom, tilecache, style): self.zoom = zoom self.lat_c = lat self.lon_c = lon @@ -220,16 +233,14 @@ class RasterTile: cr.rectangle(0, 0, self.w, self.h) cr.set_source_rgb(0.7, 0.7, 0.7) cr.fill() - latmin, lonmin = self.screen2latlon(0, self.h) - latmax, lonmax = self.screen2latlon(self.w, 0) - latkey_min = int(latmin*100) - latkey_max = int(latmax*100) - lonkey_min = int(lonmin*100) - lonkey_max = int(lonmax*100) - print(latmin, lonmin, latmax, lonmax) - print( latkey_min, latkey_max, lonkey_min, lonkey_max) + lonmin, latmin = self.screen2lonlat(0, self.h) + lonmax, latmax = self.screen2lonlat(self.w, 0) + a,d,c,b = [int(x) for x in projections.tile_by_bbox((lonmin, latmin, lonmax, latmax),self.zoomlevel, self.data_projection)] + + print(latmin, lonmin, latmax, lonmax) + print( a, b, c, d) #FIXME: add time - active_tile = set([(i,j) for i in range(latkey_min, latkey_max+1) for j in range(lonkey_min, lonkey_max+1)]) + active_tile = set([(self.zoomlevel,i,j) for i in range(a, c+1) for j in range(b, d+1)]) print(active_tile) for k in tilecache.keys(): if k not in active_tile: @@ -247,7 +258,7 @@ class RasterTile: for w in ww: cs = [] for k in range(0, len(w.coords), 2): - x, y = self.latlon2screen(w.coords[k], w.coords[k+1], lcc); + x, y = self.lonlat2screen(w.coords[k], w.coords[k+1], lcc); cs.append(x) cs.append(y) w.cs = cs @@ -272,12 +283,12 @@ class Way: self.style = style self.cs = None -def key_to_filename(k): - return "data/" + str(k[0]//100) + "/" + str(k[1]//100) + "/" + str(k[0]%100) + "/" + str(k[1]%100) + ".map" +def key_to_filename((z,x,y)): + return "tiles/z%s/%s/x%s/%s/y%s.vtile"%(z, x/1024, x, y/1024, y) if __name__ == "__main__": + comm = (Queue.Queue(), Queue.Queue()) gtk.gdk.threads_init() - comm = [Queue.Queue(), Queue.Queue()] nav = Navigator(comm) r = Renderer(comm) r.daemon = True diff --git a/src/python-osm-converter/osm2tiles.py b/src/python-osm-converter/osm2tiles.py new file mode 100644 index 0000000..eb274a6 --- /dev/null +++ b/src/python-osm-converter/osm2tiles.py @@ -0,0 +1,162 @@ +#!/usr/bin/env python +# -*- coding: utf-8 -*- +# This file is part of kothic, the realtime map renderer. + +# kothic is free software: you can redistribute it and/or modify +# it under the terms of the GNU General Public License as published by +# the Free Software Foundation, either version 3 of the License, or +# (at your option) any later version. + +# kothic is distributed in the hope that it will be useful, +# but WITHOUT ANY WARRANTY; without even the implied warranty of +# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the +# GNU General Public License for more details. + +# You should have received a copy of the GNU General Public License +# along with kothic. If not, see . + +import os +import sys +from lxml import etree +from twms import projections + +try: + import psyco + psyco.full() +except ImportError: + pass + +MAXZOOM = 18 +proj = "EPSG:4326" + +style = {} +style["L"] = { + 1: set([("highway",("primary", "motorway", "trunk"))]), + 2: set([("highway",("primary_link", "motorway_link", "trunk_link"))]), + 3: set([("highway",("secondary"))]), + 4: set([("highway",("residential", "tertiary", "living_street"))]), + 5: set([("highway",("service", "unclassified"))]), + 8: set([("highway", None)]), +} +style["P"] = { + 6: set([("building",None)]), + 7: set([("natural",("wood")), ("landuse",("forest")), ("leisure", ("park"))]), + 9: set([("landuse",("industrial"))]), + 10: set([("natural",("water")),("waterway",("riverbank"))]), + 11: set([("landuse",("residential"))]), +} + +# elsif($k eq 'highway' and $v eq 'footway' or $v eq 'path' or $v eq 'track'){ + +def tilelist_by_geometry(way, start_zoom = 0, ispoly = False): + """ + Gives a number of (z,x,y) tile numbers that geometry crosses. + """ + ret = set([]) + tiles_by_zooms = {} # zoom: set(tile,tile,tile...) + for t in xrange(0,MAXZOOM+1): + tiles_by_zooms[t] = set([]) + for point in way: + tile = projections.tile_by_coords(point, MAXZOOM, proj) + tile = (MAXZOOM, int(tile[0]),int(tile[1])) + tiles_by_zooms[MAXZOOM].add(tile) + for t in xrange(MAXZOOM-1,start_zoom-1,-1): + for tt in tiles_by_zooms[t+1]: + tiles_by_zooms[t].add((t, int(tt[1]/2), int(tt[2]/2))) + for z in tiles_by_zooms.values(): + ret.update(z) + return ret + +def pix_distance(a,b,z): + """ + Calculates onscreen disatnce between 2 points on given zoom. + """ + return 2**z*256*(((a[0]-b[0])/360.)**2+((a[1]-b[1])/180.)**2)**0.5 + +needed_ways_tags = set(['highway','building','landuse']) + +def way_interesting(tags): + res = {} + for k,v in tags.iteritems(): + if k in needed_ways_tags: + res[k] = v + return res + + +def main (): + DROPPED_POINTS = 0 + tilefiles = {} + osm_infile = open("minsk.osm", "rb") + nodes = {} + curway = [] + tags = {} + context = etree.iterparse(osm_infile) + for action, elem in context: + items = dict(elem.items()) + if elem.tag == "node": + nodes[int(items["id"])] = (float(items["lon"]), float(items["lat"])) + elif elem.tag == "nd": + curway.append(nodes[int(items["ref"])]) + elif elem.tag == "tag": + tags[items["k"]] = items["v"] + elif elem.tag == "way": + tags = way_interesting(tags) + if tags: + mzoom = 1 + + + way_simplified = {} + #for zoom in xrange(MAXZOOM,-1,-1): ######## generalize a bit + + #prev_point = curway[0] + #way = [prev_point] + #for point in curway: + #if pix_distance(point, prev_point, zoom) > 2.: + #way.append(point) + #else: + #DROPPED_POINTS += 1 + #prev_point = point + #if len(way) == 1: + #mzoom = zoom + #print zoom + #break + #if len(way) > 1: + #way_simplified[zoom] = way + ##print way + + waytype, waynum = 0, 0 + for objtype, tagset in style.iteritems(): + for tid, tagz in tagset.iteritems(): + for k, v in tagz: + if k in tags: + if v: + if tags[k] not in v: + continue + #print k, v + waytype = objtype + waynum = tid + + + for tile in tilelist_by_geometry(curway, mzoom): + if tile not in tilefiles: + z, x, y = tile + path = "tiles/z%s/%s/x%s/%s/"%(z, x/1024, x, y/1024) + if not os.path.exists(path): + os.makedirs(path) + tilefiles[tile] = "aaa" + tilefile = open(path+"y"+str(y)+".vtile","wb") + tilefile = open(path+"y"+str(y)+".vtile","ab") + + + print >>tilefile, "%s %s %s" % (waytype, items["id"], waynum), " ".join([str(x[0])+" "+str(x[1]) for x in curway])#_simplified[tile[0]]]) + tilefile.flush() + #print >>corr, "%s %s %s %s %s %s"% (curway[0][0],curway[0][1],curway[1][0],curway[1][1], user, ts ) + curway = [] + tags = {} + #user = default_user + #ts = "" + print DROPPED_POINTS + + +main() +