generated from bisco/codex-bootstrap
feat(admin): add manual reservation operations
This commit is contained in:
@@ -1,8 +1,23 @@
|
||||
from django import forms
|
||||
from django.contrib import admin, messages
|
||||
from django.http import HttpResponseRedirect
|
||||
from django.template.response import TemplateResponse
|
||||
from django.urls import path, reverse
|
||||
from django.utils.html import format_html
|
||||
|
||||
from .models import Reservation, ReservationToken
|
||||
from .services import PerformanceNotAvailable, create_pending_reservation
|
||||
from .services import (
|
||||
AlreadyConfirmedReservation,
|
||||
NotEnoughSeats,
|
||||
PerformanceNotAvailable,
|
||||
ReservationNotConfirmed,
|
||||
ReservationNotPending,
|
||||
confirm_reservation_manually,
|
||||
create_pending_reservation,
|
||||
issue_check_in_access_for_reservation,
|
||||
)
|
||||
from checkins.models import CheckIn
|
||||
from checkins.services import AlreadyCheckedIn, confirm_check_in_for_reservation
|
||||
|
||||
|
||||
class ReservationAdminForm(forms.ModelForm):
|
||||
@@ -83,6 +98,8 @@ class ReservationAdmin(admin.ModelAdmin):
|
||||
"venue_name",
|
||||
"confirmation_state_display",
|
||||
"check_in_state_display",
|
||||
"check_in_access_display",
|
||||
"operational_tools",
|
||||
"created_at",
|
||||
"updated_at",
|
||||
"confirmed_at",
|
||||
@@ -90,6 +107,27 @@ class ReservationAdmin(admin.ModelAdmin):
|
||||
)
|
||||
autocomplete_fields = ("performance",)
|
||||
|
||||
def get_urls(self):
|
||||
urls = super().get_urls()
|
||||
custom_urls = [
|
||||
path(
|
||||
"<path:object_id>/manual-confirm/",
|
||||
self.admin_site.admin_view(self.manual_confirm_view),
|
||||
name="bookings_reservation_manual_confirm",
|
||||
),
|
||||
path(
|
||||
"<path:object_id>/check-in-pass/",
|
||||
self.admin_site.admin_view(self.check_in_pass_view),
|
||||
name="bookings_reservation_check_in_pass",
|
||||
),
|
||||
path(
|
||||
"<path:object_id>/manual-check-in/",
|
||||
self.admin_site.admin_view(self.manual_check_in_view),
|
||||
name="bookings_reservation_manual_check_in",
|
||||
),
|
||||
]
|
||||
return custom_urls + urls
|
||||
|
||||
def get_queryset(self, request):
|
||||
return super().get_queryset(request).select_related(
|
||||
"performance",
|
||||
@@ -149,6 +187,8 @@ class ReservationAdmin(admin.ModelAdmin):
|
||||
"status",
|
||||
"confirmation_state_display",
|
||||
"check_in_state_display",
|
||||
"check_in_access_display",
|
||||
"operational_tools",
|
||||
"confirmed_at",
|
||||
"qr_code_generated_at",
|
||||
),
|
||||
@@ -228,6 +268,183 @@ class ReservationAdmin(admin.ModelAdmin):
|
||||
def check_in_state_display(self, obj):
|
||||
return "Checked in" if hasattr(obj, "check_in") else "Not checked in"
|
||||
|
||||
@admin.display(description="Check-in access")
|
||||
def check_in_access_display(self, obj):
|
||||
if obj.status != Reservation.Status.CONFIRMED:
|
||||
return "Available after confirmation"
|
||||
|
||||
url = reverse("admin:bookings_reservation_check_in_pass", args=[obj.pk])
|
||||
return format_html('<a href="{}">Generate operational QR / URL</a>', url)
|
||||
|
||||
@admin.display(description="Operational tools")
|
||||
def operational_tools(self, obj):
|
||||
links = []
|
||||
if obj.status == Reservation.Status.PENDING:
|
||||
confirm_url = reverse("admin:bookings_reservation_manual_confirm", args=[obj.pk])
|
||||
links.append(f'<a href="{confirm_url}">Confirm reservation now</a>')
|
||||
if obj.status == Reservation.Status.CONFIRMED and not hasattr(obj, "check_in"):
|
||||
check_in_url = reverse("admin:bookings_reservation_manual_check_in", args=[obj.pk])
|
||||
links.append(f'<a href="{check_in_url}">Mark as checked in</a>')
|
||||
if obj.status == Reservation.Status.CONFIRMED:
|
||||
pass_url = reverse("admin:bookings_reservation_check_in_pass", args=[obj.pk])
|
||||
links.append(f'<a href="{pass_url}">Show QR / check-in URL</a>')
|
||||
if not links:
|
||||
return "-"
|
||||
return format_html(" | ".join(links))
|
||||
|
||||
def manual_confirm_view(self, request, object_id):
|
||||
reservation = self.get_object(request, object_id)
|
||||
if reservation is None:
|
||||
return self._redirect_to_changelist()
|
||||
|
||||
if request.method == "POST":
|
||||
try:
|
||||
result = confirm_reservation_manually(reservation_id=reservation.pk)
|
||||
except AlreadyConfirmedReservation:
|
||||
self.message_user(request, "Reservation is already confirmed.", level=messages.WARNING)
|
||||
return HttpResponseRedirect(self._change_url(reservation.pk))
|
||||
except ReservationNotPending as exc:
|
||||
self.message_user(request, str(exc), level=messages.ERROR)
|
||||
return HttpResponseRedirect(self._change_url(reservation.pk))
|
||||
except NotEnoughSeats as exc:
|
||||
self.message_user(request, str(exc), level=messages.ERROR)
|
||||
return HttpResponseRedirect(self._change_url(reservation.pk))
|
||||
|
||||
self.message_user(
|
||||
request,
|
||||
"Reservation confirmed manually. A check-in token was generated for operations.",
|
||||
level=messages.SUCCESS,
|
||||
)
|
||||
return self._render_operation_page(
|
||||
request,
|
||||
reservation=result.reservation,
|
||||
title="Reservation confirmed manually",
|
||||
heading="Reservation confirmed manually",
|
||||
description=(
|
||||
"Use this operational QR code or check-in URL only when the guest cannot complete the normal email flow."
|
||||
),
|
||||
qr_code_image=result.qr_code_image,
|
||||
qr_code_url=result.qr_code_url,
|
||||
)
|
||||
|
||||
return self._render_operation_page(
|
||||
request,
|
||||
reservation=reservation,
|
||||
title="Confirm reservation manually",
|
||||
heading="Confirm reservation manually",
|
||||
description=(
|
||||
"This bypasses guest email confirmation, but still rechecks booking availability and generates a check-in token."
|
||||
),
|
||||
submit_label="Confirm reservation",
|
||||
)
|
||||
|
||||
def check_in_pass_view(self, request, object_id):
|
||||
reservation = self.get_object(request, object_id)
|
||||
if reservation is None:
|
||||
return self._redirect_to_changelist()
|
||||
|
||||
if request.method == "POST":
|
||||
try:
|
||||
result = issue_check_in_access_for_reservation(reservation_id=reservation.pk)
|
||||
except ReservationNotConfirmed as exc:
|
||||
self.message_user(request, str(exc), level=messages.ERROR)
|
||||
return HttpResponseRedirect(self._change_url(reservation.pk))
|
||||
|
||||
self.message_user(
|
||||
request,
|
||||
"Operational check-in access generated for this reservation.",
|
||||
level=messages.SUCCESS,
|
||||
)
|
||||
return self._render_operation_page(
|
||||
request,
|
||||
reservation=result.reservation,
|
||||
title="Operational check-in access",
|
||||
heading="Operational check-in access",
|
||||
description=(
|
||||
"This page shows a one-time operational QR code and check-in URL for manual testing or guest support."
|
||||
),
|
||||
qr_code_image=result.qr_code_image,
|
||||
qr_code_url=result.qr_code_url,
|
||||
)
|
||||
|
||||
return self._render_operation_page(
|
||||
request,
|
||||
reservation=reservation,
|
||||
title="Generate operational check-in access",
|
||||
heading="Generate operational check-in access",
|
||||
description=(
|
||||
"Generate a fresh QR code and check-in URL without exposing token hashes in normal admin screens."
|
||||
),
|
||||
submit_label="Generate QR / URL",
|
||||
)
|
||||
|
||||
def manual_check_in_view(self, request, object_id):
|
||||
reservation = self.get_object(request, object_id)
|
||||
if reservation is None:
|
||||
return self._redirect_to_changelist()
|
||||
|
||||
if request.method == "POST":
|
||||
try:
|
||||
confirm_check_in_for_reservation(
|
||||
reservation_id=reservation.pk,
|
||||
staff_user=request.user,
|
||||
source=CheckIn.Source.MANUAL,
|
||||
)
|
||||
except ReservationNotConfirmed as exc:
|
||||
self.message_user(request, str(exc), level=messages.ERROR)
|
||||
return HttpResponseRedirect(self._change_url(reservation.pk))
|
||||
except AlreadyCheckedIn:
|
||||
self.message_user(request, "Reservation has already been checked in.", level=messages.WARNING)
|
||||
return HttpResponseRedirect(self._change_url(reservation.pk))
|
||||
|
||||
self.message_user(request, "Reservation marked as checked in.", level=messages.SUCCESS)
|
||||
return HttpResponseRedirect(self._change_url(reservation.pk))
|
||||
|
||||
return self._render_operation_page(
|
||||
request,
|
||||
reservation=reservation,
|
||||
title="Mark reservation as checked in",
|
||||
heading="Mark reservation as checked in",
|
||||
description="Use this only for staff-side emergency or desk check-in workflows.",
|
||||
submit_label="Confirm check-in",
|
||||
)
|
||||
|
||||
def _change_url(self, reservation_id):
|
||||
return reverse("admin:bookings_reservation_change", args=[reservation_id])
|
||||
|
||||
def _redirect_to_changelist(self):
|
||||
return HttpResponseRedirect(reverse("admin:bookings_reservation_changelist"))
|
||||
|
||||
def _render_operation_page(
|
||||
self,
|
||||
request,
|
||||
*,
|
||||
reservation,
|
||||
title,
|
||||
heading,
|
||||
description,
|
||||
submit_label=None,
|
||||
qr_code_image=None,
|
||||
qr_code_url=None,
|
||||
):
|
||||
context = {
|
||||
**self.admin_site.each_context(request),
|
||||
"opts": self.model._meta,
|
||||
"original": reservation,
|
||||
"title": title,
|
||||
"heading": heading,
|
||||
"description": description,
|
||||
"submit_label": submit_label,
|
||||
"qr_code_image": qr_code_image,
|
||||
"qr_code_url": qr_code_url,
|
||||
"change_url": self._change_url(reservation.pk),
|
||||
}
|
||||
return TemplateResponse(
|
||||
request,
|
||||
"admin/bookings/reservation/operation.html",
|
||||
context,
|
||||
)
|
||||
|
||||
|
||||
@admin.register(ReservationToken)
|
||||
class ReservationTokenAdmin(admin.ModelAdmin):
|
||||
|
||||
Reference in New Issue
Block a user