team-3/src/c3nav/mapdata/models/geometry/base.py

124 lines
4.8 KiB
Python
Raw Normal View History

2017-10-28 21:36:52 +02:00
import math
from collections import OrderedDict
2017-05-08 21:55:45 +02:00
2017-10-28 21:36:52 +02:00
from django.utils.functional import cached_property
from shapely.geometry import LineString, Point, mapping
2017-11-16 19:24:49 +01:00
from shapely.ops import unary_union
from c3nav.mapdata.models.base import SerializableMixin
2017-11-16 19:24:49 +01:00
from c3nav.mapdata.utils.geometry import assert_multilinestring, assert_multipolygon
from c3nav.mapdata.utils.json import format_geojson
geometry_affecting_fields = ('height', 'width', 'access_restriction')
class GeometryMixin(SerializableMixin):
"""
A map feature with a geometry
"""
geometry = None
class Meta:
abstract = True
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.orig_geometry = None if 'geometry' in self.get_deferred_fields() else self.geometry
self._orig = {field.attname: (None if field.attname in self.get_deferred_fields()
else getattr(self, field.attname))
for field in self._meta.get_fields()
if field.name in geometry_affecting_fields}
def get_geojson_properties(self, *args, **kwargs) -> dict:
2017-05-27 16:19:49 +02:00
result = OrderedDict((
('type', self.__class__.__name__.lower()),
2017-06-16 18:19:52 +02:00
('id', self.pk),
))
2017-05-27 16:19:49 +02:00
if getattr(self, 'bounds', False):
result['bounds'] = True
return result
def to_geojson(self, instance=None) -> dict:
result = OrderedDict((
('type', 'Feature'),
('properties', self.get_geojson_properties(instance=instance)),
('geometry', format_geojson(mapping(self.geometry), round=False)),
))
original_geometry = getattr(self, 'original_geometry', None)
if original_geometry:
result['original_geometry'] = format_geojson(mapping(original_geometry), round=False)
return result
2017-05-11 21:30:29 +02:00
@classmethod
def serialize_type(cls, geomtype=True, **kwargs):
result = super().serialize_type()
if geomtype:
result['geomtype'] = cls._meta.get_field('geometry').geomtype
return result
2017-10-28 21:36:52 +02:00
@cached_property
def point(self):
c = self.geometry.centroid
x1, y1, x2, y2 = self.geometry.bounds
lines = (tuple(assert_multilinestring(LineString(((x1, c.y), (x2, c.y))).intersection(self.geometry))) +
tuple(assert_multilinestring(LineString(((c.x, y1), (c.x, y2))).intersection(self.geometry))))
return min(lines, key=lambda line: (line.distance(c), line.length),
default=self.geometry.representative_point).centroid
2017-10-28 21:36:52 +02:00
def serialize(self, **kwargs):
result = super().serialize(**kwargs)
if 'geometry' in result:
2017-05-11 19:36:49 +02:00
result.move_to_end('geometry')
return result
2017-10-28 21:36:52 +02:00
def _serialize(self, geometry=True, simple_geometry=False, **kwargs):
result = super()._serialize(simple_geometry=simple_geometry, **kwargs)
2017-05-11 19:36:49 +02:00
if geometry:
result['geometry'] = format_geojson(mapping(self.geometry), round=False)
2017-10-28 21:36:52 +02:00
if simple_geometry:
result['point'] = (self.level_id, ) + tuple(round(i, 2) for i in self.point.coords[0])
2017-10-28 21:36:52 +02:00
if not isinstance(self.geometry, Point):
2017-11-22 22:15:33 +01:00
minx, miny, maxx, maxy = self.geometry.bounds
result['bounds'] = ((int(math.floor(minx)), int(math.floor(miny))),
(int(math.ceil(maxx)), int(math.ceil(maxy))))
2017-05-11 19:36:49 +02:00
return result
2017-11-02 13:35:58 +01:00
def details_display(self):
result = super().details_display()
result['geometry'] = format_geojson(mapping(self.geometry), round=False)
return result
2017-05-11 19:36:49 +02:00
def get_shadow_geojson(self):
2017-05-09 09:36:08 +02:00
pass
2017-05-09 13:16:36 +02:00
def contains(self, x, y) -> bool:
return self.geometry.contains(Point(x, y))
@property
def all_geometry_changed(self):
return any(getattr(self, attname) != value for attname, value in self._orig.items())
@property
def geometry_changed(self):
2017-10-24 00:11:02 +02:00
if self.orig_geometry is None:
return True
if self.geometry is self.orig_geometry:
return False
if not self.geometry.almost_equals(self.orig_geometry, 1):
return True
field = self._meta.get_field('geometry')
rounded = field.to_python(field.get_prep_value(self.geometry))
if not rounded.almost_equals(self.orig_geometry, 2):
return True
return False
def get_changed_geometry(self):
field = self._meta.get_field('geometry')
new_geometry = field.get_final_value(self.geometry)
2017-11-16 19:24:49 +01:00
if self.orig_geometry is None:
return new_geometry
difference = new_geometry.symmetric_difference(self.orig_geometry)
2017-11-16 19:33:19 +01:00
if self._meta.get_field('geometry').geomtype in ('polygon', 'multipolygon'):
2017-11-16 19:24:49 +01:00
difference = unary_union(assert_multipolygon(difference))
return difference