Allow to download tickets with alternative layouts in backend

This commit is contained in:
Raphael Michel
2022-11-24 13:44:30 +01:00
parent 5858ed8d5c
commit e9ba9a25df
4 changed files with 101 additions and 6 deletions

View File

@@ -23,17 +23,19 @@ import copy
import json
from django.dispatch import receiver
from django.template.loader import get_template
from django.urls import reverse
from django.utils.html import escape
from django.utils.translation import gettext_lazy as _
from pretix.base.channels import get_all_sales_channels
from pretix.base.models import Event
from pretix.base.signals import ( # NOQA: legacy import
EventPluginSignal, event_copy_data, item_copy_data, layout_text_variables,
logentry_display, logentry_object_link, register_data_exporters,
register_multievent_data_exporters, register_ticket_outputs,
)
from pretix.control.signals import item_forms
from pretix.control.signals import item_forms, order_position_buttons
from pretix.plugins.ticketoutputpdf.forms import TicketLayoutItemForm
from pretix.plugins.ticketoutputpdf.models import (
TicketLayout, TicketLayoutItem,
@@ -160,6 +162,53 @@ def pdf_logentry_object_link(sender, logentry, **kwargs):
return a_text.format_map(a_map)
def _ticket_layouts_for_item(request, item):
if not hasattr(request, '_ticket_layouts_for_item'):
request._ticket_layouts_for_item = {}
if item.pk not in request._ticket_layouts_for_item:
request._ticket_layouts_for_item[item.pk] = {
tli.sales_channel: tli.layout
for tli in item.ticketlayout_assignments.select_related('layout')
}
if request._ticket_layouts_for_item[item.pk] and 'web' not in request._ticket_layouts_for_item[item.pk]:
request._ticket_layouts_for_item[item.pk]['web'] = request.event.ticket_layouts.get(default=True)
return request._ticket_layouts_for_item[item.pk]
@receiver(order_position_buttons, dispatch_uid="pretix_ticketoutputpdf_control_order_buttons")
def control_order_position_info(sender: Event, position, request, order, **kwargs):
if not position.generate_ticket:
return ''
layouts = []
seen = set()
lm = _ticket_layouts_for_item(request, position.item)
if order.sales_channel in lm:
seen.add(lm[order.sales_channel])
for k, l in lm.items():
if k == order.sales_channel or l in seen:
continue
layouts.append({
'label': str(l.name),
'channel': k,
})
seen.add(l)
if not layouts:
return ''
template = get_template('pretixplugins/ticketoutputpdf/control_order_position_buttons.html')
ctx = {
'order': position.order,
'request': request,
'event': sender,
'position': position,
'layouts': layouts,
}
return template.render(ctx, request=request).strip()
override_layout = EventPluginSignal()
"""
Arguments: ``layout``, ``orderposition``

View File

@@ -65,9 +65,10 @@ class PdfTicketOutput(BaseTicketOutput):
multi_download_button_text = _('Download tickets (PDF)')
long_download_button_text = _('Download ticket (PDF)')
def __init__(self, event, override_layout=None, override_background=None):
def __init__(self, event, override_layout=None, override_background=None, override_channel=None):
self.override_layout = override_layout
self.override_background = override_background
self.override_channel = override_channel
super().__init__(event)
@cached_property
@@ -117,7 +118,7 @@ class PdfTicketOutput(BaseTicketOutput):
for op in order.positions_with_tickets:
layout = override_layout.send_chained(
order.event, 'layout', orderposition=op, layout=self.layout_map.get(
(op.item_id, order.sales_channel),
(op.item_id, self.override_channel or order.sales_channel),
self.layout_map.get(
(op.item_id, 'web'),
self.default_layout
@@ -138,7 +139,7 @@ class PdfTicketOutput(BaseTicketOutput):
layout = override_layout.send_chained(
order.event, 'layout', orderposition=op, layout=self.layout_map.get(
(op.item_id, order.sales_channel),
(op.item_id, self.override_channel or order.sales_channel),
self.layout_map.get(
(op.item_id, 'web'),
self.default_layout

View File

@@ -27,10 +27,12 @@ from pretix.plugins.ticketoutputpdf.api import (
)
from pretix.plugins.ticketoutputpdf.views import (
LayoutCreate, LayoutDelete, LayoutEditorView, LayoutGetDefault,
LayoutListView, LayoutSetDefault,
LayoutListView, LayoutSetDefault, OrderPrintDo,
)
urlpatterns = [
re_path(r'^control/event/(?P<organizer>[^/]+)/(?P<event>[^/]+)/badges/print$',
OrderPrintDo.as_view(), name='print'),
re_path(r'^control/event/(?P<organizer>[^/]+)/(?P<event>[^/]+)/pdfoutput/$',
LayoutListView.as_view(), name='index'),
re_path(r'^control/event/(?P<organizer>[^/]+)/(?P<event>[^/]+)/pdfoutput/add$',

View File

@@ -21,6 +21,7 @@
#
import json
import logging
from datetime import timedelta
from io import BytesIO
from django.contrib import messages
@@ -29,10 +30,11 @@ from django.core.files import File
from django.core.files.storage import default_storage
from django.db import transaction
from django.http import Http404
from django.shortcuts import redirect
from django.shortcuts import get_object_or_404, redirect
from django.templatetags.static import static
from django.urls import reverse
from django.utils.functional import cached_property
from django.utils.timezone import now
from django.utils.translation import gettext, gettext_lazy as _
from django.views import View
from django.views.generic import CreateView, DeleteView, DetailView, ListView
@@ -48,7 +50,9 @@ from pretix.helpers.models import modelcopy
from pretix.plugins.ticketoutputpdf.forms import TicketLayoutForm
from pretix.plugins.ticketoutputpdf.ticketoutput import PdfTicketOutput
from ...base.views.tasks import AsyncAction
from .models import TicketLayout
from .tasks import tickets_create_pdf
logger = logging.getLogger(__name__)
@@ -286,3 +290,42 @@ class LayoutEditorView(BaseEditorView):
self.layout.background.delete()
self.layout.background.save('background.pdf', f.file)
invalidate_cache.apply_async(kwargs={'event': self.request.event.pk, 'provider': 'pdf'})
class OrderPrintDo(EventPermissionRequiredMixin, AsyncAction, View):
task = tickets_create_pdf
permission = 'can_view_orders'
known_errortypes = ['OrderError', 'ExportError']
def get_success_message(self, value):
return None
def get_success_url(self, value):
return reverse('cachedfile.download', kwargs={'id': str(value)})
def get_error_url(self):
return reverse('control:event.index', kwargs={
'organizer': self.request.organizer.slug,
'event': self.request.event.slug,
})
def get_error_message(self, exception):
if isinstance(exception, str):
return exception
return super().get_error_message(exception)
def post(self, request, *args, **kwargs):
order = get_object_or_404(self.request.event.orders, code=request.GET.get("code"))
cf = CachedFile(web_download=True, session_key=self.request.session.session_key)
cf.date = now()
cf.type = 'application/pdf'
cf.expires = now() + timedelta(days=3)
position = get_object_or_404(order.positions, pk=request.GET.get('position'))
cf.filename = f'tickets_{self.request.event.slug}_{order.code}-{position.positionid}.pdf'
cf.save()
return self.do(
self.request.event.pk,
str(cf.id),
position.pk,
request.GET.get('channel'),
)