From 8437882b145e9f458ad54e1904de8d6e150bcade Mon Sep 17 00:00:00 2001 From: Luca Date: Tue, 10 May 2022 15:43:37 +0200 Subject: [PATCH] Add shift-specific 'required_helpers' field --- shiftregister/app/admin.py | 8 +++--- .../migrations/0004_shift_required_helpers.py | 18 +++++++++++++ shiftregister/app/models.py | 6 +++-- shiftregister/app/views.py | 13 +++++++--- shiftregister/importer/importer.py | 25 ++++++++++++++++--- .../team/templates/partials/shift_box.html | 2 +- .../team/templates/shift_detail.html | 2 +- shiftregister/team/views.py | 10 +++++--- 8 files changed, 67 insertions(+), 17 deletions(-) create mode 100644 shiftregister/app/migrations/0004_shift_required_helpers.py diff --git a/shiftregister/app/admin.py b/shiftregister/app/admin.py index caab3a5..3836276 100644 --- a/shiftregister/app/admin.py +++ b/shiftregister/app/admin.py @@ -16,7 +16,9 @@ class ShiftAdmin(admin.ModelAdmin): # object.helpers.count() def free_slots(self, object): - return object.room.required_helpers - object.shiftregistration_set.count() + return ( + object.required_helpers or object.room.required_helpers + ) - object.shiftregistration_set.count() def send_login(modeladmin, request, queryset): @@ -24,7 +26,7 @@ def send_login(modeladmin, request, queryset): helper.send_confirmation() -class RegistationInline(admin.TabularInline): +class RegistrationInline(admin.TabularInline): model = ShiftRegistration @@ -33,7 +35,7 @@ class HelperAdmin(admin.ModelAdmin): readonly_fields = ("phone",) fields = ("name", "phone", "number_validated") list_display = ("name",) - inlines = (RegistationInline,) + inlines = (RegistrationInline,) actions = (send_login,) diff --git a/shiftregister/app/migrations/0004_shift_required_helpers.py b/shiftregister/app/migrations/0004_shift_required_helpers.py new file mode 100644 index 0000000..7d42bfc --- /dev/null +++ b/shiftregister/app/migrations/0004_shift_required_helpers.py @@ -0,0 +1,18 @@ +# Generated by Django 4.0.4 on 2022-05-10 13:03 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ("app", "0003_logintoken_send_count"), + ] + + operations = [ + migrations.AddField( + model_name="shift", + name="required_helpers", + field=models.IntegerField(default=0), + ), + ] diff --git a/shiftregister/app/models.py b/shiftregister/app/models.py index eeab6a3..14509f1 100644 --- a/shiftregister/app/models.py +++ b/shiftregister/app/models.py @@ -22,8 +22,10 @@ class Shift(models.Model): room = models.ForeignKey(Room, on_delete=models.RESTRICT) start_at = models.DateTimeField() duration = models.DurationField() + required_helpers = models.IntegerField( + default=0, help_text="When this is set to zero, the room value is used instead." + ) deleted = models.BooleanField(default=False) - # todo: add helper amount override field def __str__(self): return f"{self.room.name}: {self.start_at}" @@ -82,7 +84,7 @@ class ShiftRegistration(models.Model): ) def send_reminder(self): - text = f"Deine Kontakt-Schicht begint um {self.shift.start_at.strftime('%H:%M')}, bitte komme eine halbe Stunde vorher an den Infopoint." + text = f"Deine kontakt-Schicht beginnt um {self.shift.start_at.strftime('%H:%M')}, bitte komm eine halbe Stunde vorher an den Infopoint." msg = Message(to=self.helper, text=text) msg.save() self.reminder_sent = True diff --git a/shiftregister/app/views.py b/shiftregister/app/views.py index ee1e11a..4366193 100644 --- a/shiftregister/app/views.py +++ b/shiftregister/app/views.py @@ -35,11 +35,14 @@ def index(request): if imp_shift and imp_shift.is_running(): context["current_shift"] = imp_shift + help_wanted = Q(required_helpers__gt=F("reg_count")) | Q(required_helpers=0) & Q( + room__required_helpers__gt=F("reg_count") + ) free_shifts = ( Shift.objects.annotate(reg_count=Count("shiftregistration")) .filter( + help_wanted, start_at__gt=timezone.now(), - room__required_helpers__gt=F("reg_count"), deleted=False, ) .order_by("start_at") @@ -49,8 +52,8 @@ def index(request): free_shifts = ( Shift.objects.annotate(reg_count=Count("shiftregistration")) .filter( + help_wanted, start_at__gt=timezone.now(), - room__required_helpers__gt=F("reg_count"), deleted=False, ) .filter(~Q(shiftregistration__helper=request.helper)) @@ -161,7 +164,11 @@ def shift(request, shiftid): # this currently ignores date/time request.session["last_seen_shift"] = shiftid - if shift.room.required_helpers > shift.shiftregistration_set.count(): + if ( + shift.required_helpers > shift.shiftregistration_set.count() + or shift.required_helpers == 0 + and shift.room.required_helpers > shift.shiftregistration_set.count() + ): context["can_register"] = True if helper: diff --git a/shiftregister/importer/importer.py b/shiftregister/importer/importer.py index cc13b49..ee35dea 100644 --- a/shiftregister/importer/importer.py +++ b/shiftregister/importer/importer.py @@ -25,11 +25,26 @@ def import_calendar(calendar): events = {} for event in cal.walk("vevent"): uid = event.decoded("uid").decode() - room = ( - event.decoded("location", None) or event.decoded("summary") - ).decode() + summary = event.decoded("summary").decode() start = event.decoded("dtstart").astimezone(timezone.utc) end = event.decoded("dtend").astimezone(timezone.utc) + location = event.decoded("location", None) + if location is not None: + location = location.decode() + + if location: + room = location + try: + required_helpers = int(summary) + except ValueError: + required_helpers = 0 + else: + try: + room, required_helpers = tuple(summary.split()) + required_helpers = int(required_helpers) + except ValueError: + room = summary + required_helpers = 0 if not uid or not room: return False @@ -40,6 +55,7 @@ def import_calendar(calendar): { "start_at": start, "duration": end - start, + "required_helpers": required_helpers, "uuid": uid, "calendar": calendar, }, @@ -52,7 +68,7 @@ def import_calendar(calendar): if r == None: rooms[room] = Room( name=room, required_helpers=0 - ) # required_helpers=0 ensures a shift in a new room is not displayed until the correct number of required helpers is set + ) # required_helpers=0 ensures a shift in a new room is not displayed unless the correct number of required helpers is set or the shift itself specifies it rooms[room].save() for e in Event.objects.filter(calendar=calendar, uuid__in=events): @@ -62,6 +78,7 @@ def import_calendar(calendar): e.room = rooms[room] e.start_at = event["start_at"] e.duration = event["duration"] + e.required_helpers = event["required_helpers"] e.save() events[uuid] = (room, e) diff --git a/shiftregister/team/templates/partials/shift_box.html b/shiftregister/team/templates/partials/shift_box.html index 77ea0c7..7d6fc22 100644 --- a/shiftregister/team/templates/partials/shift_box.html +++ b/shiftregister/team/templates/partials/shift_box.html @@ -6,7 +6,7 @@ Ort: {{ shift.room.name }}
Beginn: {{ shift.start_at }}
Dauer: {{ shift.duration }}
- Belegung: {{ shift.shiftregistration_set.count }}/{{ shift.room.required_helpers }} + Belegung: {{ shift.shiftregistration_set.count }}/{{ shift.required_helpers|default:shift.room.required_helpers }}
Details diff --git a/shiftregister/team/templates/shift_detail.html b/shiftregister/team/templates/shift_detail.html index e5a92a0..e6581cb 100644 --- a/shiftregister/team/templates/shift_detail.html +++ b/shiftregister/team/templates/shift_detail.html @@ -2,7 +2,7 @@ {% block title %}Schichtdetails{% endblock %} {% block content %} -

{% if shift.deleted %}(gelöscht) {% endif %}{{ shift.room.name }} {{ shift.start_at }} ({{ shift.shiftregistration_set.count }}/{{ shift.room.required_helpers }})

+

{% if shift.deleted %}(gelöscht) {% endif %}{{ shift.room.name }} {{ shift.start_at }} ({{ shift.shiftregistration_set.count }}/{{ shift.required_helpers|default:shift.room.required_helpers }})

{% if shift.shiftregistration_set.all %}
Helfer*innen
diff --git a/shiftregister/team/views.py b/shiftregister/team/views.py index aad3907..5fe9667 100644 --- a/shiftregister/team/views.py +++ b/shiftregister/team/views.py @@ -25,13 +25,13 @@ def shift_overview(request): output_field=DateTimeField(), ) ) - .filter(start_at__lte=timezone.now(), end_at__gte=timezone.now()) + .filter(start_at__lte=timezone.now(), end_at__gte=timezone.now(), deleted=False) .order_by("start_at") ] # probably can do some distinct/group by stuff but not sure how tih django queries context["next_shifts"] = [ - Shift.objects.filter(room=room, start_at__gt=timezone.now()) + Shift.objects.filter(room=room, start_at__gt=timezone.now(), deleted=False) .order_by("start_at") .first() for room in Room.objects.all() @@ -97,6 +97,10 @@ class FreeShiftList(ShiftList): title = "Freie Schichten" def get_queryset(self): + help_wanted = Q(required_helpers__gt=F("reg_count")) | Q( + required_helpers=0 + ) & Q(room__required_helpers__gt=F("reg_count")) return Shift.objects.annotate(reg_count=Count("shiftregistration")).filter( - start_at__gt=timezone.now(), room__required_helpers__gt=F("reg_count") + help_wanted, + start_at__gt=timezone.now(), )