2016-11-27 14:03:39 +01:00
|
|
|
from collections import OrderedDict
|
|
|
|
|
2016-10-12 15:25:00 +02:00
|
|
|
from django.db import models
|
2016-12-18 00:40:10 +01:00
|
|
|
from django.utils.functional import cached_property
|
2016-10-12 15:25:00 +02:00
|
|
|
from django.utils.translation import ugettext_lazy as _
|
2016-12-21 00:24:56 +01:00
|
|
|
from shapely.geometry import CAP_STYLE, JOIN_STYLE, Point
|
2017-04-28 11:14:01 +02:00
|
|
|
from shapely.geometry.geo import mapping
|
2016-10-12 15:25:00 +02:00
|
|
|
|
|
|
|
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-10-12 15:25:00 +02:00
|
|
|
|
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-10-12 15:25:00 +02:00
|
|
|
|
2016-11-27 14:03:39 +01:00
|
|
|
class GeometryMapItem(MapItem, metaclass=GeometryMapItemMeta):
|
2016-10-12 15:25:00 +02:00
|
|
|
"""
|
|
|
|
A map feature
|
|
|
|
"""
|
|
|
|
geometry = GeometryField()
|
|
|
|
|
2016-11-28 16:38:22 +01:00
|
|
|
geomtype = None
|
|
|
|
|
2016-10-12 15:25:00 +02:00
|
|
|
class Meta:
|
|
|
|
abstract = True
|
|
|
|
|
2016-11-27 14:03:39 +01:00
|
|
|
def get_geojson_properties(self):
|
|
|
|
return OrderedDict((
|
|
|
|
('type', self.__class__.__name__.lower()),
|
|
|
|
('name', self.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
|
|
|
|
2016-12-08 12:36:09 +01:00
|
|
|
def get_shadow_geojson(self):
|
|
|
|
return None
|
|
|
|
|
2016-12-21 00:24:56 +01:00
|
|
|
def contains(self, x, y):
|
|
|
|
return self.geometry.contains(Point(x, y))
|
|
|
|
|
2016-10-12 15:25:00 +02:00
|
|
|
|
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
|
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
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'
|
|
|
|
|
|
|
|
|
2017-05-04 12:08:17 +02:00
|
|
|
class Area(GeometryMapItemWithLevel):
|
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-01 18:10:46 +02:00
|
|
|
|
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
|
|
|
|
2017-05-01 17:35:41 +02:00
|
|
|
public = models.BooleanField(verbose_name=_('public'))
|
2017-05-04 12:10:41 +02:00
|
|
|
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
|
|
|
|
2017-05-01 18:10:46 +02:00
|
|
|
def get_geojson_properties(self):
|
|
|
|
result = super().get_geojson_properties()
|
|
|
|
result['public'] = self.public
|
|
|
|
return result
|
|
|
|
|
2016-12-04 01:49:49 +01:00
|
|
|
|
2016-12-27 00:52:22 +01:00
|
|
|
class StuffedArea(GeometryMapItemWithLevel):
|
|
|
|
"""
|
|
|
|
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'
|
|
|
|
|
|
|
|
|
2016-12-18 01:15:25 +01:00
|
|
|
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'
|
|
|
|
|
|
|
|
def get_geojson_properties(self):
|
|
|
|
result = super().get_geojson_properties()
|
|
|
|
result['direction'] = 'up' if self.direction else 'down'
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
2017-05-04 12:28:17 +02:00
|
|
|
class Stair(GeometryMapItemWithLevel):
|
2016-12-08 18:12:07 +01:00
|
|
|
"""
|
|
|
|
A stair
|
|
|
|
"""
|
2017-05-04 12:28:17 +02:00
|
|
|
geomtype = 'polyline'
|
|
|
|
|
2016-12-08 18:12:07 +01:00
|
|
|
class Meta:
|
|
|
|
verbose_name = _('Stair')
|
|
|
|
verbose_name_plural = _('Stairs')
|
|
|
|
default_related_name = 'stairs'
|
|
|
|
|
2017-05-04 12:28:17 +02:00
|
|
|
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),
|
|
|
|
('level', self.level.name),
|
|
|
|
))),
|
|
|
|
('geometry', format_geojson(mapping(shadow), round=False)),
|
|
|
|
))
|
|
|
|
|
2016-12-08 18:12:07 +01:00
|
|
|
|
2016-12-09 14:49:20 +01:00
|
|
|
class Obstacle(GeometryMapItemWithLevel):
|
2016-12-08 22:23:42 +01:00
|
|
|
"""
|
2016-12-09 14:49:20 +01:00
|
|
|
An obstacle
|
2016-12-08 22:23:42 +01:00
|
|
|
"""
|
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-08 22:23:42 +01:00
|
|
|
|
2016-12-09 14:49:20 +01:00
|
|
|
geomtype = 'polygon'
|
2016-12-08 22:23:42 +01:00
|
|
|
|
|
|
|
class Meta:
|
2016-12-09 14:49:20 +01:00
|
|
|
verbose_name = _('Obstacle')
|
|
|
|
verbose_name_plural = _('Obstacles')
|
|
|
|
default_related_name = 'obstacles'
|
2016-12-08 22:23:42 +01:00
|
|
|
|
|
|
|
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
|
2016-12-08 22:23:42 +01:00
|
|
|
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
|
|
|
|
|
|
|
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
|
|
|
|
|
|
|
|
|
|
|
|
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)
|
2016-12-26 01:08:45 +01:00
|
|
|
override_altitude = models.DecimalField(_('override level altitude'),
|
|
|
|
blank=True, null=True, max_digits=6, decimal_places=2)
|
2017-05-01 17:35:41 +02:00
|
|
|
public = models.BooleanField(verbose_name=_('public'))
|
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()
|
2017-05-01 18:10:46 +02:00
|
|
|
result['public'] = self.public
|
2016-11-28 20:56:52 +01:00
|
|
|
result['elevator'] = self.elevator.name
|
|
|
|
result['button'] = self.button
|
|
|
|
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
|