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

242 lines
9.9 KiB
Python
Raw Normal View History

import io
import math
import re
import subprocess
2017-05-12 17:22:10 +02:00
import xml.etree.ElementTree as ET
from itertools import chain
2017-05-12 17:22:10 +02:00
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 scale, translate
2017-10-16 17:39:25 +02:00
from shapely.ops import unary_union
2017-05-12 17:22:10 +02:00
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:
def __init__(self, bounds, scale: float=1, buffer=0):
(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
self.buffer_px = int(math.ceil(buffer*self.scale))
self.g = ET.Element('g', {})
2017-05-12 23:37:03 +02:00
self.defs = ET.Element('defs')
self.def_i = 0
2017-10-16 17:39:25 +02:00
self.altitudes = {}
self.last_altitude = None
self.blurs = set()
2017-05-12 23:37:03 +02:00
def get_dimensions_px(self, 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
2017-10-17 12:41:22 +02:00
def get_element(self, buffer=False):
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
root = ET.Element('svg', {
2017-10-17 12:41:22 +02:00
'width': width_px,
'height': height_px,
2017-05-12 17:22:10 +02:00
'xmlns:svg': 'http://www.w3.org/2000/svg',
'xmlns': 'http://www.w3.org/2000/svg',
'xmlns:xlink': 'http://www.w3.org/1999/xlink',
})
2017-10-17 12:41:22 +02:00
if buffer:
root.attrib['viewBox'] = ' '.join((offset_px, offset_px, width_px, height_px))
2017-05-12 23:37:03 +02:00
root.append(self.defs)
2017-05-12 17:22:10 +02:00
root.append(self.g)
return root
2017-10-17 12:41:22 +02:00
def get_xml(self, buffer=False):
return ET.tostring(self.get_element(buffer=buffer)).decode()
def get_png(self, f=None):
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)
2017-05-12 23:37:03 +02:00
def new_defid(self):
defid = 's'+str(self.def_i)
self.def_i += 1
return defid
2017-05-12 17:22:10 +02:00
def _trim_decimals(self, data):
return re.sub(r'([0-9]+)\.0', r'\1', re.sub(r'([0-9]+\.[0-9])[0-9]+', r'\1', data))
2017-05-13 20:10:12 +02:00
def _create_geometry(self, geometry):
2017-10-17 12:41:22 +02:00
geometry = translate(geometry, xoff=0-self.left, yoff=0-self.bottom)
2017-05-13 20:10:12 +02:00
geometry = scale(geometry, xfact=1, yfact=-1, origin=(self.width / 2, self.height / 2))
geometry = scale(geometry, xfact=self.scale, yfact=self.scale, origin=(0, 0))
element = ET.fromstring(self._trim_decimals(geometry.svg(0, '#FFFFFF')))
2017-05-12 17:22:10 +02:00
if element.tag != 'g':
new_element = ET.Element('g')
new_element.append(element)
element = new_element
for elem in chain(element.findall('polyline'), element.findall('path')):
elem.attrib.pop('opacity', None)
elem.attrib.pop('fill', None)
elem.attrib.pop('fill-rule', None)
elem.attrib.pop('stroke', None)
elem.attrib.pop('stroke-width', None)
2017-05-13 20:10:12 +02:00
return element
def register_geometry(self, geometry, defid=None, as_clip_path=False, comment=None):
if defid is None:
defid = self.new_defid()
2017-05-12 23:37:03 +02:00
2017-05-13 20:10:12 +02:00
element = self._create_geometry(geometry)
if as_clip_path:
element.tag = 'clipPath'
2017-05-12 23:37:03 +02:00
element.set('id', defid)
self.defs.append(element)
return defid
def get_blur(self, elevation):
blur_id = 'blur'+str(elevation*100)
if elevation not in self.blurs:
blur_filter = ET.Element('filter', {'id': blur_id,
'width': '200%',
'height': '200%',
'x': '-50%',
'y': '-50%'})
blur_filter.append(ET.Element('feGaussianBlur',
{'in': 'SourceGraphic',
'stdDeviation': str(elevation*self.scale)}))
self.defs.append(blur_filter)
self.blurs.add(elevation)
return blur_id
2017-05-13 19:19:28 +02:00
def add_clip_path(self, *geometries, inverted=False, subtract=False, defid=None):
2017-05-12 23:37:03 +02:00
if defid is None:
defid = self.new_defid()
2017-05-13 19:19:28 +02:00
clippath = ET.Element('clipPath', {'id': defid})
clippath.append(ET.Element('use', {'xlink:href': '#' + geometries[0]}))
self.defs.append(clippath)
2017-05-12 23:37:03 +02:00
return defid
2017-10-16 17:39:25 +02:00
def clip_altitudes(self, new_geometry, new_altitude=None):
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):
if geometry is not None:
2017-05-13 20:48:27 +02:00
if not geometry:
return
2017-05-13 20:10:12 +02:00
if isinstance(geometry, str):
element = ET.Element('use', {'xlink:href': '#'+geometry})
else:
element = self._create_geometry(geometry)
2017-10-16 17:39:25 +02:00
if altitude is not None or elevation is not None:
blur_radius = float(1 if elevation is None else elevation)
2017-10-17 19:31:28 +02:00
if blur_radius:
buffered_geometry = translate(geometry.buffer(blur_radius/20),
xoff=blur_radius/40, yoff=-blur_radius/40)
shadow_element = self._create_geometry(buffered_geometry)
shadow_element.set('fill', '#000000')
shadow_element.set('fill-opacity', '0.14')
shadow_element.set('filter', 'url(#'+self.get_blur(blur_radius/15)+')')
self.g.append(shadow_element)
2017-10-16 17:39:25 +02:00
self.clip_altitudes(geometry, altitude)
2017-05-13 19:19:28 +02:00
else:
element = ET.Element('rect', {'width': '100%', 'height': '100%'})
2017-05-12 23:37:03 +02:00
element.set('fill', fill_color or 'none')
if fill_opacity:
element.set('fill-opacity', str(fill_opacity)[:4])
if stroke_px:
element.set('stroke-width', self._trim_decimals(str(stroke_px)))
elif stroke_width:
element.set('stroke-width', self._trim_decimals(str(stroke_width * self.scale)))
2017-05-12 23:37:03 +02:00
if stroke_color:
element.set('stroke', stroke_color)
if stroke_opacity:
element.set('stroke-opacity', str(stroke_opacity)[:4])
2017-05-12 23:37:03 +02:00
if stroke_linejoin:
element.set('stroke-linejoin', stroke_linejoin)
2017-05-12 23:37:03 +02:00
if opacity:
element.set('opacity', str(opacity)[:4])
2017-05-12 23:37:03 +02:00
if filter:
element.set('filter', 'url(#'+filter+')')
2017-05-13 19:19:28 +02:00
if clip_path:
element.set('clip-path', 'url(#'+clip_path+')')
2017-05-12 17:22:10 +02:00
2017-05-12 23:37:03 +02:00
self.g.append(element)
2017-05-12 17:22:10 +02:00
return element