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

198 lines
5.4 KiB
Python
Raw Normal View History

2016-11-27 14:03:39 +01:00
from collections import OrderedDict
from django.db import models
from django.utils.translation import ugettext_lazy as _
2017-05-05 12:32:35 +02:00
from shapely.geometry import CAP_STYLE, JOIN_STYLE
2017-04-28 11:14:01 +02:00
from shapely.geometry.geo import mapping
2017-05-05 12:32:35 +02:00
from c3nav.mapdata.models.base import GeometryFeature
2016-12-07 16:11:33 +01:00
from c3nav.mapdata.utils.json import format_geojson
2017-05-05 12:32:35 +02:00
class LevelFeature(GeometryFeature):
2016-12-01 12:25:02 +01:00
"""
2017-05-05 12:32:35 +02:00
a map feature that has a geometry and belongs to a level
2016-12-01 12:25:02 +01:00
"""
level = models.ForeignKey('mapdata.Level', on_delete=models.CASCADE, verbose_name=_('level'))
class Meta:
abstract = True
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
2017-05-05 12:32:35 +02:00
class AreaFeature(GeometryFeature):
2017-05-04 15:44:32 +02:00
"""
2017-05-05 12:32:35 +02:00
a map feature that has a geometry and belongs to an area
2017-05-04 15:44:32 +02:00
"""
area = models.ForeignKey('mapdata.Area', on_delete=models.CASCADE, verbose_name=_('area'))
class Meta:
abstract = True
def get_geojson_properties(self):
result = super().get_geojson_properties()
result['area'] = self.area.name
return result
2017-05-05 12:32:35 +02:00
class Building(LevelFeature):
2016-12-04 01:49:49 +01:00
"""
The outline of a building on a specific level
"""
geomtype = 'polygon'
class Meta:
verbose_name = _('Building')
verbose_name_plural = _('Buildings')
default_related_name = 'buildings'
2017-05-05 12:32:35 +02:00
class Area(LevelFeature):
2016-12-04 01:49:49 +01:00
"""
2017-05-04 12:08:17 +02:00
An accessible area. Shouldn't overlap.
2016-12-04 01:49:49 +01:00
"""
geomtype = 'polygon'
2017-05-04 12:08:17 +02:00
CATEGORIES = (
('', _('normal')),
('stairs', _('stairs')),
('escalator', _('escalator')),
('elevator', _('elevator')),
)
LAYERS = (
('', _('normal')),
('upper', _('upper')),
('lowerr', _('lower')),
)
2016-12-04 01:49:49 +01:00
public = models.BooleanField(verbose_name=_('public'))
category = models.CharField(verbose_name=_('category'), choices=CATEGORIES, max_length=16)
2017-05-04 12:08:17 +02:00
layer = models.CharField(verbose_name=_('layer'), choices=LAYERS, max_length=16)
2016-12-04 01:49:49 +01:00
class Meta:
2017-05-04 12:08:17 +02:00
verbose_name = _('Area')
verbose_name_plural = _('Areas')
default_related_name = 'areas'
2016-12-04 01:49:49 +01:00
def get_geojson_properties(self):
result = super().get_geojson_properties()
2017-05-05 14:20:32 +02:00
result['category'] = self.category
result['layer'] = self.layer
result['public'] = self.public
return result
2016-12-04 01:49:49 +01:00
2017-05-05 12:32:35 +02:00
class StuffedArea(AreaFeature):
"""
A slow area with many tables or similar. Avoid it from routing by slowing it a bit down
"""
geomtype = 'polygon'
class Meta:
verbose_name = _('Stuffed Area')
verbose_name_plural = _('Stuffed Areas')
default_related_name = 'stuffedareas'
2017-05-05 12:32:35 +02:00
class Stair(AreaFeature):
2016-12-08 18:12:07 +01:00
"""
A stair
"""
geomtype = 'polyline'
2016-12-08 18:12:07 +01:00
class Meta:
verbose_name = _('Stair')
verbose_name_plural = _('Stairs')
default_related_name = 'stairs'
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)
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),
2017-05-05 14:20:32 +02:00
('area', self.area.name),
))),
('geometry', format_geojson(mapping(shadow), round=False)),
))
2016-12-08 18:12:07 +01:00
2017-05-05 12:32:35 +02:00
class Obstacle(AreaFeature):
"""
2016-12-09 14:49:20 +01:00
An obstacle
"""
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'
2017-05-05 12:32:35 +02:00
class LineObstacle(AreaFeature):
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'
2017-05-04 15:44:32 +02:00
@property
def buffered_geometry(self):
return self.geometry.buffer(self.width / 2, join_style=JOIN_STYLE.mitre, cap_style=CAP_STYLE.flat)
2016-12-09 14:49:20 +01:00
def to_geojson(self):
result = super().to_geojson()
original_geometry = result['geometry']
2017-05-04 15:44:32 +02:00
result['geometry'] = format_geojson(mapping(self.buffered_geometry))
2016-12-09 14:49:20 +01:00
result['original_geometry'] = original_geometry
return result
2016-12-01 12:25:02 +01:00
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
2017-05-05 12:32:35 +02:00
class Door(LevelFeature):
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
2017-05-05 12:32:35 +02:00
class Hole(LevelFeature):
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'