from decimal import Decimal import pytz from django.contrib.contenttypes.models import ContentType from django.core.urlresolvers import reverse from django.db.models import ( Count, Exists, IntegerField, Max, Min, OuterRef, Q, Subquery, Sum, ) from django.db.models.functions import Coalesce, Greatest from django.dispatch import receiver from django.shortcuts import render from django.template.loader import get_template from django.utils import formats from django.utils.formats import date_format from django.utils.html import escape from django.utils.timezone import now from django.utils.translation import ugettext_lazy as _, ungettext from pretix.base.models import ( Event, Item, Order, OrderPosition, RequiredAction, SubEvent, Voucher, WaitingListEntry, ) from pretix.control.forms.event import CommentForm from pretix.control.signals import ( event_dashboard_widgets, user_dashboard_widgets, ) from pretix.helpers.daterange import daterange from ..logdisplay import OVERVIEW_BLACKLIST NUM_WIDGET = '
{num}{text}
' @receiver(signal=event_dashboard_widgets) def base_widgets(sender, subevent=None, **kwargs): prodc = Item.objects.filter( event=sender, active=True, ).filter( (Q(available_until__isnull=True) | Q(available_until__gte=now())) & (Q(available_from__isnull=True) | Q(available_from__lte=now())) ).count() if subevent: opqs = OrderPosition.objects.filter(subevent=subevent) else: opqs = OrderPosition.objects tickc = opqs.filter( order__event=sender, item__admission=True, order__status__in=(Order.STATUS_PAID, Order.STATUS_PENDING), ).count() paidc = opqs.filter( order__event=sender, item__admission=True, order__status=Order.STATUS_PAID, ).count() if subevent: rev = opqs.filter( order__event=sender, order__status=Order.STATUS_PAID ).aggregate( sum=Sum('price') )['sum'] or Decimal('0.00') else: rev = Order.objects.filter( event=sender, status=Order.STATUS_PAID ).aggregate(sum=Sum('total'))['sum'] or Decimal('0.00') return [ { 'content': NUM_WIDGET.format(num=tickc, text=_('Attendees (ordered)')), 'display_size': 'small', 'priority': 100, 'url': reverse('control:event.orders', kwargs={ 'event': sender.slug, 'organizer': sender.organizer.slug }) + ('?subevent={}'.format(subevent.pk) if subevent else '') }, { 'content': NUM_WIDGET.format(num=paidc, text=_('Attendees (paid)')), 'display_size': 'small', 'priority': 100, 'url': reverse('control:event.orders.overview', kwargs={ 'event': sender.slug, 'organizer': sender.organizer.slug }) + ('?subevent={}'.format(subevent.pk) if subevent else '') }, { 'content': NUM_WIDGET.format( num=formats.localize(rev), text=_('Total revenue ({currency})').format(currency=sender.currency)), 'display_size': 'small', 'priority': 100, 'url': reverse('control:event.orders.overview', kwargs={ 'event': sender.slug, 'organizer': sender.organizer.slug }) + ('?subevent={}'.format(subevent.pk) if subevent else '') }, { 'content': NUM_WIDGET.format(num=prodc, text=_('Active products')), 'display_size': 'small', 'priority': 100, 'url': reverse('control:event.items', kwargs={ 'event': sender.slug, 'organizer': sender.organizer.slug }) }, ] @receiver(signal=event_dashboard_widgets) def waitinglist_widgets(sender, subevent=None, **kwargs): widgets = [] wles = WaitingListEntry.objects.filter(event=sender, subevent=subevent, voucher__isnull=True) if wles.count(): quota_cache = {} itemvar_cache = {} happy = 0 for wle in wles: if (wle.item, wle.variation) not in itemvar_cache: itemvar_cache[(wle.item, wle.variation)] = ( wle.variation.check_quotas(subevent=wle.subevent, count_waitinglist=False, _cache=quota_cache) if wle.variation else wle.item.check_quotas(subevent=wle.subevent, count_waitinglist=False, _cache=quota_cache) ) row = itemvar_cache.get((wle.item, wle.variation)) if row[1] > 0: itemvar_cache[(wle.item, wle.variation)] = (row[0], row[1] - 1) happy += 1 widgets.append({ 'content': NUM_WIDGET.format(num=str(happy), text=_('available to give to people on waiting list')), 'priority': 50, 'url': reverse('control:event.orders.waitinglist', kwargs={ 'event': sender.slug, 'organizer': sender.organizer.slug, }) }) widgets.append({ 'content': NUM_WIDGET.format(num=str(wles.count()), text=_('total waiting list length')), 'display_size': 'small', 'priority': 50, 'url': reverse('control:event.orders.waitinglist', kwargs={ 'event': sender.slug, 'organizer': sender.organizer.slug, }) }) return widgets @receiver(signal=event_dashboard_widgets) def quota_widgets(sender, subevent=None, **kwargs): widgets = [] for q in sender.quotas.filter(subevent=subevent): status, left = q.availability(allow_cache=True) widgets.append({ 'content': NUM_WIDGET.format(num='{}/{}'.format(left, q.size) if q.size is not None else '\u221e', text=_('{quota} left').format(quota=escape(q.name))), 'display_size': 'small', 'priority': 50, 'url': reverse('control:event.items.quotas.show', kwargs={ 'event': sender.slug, 'organizer': sender.organizer.slug, 'quota': q.id }) }) return widgets @receiver(signal=event_dashboard_widgets) def shop_state_widget(sender, **kwargs): return [{ 'display_size': 'small', 'priority': 1000, 'content': '
{t1}
{state}{t2}
'.format( t1=_('Your ticket shop is'), t2=_('Click here to change'), state=_('live') if sender.live else _('not yet public'), icon='fa-check-circle' if sender.live else 'fa-times-circle', cls='live' if sender.live else 'off' ), 'url': reverse('control:event.live', kwargs={ 'event': sender.slug, 'organizer': sender.organizer.slug }) }] @receiver(signal=event_dashboard_widgets) def checkin_widget(sender, **kwargs): size_qs = OrderPosition.objects.filter(order__event=sender, order__status='p') checked_qs = OrderPosition.objects.filter(order__event=sender, order__status='p', checkins__isnull=False) # if this setting is False, we check only items for admission if not sender.settings.ticket_download_nonadm: size_qs = size_qs.filter(item__admission=True) checked_qs = checked_qs.filter(item__admission=True) return [{ 'content': NUM_WIDGET.format(num='{}/{}'.format(checked_qs.count(), size_qs.count()), text=_('Checked in')), 'display_size': 'small', 'priority': 50, 'url': reverse('control:event.orders.checkins', kwargs={ 'event': sender.slug, 'organizer': sender.organizer.slug }) }] @receiver(signal=event_dashboard_widgets) def welcome_wizard_widget(sender, **kwargs): template = get_template('pretixcontrol/event/dashboard_widget_welcome.html') ctx = { 'title': _('Welcome to pretix!') } kwargs = {'event': sender.slug, 'organizer': sender.organizer.slug} if not sender.items.exists(): ctx.update({ 'subtitle': _('Get started by creating a product'), 'text': _('The first thing you need for selling tickets to your event is one or more "products" your ' 'participants can choose from. A product can be a ticket or anything else that you want to sell, ' 'e.g. additional merchandise in form of t-shirts.'), 'button_text': _('Create a first product'), 'button_url': reverse('control:event.items.add', kwargs=kwargs) }) elif not sender.quotas.exists(): ctx.update({ 'subtitle': _('Create quotas that apply to your products'), 'text': _('Your tickets will only be available for sale if you create a matching quota, i.e. if you tell ' 'pretix how many tickets it should sell for your event.'), 'button_text': _('Create a first quota'), 'button_url': reverse('control:event.items.quotas.add', kwargs=kwargs) }) else: return [] return [{ 'display_size': 'full', 'priority': 2000, 'content': template.render(ctx) }] def event_index(request, organizer, event): subevent = None if request.GET.get("subevent", "") != "" and request.event.has_subevents: i = request.GET.get("subevent", "") try: subevent = request.event.subevents.get(pk=i) except SubEvent.DoesNotExist: pass widgets = [] for r, result in event_dashboard_widgets.send(sender=request.event, subevent=subevent): widgets.extend(result) can_change_orders = request.user.has_event_permission(request.organizer, request.event, 'can_change_orders') qs = request.event.logentry_set.all().select_related('user', 'content_type').order_by('-datetime') qs = qs.exclude(action_type__in=OVERVIEW_BLACKLIST) if not request.user.has_event_permission(request.organizer, request.event, 'can_view_orders'): qs = qs.exclude(content_type=ContentType.objects.get_for_model(Order)) if not request.user.has_event_permission(request.organizer, request.event, 'can_view_vouchers'): qs = qs.exclude(content_type=ContentType.objects.get_for_model(Voucher)) a_qs = request.event.requiredaction_set.filter(done=False) ctx = { 'widgets': rearrange(widgets), 'logs': qs[:5], 'actions': a_qs[:5] if can_change_orders else [], 'comment_form': CommentForm(initial={'comment': request.event.comment}) } for a in ctx['actions']: a.display = a.display(request) return render(request, 'pretixcontrol/event/index.html', ctx) @receiver(signal=user_dashboard_widgets) def user_event_widgets(**kwargs): user = kwargs.pop('user') widgets = [] tpl = """
{event}
{daterange}
{times}
{orders} {status}
""" active_orders = Order.objects.filter( event=OuterRef('pk'), status__in=[Order.STATUS_PENDING, Order.STATUS_PAID] ).order_by().values('event').annotate( c=Count('*') ).values( 'c' ) required_actions = RequiredAction.objects.filter( event=OuterRef('pk'), done=False ) # Get set of events where we have the permission to show the # of orders events_with_orders = set(Event.objects.filter( Q(organizer_id__in=user.teams.filter(all_events=True, can_view_orders=True).values_list('organizer', flat=True)) | Q(id__in=user.teams.filter(can_view_orders=True).values_list('limit_events__id', flat=True)) ).values_list('id', flat=True)) events = user.get_events_with_any_permission().annotate( order_count=Subquery(active_orders, output_field=IntegerField()), has_ra=Exists(required_actions) ).annotate( min_from=Min('subevents__date_from'), max_from=Max('subevents__date_from'), max_to=Max('subevents__date_to'), max_fromto=Greatest(Max('subevents__date_to'), Max('subevents__date_from')) ).annotate( order_from=Coalesce('min_from', 'date_from'), order_to=Coalesce('max_fromto', 'max_to', 'max_from', 'date_to'), ).order_by( '-order_from', 'name' ).prefetch_related( '_settings_objects', 'organizer___settings_objects' ).select_related('organizer')[:100] for event in events: dr = event.get_date_range_display() tz = pytz.timezone(event.settings.timezone) if event.has_subevents: dr = daterange( (event.min_from).astimezone(tz), (event.max_fromto or event.max_to or event.max_from).astimezone(tz) ) if event.has_ra: status = ('danger', _('Action required')) elif not event.live: status = ('warning', _('Shop disabled')) elif event.presale_has_ended: status = ('default', _('Sale over')) elif not event.presale_is_running: status = ('default', _('Soon')) else: status = ('success', _('On sale')) widgets.append({ 'content': tpl.format( event=escape(event.name), times=_('Event series') if event.has_subevents else ( ((date_format(event.date_admission.astimezone(tz), 'TIME_FORMAT') + ' / ') if event.date_admission and event.date_admission != event.date_from else '') + (date_format(event.date_from.astimezone(tz), 'TIME_FORMAT') if event.date_from else '') ), url=reverse('control:event.index', kwargs={ 'event': event.slug, 'organizer': event.organizer.slug }), orders=( '{orders_text}'.format( orders_url=reverse('control:event.orders', kwargs={ 'event': event.slug, 'organizer': event.organizer.slug }), orders_text=ungettext('{num} order', '{num} orders', event.order_count or 0).format( num=event.order_count or 0 ) ) if user.is_superuser or event.pk in events_with_orders else '' ), daterange=dr, status=status[1], statusclass=status[0], ), 'display_size': 'small', 'priority': 100, 'container_class': 'widget-container widget-container-event', }) """ {% if not e.live %} {% trans "Shop disabled" %} {% elif e.presale_has_ended %} {% trans "Presale over" %} {% elif not e.presale_is_running %} {% trans "Presale not started" %} {% else %} {% trans "On sale" %} {% endif %} """ return widgets @receiver(signal=user_dashboard_widgets) def new_event_widgets(**kwargs): return [ { 'content': '
{t}
'.format( t=_('Create a new event') ), 'display_size': 'small', 'priority': 50, 'url': reverse('control:events.add') } ] def user_index(request): widgets = [] for r, result in user_dashboard_widgets.send(request, user=request.user): widgets.extend(result) ctx = { 'widgets': rearrange(widgets), } return render(request, 'pretixcontrol/dashboard.html', ctx) def rearrange(widgets: list): """ Sort widget boxes according to priority. """ mapping = { 'small': 1, 'big': 2, 'full': 3, } def sort_key(element): return ( element.get('priority', 1), mapping.get(element.get('display_size', 'small'), 1), ) return sorted(widgets, key=sort_key, reverse=True)