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()
+