team-3/src/c3nav/mapdata/render/renderer.py

193 lines
7.1 KiB
Python
Raw Normal View History

2016-11-26 13:48:44 +01:00
import os
2016-12-02 18:53:16 +01:00
import subprocess
2016-11-26 13:48:44 +01:00
import xml.etree.ElementTree as ET
from django.conf import settings
from django.db.models import Max, Min
from shapely.affinity import scale
2016-12-04 01:49:49 +01:00
from shapely.geometry import JOIN_STYLE, box
2016-11-26 13:48:44 +01:00
from c3nav.mapdata.models import Package
2016-12-04 01:49:49 +01:00
def get_render_path(filename):
return os.path.join(settings.RENDER_ROOT, filename)
def get_dimensions():
aggregate = Package.objects.all().aggregate(Max('right'), Min('left'), Max('top'), Min('bottom'))
return (
float(aggregate['right__max'] - aggregate['left__min']),
float(aggregate['top__max'] - aggregate['bottom__min']),
)
2016-11-26 13:48:44 +01:00
class LevelRenderer():
def __init__(self, level):
self.level = level
@staticmethod
def get_dimensions():
2016-12-04 01:49:49 +01:00
width, height = get_dimensions()
return (width * settings.RENDER_SCALE, height * settings.RENDER_SCALE)
2016-11-26 13:48:44 +01:00
@staticmethod
def polygon_svg(geometry, fill_color=None, fill_opacity=None, stroke_width=0.0, stroke_color=None, filter=None):
2016-12-02 18:51:42 +01:00
scaled = scale(geometry, xfact=settings.RENDER_SCALE, yfact=settings.RENDER_SCALE, origin=(0, 0))
2016-12-02 18:50:19 +01:00
element = ET.fromstring(scaled.svg(0, fill_color or '#FFFFFF'))
2016-11-26 13:48:44 +01:00
if element.tag != 'g':
new_element = ET.Element('g')
new_element.append(element)
element = new_element
for path in element.findall('path'):
path.attrib.pop('opacity')
2016-12-02 18:50:19 +01:00
path.set('stroke-width', str(stroke_width * settings.RENDER_SCALE))
2016-11-26 13:48:44 +01:00
if fill_color is None and 'fill' in path.attrib:
path.attrib.pop('fill')
path.set('fill-opacity', '0')
if fill_opacity is not None:
path.set('fill-opacity', str(fill_opacity))
2016-11-26 13:48:44 +01:00
if stroke_color is not None:
path.set('stroke', stroke_color)
elif 'stroke' in path.attrib:
path.attrib.pop('stroke')
if filter is not None:
path.set('filter', filter)
return element
2016-12-04 01:49:49 +01:00
def create_svg(self):
2016-11-26 13:48:44 +01:00
width, height = self.get_dimensions()
svg = ET.Element('svg', {
'width': str(width),
'height': str(height),
'xmlns:svg': 'http://www.w3.org/2000/svg',
'xmlns': 'http://www.w3.org/2000/svg',
2016-12-04 01:49:49 +01:00
'xmlns:xlink': 'http://www.w3.org/1999/xlink',
2016-11-26 13:48:44 +01:00
})
2016-12-04 01:49:49 +01:00
return svg
2016-11-26 13:48:44 +01:00
2016-12-04 01:49:49 +01:00
def add_svg_content(self, svg):
width, height = self.get_dimensions()
2016-11-26 13:48:44 +01:00
contents = ET.Element('g', {
'transform': 'scale(1 -1) translate(0 -%d)' % (height),
})
svg.append(contents)
2016-12-04 01:49:49 +01:00
return contents
def add_svg_image(self, svg, image):
width, height = self.get_dimensions()
contents = ET.Element('image', {
'x': '0',
'y': '0',
'width': str(width),
'height': str(height),
'xlink:href': image
})
svg.append(contents)
def render_base(self, png=True):
svg = self.create_svg()
contents = self.add_svg_content(svg)
if not self.level.intermediate:
width, height = get_dimensions()
holes = self.level.geometries.holes.buffer(0.1, join_style=JOIN_STYLE.mitre)
contents.append(self.polygon_svg(box(0, 0, width, height).difference(holes),
fill_color='#000000'))
2016-11-26 13:48:44 +01:00
2016-11-29 00:47:37 +01:00
contents.append(self.polygon_svg(self.level.geometries.buildings_with_holes,
fill_color=('#EBEBEB' if self.level.intermediate else '#D5D5D5')))
contents.append(self.polygon_svg(self.level.geometries.outsides,
fill_color='#DCE6DC'))
contents.append(self.polygon_svg(self.level.geometries.walls_shadow,
fill_color='#000000',
2016-11-29 17:47:05 +01:00
fill_opacity=0.06))
contents.append(self.polygon_svg(self.level.geometries.elevatorlevels,
fill_color='#9EF8FB'))
contents.append(self.polygon_svg(self.level.geometries.doors,
fill_color='#FFFFFF',
2016-11-29 17:47:05 +01:00
stroke_color='#3c3c3c',
2016-12-02 18:50:19 +01:00
stroke_width=0.05))
contents.append(self.polygon_svg(self.level.geometries.obstacles,
fill_color='#BDBDBD',
stroke_color='#9E9E9E',
2016-12-02 18:50:19 +01:00
stroke_width=0.05))
2016-11-28 20:56:52 +01:00
2016-11-29 00:47:37 +01:00
contents.append(self.polygon_svg(self.level.geometries.walls,
2016-11-26 13:48:44 +01:00
fill_color='#949494',
2016-11-29 17:47:05 +01:00
stroke_color='#3c3c3c',
2016-12-02 18:50:19 +01:00
stroke_width=0.05))
2016-11-26 13:48:44 +01:00
2016-12-04 01:49:49 +01:00
filename = get_render_path('level-%s.base.svg' % self.level.name)
with open(filename, 'w') as f:
f.write(ET.tostring(svg).decode())
if png:
png_filename = get_render_path('level-%s.base.png' % self.level.name)
subprocess.call(['rsvg-convert', filename, '-o', png_filename])
def render_simple(self, png=True):
svg = self.create_svg()
lower = []
for level in self.level.lower():
lower.append(level)
if not level.intermediate:
lower = []
lower.append(self.level)
width, height = get_dimensions()
contents = self.add_svg_content(svg)
contents.append(self.polygon_svg(box(0, 0, width, height),
fill_color='#000000'))
for level in lower:
self.add_svg_image(svg, 'file://'+get_render_path('level-%s.base.png' % level.name))
filename = get_render_path('level-%s.simple.svg' % self.level.name)
with open(filename, 'w') as f:
f.write(ET.tostring(svg).decode())
if png:
png_filename = get_render_path('level-%s.simple.png' % self.level.name)
subprocess.call(['rsvg-convert', filename, '-o', png_filename])
def render_full(self, png=True):
svg = self.create_svg()
self.add_svg_image(svg, 'file://' + get_render_path('level-%s.simple.png' % self.level.name))
higher = []
for level in self.level.higher():
if not level.intermediate:
break
higher.append(level)
2016-11-26 13:48:44 +01:00
contents = self.add_svg_content(svg)
for level in higher:
contents.append(self.polygon_svg(level.geometries.intermediate_shadows(to_level=self.level),
2016-12-04 02:42:00 +01:00
stroke_color='#9E9E9E',
stroke_width=0.1))
2016-12-04 01:49:49 +01:00
for level in higher:
self.add_svg_image(svg, 'file://'+get_render_path('level-%s.base.png' % level.name))
2016-12-02 18:50:49 +01:00
2016-12-04 01:49:49 +01:00
filename = get_render_path('level-%s.full.svg' % self.level.name)
2016-11-26 13:48:44 +01:00
with open(filename, 'w') as f:
2016-12-04 01:49:49 +01:00
f.write(ET.tostring(svg).decode())
2016-12-02 18:50:49 +01:00
2016-12-04 01:49:49 +01:00
if png:
png_filename = get_render_path('level-%s.full.png' % self.level.name)
subprocess.call(['rsvg-convert', filename, '-o', png_filename])