2016-11-27 14:03:39 +01:00
|
|
|
from collections import OrderedDict
|
|
|
|
|
2016-12-06 22:48:07 +01:00
|
|
|
from django.conf import settings
|
2016-10-12 15:25:00 +02:00
|
|
|
from django.db import models
|
|
|
|
from django.utils.translation import ugettext_lazy as _
|
2016-10-13 13:55:02 +02:00
|
|
|
from shapely.geometry.geo import mapping, shape
|
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-10-12 15:25:00 +02:00
|
|
|
from c3nav.mapdata.utils 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-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-12-06 22:48:07 +01:00
|
|
|
cached_geojson = {}
|
2016-10-12 15:25:00 +02:00
|
|
|
|
2016-11-28 16:38:22 +01:00
|
|
|
geomtype = None
|
|
|
|
|
2016-10-12 15:25:00 +02:00
|
|
|
class Meta:
|
|
|
|
abstract = True
|
|
|
|
|
2016-10-13 15:55:15 +02:00
|
|
|
@property
|
|
|
|
def title(self):
|
|
|
|
return self.name
|
|
|
|
|
2016-10-12 15:25:00 +02:00
|
|
|
@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
|
|
|
@classmethod
|
|
|
|
def get_styles(cls):
|
|
|
|
return {
|
|
|
|
cls.__name__.lower(): cls.color
|
|
|
|
}
|
|
|
|
|
|
|
|
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-06 22:48:07 +01:00
|
|
|
if settings.DIRECT_EDITING:
|
|
|
|
return self._to_geojson()
|
|
|
|
key = (self.__class__, self.name)
|
|
|
|
if key not in self.cached_geojson:
|
|
|
|
self.cached_geojson[key] = self._to_geojson()
|
|
|
|
return self.cached_geojson[key]
|
|
|
|
|
|
|
|
def _to_geojson(self):
|
2016-11-27 14:03:39 +01:00
|
|
|
return [OrderedDict((
|
|
|
|
('type', 'Feature'),
|
|
|
|
('properties', self.get_geojson_properties()),
|
|
|
|
('geometry', format_geojson(mapping(self.geometry), round=False)),
|
|
|
|
))]
|
|
|
|
|
2016-10-12 15:25:00 +02:00
|
|
|
def tofile(self):
|
|
|
|
result = super().tofile()
|
|
|
|
result['geometry'] = format_geojson(mapping(self.geometry))
|
|
|
|
return result
|
|
|
|
|
|
|
|
|
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):
|
|
|
|
result = super().tofile()
|
|
|
|
result['level'] = self.level.name
|
|
|
|
result.move_to_end('geometry')
|
|
|
|
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'
|
|
|
|
|
|
|
|
|
|
|
|
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'
|
|
|
|
|
|
|
|
|
2016-12-04 20:01:37 +01:00
|
|
|
class Obstacle(GeometryMapItemWithLevel):
|
2016-12-01 12:25:02 +01:00
|
|
|
"""
|
2016-12-04 20:01:37 +01:00
|
|
|
An obstacle
|
2016-12-01 12:25:02 +01:00
|
|
|
"""
|
2016-12-04 20:01:37 +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-01 12:25:02 +01:00
|
|
|
geomtype = 'polygon'
|
|
|
|
|
|
|
|
class Meta:
|
2016-12-04 20:01:37 +01:00
|
|
|
verbose_name = _('Obstacle')
|
|
|
|
verbose_name_plural = _('Obstacles')
|
|
|
|
default_related_name = 'obstacles'
|
2016-12-01 12:25:02 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def fromfile(cls, data, file_path):
|
|
|
|
kwargs = super().fromfile(data, file_path)
|
|
|
|
|
2016-12-04 20:01:37 +01:00
|
|
|
if 'crop_to_level' in data:
|
|
|
|
kwargs['crop_to_level'] = data['crop_to_level']
|
2016-12-01 12:25:02 +01:00
|
|
|
|
|
|
|
return kwargs
|
|
|
|
|
|
|
|
def get_geojson_properties(self):
|
|
|
|
result = super().get_geojson_properties()
|
2016-12-04 20:01:37 +01:00
|
|
|
if self.crop_to_level is not None:
|
|
|
|
result['crop_to_level'] = self.crop_to_level.name
|
2016-12-01 12:25:02 +01:00
|
|
|
return result
|
|
|
|
|
|
|
|
def tofile(self):
|
|
|
|
result = super().tofile()
|
2016-12-04 20:01:37 +01:00
|
|
|
if self.crop_to_level is not None:
|
|
|
|
result['crop_to_level'] = self.crop_to_level.name
|
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
|
|
|
|
|
2016-12-04 11:43:12 +01:00
|
|
|
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))
|
2016-12-04 11:43:12 +01:00
|
|
|
return result
|
|
|
|
|
2016-10-16 13:20:34 +02:00
|
|
|
def tofile(self):
|
|
|
|
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)
|
|
|
|
|
|
|
|
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']
|
|
|
|
|
|
|
|
return kwargs
|
|
|
|
|
|
|
|
def tofile(self):
|
|
|
|
result = super().tofile()
|
|
|
|
result['elevator'] = self.elevator.name
|
|
|
|
result['button'] = self.button
|
|
|
|
return result
|