mirror of
https://github.com/pretix/pretix.git
synced 2026-05-06 15:24:02 +00:00
208 lines
7.5 KiB
Python
208 lines
7.5 KiB
Python
#
|
|
# This file is part of pretix (Community Edition).
|
|
#
|
|
# Copyright (C) 2014-2020 Raphael Michel and contributors
|
|
# Copyright (C) 2020-2021 rami.io GmbH and contributors
|
|
#
|
|
# This program is free software: you can redistribute it and/or modify it under the terms of the GNU Affero General
|
|
# Public License as published by the Free Software Foundation in version 3 of the License.
|
|
#
|
|
# ADDITIONAL TERMS APPLY: Pursuant to Section 7 of the GNU Affero General Public License, additional terms are
|
|
# applicable granting you additional permissions and placing additional restrictions on your usage of this software.
|
|
# Please refer to the pretix LICENSE file to obtain the full terms applicable to this work. If you did not receive
|
|
# this file, see <https://pretix.eu/about/en/license>.
|
|
#
|
|
# This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied
|
|
# warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for more
|
|
# details.
|
|
#
|
|
# You should have received a copy of the GNU Affero General Public License along with this program. If not, see
|
|
# <https://www.gnu.org/licenses/>.
|
|
#
|
|
import copy
|
|
import json
|
|
from collections import defaultdict
|
|
|
|
from django.dispatch import receiver
|
|
from django.template.loader import get_template
|
|
from django.urls import resolve, reverse
|
|
from django.utils.html import escape
|
|
from django.utils.translation import gettext_lazy as _
|
|
|
|
from pretix.base.models import Event, Order
|
|
from pretix.base.signals import (
|
|
event_copy_data, item_copy_data, logentry_display, logentry_object_link,
|
|
register_data_exporters,
|
|
)
|
|
from pretix.control.signals import (
|
|
item_forms, nav_event, order_info, order_position_buttons,
|
|
)
|
|
from pretix.plugins.badges.forms import BadgeItemForm
|
|
from pretix.plugins.badges.models import BadgeItem, BadgeLayout
|
|
|
|
|
|
@receiver(nav_event, dispatch_uid="badges_nav")
|
|
def control_nav_import(sender, request=None, **kwargs):
|
|
url = resolve(request.path_info)
|
|
p = (
|
|
request.user.has_event_permission(request.organizer, request.event, 'can_change_settings', request)
|
|
or request.user.has_event_permission(request.organizer, request.event, 'can_view_orders', request)
|
|
)
|
|
if not p:
|
|
return []
|
|
return [
|
|
{
|
|
'label': _('Badges'),
|
|
'url': reverse('plugins:badges:index', kwargs={
|
|
'event': request.event.slug,
|
|
'organizer': request.event.organizer.slug,
|
|
}),
|
|
'active': url.namespace == 'plugins:badges',
|
|
'icon': 'id-card',
|
|
}
|
|
]
|
|
|
|
|
|
@receiver(item_forms, dispatch_uid="badges_item_forms")
|
|
def control_item_forms(sender, request, item, **kwargs):
|
|
try:
|
|
inst = BadgeItem.objects.get(item=item)
|
|
except BadgeItem.DoesNotExist:
|
|
inst = BadgeItem(item=item)
|
|
return BadgeItemForm(
|
|
instance=inst,
|
|
event=sender,
|
|
data=(request.POST if request.method == "POST" else None),
|
|
prefix="badgeitem"
|
|
)
|
|
|
|
|
|
@receiver(item_copy_data, dispatch_uid="badges_item_copy")
|
|
def copy_item(sender, source, target, **kwargs):
|
|
try:
|
|
inst = BadgeItem.objects.get(item=source)
|
|
BadgeItem.objects.create(item=target, layout=inst.layout)
|
|
except BadgeItem.DoesNotExist:
|
|
pass
|
|
|
|
|
|
@receiver(signal=event_copy_data, dispatch_uid="badges_copy_data")
|
|
def event_copy_data_receiver(sender, other, question_map, item_map, **kwargs):
|
|
layout_map = {}
|
|
for bl in other.badge_layouts.all():
|
|
oldid = bl.pk
|
|
bl = copy.copy(bl)
|
|
bl.pk = None
|
|
bl.event = sender
|
|
|
|
layout = json.loads(bl.layout)
|
|
for o in layout:
|
|
if o['type'] == 'textarea':
|
|
if o['content'].startswith('question_'):
|
|
try:
|
|
newq = question_map.get(int(o['content'][9:]))
|
|
except ValueError:
|
|
# int cannot convert new placeholders question_{identifier}
|
|
# can be safely ignored as only old format questions_{pk} should be converted
|
|
pass
|
|
else:
|
|
if newq:
|
|
o['content'] = 'question_{}'.format(newq.pk)
|
|
|
|
bl.layout = json.dumps(layout)
|
|
bl.save()
|
|
|
|
if bl.background and bl.background.name:
|
|
bl.background.save('background.pdf', bl.background)
|
|
|
|
layout_map[oldid] = bl
|
|
|
|
for bi in BadgeItem.objects.filter(item__event=other):
|
|
BadgeItem.objects.create(item=item_map.get(bi.item_id), layout=layout_map.get(bi.layout_id))
|
|
|
|
|
|
@receiver(register_data_exporters, dispatch_uid="badges_export_all")
|
|
def register_pdf(sender, **kwargs):
|
|
from .exporters import BadgeExporter
|
|
return BadgeExporter
|
|
|
|
|
|
def _cached_rendermap(event):
|
|
if hasattr(event, '_cached_renderermap'):
|
|
return event._cached_renderermap
|
|
renderermap = {
|
|
bi.item_id: bi.layout_id
|
|
for bi in BadgeItem.objects.select_related('layout').filter(item__event=event)
|
|
}
|
|
try:
|
|
default_renderer = event.badge_layouts.get(default=True).pk
|
|
except BadgeLayout.DoesNotExist:
|
|
default_renderer = None
|
|
event._cached_renderermap = defaultdict(lambda: default_renderer)
|
|
event._cached_renderermap.update(renderermap)
|
|
return event._cached_renderermap
|
|
|
|
|
|
@receiver(order_position_buttons, dispatch_uid="badges_control_order_buttons")
|
|
def control_order_position_info(sender: Event, position, request, order: Order, **kwargs):
|
|
if _cached_rendermap(sender)[position.item_id] is None:
|
|
return ''
|
|
template = get_template('pretixplugins/badges/control_order_position_buttons.html')
|
|
ctx = {
|
|
'order': order,
|
|
'request': request,
|
|
'event': sender,
|
|
'position': position
|
|
}
|
|
return template.render(ctx, request=request).strip()
|
|
|
|
|
|
@receiver(order_info, dispatch_uid="badges_control_order_info")
|
|
def control_order_info(sender: Event, request, order: Order, **kwargs):
|
|
cm = _cached_rendermap(sender)
|
|
if all(cm[p.item_id] is None for p in order.positions.all()):
|
|
return ''
|
|
|
|
template = get_template('pretixplugins/badges/control_order_info.html')
|
|
|
|
ctx = {
|
|
'order': order,
|
|
'request': request,
|
|
'event': sender,
|
|
}
|
|
return template.render(ctx, request=request)
|
|
|
|
|
|
@receiver(signal=logentry_display, dispatch_uid="badges_logentry_display")
|
|
def badges_logentry_display(sender, logentry, **kwargs):
|
|
if not logentry.action_type.startswith('pretix.plugins.badges'):
|
|
return
|
|
|
|
plains = {
|
|
'pretix.plugins.badges.layout.added': _('Badge layout created.'),
|
|
'pretix.plugins.badges.layout.deleted': _('Badge layout deleted.'),
|
|
'pretix.plugins.badges.layout.changed': _('Badge layout changed.'),
|
|
}
|
|
|
|
if logentry.action_type in plains:
|
|
return plains[logentry.action_type]
|
|
|
|
|
|
@receiver(signal=logentry_object_link, dispatch_uid="badges_logentry_object_link")
|
|
def badges_logentry_object_link(sender, logentry, **kwargs):
|
|
if not logentry.action_type.startswith('pretix.plugins.badges.layout') or not isinstance(logentry.content_object,
|
|
BadgeLayout):
|
|
return
|
|
|
|
a_text = _('Badge layout {val}')
|
|
a_map = {
|
|
'href': reverse('plugins:badges:edit', kwargs={
|
|
'event': sender.slug,
|
|
'organizer': sender.organizer.slug,
|
|
'layout': logentry.content_object.id
|
|
}),
|
|
'val': escape(logentry.content_object.name),
|
|
}
|
|
a_map['val'] = '<a href="{href}">{val}</a>'.format_map(a_map)
|
|
return a_text.format_map(a_map)
|