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

501 lines
19 KiB
Python
Raw Normal View History

2024-02-07 18:34:28 +01:00
import typing
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
2023-11-10 18:59:37 +01:00
from django.core.validators import MinValueValidator, RegexValidator
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 gettext_lazy as _
2017-05-07 12:06:13 +02:00
from shapely.geometry import CAP_STYLE, JOIN_STYLE, mapping
2022-04-03 19:40:12 +02:00
from c3nav.mapdata.fields import GeometryField, I18nField
2018-12-10 19:06:38 +01:00
from c3nav.mapdata.grid import grid
2024-09-06 15:07:54 +02:00
from c3nav.mapdata.models import Space, Level
from c3nav.mapdata.models.access import AccessRestrictionMixin
2024-01-06 13:26:34 +01:00
from c3nav.mapdata.models.base import SerializableMixin, TitledMixin
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
2023-12-01 01:56:23 +01:00
from c3nav.mapdata.utils.geometry import unwrap_geom
2016-12-07 16:11:33 +01:00
from c3nav.mapdata.utils.json import format_geojson
2024-02-07 18:34:28 +01:00
if typing.TYPE_CHECKING:
from c3nav.mapdata.render.theme import ThemeColorManager
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, **kwargs) -> dict:
result = super().get_geojson_properties(*args, **kwargs)
if hasattr(self, 'get_color'):
2024-01-06 13:26:34 +01:00
from c3nav.mapdata.render.theme import ColorManager
color = self.get_color(ColorManager.for_theme(None))
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
2024-12-05 22:06:27 +01:00
if force or self._state.adding or self.all_geometry_changed or self.geometry_changed:
changed_geometries.register(space.level_id, space.geometry.intersection(
2024-12-05 22:45:16 +01:00
unwrap_geom(self.geometry if force or self._state.adding else self.get_changed_geometry())
))
2018-09-19 19:08:47 +02:00
def details_display(self, **kwargs):
result = super().details_display(**kwargs)
result['display'].insert(3, (
_('Space'),
{
'id': self.space_id,
2024-12-03 16:07:07 +01:00
'slug': self.space.effective_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
2023-12-01 01:56:23 +01:00
changed_geometries.register(space.level_id, space.geometry.intersection(unwrap_geom(self.geometry)))
def save(self, *args, **kwargs):
self.register_change()
super().save(*args, **kwargs)
2017-05-04 15:44:32 +02:00
class Column(SpaceGeometryMixin, AccessRestrictionMixin, models.Model):
2017-06-09 15:22:30 +02:00
"""
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,
2018-12-23 17:44:47 +01:00
validators=[MinValueValidator(Decimal('0.01'))],
help_text=_('values of overlapping areas get multiplied!'))
2023-12-20 00:13:36 +01:00
main_point = GeometryField('point', null=True, blank=True,
2023-12-19 15:44:00 +01:00
help_text=_('main routing point (optional)'))
class Meta:
verbose_name = _('Area')
verbose_name_plural = _('Areas')
2017-05-10 15:27:37 +02:00
default_related_name = 'areas'
@property
def grid_square(self):
if "geometry" in self.get_deferred_fields():
return None
return grid.get_squares_for_bounds(self.geometry.bounds) or ''
2018-09-19 19:08:47 +02:00
def details_display(self, editor_url=True, **kwargs):
result = super().details_display(**kwargs)
if editor_url:
result['editor_url'] = reverse('editor.areas.edit', kwargs={'space': self.space_id, 'pk': self.pk})
2017-11-02 13:35:58 +01:00
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'
2024-12-04 15:53:00 +01:00
# todo: move to other file? this is NOT a geometry!
2024-01-06 13:26:34 +01:00
class ObstacleGroup(TitledMixin, models.Model):
color = models.CharField(max_length=32, null=True, blank=True)
2024-09-06 15:07:54 +02:00
in_legend = models.BooleanField(default=False, verbose_name=_('show in legend (if color set)'))
2024-01-06 13:26:34 +01:00
class Meta:
verbose_name = _('Obstacle Group')
verbose_name_plural = _('Obstacle Groups')
default_related_name = 'groups'
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self._orig = {"color": self.color}
2024-01-06 13:26:34 +01:00
def save(self, *args, **kwargs):
2024-12-05 20:54:19 +01:00
if not self._state.adding and any(getattr(self, attname) != value for attname, value in self._orig.items()):
2024-01-06 13:26:34 +01:00
self.register_changed_geometries()
super().save(*args, **kwargs)
def delete(self, *args, **kwargs):
self.register_changed_geometries()
super().delete(*args, **kwargs)
def register_changed_geometries(self):
2024-01-06 13:26:34 +01:00
for obj in self.obstacles.select_related('space'):
obj.register_change(force=True)
for obj in self.lineobstacles.select_related('space'):
obj.register_change(force=True)
2024-01-06 13:26:34 +01:00
2017-05-08 16:40:22 +02:00
class Obstacle(SpaceGeometryMixin, models.Model):
"""
2016-12-09 14:49:20 +01:00
An obstacle
"""
2024-01-06 13:26:34 +01:00
group = models.ForeignKey(ObstacleGroup, null=True, blank=True, on_delete=models.SET_NULL)
geometry = GeometryField('polygon')
height = models.DecimalField(_('height'), max_digits=6, decimal_places=2, default=0.8,
validators=[MinValueValidator(Decimal('0'))])
2019-12-22 20:57:32 +01:00
altitude = models.DecimalField(_('altitude above ground'), max_digits=6, decimal_places=2, default=0,
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'
2019-12-22 21:21:53 +01:00
ordering = ('altitude', 'height')
def get_geojson_properties(self, *args, **kwargs) -> dict:
result = super().get_geojson_properties(*args, **kwargs)
2024-01-06 13:26:34 +01:00
from c3nav.mapdata.render.theme import ColorManager
color = self.get_color(ColorManager.for_theme(None))
2024-01-06 13:26:34 +01:00
if color:
result['color'] = color
return result
@property
def color(self):
2024-01-06 13:26:34 +01:00
from c3nav.mapdata.render.theme import ColorManager
return self.get_color(ColorManager.for_theme(None))
2017-08-07 15:29:52 +02:00
def get_color(self, color_manager: 'ThemeColorManager'):
2024-02-07 18:34:28 +01:00
return (
color_manager.obstaclegroup_fill_color(self.group)
if self.group is not None
2024-02-07 18:34:28 +01:00
else color_manager.obstacles_default_fill
)
2024-01-06 13:26:34 +01:00
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
"""
2024-01-06 13:26:34 +01:00
group = models.ForeignKey(ObstacleGroup, null=True, blank=True, on_delete=models.SET_NULL)
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'))])
2019-12-22 20:57:32 +01:00
altitude = models.DecimalField(_('altitude above ground'), max_digits=6, decimal_places=2, default=0,
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'
2019-12-22 21:21:53 +01:00
ordering = ('altitude', 'height')
2016-12-09 14:49:20 +01:00
def get_geojson_properties(self, *args, **kwargs) -> dict:
result = super().get_geojson_properties(*args, **kwargs)
2024-01-06 13:26:34 +01:00
from c3nav.mapdata.render.theme import ColorManager
color = self.get_color(ColorManager.for_theme(None))
2024-01-06 13:26:34 +01:00
if color:
result['color'] = color
return result
@property
def color(self):
from c3nav.mapdata.render.theme import ColorManager
return self.get_color(ColorManager.for_theme(None))
def get_color(self, color_manager: 'ThemeColorManager'):
2024-01-06 13:26:34 +01:00
# TODO: should line obstacles use border color?
2024-02-07 18:34:28 +01:00
return (
color_manager.obstaclegroup_fill_color(self.group)
if self.group is not None
2024-02-07 18:34:28 +01:00
else color_manager.obstacles_default_fill
)
2024-01-06 13:26:34 +01:00
2017-05-04 15:44:32 +02:00
@property
def buffered_geometry(self):
return self.geometry.buffer(float(self.width / 2), join_style=JOIN_STYLE.mitre, cap_style=CAP_STYLE.flat)
2017-05-04 15:44:32 +02:00
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
"""
2024-12-03 18:42:33 +01:00
A 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'
2018-09-19 19:08:47 +02:00
def details_display(self, editor_url=True, **kwargs):
result = super().details_display(**kwargs)
if editor_url:
result['editor_url'] = reverse('editor.pois.edit', kwargs={'space': self.space_id, 'pk': self.pk})
2017-11-02 13:35:58 +01:00
return result
2018-12-10 19:06:38 +01:00
@property
def grid_square(self):
if "geometry" in self.get_deferred_fields():
return None
return grid.get_square_for_point(self.x, self.y) or ''
2018-12-10 19:06:38 +01:00
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')
groundaltitude = models.ForeignKey('mapdata.GroundAltitude', on_delete=models.CASCADE,
verbose_name=_('altitude'))
2017-08-05 11:56:29 +02:00
class Meta:
verbose_name = _('Altitude Marker')
verbose_name_plural = _('Altitude Markers')
default_related_name = 'altitudemarkers'
@property
def altitude(self) -> Decimal:
return self.groundaltitude.altitude
@property
def title(self):
return f'#{self.pk}: {self.groundaltitude.title}'
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'), plural_name='descriptions')
class Meta:
verbose_name = _('Leave description')
verbose_name_plural = _('Leave descriptions')
default_related_name = 'leave_descriptions'
unique_together = (
('space', 'target_space')
)
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'), plural_name='descriptions')
class Meta:
verbose_name = _('Cross description')
verbose_name_plural = _('Cross descriptions')
default_related_name = 'cross_descriptions'
unique_together = (
('space', 'origin_space', 'target_space')
)
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)
)
class BeaconMeasurement(SpaceGeometryMixin, models.Model):
2017-12-16 12:43:14 +01:00
"""
A Beacon (WiFI / iBeacon) measurement
2017-12-16 12:43:14 +01:00
"""
geometry = GeometryField('point')
author = models.ForeignKey(settings.AUTH_USER_MODEL, on_delete=models.SET_NULL, null=True, blank=True,
verbose_name=_('author'))
2017-12-16 12:43:14 +01:00
comment = models.TextField(null=True, blank=True, verbose_name=_('comment'))
2024-08-17 17:16:01 +02:00
data = models.JSONField(_('Measurement list'), default=dict) # todo: would be nice if this used pydantic
2017-12-16 12:43:14 +01:00
class Meta:
verbose_name = _('Beacon Measurement')
verbose_name_plural = _('Beacon Measurements')
default_related_name = 'beacon_measurements'
@property
def all_geometry_changed(self):
return False
@property
def geometry_changed(self):
return False
2023-11-10 18:59:37 +01:00
class RangingBeacon(SpaceGeometryMixin, models.Model):
"""
A ranging beacon
"""
geometry = GeometryField('point')
node_number = models.PositiveSmallIntegerField(_('Node Number'), unique=True, null=True, blank=True)
wifi_bssid = models.CharField(_('WiFi BSSID'), unique=True, null=True, blank=True,
max_length=17,
validators=[RegexValidator(
regex='^([a-f0-9]{2}:){5}[a-f0-9]{2}$',
message='Must be a lower-case bssid',
code='invalid_bssid'
)],
help_text=_("uses node's value if not set"))
bluetooth_address = models.CharField(_('Bluetooth Address'), unique=True, null=True, blank=True,
max_length=17,
validators=[RegexValidator(
regex='^([a-f0-9]{2}:){5}[a-f0-9]{2}$',
message='Must be a lower-case mac address',
code='invalid_bluetooth_address'
)],
help_text=_("uses node's value if not set"))
ibeacon_uuid = models.UUIDField(_('iBeacon UUID'), null=True, blank=True,
help_text=_("uses node's value if not set"))
ibeacon_major = models.PositiveIntegerField(_('iBeacon major value'), null=True, blank=True,
help_text=_("uses node's value if not set"))
ibeacon_minor = models.PositiveIntegerField(_('iBeacon minor value'), null=True, blank=True,
help_text=_("uses node's value if not set"))
uwb_address = models.CharField(_('UWB Address'), unique=True, null=True, blank=True,
max_length=23,
validators=[RegexValidator(
regex='^([a-f0-9]{2}:){7}[a-f0-9]{2}$',
message='Must be a lower-case 8-byte UWB address',
code='invalid_uwb_address'
)],
help_text=_("uses node's value if not set"))
2023-11-10 18:59:37 +01:00
altitude = models.DecimalField(_('altitude above ground'), max_digits=6, decimal_places=2, default=0,
validators=[MinValueValidator(Decimal('0'))])
comment = models.TextField(null=True, blank=True, verbose_name=_('comment'))
class Meta:
verbose_name = _('Ranging beacon')
verbose_name_plural = _('Ranging beacons')
default_related_name = 'ranging_beacons'
@property
def all_geometry_changed(self):
return False
@property
def geometry_changed(self):
return False
@property
def title(self):
if self.comment:
return f'{self.node_number} {self.wifi_bssid} ({self.comment})'.strip()
return f'{self.node_number} {self.wifi_bssid}'.strip()