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

369 lines
12 KiB
Python
Raw Normal View History

from decimal import Decimal
2017-12-16 12:43:14 +01:00
from django.conf import settings
2017-12-20 18:12:27 +01:00
from django.core.exceptions import ObjectDoesNotExist
from django.core.validators import MinValueValidator
from django.db import models
2017-11-02 13:35:58 +01:00
from django.urls import reverse
2017-10-28 21:36:52 +02:00
from django.utils.functional import cached_property
from django.utils.text import format_lazy
from django.utils.translation import ugettext_lazy as _
2017-05-07 12:06:13 +02:00
from shapely.geometry import CAP_STYLE, JOIN_STYLE, mapping
from c3nav.mapdata.fields import GeometryField, I18nField, JSONField
from c3nav.mapdata.models import Space
from c3nav.mapdata.models.base import SerializableMixin
2017-05-08 16:40:22 +02:00
from c3nav.mapdata.models.geometry.base import GeometryMixin
2017-05-10 18:03:57 +02:00
from c3nav.mapdata.models.locations import SpecificLocation
from c3nav.mapdata.utils.cache.changes import changed_geometries
2016-12-07 16:11:33 +01:00
from c3nav.mapdata.utils.json import format_geojson
2016-12-01 12:25:02 +01:00
2017-05-08 16:40:22 +02:00
class SpaceGeometryMixin(GeometryMixin):
space = models.ForeignKey('mapdata.Space', on_delete=models.CASCADE, verbose_name=_('space'))
class Meta:
abstract = True
2017-10-28 21:36:52 +02:00
@cached_property
def level_id(self):
2017-12-20 18:12:27 +01:00
try:
return self.space.level_id
except ObjectDoesNotExist:
return None
2017-05-04 15:44:32 +02:00
def get_geojson_properties(self, *args, instance=None, **kwargs) -> dict:
result = super().get_geojson_properties(*args, **kwargs)
if hasattr(self, 'get_color'):
color = self.get_color(instance=instance)
if color:
result['color'] = color
2017-12-14 22:00:22 +01:00
if hasattr(self, 'opacity'):
result['opacity'] = self.opacity
return result
@property
def subtitle(self):
base_subtitle = super().subtitle
2017-11-27 15:39:42 +01:00
space = getattr(self, '_space_cache', None)
if space is not None:
2017-11-27 15:39:42 +01:00
level = getattr(space, '_level_cache', None)
if level is not None:
return format_lazy(_('{category}, {space}, {level}'),
category=base_subtitle,
space=space.title,
level=level.title)
return format_lazy(_('{category}, {space}'),
category=base_subtitle,
level=space.title)
return base_subtitle
@classmethod
def q_for_request(cls, request, prefix='', allow_none=False):
return (
super().q_for_request(request, prefix=prefix, allow_none=allow_none) &
Space.q_for_request(request, prefix=prefix + 'space__', allow_none=allow_none)
)
def register_change(self, force=False):
space = self.space
force = force or self.all_geometry_changed
if force or self.geometry_changed:
changed_geometries.register(space.level_id, space.geometry.intersection(
self.geometry if force else self.get_changed_geometry()
))
2017-11-02 13:35:58 +01:00
def details_display(self):
result = super().details_display()
result['display'].insert(3, (
_('Space'),
{
'id': self.space_id,
'slug': self.space.get_slug(),
'title': self.space.title,
'can_search': self.space.can_search,
},
))
2017-12-23 03:57:54 +01:00
result['level'] = self.level_id
2017-11-02 13:35:58 +01:00
return result
def register_delete(self):
space = self.space
changed_geometries.register(space.level_id, space.geometry.intersection(self.geometry))
def save(self, *args, **kwargs):
self.register_change()
super().save(*args, **kwargs)
2017-05-04 15:44:32 +02:00
2017-06-09 15:22:30 +02:00
class Column(SpaceGeometryMixin, models.Model):
"""
An column in a space, also used to be able to create rooms within rooms.
"""
geometry = GeometryField('polygon')
class Meta:
verbose_name = _('Column')
verbose_name_plural = _('Columns')
default_related_name = 'columns'
2017-10-10 17:49:53 +02:00
class Area(SpaceGeometryMixin, SpecificLocation, models.Model):
"""
An area in a space.
"""
geometry = GeometryField('polygon')
2017-12-22 15:16:28 +01:00
slow_down_factor = models.DecimalField(_('slow down factor'), max_digits=6, decimal_places=2, default=1,
validators=[MinValueValidator(Decimal('0.01'))])
class Meta:
verbose_name = _('Area')
verbose_name_plural = _('Areas')
2017-05-10 15:27:37 +02:00
default_related_name = 'areas'
2017-05-11 19:36:49 +02:00
def _serialize(self, **kwargs):
result = super()._serialize(**kwargs)
return result
2017-11-02 13:35:58 +01:00
def details_display(self):
result = super().details_display()
result['editor_url'] = reverse('editor.areas.edit', kwargs={'space': self.space_id, 'pk': self.pk})
return result
2017-05-08 16:40:22 +02:00
class Stair(SpaceGeometryMixin, models.Model):
2016-12-08 18:12:07 +01:00
"""
A stair
"""
geometry = GeometryField('linestring')
2016-12-08 18:12:07 +01:00
class Meta:
verbose_name = _('Stair')
verbose_name_plural = _('Stairs')
default_related_name = 'stairs'
2017-11-17 20:31:29 +01:00
class Ramp(SpaceGeometryMixin, models.Model):
"""
A ramp
"""
geometry = GeometryField('polygon')
class Meta:
verbose_name = _('Ramp')
verbose_name_plural = _('Ramps')
default_related_name = 'ramps'
2017-05-08 16:40:22 +02:00
class Obstacle(SpaceGeometryMixin, models.Model):
"""
2016-12-09 14:49:20 +01:00
An obstacle
"""
geometry = GeometryField('polygon')
height = models.DecimalField(_('height'), max_digits=6, decimal_places=2, default=0.8,
validators=[MinValueValidator(Decimal('0'))])
class Meta:
2016-12-09 14:49:20 +01:00
verbose_name = _('Obstacle')
verbose_name_plural = _('Obstacles')
default_related_name = 'obstacles'
2017-08-07 15:29:52 +02:00
def _serialize(self, geometry=True, **kwargs):
result = super()._serialize(geometry=geometry, **kwargs)
result['height'] = float(str(self.height))
return result
2017-05-08 16:40:22 +02:00
class LineObstacle(SpaceGeometryMixin, models.Model):
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
"""
geometry = GeometryField('linestring')
2017-08-07 15:29:52 +02:00
width = models.DecimalField(_('width'), max_digits=4, decimal_places=2, default=0.15)
height = models.DecimalField(_('height'), max_digits=6, decimal_places=2, default=0.8,
validators=[MinValueValidator(Decimal('0'))])
2016-12-04 20:01:37 +01:00
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-11 19:36:49 +02:00
def serialize(self, geometry=True, **kwargs):
result = super().serialize(geometry=geometry, **kwargs)
if geometry:
result.move_to_end('buffered_geometry')
return result
def _serialize(self, geometry=True, **kwargs):
result = super()._serialize(geometry=geometry, **kwargs)
result['width'] = float(str(self.width))
2017-08-07 15:29:52 +02:00
result['height'] = float(str(self.height))
2017-05-11 19:36:49 +02:00
if geometry:
result['buffered_geometry'] = format_geojson(mapping(self.buffered_geometry))
return result
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)
def to_geojson(self, *args, **kwargs):
result = super().to_geojson(*args, **kwargs)
2017-05-21 23:34:09 +02:00
result['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
return result
2016-12-01 12:25:02 +01:00
2017-05-10 15:30:54 +02:00
2017-10-10 17:49:53 +02:00
class POI(SpaceGeometryMixin, SpecificLocation, models.Model):
2017-05-10 15:30:54 +02:00
"""
2017-07-08 16:29:12 +02:00
An point of interest
2017-05-10 15:30:54 +02:00
"""
geometry = GeometryField('point')
class Meta:
2017-07-08 16:29:12 +02:00
verbose_name = _('Point of Interest')
verbose_name_plural = _('Points of Interest')
default_related_name = 'pois'
2017-11-02 13:35:58 +01:00
def details_display(self):
result = super().details_display()
result['editor_url'] = reverse('editor.pois.edit', kwargs={'space': self.space_id, 'pk': self.pk})
return result
2017-12-24 20:39:13 +01:00
@property
def x(self):
return self.geometry.x
@property
def y(self):
return self.geometry.y
2017-06-08 15:19:12 +02:00
class Hole(SpaceGeometryMixin, models.Model):
"""
A hole in the ground of a space, e.g. for stairs.
"""
geometry = GeometryField('polygon')
class Meta:
verbose_name = _('Hole')
verbose_name_plural = _('Holes')
default_related_name = 'holes'
2017-08-05 11:56:29 +02:00
class AltitudeMarker(SpaceGeometryMixin, models.Model):
"""
An altitude marker
"""
geometry = GeometryField('point')
altitude = models.DecimalField(_('altitude'), null=False, max_digits=6, decimal_places=2)
class Meta:
verbose_name = _('Altitude Marker')
verbose_name_plural = _('Altitude Markers')
default_related_name = 'altitudemarkers'
@property
def title(self):
return '%s (%sm)' % (super().title, self.altitude)
2017-12-16 12:43:14 +01:00
class LeaveDescription(SerializableMixin):
"""
A description for leaving a space to another space
"""
space = models.ForeignKey('mapdata.Space', on_delete=models.CASCADE, verbose_name=_('space'))
target_space = models.ForeignKey('mapdata.Space', on_delete=models.CASCADE, verbose_name=_('target space'),
related_name='enter_descriptions')
description = I18nField(_('description'))
class Meta:
verbose_name = _('Leave description')
verbose_name_plural = _('Leave descriptions')
default_related_name = 'leave_descriptions'
unique_together = (
('space', 'target_space')
)
def _serialize(self, **kwargs):
result = super()._serialize(**kwargs)
result['space'] = self.space_id
result['target_space'] = self.target_space_id
result['description_i18n'] = self.description_i18n
result['description'] = self.description
return result
2017-12-19 01:50:27 +01:00
@cached_property
def title(self):
return self.target_space.title
@classmethod
def q_for_request(cls, request, prefix='', allow_none=False):
return (
Space.q_for_request(request, prefix='space__', allow_none=allow_none) &
Space.q_for_request(request, prefix='target_space__', allow_none=allow_none)
)
class CrossDescription(SerializableMixin):
"""
A description for crossing a space from one space to another space
"""
space = models.ForeignKey('mapdata.Space', on_delete=models.CASCADE, verbose_name=_('space'))
origin_space = models.ForeignKey('mapdata.Space', on_delete=models.CASCADE, verbose_name=_('origin space'),
related_name='leave_cross_descriptions')
target_space = models.ForeignKey('mapdata.Space', on_delete=models.CASCADE, verbose_name=_('target space'),
related_name='cross_enter_descriptions')
description = I18nField(_('description'))
class Meta:
verbose_name = _('Cross description')
verbose_name_plural = _('Cross descriptions')
default_related_name = 'cross_descriptions'
unique_together = (
('space', 'origin_space', 'target_space')
)
def _serialize(self, **kwargs):
result = super()._serialize(**kwargs)
result['space'] = self.space_id
result['origin_space'] = self.origin_space_id
result['target_space'] = self.target_space_id
result['description_i18n'] = self.description_i18n
result['description'] = self.description
return result
2017-12-19 01:50:27 +01:00
@cached_property
def title(self):
return '%s%s' % (self.origin_space.title, self.target_space.title)
@classmethod
def q_for_request(cls, request, prefix='', allow_none=False):
return (
Space.q_for_request(request, prefix='space__', allow_none=allow_none) &
Space.q_for_request(request, prefix='origin_space__', allow_none=allow_none) &
Space.q_for_request(request, prefix='target_space__', allow_none=allow_none)
)
2017-12-16 12:43:14 +01:00
class WifiMeasurement(SpaceGeometryMixin, models.Model):
"""
A Wi-Fi measurement
"""
geometry = GeometryField('point')
author = models.ForeignKey(settings.AUTH_USER_MODEL, on_delete=models.PROTECT, verbose_name=_('author'))
comment = models.TextField(null=True, blank=True, verbose_name=_('comment'))
data = JSONField(_('Measurement list'))
class Meta:
verbose_name = _('Wi-Fi Measurement')
verbose_name_plural = _('Wi-Fi Measurements')
default_related_name = 'wifi_measurements'
@property
def all_geometry_changed(self):
return False
@property
def geometry_changed(self):
return False