2017-10-19 17:20:55 +02:00
|
|
|
from django.utils.functional import cached_property
|
2017-10-10 17:49:53 +02:00
|
|
|
from shapely.geometry import box
|
2017-10-19 16:33:32 +02:00
|
|
|
from shapely.ops import unary_union
|
2017-10-10 17:49:53 +02:00
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
from c3nav.mapdata.render.base import get_level_render_data
|
2017-10-10 14:39:11 +02:00
|
|
|
from c3nav.mapdata.utils.svg import SVGImage
|
|
|
|
|
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
class SVGRenderer:
|
|
|
|
def __init__(self, level, miny, minx, maxy, maxx, scale=1, user=None):
|
|
|
|
self.level = level
|
|
|
|
self.miny = miny
|
|
|
|
self.minx = minx
|
|
|
|
self.maxy = maxy
|
|
|
|
self.maxx = maxx
|
|
|
|
self.scale = scale
|
|
|
|
self.user = user
|
2017-10-10 14:39:11 +02:00
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
@cached_property
|
|
|
|
def bbox(self):
|
|
|
|
return box(self.minx-1, self.miny-1, self.maxx+1, self.maxy+1)
|
2017-10-10 17:49:53 +02:00
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
@cached_property
|
|
|
|
def level_render_data(self):
|
|
|
|
return get_level_render_data(self.level)
|
2017-10-19 15:31:30 +02:00
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
def check_level(self):
|
|
|
|
return self.level_render_data
|
2017-10-19 15:31:30 +02:00
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
@cached_property
|
|
|
|
def affected_access_restrictions(self):
|
|
|
|
access_restrictions = set()
|
|
|
|
for geoms, default_height in self.level_render_data:
|
|
|
|
for access_restriction, area in geoms.access_restriction_affected.items():
|
|
|
|
if access_restriction not in access_restrictions and area.intersects(self.bbox):
|
|
|
|
access_restrictions.add(access_restriction)
|
|
|
|
return access_restrictions
|
2017-10-19 15:31:30 +02:00
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
@cached_property
|
|
|
|
def unlocked_access_restrictions(self):
|
|
|
|
# todo access_restriction
|
|
|
|
return set(access_restriction for access_restriction in self.affected_access_restrictions
|
|
|
|
if self.user is not None and self.user.is_superuser)
|
2017-10-19 15:31:30 +02:00
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
@cached_property
|
|
|
|
def access_cache_key(self):
|
|
|
|
return '_'.join(str(i) for i in sorted(self.unlocked_access_restrictions)) or '0'
|
2017-10-19 15:31:30 +02:00
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
def render(self):
|
|
|
|
svg = SVGImage(bounds=((self.miny, self.minx), (self.maxy, self.maxx)), scale=self.scale, buffer=1)
|
2017-10-19 16:33:32 +02:00
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
unlocked_access_restrictions = self.unlocked_access_restrictions | set([None])
|
2017-10-16 17:10:32 +02:00
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
crop_to = None
|
|
|
|
primary_level_count = 0
|
|
|
|
for geoms, default_height in reversed(self.level_render_data):
|
|
|
|
if geoms.holes is not None:
|
|
|
|
primary_level_count += 1
|
2017-10-16 17:10:32 +02:00
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
geoms.crop_to = crop_to if primary_level_count > 1 else None
|
2017-10-17 19:31:19 +02:00
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
if geoms.holes is not None:
|
|
|
|
if crop_to is None:
|
|
|
|
crop_to = geoms.holes
|
|
|
|
else:
|
|
|
|
crop_to = crop_to.intersection(geoms.holes)
|
|
|
|
|
|
|
|
for geoms, default_height in self.level_render_data:
|
|
|
|
crop_to = self.bbox
|
|
|
|
if geoms.crop_to is not None:
|
|
|
|
crop_to = crop_to.intersection(geoms.crop_to)
|
|
|
|
|
2017-10-19 17:48:12 +02:00
|
|
|
add_walls = unary_union(tuple(area for access_restriction, area in geoms.restricted_spaces_indoors.items()
|
|
|
|
if access_restriction not in unlocked_access_restrictions))
|
|
|
|
crop_areas = unary_union(
|
|
|
|
tuple(area for access_restriction, area in geoms.restricted_spaces_outdoors.items()
|
|
|
|
if access_restriction not in unlocked_access_restrictions)
|
|
|
|
).union(add_walls)
|
|
|
|
|
2017-10-19 17:20:55 +02:00
|
|
|
for altitudearea in geoms.altitudeareas:
|
2017-10-19 17:48:12 +02:00
|
|
|
svg.add_geometry(crop_to.intersection(altitudearea.geometry.difference(crop_areas)),
|
2017-10-19 17:20:55 +02:00
|
|
|
fill_color='#eeeeee', altitude=altitudearea.altitude)
|
|
|
|
|
|
|
|
for color, areas in altitudearea.colors.items():
|
|
|
|
areas = tuple(area for access_restriction, area in areas.items()
|
|
|
|
if access_restriction in unlocked_access_restrictions)
|
|
|
|
if areas:
|
|
|
|
svg.add_geometry(crop_to.intersection(unary_union(areas)), fill_color=color, elevation=0)
|
|
|
|
|
2017-10-19 17:48:12 +02:00
|
|
|
svg.add_geometry(crop_to.intersection(geoms.walls.union(add_walls)),
|
2017-10-19 17:20:55 +02:00
|
|
|
fill_color='#aaaaaa', stroke_px=0.5, stroke_color='#aaaaaa', elevation=default_height)
|
|
|
|
|
|
|
|
svg.add_geometry(crop_to.intersection(geoms.doors), fill_color='#ffffff', elevation=0)
|
|
|
|
|
|
|
|
return svg
|