diff --git a/.hgignore b/.hgignore
new file mode 100644
index 0000000..51860e6
--- /dev/null
+++ b/.hgignore
@@ -0,0 +1,4 @@
+src/data/
+src/tiles/
+.*~
+.*.pyc
diff --git a/src/kothic.py b/src/kothic.py
index bb35535..6b95d7c 100644
--- a/src/kothic.py
+++ b/src/kothic.py
@@ -23,15 +23,11 @@ import string
import threading
import time
import Queue
-from twms import projections
-
-
-# Ucomment one of the following lines depending on current mode:
from debug import debug, Timer
-#from production import debug, Timer
+from vtiles_backend import QuadTileBackend as DataBackend
@@ -56,7 +52,7 @@ class Renderer(threading.Thread):
break
#debug (" got request:", request)
t = Timer("Rendering screen")
- res = RasterTile(request.size[0], request.size[1], request.zoomlevel, request.data_projection)
+ res = RasterTile(request.size[0], request.size[1], request.zoomlevel, request.data_backend)
res.update_surface(request.center_lonlat, request.zoom, self.tc, request.style)
t.stop()
comm[1].put(res)
@@ -68,11 +64,12 @@ class Navigator:
self.comm = comm
self.center_coord = (27.6749791, 53.8621394)
self.width, self.height = 800, 480
- self.zoomlevel = 16
+ self.zoomlevel = 15
self.data_projection = "EPSG:4326"
self.zoom = self.width/0.02;
self.request_d = (0,0)
self.window = gtk.Window(gtk.WINDOW_TOPLEVEL)
+ self.data = DataBackend()
self.dx = 0
self.dy = 0
self.drag_x = 0
@@ -125,10 +122,10 @@ class Navigator:
if self.drag:
self.dx = event.x - self.drag_x
self.dy = event.y - self.drag_y
- #if((abs(self.dx) > 150 or abs(self.dy) > 150) and self.f):
- # self.redraw()
+ if((abs(self.dx) > 150 or abs(self.dy) > 150) and self.f):
+ self.redraw()
# self.request_d = (self.dx, self.dy)
- # self.f = False
+ self.f = False
widget.queue_draw()
def delete_ev(self, widget, event):
gtk.main_quit()
@@ -162,10 +159,10 @@ class Navigator:
self.zoomlevel += 1
debug("Zoom in")
elif event.direction == gtk.gdk.SCROLL_DOWN:
- self.zoom /= 2
- self.zoomlevel -= 1
-
- debug("Zoom out")
+ if self.zoomlevel >= 0: ## negative zooms are nonsense
+ self.zoom /= 2
+ self.zoomlevel -= 1
+ debug("Zoom out")
self.redraw()
# widget.queue_draw()
def redraw(self):
@@ -174,7 +171,7 @@ class Navigator:
"""
com = MessageContainer()
com.center_lonlat = self.center_coord
- com.data_projection = self.data_projection
+ com.data_backend = self.data
com.zoomlevel = self.zoomlevel
com.zoom = self.zoom
com.size = (self.width + self.border*2, self.height + self.border*2)
@@ -191,7 +188,7 @@ 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.zoomlevel, self.data_projection)
+ self.rastertile = RasterTile(self.width + self.border*2, self.height + self.border*2, self.zoomlevel, self.data)
self.rastertile.update_surface(self.center_coord, self.zoom, self.tilecache, self.style, None)
nrt = None
while(not self.comm[1].empty()):
@@ -244,33 +241,10 @@ def poly(cr, c):
cr.line_to(c[k], c[k + 1])
cr.fill()
-def ways(t):
-# return [y for x in t.itervalues() for y in x.itervalues()]
- r = {}
- for i in t.values():
- r.update(i)
- return r.values()
-def load_tile(k,lock = None):
- #debug("loading tile: ", k)
- try:
- f = open(key_to_filename(k))
- except IOError:
- # debug ( "Failed open: %s" % key_to_filename(k) )
- return {}
- t = {}
- 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()
- if lock is not None:
- lock.acquire()
- lock.release()
- return t
class RasterTile:
- def __init__(self, width, height, zoom, data_projection):
+ def __init__(self, width, height, zoom, data_backend):
self.w = width
self.h = height
self.surface = cairo.ImageSurface(cairo.FORMAT_RGB24, self.w, self.h)
@@ -279,7 +253,7 @@ class RasterTile:
self.center_coord = None
self.zoomlevel = zoom
self.zoom = None
- self.data_projection = data_projection
+ self.data = data_backend
def screen2lonlat(self, x, y):
return (x - self.w/2)/(math.cos(self.center_coord[1]*math.pi/180)*self.zoom) + self.center_coord[0], -(y - self.h/2)/self.zoom + self.center_coord[1]
def lonlat2screen(self, (lon, lat)):
@@ -293,23 +267,13 @@ class RasterTile:
cr.fill()
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)]
-
- #debug((latmin, lonmin, latmax, lonmax))
- debug(( a, b, c, d))
+###########################################3
#FIXME: add time
- active_tile = set([(self.zoomlevel,i,j) for i in range(a, c+1) for j in range(b, d+1)])
- debug("Active tiles in memory: %s" % len(active_tile))
- for k in tilecache.keys():
- if k not in active_tile:
- del tilecache[k]
- debug("del tile: %s" % (k,))
- for k in active_tile:
- if k not in tilecache:
- tilecache[k] = load_tile(k)
+
#FIXME add time2
- ww = ways(tilecache)
- debug("ways: %s" % len(ww))
+ #ww = ways(tilecache)
+ #debug("ways: %s" % len(ww))
+ ww = self.data.get_vectors((lonmin,latmin,lonmax,latmax),self.zoomlevel).values()
if lock is not None:
lock.acquire()
lock.release()
@@ -339,16 +303,7 @@ class RasterTile:
elif w.type == "P":
poly(cr, w.cs)
-class Way:
- def __init__(self, type, id, style, coords):
- self.type = type
- self.id = id
- self.coords = coords
- self.style = style
- self.cs = None
-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__":
diff --git a/src/python-osm-converter/osm2tiles.py b/src/python-osm-converter/osm2tiles.py
index 9557d6a..e1688de 100644
--- a/src/python-osm-converter/osm2tiles.py
+++ b/src/python-osm-converter/osm2tiles.py
@@ -26,7 +26,7 @@ try:
except ImportError:
pass
-MAXZOOM = 18
+MAXZOOM = 16
proj = "EPSG:4326"
style = {}
@@ -106,26 +106,6 @@ def main ():
elif elem.tag == "way":
mzoom = 1
-
- way_simplified = {}
- for zoom in xrange(MAXZOOM,-1,-1): ######## generalize a bit
- # TODO: Douglas-Peucker
- 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():
@@ -140,6 +120,25 @@ def main ():
waynum = tid
if waytype is not 0:
+ way_simplified = {MAXZOOM: curway}
+
+ for zoom in xrange(MAXZOOM+1,-1,-1): ######## generalize a bit
+ # TODO: Douglas-Peucker
+ 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
for tile in tilelist_by_geometry(curway, mzoom+1):
z, x, y = tile
path = "../tiles/z%s/%s/x%s/%s/"%(z, x/1024, x, y/1024)
diff --git a/src/render.py b/src/render.py
new file mode 100644
index 0000000..f725416
--- /dev/null
+++ b/src/render.py
@@ -0,0 +1,89 @@
+#!/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 cairo
+
+
+class Renderer:
+ 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.center_coord = None
+ self.zoomlevel = zoom
+ self.zoom = None
+ self.data_projection = data_projection
+ def screen2lonlat(self, x, y):
+ return (x - self.w/2)/(math.cos(self.center_coord[1]*math.pi/180)*self.zoom) + self.center_coord[0], -(y - self.h/2)/self.zoom + self.center_coord[1]
+ def lonlat2screen(self, (lon, lat)):
+ return (lon - self.center_coord[0])*self.lcc*self.zoom + self.w/2, -(lat - self.center_coord[1])*self.zoom + self.h/2
+ def update_surface(self, lonlat, zoom, tilecache, style, lock = None):
+ self.zoom = zoom
+ self.center_coord = lonlat
+ cr = cairo.Context(self.surface)
+ cr.rectangle(0, 0, self.w, self.h)
+ cr.set_source_rgb(0.7, 0.7, 0.7)
+ cr.fill()
+ 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)]
+
+ #debug((latmin, lonmin, latmax, lonmax))
+ debug(( a, b, c, d))
+#FIXME: add time
+ active_tile = set([(self.zoomlevel,i,j) for i in range(a, c+1) for j in range(b, d+1)])
+ debug("Active tiles in memory: %s" % len(active_tile))
+ for k in tilecache.keys():
+ if k not in active_tile:
+ del tilecache[k]
+ debug("del tile: %s" % (k,))
+ for k in active_tile:
+ if k not in tilecache:
+ tilecache[k] = load_tile(k)
+ #FIXME add time2
+ ww = ways(tilecache)
+ debug("ways: %s" % len(ww))
+ if lock is not None:
+ lock.acquire()
+ lock.release()
+ self.lcc = math.cos(self.center_coord[1]*math.pi/180)
+ ww.sort(key=lambda x: style[x.style][3])
+ lcc = math.cos(self.center_coord[1]*math.pi/180)
+ for w in ww:
+ cs = []
+ for k in range(0, len(w.coords), 2):
+ x, y = self.lonlat2screen((w.coords[k], w.coords[k+1]));
+ cs.append(x)
+ cs.append(y)
+ w.cs = cs
+ for passn in range(1, 4):
+ debug("pass %s" % passn)
+ for w in ww:
+ stn = w.style
+ #if lock is not None:
+ #lock.acquire()
+ #lock.release()
+ if stn < len(style) and style[stn] is not None and style[stn][passn-1] is not None:
+ st = style[w.style][passn-1]
+ cr.set_line_width(st[0])
+ cr.set_source_rgb(st[1][0], st[1][1], st[1][2])
+ if w.type == "L":
+ line(cr, w.cs)
+ elif w.type == "P":
+ poly(cr, w.cs)
\ No newline at end of file
diff --git a/src/vtiles_backend.py b/src/vtiles_backend.py
new file mode 100644
index 0000000..0f42c31
--- /dev/null
+++ b/src/vtiles_backend.py
@@ -0,0 +1,90 @@
+#!/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 .
+
+from debug import debug
+from twms import projections
+
+class Way:
+ def __init__(self, type, style, coords):
+ self.type = type
+ self.coords = coords
+ self.style = style
+ self.cs = None
+
+class QuadTileBackend:
+ """
+ A class that gives out vector data on demand.
+ """
+
+
+ def __init__(self,max_zoom = 16,proj = "EPSG:4326", path = "tiles", lang = "ru"):
+
+ debug("Bakend created")
+ self.max_zoom = max_zoom # no better tiles available
+ self.path = path # path to tile files
+ self.lang = lang # map language to use
+ self.tiles = {} # loaded vector tiles go here
+ self.data_projection = proj # which projection used to cut map in tiles
+ self.keep_tiles = 90 # a number of tiles to cache in memory
+ self.tile_load_log = [] # used when selecting which tile to unload
+
+ def filename(self, (z,x,y)):
+ return "%s/z%s/%s/x%s/%s/y%s.vtile"%(self.path, z, x/1024, x, y/1024, y)
+ def load_tile(self, k):
+ debug("loading tile: %s"% (k,))
+ try:
+ f = open(self.filename(k))
+ except IOError:
+ debug ( "Failed open: %s" % self.filename(k) )
+ return {}
+ t = {}
+ for line in f:
+ a = line.split(" ")
+ w = Way(a[0], int(a[2]), [float(x) for x in a[3:]])
+ t[int(a[1])] = w
+ f.close()
+ return t
+ def collect_garbage(self):
+ """
+ Cleans up some RAM by removing least accessed tiles.
+ """
+ if len(self.tiles) > self.keep_tiles:
+ debug("Now %s tiles cached, trying to kill %s"%(len(self.tiles),len(self.tiles)-self.keep_tiles))
+ for tile in self.tile_load_log[0:len(self.tiles)-self.keep_tiles]:
+ try:
+ del self.tiles[tile]
+ self.tile_load_log.remove(tile)
+ debug ("killed tile: %s" % (tile,))
+ except KeyError, ValueError:
+ debug ("tile killed not by us: %s" % (tile,))
+
+ def get_vectors (self, bbox, zoom):
+ zoom = min(zoom, self.max_zoom) ## If requested zoom is better than the best, take the best
+ zoom = max(zoom, 0) ## Negative zooms are nonsense
+ a,d,c,b = [int(x) for x in projections.tile_by_bbox(bbox,zoom, self.data_projection)]
+ resp = {}
+ for tile in set([(zoom,i,j) for i in range(a, c+1) for j in range(b, d+1)]):
+ if tile not in self.tiles:
+ self.tiles[tile] = self.load_tile(tile)
+ try:
+ self.tile_load_log.remove(tile)
+ except ValueError:
+ pass
+ self.tile_load_log.append(tile)
+ resp.update(self.tiles[tile])
+ self.collect_garbage()
+ return resp
\ No newline at end of file