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

432 lines
12 KiB
Python
Raw Normal View History

2016-11-27 14:03:39 +01:00
from collections import OrderedDict
from django.db import models
2016-12-18 00:40:10 +01:00
from django.utils.functional import cached_property
from django.utils.translation import ugettext_lazy as _
from shapely.geometry import CAP_STYLE, JOIN_STYLE, Point
2016-10-13 13:55:02 +02:00
from shapely.geometry.geo import mapping, shape
from c3nav.mapdata.fields import GeometryField
2016-11-28 20:56:52 +01:00
from c3nav.mapdata.models import Elevator
2016-11-27 23:51:44 +01:00
from c3nav.mapdata.models.base import MapItem, MapItemMeta
2016-12-07 16:11:33 +01:00
from c3nav.mapdata.utils.json import format_geojson
2016-11-27 14:03:39 +01:00
GEOMETRY_MAPITEM_TYPES = OrderedDict()
2016-11-27 23:51:44 +01:00
class GeometryMapItemMeta(MapItemMeta):
2016-11-27 14:03:39 +01:00
def __new__(mcs, name, bases, attrs):
cls = super().__new__(mcs, name, bases, attrs)
if not cls._meta.abstract:
GEOMETRY_MAPITEM_TYPES[name.lower()] = cls
return cls
2016-11-27 14:03:39 +01:00
class GeometryMapItem(MapItem, metaclass=GeometryMapItemMeta):
"""
A map feature
"""
geometry = GeometryField()
geomtype = None
class Meta:
abstract = True
@classmethod
def fromfile(cls, data, file_path):
kwargs = super().fromfile(data, file_path)
if 'geometry' not in data:
raise ValueError('missing geometry.')
try:
kwargs['geometry'] = shape(data['geometry'])
except:
raise ValueError(_('Invalid GeoJSON.'))
return kwargs
2016-11-27 14:03:39 +01:00
def get_geojson_properties(self):
return OrderedDict((
('type', self.__class__.__name__.lower()),
('name', self.name),
('package', self.package.name),
))
def to_geojson(self):
2016-12-08 12:36:09 +01:00
return OrderedDict((
2016-11-27 14:03:39 +01:00
('type', 'Feature'),
('properties', self.get_geojson_properties()),
('geometry', format_geojson(mapping(self.geometry), round=False)),
2016-12-08 12:36:09 +01:00
))
2016-11-27 14:03:39 +01:00
def tofile(self, form=None):
result = super().tofile()
result['geometry'] = format_geojson(mapping(self.geometry))
return result
2016-12-08 12:36:09 +01:00
def get_shadow_geojson(self):
return None
def contains(self, x, y):
return self.geometry.contains(Point(x, y))
2016-12-01 12:25:02 +01:00
class GeometryMapItemWithLevel(GeometryMapItem):
"""
A map feature
"""
level = models.ForeignKey('mapdata.Level', on_delete=models.CASCADE, verbose_name=_('level'))
class Meta:
abstract = True
@classmethod
def fromfile(cls, data, file_path):
kwargs = super().fromfile(data, file_path)
if 'level' not in data:
raise ValueError('missing level.')
kwargs['level'] = data['level']
return kwargs
def get_geojson_properties(self):
result = super().get_geojson_properties()
2016-12-04 01:49:49 +01:00
result['level'] = self.level.name
2016-12-01 12:25:02 +01:00
return result
def tofile(self, form=None):
2016-12-01 12:25:02 +01:00
result = super().tofile()
result['level'] = self.level.name
result.move_to_end('geometry')
return result
class DirectedLineGeometryMapItemWithLevel(GeometryMapItemWithLevel):
2016-12-08 18:12:07 +01:00
geomtype = 'polyline'
class Meta:
abstract = True
def to_geojson(self):
result = super().to_geojson()
original_geometry = result['geometry']
draw = self.geometry.buffer(0.05, join_style=JOIN_STYLE.mitre, cap_style=CAP_STYLE.flat)
result['geometry'] = format_geojson(mapping(draw))
result['original_geometry'] = original_geometry
return result
def to_shadow_geojson(self):
shadow = self.geometry.parallel_offset(0.03, 'right', join_style=JOIN_STYLE.mitre)
2016-12-08 18:12:07 +01:00
shadow = shadow.buffer(0.019, join_style=JOIN_STYLE.mitre, cap_style=CAP_STYLE.flat)
return OrderedDict((
('type', 'Feature'),
('properties', OrderedDict((
('type', 'shadow'),
('original_type', self.__class__.__name__.lower()),
('original_name', self.name),
('level', self.level.name),
))),
('geometry', format_geojson(mapping(shadow), round=False)),
))
2016-12-04 01:49:49 +01:00
class Building(GeometryMapItemWithLevel):
"""
The outline of a building on a specific level
"""
geomtype = 'polygon'
class Meta:
verbose_name = _('Building')
verbose_name_plural = _('Buildings')
default_related_name = 'buildings'
class Room(GeometryMapItemWithLevel):
"""
An accessible area like a room. Can overlap.
"""
geomtype = 'polygon'
class Meta:
verbose_name = _('Room')
verbose_name_plural = _('Rooms')
default_related_name = 'rooms'
class Outside(GeometryMapItemWithLevel):
"""
An accessible outdoor area like a court. Can overlap.
"""
geomtype = 'polygon'
class Meta:
verbose_name = _('Outside Area')
verbose_name_plural = _('Outside Areas')
default_related_name = 'outsides'
class Escalator(GeometryMapItemWithLevel):
"""
An escalator area
"""
DIRECTIONS = (
(True, _('up')),
(False, _('down')),
)
direction = models.BooleanField(verbose_name=_('direction'), choices=DIRECTIONS)
geomtype = 'polygon'
class Meta:
verbose_name = _('Escalator')
verbose_name_plural = _('Escalators')
default_related_name = 'escalators'
@classmethod
def fromfile(cls, data, file_path):
kwargs = super().fromfile(data, file_path)
if 'direction' not in data:
raise ValueError('missing direction.')
kwargs['direction'] = data['direction']
return kwargs
def get_geojson_properties(self):
result = super().get_geojson_properties()
result['direction'] = 'up' if self.direction else 'down'
return result
def tofile(self, form=None):
result = super().tofile()
result['direction'] = self.direction
return result
class EscalatorSlope(DirectedLineGeometryMapItemWithLevel):
"""
An escalator slope, indicating which side of the escalator is up
"""
class Meta:
verbose_name = _('Escalator Slope')
verbose_name_plural = _('Escalator Slopes')
default_related_name = 'escalatorslopes'
class Stair(DirectedLineGeometryMapItemWithLevel):
2016-12-08 18:12:07 +01:00
"""
A stair
"""
class Meta:
verbose_name = _('Stair')
verbose_name_plural = _('Stairs')
default_related_name = 'stairs'
class OneWay(DirectedLineGeometryMapItemWithLevel):
"""
A one way obstacle
"""
class Meta:
verbose_name = _('Oneway')
verbose_name_plural = _('Oneways')
default_related_name = 'oneways'
2016-12-09 14:49:20 +01:00
class Obstacle(GeometryMapItemWithLevel):
"""
2016-12-09 14:49:20 +01:00
An obstacle
"""
2016-12-09 14:49:20 +01:00
crop_to_level = models.ForeignKey('mapdata.Level', on_delete=models.CASCADE, null=True, blank=True,
verbose_name=_('crop to other level'), related_name='crops_obstacles')
2016-12-09 14:49:20 +01:00
geomtype = 'polygon'
class Meta:
2016-12-09 14:49:20 +01:00
verbose_name = _('Obstacle')
verbose_name_plural = _('Obstacles')
default_related_name = 'obstacles'
@classmethod
def fromfile(cls, data, file_path):
kwargs = super().fromfile(data, file_path)
2016-12-09 14:49:20 +01:00
if 'crop_to_level' in data:
kwargs['crop_to_level'] = data['crop_to_level']
return kwargs
def get_geojson_properties(self):
result = super().get_geojson_properties()
2016-12-09 14:49:20 +01:00
if self.crop_to_level is not None:
result['crop_to_level'] = self.crop_to_level.name
return result
def tofile(self, form=None):
result = super().tofile()
2016-12-09 14:49:20 +01:00
if self.crop_to_level is not None:
result['crop_to_level'] = self.crop_to_level.name
return result
2016-12-09 14:49:20 +01:00
class LineObstacle(GeometryMapItemWithLevel):
2016-12-01 12:25:02 +01:00
"""
2016-12-09 14:49:20 +01:00
An obstacle that is a line with a specific width
2016-12-01 12:25:02 +01:00
"""
2016-12-09 14:49:20 +01:00
width = models.DecimalField(_('obstacle width'), max_digits=4, decimal_places=2, default=0.15)
2016-12-04 20:01:37 +01:00
2016-12-09 14:49:20 +01:00
geomtype = 'polyline'
2016-12-01 12:25:02 +01:00
class Meta:
2016-12-09 14:49:20 +01:00
verbose_name = _('Line Obstacle')
verbose_name_plural = _('Line Obstacles')
default_related_name = 'lineobstacles'
def to_geojson(self):
result = super().to_geojson()
original_geometry = result['geometry']
draw = self.geometry.buffer(self.width/2, join_style=JOIN_STYLE.mitre, cap_style=CAP_STYLE.flat)
result['geometry'] = format_geojson(mapping(draw))
result['original_geometry'] = original_geometry
return result
2016-12-01 12:25:02 +01:00
@classmethod
def fromfile(cls, data, file_path):
kwargs = super().fromfile(data, file_path)
2016-12-09 14:49:20 +01:00
if 'width' not in data:
raise ValueError('missing width.')
kwargs['width'] = data['width']
2016-12-01 12:25:02 +01:00
return kwargs
def get_geojson_properties(self):
result = super().get_geojson_properties()
2016-12-09 14:49:20 +01:00
result['width'] = float(self.width)
2016-12-01 12:25:02 +01:00
return result
def tofile(self, form=None):
2016-12-01 12:25:02 +01:00
result = super().tofile()
2016-12-09 14:49:20 +01:00
result['width'] = float(self.width)
2016-12-01 12:25:02 +01:00
return result
2016-12-04 20:01:37 +01:00
class LevelConnector(GeometryMapItem):
2016-10-16 13:20:34 +02:00
"""
2016-12-04 20:01:37 +01:00
A connector connecting levels
2016-10-16 13:20:34 +02:00
"""
geomtype = 'polygon'
2016-12-04 20:01:37 +01:00
levels = models.ManyToManyField('mapdata.Level', verbose_name=_('levels'))
2016-10-16 13:20:34 +02:00
class Meta:
2016-12-04 20:01:37 +01:00
verbose_name = _('Level Connector')
verbose_name_plural = _('Level Connectors')
default_related_name = 'levelconnectors'
2016-10-16 13:20:34 +02:00
@classmethod
def fromfile(cls, data, file_path):
kwargs = super().fromfile(data, file_path)
2016-12-04 20:01:37 +01:00
if 'levels' not in data:
raise ValueError('missing levels.')
levels = data.get('levels', None)
if not isinstance(levels, list):
raise TypeError('levels has to be a list')
if len(levels) < 2:
raise ValueError('a level connector needs at least two levels')
kwargs['levels'] = levels
2016-10-16 13:20:34 +02:00
return kwargs
def get_geojson_properties(self):
result = super().get_geojson_properties()
2016-12-04 20:01:37 +01:00
result['levels'] = tuple(self.levels.all().order_by('name').values_list('name', flat=True))
return result
def tofile(self, form=None):
2016-10-16 13:20:34 +02:00
result = super().tofile()
2016-12-04 20:01:37 +01:00
result['levels'] = sorted(self.levels.all().order_by('name').values_list('name', flat=True))
result.move_to_end('geometry')
2016-10-16 13:20:34 +02:00
return result
2016-12-01 12:25:02 +01:00
class Door(GeometryMapItemWithLevel):
2016-10-16 13:20:34 +02:00
"""
A connection between two rooms
"""
geomtype = 'polygon'
class Meta:
verbose_name = _('Door')
verbose_name_plural = _('Doors')
default_related_name = 'doors'
2016-11-28 20:56:52 +01:00
2016-12-01 12:25:02 +01:00
class Hole(GeometryMapItemWithLevel):
2016-11-29 00:47:37 +01:00
"""
A hole in the ground of a room, e.g. for stairs.
"""
geomtype = 'polygon'
class Meta:
verbose_name = _('Hole')
verbose_name_plural = _('Holes')
default_related_name = 'holes'
2016-12-01 12:25:02 +01:00
class ElevatorLevel(GeometryMapItemWithLevel):
2016-11-28 20:56:52 +01:00
"""
An elevator Level
"""
2016-12-01 12:25:02 +01:00
elevator = models.ForeignKey(Elevator, on_delete=models.PROTECT)
2016-11-28 20:56:52 +01:00
button = models.SlugField(_('Button label'), max_length=10)
override_altitude = models.DecimalField(_('override level altitude'),
blank=True, null=True, max_digits=6, decimal_places=2)
2016-11-28 20:56:52 +01:00
geomtype = 'polygon'
class Meta:
verbose_name = _('Elevator Level')
verbose_name_plural = _('Elevator Levels')
default_related_name = 'elevatorlevels'
def get_geojson_properties(self):
result = super().get_geojson_properties()
result['elevator'] = self.elevator.name
result['button'] = self.button
return result
@classmethod
def fromfile(cls, data, file_path):
kwargs = super().fromfile(data, file_path)
if 'elevator' not in data:
raise ValueError('missing elevator.')
kwargs['elevator'] = data['elevator']
if 'button' not in data:
raise ValueError('missing button.')
kwargs['button'] = data['button']
2016-12-18 00:40:10 +01:00
if 'override_altitude' in data:
kwargs['override_altitude'] = data['override_altitude']
2016-11-28 20:56:52 +01:00
return kwargs
def tofile(self, form=None):
2016-11-28 20:56:52 +01:00
result = super().tofile()
result['elevator'] = self.elevator.name
result['button'] = self.button
2016-12-18 00:40:10 +01:00
if self.override_altitude is not None:
result['override_altitude'] = float(self.override_altitude)
2016-11-28 20:56:52 +01:00
return result
2016-12-18 00:40:10 +01:00
@cached_property
def altitude(self):
if self.override_altitude is not None:
return self.override_altitude
return self.level.altitude