add crossdescription quests
This commit is contained in:
parent
c0a412297d
commit
9c58d73036
1 changed files with 123 additions and 26 deletions
|
@ -1,16 +1,17 @@
|
||||||
|
import operator
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from operator import attrgetter
|
from functools import reduce
|
||||||
from typing import ClassVar
|
from typing import ClassVar, Optional
|
||||||
|
|
||||||
from django.core.exceptions import ValidationError
|
from django.core.exceptions import ValidationError
|
||||||
from django.db.models import F
|
from django.db.models import F, Q
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
from shapely import Point
|
from shapely import Point, LineString
|
||||||
from shapely.geometry import mapping
|
from shapely.geometry import mapping
|
||||||
|
|
||||||
from c3nav.mapdata.forms import I18nModelFormMixin
|
from c3nav.mapdata.forms import I18nModelFormMixin
|
||||||
from c3nav.mapdata.models import GraphEdge, Space
|
from c3nav.mapdata.models import GraphEdge, Space
|
||||||
from c3nav.mapdata.models.geometry.space import RangingBeacon, LeaveDescription
|
from c3nav.mapdata.models.geometry.space import RangingBeacon, LeaveDescription, CrossDescription
|
||||||
from c3nav.mapdata.quests.base import register_quest, Quest, ChangeSetModelForm
|
from c3nav.mapdata.quests.base import register_quest, Quest, ChangeSetModelForm
|
||||||
from c3nav.mapdata.utils.geometry import unwrap_geom
|
from c3nav.mapdata.utils.geometry import unwrap_geom
|
||||||
|
|
||||||
|
@ -60,6 +61,31 @@ class RangingBeaconAltitudeQuest(Quest):
|
||||||
'space__level').filter(altitude_quest=True)
|
'space__level').filter(altitude_quest=True)
|
||||||
|
|
||||||
|
|
||||||
|
def get_door_edges_for_request(request, space_ids: Optional[list[int]] = None):
|
||||||
|
qs = Space.qs_for_request(request)
|
||||||
|
if space_ids:
|
||||||
|
qs = qs.filter(pk__in=space_ids)
|
||||||
|
spaces = {space.pk: space for space in qs.select_related("level")}
|
||||||
|
existing = set(tuple(item) for item in LeaveDescription.objects.values_list("space_id", "target_space_id"))
|
||||||
|
|
||||||
|
|
||||||
|
qs = GraphEdge.objects.filter(
|
||||||
|
from_node__space__in=spaces,
|
||||||
|
to_node__space__in=spaces,
|
||||||
|
waytype=None,
|
||||||
|
)
|
||||||
|
if space_ids:
|
||||||
|
qs = qs.filter(reduce(operator.or_, (Q(from_node__space_id=space_ids[i], to_node__space_id=space_ids[i+1])
|
||||||
|
for i in range(len(space_ids) - 1))))
|
||||||
|
return spaces, {
|
||||||
|
(from_space, to_space): (from_point, to_point)
|
||||||
|
for from_space, to_space, from_point, to_point in qs.exclude(
|
||||||
|
from_node__space=F("to_node__space")
|
||||||
|
).values_list("from_node__space_id", "to_node__space_id", "from_node__geometry", "to_node__geometry")
|
||||||
|
if (from_space, to_space) not in existing
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class LeaveDescriptionQuestForm(I18nModelFormMixin, ChangeSetModelForm):
|
class LeaveDescriptionQuestForm(I18nModelFormMixin, ChangeSetModelForm):
|
||||||
class Meta:
|
class Meta:
|
||||||
model = LeaveDescription
|
model = LeaveDescription
|
||||||
|
@ -89,8 +115,10 @@ class LeaveDescriptionQuest(Quest):
|
||||||
"from_space": self.space.title,
|
"from_space": self.space.title,
|
||||||
"to_space": self.target_space.title,
|
"to_space": self.target_space.title,
|
||||||
},
|
},
|
||||||
_("This will be used all doors that lead from this space to the other, not just the highlighted one! "
|
_("This will be used all for all connections that lead from this space to the other, not just the "
|
||||||
"So please be generic if there is more then one."),
|
"highlighted one! So, if there is more than one connection between these two rooms, please be generic."),
|
||||||
|
_("The description should make it possible to find the room exit no matter where in the room you are."),
|
||||||
|
_("Examples: “Walk through the red door.”, „Walk through the doors with the Sign “Hall 3” above it.”"),
|
||||||
]
|
]
|
||||||
|
|
||||||
@property
|
@property
|
||||||
|
@ -107,24 +135,7 @@ class LeaveDescriptionQuest(Quest):
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def get_all_for_request(cls, request, space_ids: tuple[int, int] = ()):
|
def get_all_for_request(cls, request, space_ids: tuple[int, int] = ()):
|
||||||
qs = Space.qs_for_request(request)
|
spaces, edges = get_door_edges_for_request(request, space_ids)
|
||||||
if space_ids:
|
|
||||||
qs = qs.filter(pk__in=space_ids)
|
|
||||||
spaces = {space.pk: space for space in qs.select_related("level")}
|
|
||||||
existing = set(tuple(item) for item in LeaveDescription.objects.values_list("space_id", "target_space_id"))
|
|
||||||
more_filter = {} if not space_ids else {"from_node__space_id": space_ids[0], "to_node__space_id": space_ids[1]}
|
|
||||||
edges = {
|
|
||||||
(from_space, to_space): (from_point, to_point)
|
|
||||||
for from_space, to_space, from_point, to_point in GraphEdge.objects.filter(
|
|
||||||
from_node__space__in=spaces,
|
|
||||||
to_node__space__in=spaces,
|
|
||||||
waytype=None,
|
|
||||||
**more_filter,
|
|
||||||
).exclude(
|
|
||||||
from_node__space=F("to_node__space")
|
|
||||||
).values_list("from_node__space_id", "to_node__space_id", "from_node__geometry", "to_node__geometry")
|
|
||||||
if (from_space, to_space) not in existing
|
|
||||||
}
|
|
||||||
return [
|
return [
|
||||||
cls(
|
cls(
|
||||||
space=spaces[from_space],
|
space=spaces[from_space],
|
||||||
|
@ -137,7 +148,92 @@ class LeaveDescriptionQuest(Quest):
|
||||||
@classmethod
|
@classmethod
|
||||||
def get_for_request(cls, request, identifier: str):
|
def get_for_request(cls, request, identifier: str):
|
||||||
space_ids = identifier.split('-')
|
space_ids = identifier.split('-')
|
||||||
if len(space_ids) != 2 or not (space_ids[0].isdigit() and space_ids[1].isdigit()):
|
if len(space_ids) != 2 or not all(i.isdigit() for i in space_ids):
|
||||||
|
return None
|
||||||
|
|
||||||
|
results = cls.get_all_for_request(request, space_ids=[int(i) for i in space_ids])
|
||||||
|
return results[0] if results else None
|
||||||
|
|
||||||
|
def get_form_kwargs(self, request):
|
||||||
|
instance = LeaveDescription()
|
||||||
|
instance.space = self.space
|
||||||
|
instance.target_space = self.target_space
|
||||||
|
return {"instance": instance}
|
||||||
|
|
||||||
|
|
||||||
|
class CrossDescriptionQuestForm(I18nModelFormMixin, ChangeSetModelForm):
|
||||||
|
class Meta:
|
||||||
|
model = CrossDescription
|
||||||
|
fields = ("description", )
|
||||||
|
|
||||||
|
@property
|
||||||
|
def changeset_title(self):
|
||||||
|
return f'CrossDesscription Quest: {self.instance.origin_space.title} → {self.instance.space.title} → {self.instance.target_space.title}'
|
||||||
|
|
||||||
|
|
||||||
|
@register_quest
|
||||||
|
@dataclass
|
||||||
|
class CrossDescriptionQuest(Quest):
|
||||||
|
quest_type = "cross_description"
|
||||||
|
quest_type_label = _('Cross Description')
|
||||||
|
quest_type_icon = "roundabout_right"
|
||||||
|
form_class = CrossDescriptionQuestForm
|
||||||
|
obj: ClassVar
|
||||||
|
space: Space
|
||||||
|
origin_space: Space
|
||||||
|
target_space: Space
|
||||||
|
the_point: Point
|
||||||
|
|
||||||
|
@property
|
||||||
|
def quest_description(self) -> list[str]:
|
||||||
|
return [
|
||||||
|
_("Please provide a description to be used when coming from “%(from_space)s” into “%(space)s” and exiting towardss “%(to_space)s”.") % {
|
||||||
|
"from_space": self.origin_space.title,
|
||||||
|
"space": self.space.title,
|
||||||
|
"to_space": self.target_space.title,
|
||||||
|
},
|
||||||
|
_("This will be used combination of space connections that match this description, not just the "
|
||||||
|
"highlighted ones! So, if there is more than connection between these two rooms, please be generic."),
|
||||||
|
_("This description will replace the entire route descripting when passing through the room this way."),
|
||||||
|
_("Examples: “Go straight ahead into the room right across.” “Turn right and go through the big doors.”"),
|
||||||
|
]
|
||||||
|
|
||||||
|
@property
|
||||||
|
def point(self) -> dict:
|
||||||
|
return mapping(self.the_point)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def level_id(self) -> int:
|
||||||
|
return self.space.level.on_top_of_id or self.space.level_id
|
||||||
|
|
||||||
|
@property
|
||||||
|
def identifier(self) -> str:
|
||||||
|
return f"{self.origin_space.pk}-{self.space.pk}-{self.target_space.pk}"
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def get_all_for_request(cls, request, space_ids: Optional[list[int]] = None):
|
||||||
|
spaces, edges = get_door_edges_for_request(request, space_ids)
|
||||||
|
from_space_conns = {}
|
||||||
|
for (from_space, to_space), (from_point, to_point) in edges.items():
|
||||||
|
from_space_conns.setdefault(from_space, []).append((to_space, from_point))
|
||||||
|
|
||||||
|
results = []
|
||||||
|
for (origin_space, space), (first_point, origin_point) in edges.items():
|
||||||
|
for target_space, target_point in from_space_conns.get(space, ()):
|
||||||
|
line = LineString([origin_point, target_point])
|
||||||
|
the_point = line.interpolate(0.33, normalized=True) if line.length < 3 else line.interpolate(1)
|
||||||
|
results.append(cls(
|
||||||
|
space=spaces[space],
|
||||||
|
origin_space=spaces[origin_space],
|
||||||
|
target_space=spaces[target_space],
|
||||||
|
the_point=the_point,
|
||||||
|
))
|
||||||
|
return results
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def get_for_request(cls, request, identifier: str):
|
||||||
|
space_ids = identifier.split('-')
|
||||||
|
if len(space_ids) != 3 or not (space_ids[0].isdigit() and space_ids[1].isdigit()):
|
||||||
return None
|
return None
|
||||||
|
|
||||||
results = cls.get_all_for_request(request, space_ids=tuple(int(i) for i in space_ids))
|
results = cls.get_all_for_request(request, space_ids=tuple(int(i) for i in space_ids))
|
||||||
|
@ -146,5 +242,6 @@ class LeaveDescriptionQuest(Quest):
|
||||||
def get_form_kwargs(self, request):
|
def get_form_kwargs(self, request):
|
||||||
instance = LeaveDescription()
|
instance = LeaveDescription()
|
||||||
instance.space = self.space
|
instance.space = self.space
|
||||||
|
instance.origin_space = self.origin_space
|
||||||
instance.target_space = self.target_space
|
instance.target_space = self.target_space
|
||||||
return {"instance": instance}
|
return {"instance": instance}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue