team-3/src/c3nav/mapdata/utils/svg.py

269 lines
12 KiB
Python
Raw Normal View History

import io
import math
import re
import subprocess
from itertools import chain
2017-05-12 17:22:10 +02:00
import numpy as np
from django.conf import settings
from django.core.checks import Error, register
2017-10-17 14:53:05 +02:00
from PIL import Image
from shapely.affinity import translate
from shapely.geometry import LineString, Polygon
2017-10-16 17:39:25 +02:00
from shapely.ops import unary_union
2017-05-12 17:22:10 +02:00
2017-10-19 18:10:12 +02:00
# import gobject-inspect, cairo and rsvg if the native rsvg SVG_RENDERER should be used
if settings.SVG_RENDERER == 'rsvg':
import pgi
import cairocffi
pgi.require_version('Rsvg', '2.0')
from pgi.repository import Rsvg
2017-05-12 17:22:10 +02:00
@register()
def check_svg_renderer(app_configs, **kwargs):
errors = []
if settings.SVG_RENDERER not in ('rsvg', 'rsvg-convert', 'inkscape'):
errors.append(
Error(
'Invalid SVG renderer: '+settings.SVG_RENDERER,
obj='settings.SVG_RENDERER',
id='c3nav.mapdata.E001',
)
)
return errors
class SVGImage:
2017-10-19 18:10:12 +02:00
# draw an svg image. supports pseudo-3D shadow-rendering
def __init__(self, bounds, scale: float=1, buffer=0):
2017-10-19 18:10:12 +02:00
# get image dimensions.
# note that these values describe the „viewport“ of the image, not its dimensions in pixels.
(self.bottom, self.left), (self.top, self.right) = bounds
self.width = self.right-self.left
self.height = self.top-self.bottom
2017-05-12 17:22:10 +02:00
self.scale = scale
2017-10-19 18:10:12 +02:00
# how many pixels around the image should be added and later cropped (otherwise rsvg does not blur correctly)
self.buffer_px = int(math.ceil(buffer*self.scale))
2017-10-19 18:10:12 +02:00
# create base elements and counter for clip path ids
self.g = ''
self.defs = ''
self.clip_path_i = 0
2017-10-19 18:10:12 +02:00
# keep track which area of the image has which altitude currently
2017-10-16 17:39:25 +02:00
self.altitudes = {}
self.last_altitude = None
2017-10-19 18:10:12 +02:00
# for fast numpy operations
self.np_scale = np.array((self.scale, -self.scale))
self.np_offset = np.array((-self.left*self.scale, self.top*self.scale))
2017-10-19 18:10:12 +02:00
# keep track of created blur filters to avoid duplicates
self.blurs = set()
2017-05-12 23:37:03 +02:00
def get_dimensions_px(self, buffer):
2017-10-19 18:10:12 +02:00
# get dimensions of the image in pixels, with or without buffer
width_px = self.width * self.scale + (self.buffer_px * 2 if buffer else 0)
height_px = self.height * self.scale + (self.buffer_px * 2 if buffer else 0)
return height_px, width_px
def get_xml(self, buffer=False):
2017-10-19 18:10:12 +02:00
# get the root <svg> element as an ElementTree element, with or without buffer
height_px, width_px = (self._trim_decimals(str(i)) for i in self.get_dimensions_px(buffer))
2017-10-17 12:41:22 +02:00
offset_px = self._trim_decimals(str(-self.buffer_px)) if buffer else '0'
2017-05-12 17:22:10 +02:00
attribs = ' viewBox="'+' '.join((offset_px, offset_px, width_px, height_px))+'"' if buffer else ''
2017-10-20 01:00:27 +02:00
result = ('<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" '
'width="'+width_px+'" height="'+height_px+'"'+attribs+'>')
if self.defs:
result += '<defs>'+self.defs+'</defs>'
if self.g:
result += '<g>'+self.g+'</g>'
result += '</svg>'
return result
def get_png(self, f=None):
2017-10-19 18:10:12 +02:00
# render the image to png. returns bytes if f is None, otherwise it calls f.write()
if settings.SVG_RENDERER == 'rsvg':
# create buffered surfaces
buffered_surface = cairocffi.SVGSurface(None, *(int(i) for i in self.get_dimensions_px(buffer=True)))
buffered_context = cairocffi.Context(buffered_surface)
# draw svg with rsvg
handle = Rsvg.Handle()
svg = handle.new_from_data(self.get_xml(buffer=True).encode())
svg.render_cairo(buffered_context)
# crop resulting immage
surface = buffered_surface.create_similar(buffered_surface.get_content(),
*(int(i) for i in self.get_dimensions_px(buffer=False)))
context = cairocffi.Context(surface)
context.set_source_surface(buffered_surface, -self.buffer_px, -self.buffer_px)
context.paint()
if f is None:
return surface.write_to_png()
2017-10-17 15:36:15 +02:00
f.write(surface.write_to_png())
elif settings.SVG_RENDERER == 'rsvg-convert':
p = subprocess.run(('rsvg-convert', '--format', 'png'),
input=self.get_xml(buffer=True).encode(), stdout=subprocess.PIPE, check=True)
png = io.BytesIO(p.stdout)
img = Image.open(png)
img = img.crop((self.buffer_px, self.buffer_px,
self.buffer_px + int(self.width * self.scale),
self.buffer_px + int(self.height * self.scale)))
if f is None:
f = io.BytesIO()
img.save(f, 'PNG')
f.seek(0)
return f.read()
img.save(f, 'PNG')
elif settings.SVG_RENDERER == 'inkscape':
p = subprocess.run(('inkscape', '-z', '-e', '/dev/stderr', '/dev/stdin'), input=self.get_xml().encode(),
stdout=subprocess.PIPE, stderr=subprocess.PIPE, check=True)
2017-10-17 14:53:05 +02:00
png = p.stderr[p.stderr.index(b'\x89PNG'):]
if f is None:
return png
f.write(png)
def _trim_decimals(self, data):
2017-10-20 00:58:19 +02:00
# remove trailing zeros from a decimal yes this is slow, but it greatly speeds up cairo rendering
return re.sub(r'([0-9]+)((\.[1-9])[0-9]+|\.[0-9]+)?', r'\1\3', data)
def _trim_lineto(self, data):
return re.sub(r'L (-?([0-9]+)(.[0-9]+)?,(-?[0-9]+)(.[0-9]+)?) L', r'L \1 ', data)
def _geometry_to_svg(self, geom):
# scale and move geometry geometry and create svg code for it
if isinstance(geom, Polygon):
return ('<path d="' +
' '.join((('M %.1f %.1f L'+(' %.1f %.1f'*(len(ring.coords)-1))) %
tuple((np.array(ring)*self.np_scale+self.np_offset).flatten()))
2017-10-20 11:31:22 +02:00
for ring in chain((geom.exterior,), geom.interiors))
2017-10-20 01:59:53 +02:00
+ '"/>').replace('.0 ', ' ')
if isinstance(geom, LineString):
2017-10-20 11:31:22 +02:00
return (('<path d="M %.1f %.1f L'+(' %.1f %.1f'*(geom.coords-1))+'"/>') %
tuple((np.array(geom)*self.np_scale+self.np_offset).flatten())).replace('.0 ', ' ')
try:
geoms = geom.geoms
except AttributeError:
return ''
return ''.join(self._geometry_to_svg(g) for g in geoms)
def _create_geometry(self, geometry, attribs='', tag='g'):
2017-10-19 18:10:12 +02:00
# convert a shapely geometry into an svg xml element
return '<'+tag+attribs+'>'+self._geometry_to_svg(geometry)+'</'+tag+'>'
2017-05-13 20:10:12 +02:00
def register_clip_path(self, geometry):
2017-10-20 01:05:35 +02:00
defid = 'clip'+str(self.clip_path_i)
self.defs += self._create_geometry(geometry, ' id="'+defid+'"', tag='clipPath')
self.clip_path_i += 1
2017-05-12 23:37:03 +02:00
return defid
2017-10-19 12:08:18 +02:00
def add_shadow(self, geometry, elevation, clip_path=None):
2017-10-19 18:10:12 +02:00
# add a shadow for the given geometry with the given elevation and, optionally, a clip path
2017-10-18 16:28:38 +02:00
elevation = min(elevation, 2)
2017-10-19 12:08:18 +02:00
blur_radius = elevation / 3 * 0.25
shadow_geom = translate(geometry.buffer(blur_radius),
xoff=(elevation / 3 * 0.12), yoff=-(elevation / 3 * 0.12))
if clip_path is not None:
if shadow_geom.distance(clip_path) >= blur_radius:
return
2017-10-18 16:28:38 +02:00
blur_id = 'blur'+str(int(elevation*100))
if elevation not in self.blurs:
self.defs += ('<filter id="'+blur_id+'" width="200%" height="200%" x="-50%" y="-50%">'
2017-10-20 02:18:47 +02:00
'<feGaussianBlur stdDeviation="'+str(int(math.ceil(blur_radius * self.scale)))+'"/>'
'</filter>')
self.blurs.add(elevation)
2017-10-18 16:28:38 +02:00
attribs = ' filter="url(#'+blur_id+')" fill="#000" fill-opacity="0.2"'
2017-10-19 12:08:18 +02:00
if clip_path:
2017-10-20 01:04:03 +02:00
attribs += ' clip-path="url(#'+self.register_clip_path(clip_path)+')"'
shadow = self._create_geometry(shadow_geom, attribs)
self.g += shadow
2017-10-16 17:39:25 +02:00
def clip_altitudes(self, new_geometry, new_altitude=None):
2017-10-19 18:10:12 +02:00
# registrer new geometry with specific (or no) altitude
# a geometry with no altitude will reset the altitude information of its area as if nothing was ever there
for altitude, geometry in tuple(self.altitudes.items()):
2017-10-16 17:39:25 +02:00
if altitude != new_altitude:
self.altitudes[altitude] = geometry.difference(new_geometry)
if self.altitudes[altitude].is_empty:
self.altitudes.pop(altitude)
2017-10-16 17:39:25 +02:00
if new_altitude is not None:
if self.last_altitude is not None and self.last_altitude > new_altitude:
raise ValueError('Altitudes have to be ascending.')
self.last_altitude = new_altitude
if new_altitude in self.altitudes:
self.altitudes[new_altitude] = unary_union([self.altitudes[new_altitude], new_geometry])
else:
self.altitudes[new_altitude] = new_geometry
2017-05-13 20:12:42 +02:00
def add_geometry(self, geometry=None, fill_color=None, fill_opacity=None, opacity=None, filter=None,
stroke_px=0.0, stroke_width=0.0, stroke_color=None, stroke_opacity=None, stroke_linejoin=None,
2017-10-16 17:39:25 +02:00
clip_path=None, altitude=None, elevation=None):
2017-10-19 18:10:12 +02:00
# draw a shapely geometry with a given style
# if altitude is set, the geometry will get a calculated shadow relative to the other geometries
# if elevation is set, the geometry will get a shadow with exactly this elevation
2017-10-20 00:51:30 +02:00
# if fill_color is set, filter out geometries that cannot be filled
if fill_color is not None:
try:
geometry.geoms
except AttributeError:
pass
else:
geometry = type(geometry)(tuple(geom for geom in geometry.geoms if hasattr(geom, 'exterior')))
if geometry.is_empty:
pass
attribs = ' fill="'+(fill_color or 'none')+'"'
if fill_opacity:
attribs += ' fill-opacity="'+str(fill_opacity)[:4]+'"'
if stroke_px:
attribs += ' stroke-width="'+self._trim_decimals(str(stroke_px))+'"'
elif stroke_width:
attribs += ' stroke-width="'+self._trim_decimals(str(stroke_width * self.scale))+'"'
if stroke_color:
attribs += ' stroke="'+stroke_color+'"'
if stroke_opacity:
attribs += ' stroke-opacity="'+str(stroke_opacity)[:4]+'"'
if stroke_linejoin:
attribs += ' stroke-linejoin="'+stroke_linejoin+'"'
if opacity:
attribs += ' opacity="'+str(opacity)[:4]+'"'
if filter:
attribs += ' filter="url(#'+filter+')"'
if clip_path:
attribs += ' clip-path="url(#'+clip_path+')"'
if geometry is not None:
2017-05-13 20:48:27 +02:00
if not geometry:
return
2017-10-16 17:39:25 +02:00
if altitude is not None or elevation is not None:
2017-10-19 11:36:53 +02:00
if elevation is not None:
elevation = float(1 if elevation is None else elevation)
if elevation:
2017-10-19 12:08:18 +02:00
self.add_shadow(geometry, elevation)
2017-10-19 11:36:53 +02:00
else:
for other_altitude, other_geom in self.altitudes.items():
2017-10-19 12:08:18 +02:00
self.add_shadow(geometry, altitude-other_altitude, clip_path=other_geom)
2017-10-16 17:39:25 +02:00
self.clip_altitudes(geometry, altitude)
element = self._create_geometry(geometry, attribs)
2017-10-18 16:28:38 +02:00
2017-05-13 19:19:28 +02:00
else:
element = '<rect width="100%" height="100%"'+attribs+'>'
2017-05-12 17:22:10 +02:00
self.g += element
2017-05-12 17:22:10 +02:00
return element