forked from CGM_Public/pretix_original
Allow to customize description of calendar files (#2415)
Co-authored-by: Martin Gross <gross@rami.io>
This commit is contained in:
@@ -659,12 +659,13 @@ class OrderViewSet(viewsets.ModelViewSet):
|
|||||||
|
|
||||||
_order_placed_email(
|
_order_placed_email(
|
||||||
request.event, order, payment.payment_provider if payment else None, email_template,
|
request.event, order, payment.payment_provider if payment else None, email_template,
|
||||||
log_entry, invoice, payment
|
log_entry, invoice, payment, is_free=free_flow
|
||||||
)
|
)
|
||||||
if email_attendees:
|
if email_attendees:
|
||||||
for p in order.positions.all():
|
for p in order.positions.all():
|
||||||
if p.addon_to_id is None and p.attendee_email and p.attendee_email != order.email:
|
if p.addon_to_id is None and p.attendee_email and p.attendee_email != order.email:
|
||||||
_order_placed_email_attendee(request.event, order, p, email_attendees_template, log_entry)
|
_order_placed_email_attendee(request.event, order, p, email_attendees_template, log_entry,
|
||||||
|
is_free=free_flow)
|
||||||
|
|
||||||
if not free_flow and order.status == Order.STATUS_PAID and payment:
|
if not free_flow and order.status == Order.STATUS_PAID and payment:
|
||||||
payment._send_paid_mail(invoice, None, '')
|
payment._send_paid_mail(invoice, None, '')
|
||||||
|
|||||||
@@ -33,6 +33,7 @@ from django.core.mail.backends.smtp import EmailBackend
|
|||||||
from django.db.models import Count
|
from django.db.models import Count
|
||||||
from django.dispatch import receiver
|
from django.dispatch import receiver
|
||||||
from django.template.loader import get_template
|
from django.template.loader import get_template
|
||||||
|
from django.utils.formats import date_format
|
||||||
from django.utils.timezone import now
|
from django.utils.timezone import now
|
||||||
from django.utils.translation import (
|
from django.utils.translation import (
|
||||||
get_language, gettext_lazy as _, pgettext_lazy,
|
get_language, gettext_lazy as _, pgettext_lazy,
|
||||||
@@ -453,6 +454,15 @@ def base_placeholders(sender, **kwargs):
|
|||||||
}
|
}
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
SimpleFunctionalMailTextPlaceholder(
|
||||||
|
'event_location', ['event_or_subevent'], lambda event_or_subevent: str(event_or_subevent.location or ''),
|
||||||
|
lambda event: str(event.location or ''),
|
||||||
|
),
|
||||||
|
SimpleFunctionalMailTextPlaceholder(
|
||||||
|
'event_admission_time', ['event_or_subevent'],
|
||||||
|
lambda event_or_subevent: date_format(event_or_subevent.date_admission, 'TIME_FORMAT') if event_or_subevent.date_admission else '',
|
||||||
|
lambda event: date_format(event.date_admission, 'TIME_FORMAT') if event.date_admission else '',
|
||||||
|
),
|
||||||
SimpleFunctionalMailTextPlaceholder(
|
SimpleFunctionalMailTextPlaceholder(
|
||||||
'subevent', ['waiting_list_entry', 'event'],
|
'subevent', ['waiting_list_entry', 'event'],
|
||||||
lambda waiting_list_entry, event: str(waiting_list_entry.subevent or event),
|
lambda waiting_list_entry, event: str(waiting_list_entry.subevent or event),
|
||||||
@@ -622,6 +632,10 @@ def base_placeholders(sender, **kwargs):
|
|||||||
'meta_%s' % k, ['event'], lambda event, k=k: event.meta_data[k],
|
'meta_%s' % k, ['event'], lambda event, k=k: event.meta_data[k],
|
||||||
v
|
v
|
||||||
))
|
))
|
||||||
|
ph.append(SimpleFunctionalMailTextPlaceholder(
|
||||||
|
'meta_%s' % k, ['event_or_subevent'], lambda event_or_subevent, k=k: event_or_subevent.meta_data[k],
|
||||||
|
v
|
||||||
|
))
|
||||||
|
|
||||||
return ph
|
return ph
|
||||||
|
|
||||||
|
|||||||
@@ -77,7 +77,7 @@ from pretix.base.signals import email_filter, global_email_filter
|
|||||||
from pretix.celery_app import app
|
from pretix.celery_app import app
|
||||||
from pretix.helpers.hierarkey import clean_filename
|
from pretix.helpers.hierarkey import clean_filename
|
||||||
from pretix.multidomain.urlreverse import build_absolute_uri
|
from pretix.multidomain.urlreverse import build_absolute_uri
|
||||||
from pretix.presale.ical import get_ical
|
from pretix.presale.ical import get_private_icals
|
||||||
|
|
||||||
logger = logging.getLogger('pretix.base.mail')
|
logger = logging.getLogger('pretix.base.mail')
|
||||||
INVALID_ADDRESS = 'invalid-pretix-mail-address'
|
INVALID_ADDRESS = 'invalid-pretix-mail-address'
|
||||||
@@ -430,18 +430,7 @@ def mail_send_task(self, *args, to: List[str], subject: str, body: str, html: st
|
|||||||
}
|
}
|
||||||
)
|
)
|
||||||
if attach_ical:
|
if attach_ical:
|
||||||
ical_events = set()
|
for i, cal in enumerate(get_private_icals(event, [position] if position else order.positions.all())):
|
||||||
if event.has_subevents:
|
|
||||||
if position:
|
|
||||||
ical_events.add(position.subevent)
|
|
||||||
else:
|
|
||||||
for p in order.positions.all():
|
|
||||||
ical_events.add(p.subevent)
|
|
||||||
else:
|
|
||||||
ical_events.add(order.event)
|
|
||||||
|
|
||||||
for i, e in enumerate(ical_events):
|
|
||||||
cal = get_ical([e])
|
|
||||||
email.attach('event-{}.ics'.format(i), cal.serialize(), 'text/calendar')
|
email.attach('event-{}.ics'.format(i), cal.serialize(), 'text/calendar')
|
||||||
|
|
||||||
email = email_filter.send_chained(event, 'message', message=email, order=order, user=user)
|
email = email_filter.send_chained(event, 'message', message=email, order=order, user=user)
|
||||||
|
|||||||
@@ -932,7 +932,7 @@ def _create_order(event: Event, email: str, positions: List[CartPosition], now_d
|
|||||||
|
|
||||||
|
|
||||||
def _order_placed_email(event: Event, order: Order, pprov: BasePaymentProvider, email_template, log_entry: str,
|
def _order_placed_email(event: Event, order: Order, pprov: BasePaymentProvider, email_template, log_entry: str,
|
||||||
invoice, payment: OrderPayment):
|
invoice, payment: OrderPayment, is_free=False):
|
||||||
email_context = get_email_context(event=event, order=order, payment=payment if pprov else None)
|
email_context = get_email_context(event=event, order=order, payment=payment if pprov else None)
|
||||||
email_subject = _('Your order: %(code)s') % {'code': order.code}
|
email_subject = _('Your order: %(code)s') % {'code': order.code}
|
||||||
try:
|
try:
|
||||||
@@ -941,7 +941,7 @@ def _order_placed_email(event: Event, order: Order, pprov: BasePaymentProvider,
|
|||||||
log_entry,
|
log_entry,
|
||||||
invoices=[invoice] if invoice and event.settings.invoice_email_attachment else [],
|
invoices=[invoice] if invoice and event.settings.invoice_email_attachment else [],
|
||||||
attach_tickets=True,
|
attach_tickets=True,
|
||||||
attach_ical=event.settings.mail_attach_ical,
|
attach_ical=event.settings.mail_attach_ical and (not event.settings.mail_attach_ical_paid_only or is_free),
|
||||||
attach_other_files=[a for a in [
|
attach_other_files=[a for a in [
|
||||||
event.settings.get('mail_attachment_new_order', as_type=str, default='')[len('file://'):]
|
event.settings.get('mail_attachment_new_order', as_type=str, default='')[len('file://'):]
|
||||||
] if a],
|
] if a],
|
||||||
@@ -950,7 +950,7 @@ def _order_placed_email(event: Event, order: Order, pprov: BasePaymentProvider,
|
|||||||
logger.exception('Order received email could not be sent')
|
logger.exception('Order received email could not be sent')
|
||||||
|
|
||||||
|
|
||||||
def _order_placed_email_attendee(event: Event, order: Order, position: OrderPosition, email_template, log_entry: str):
|
def _order_placed_email_attendee(event: Event, order: Order, position: OrderPosition, email_template, log_entry: str, is_free=False):
|
||||||
email_context = get_email_context(event=event, order=order, position=position)
|
email_context = get_email_context(event=event, order=order, position=position)
|
||||||
email_subject = _('Your event registration: %(code)s') % {'code': order.code}
|
email_subject = _('Your event registration: %(code)s') % {'code': order.code}
|
||||||
|
|
||||||
@@ -961,7 +961,7 @@ def _order_placed_email_attendee(event: Event, order: Order, position: OrderPosi
|
|||||||
invoices=[],
|
invoices=[],
|
||||||
attach_tickets=True,
|
attach_tickets=True,
|
||||||
position=position,
|
position=position,
|
||||||
attach_ical=event.settings.mail_attach_ical,
|
attach_ical=event.settings.mail_attach_ical and (not event.settings.mail_attach_ical_paid_only or is_free),
|
||||||
attach_other_files=[a for a in [
|
attach_other_files=[a for a in [
|
||||||
event.settings.get('mail_attachment_new_order', as_type=str, default='')[len('file://'):]
|
event.settings.get('mail_attachment_new_order', as_type=str, default='')[len('file://'):]
|
||||||
] if a],
|
] if a],
|
||||||
@@ -1070,11 +1070,13 @@ def _perform_order(event: Event, payment_provider: str, position_ids: List[str],
|
|||||||
email_attendees_template = event.settings.mail_text_order_placed_attendee
|
email_attendees_template = event.settings.mail_text_order_placed_attendee
|
||||||
|
|
||||||
if sales_channel in event.settings.mail_sales_channel_placed_paid:
|
if sales_channel in event.settings.mail_sales_channel_placed_paid:
|
||||||
_order_placed_email(event, order, pprov, email_template, log_entry, invoice, payment)
|
_order_placed_email(event, order, pprov, email_template, log_entry, invoice, payment,
|
||||||
|
is_free=free_order_flow)
|
||||||
if email_attendees:
|
if email_attendees:
|
||||||
for p in order.positions.all():
|
for p in order.positions.all():
|
||||||
if p.addon_to_id is None and p.attendee_email and p.attendee_email != order.email:
|
if p.addon_to_id is None and p.attendee_email and p.attendee_email != order.email:
|
||||||
_order_placed_email_attendee(event, order, p, email_attendees_template, log_entry)
|
_order_placed_email_attendee(event, order, p, email_attendees_template, log_entry,
|
||||||
|
is_free=free_order_flow)
|
||||||
|
|
||||||
return order.id
|
return order.id
|
||||||
|
|
||||||
|
|||||||
@@ -1573,6 +1573,32 @@ DEFAULTS = {
|
|||||||
help_text=_("If enabled, we will attach an .ics calendar file to order confirmation emails."),
|
help_text=_("If enabled, we will attach an .ics calendar file to order confirmation emails."),
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
|
'mail_attach_ical_paid_only': {
|
||||||
|
'default': 'False',
|
||||||
|
'type': bool,
|
||||||
|
'form_class': forms.BooleanField,
|
||||||
|
'serializer_class': serializers.BooleanField,
|
||||||
|
'form_kwargs': dict(
|
||||||
|
label=_("Attach calendar files only after order has been paid"),
|
||||||
|
help_text=_("Use this if you e.g. put a private access link into the calendar file to make sure people only "
|
||||||
|
"receive it after their payment was confirmed."),
|
||||||
|
)
|
||||||
|
},
|
||||||
|
'mail_attach_ical_description': {
|
||||||
|
'default': '',
|
||||||
|
'type': LazyI18nString,
|
||||||
|
'form_class': I18nFormField,
|
||||||
|
'form_kwargs': dict(
|
||||||
|
label=_("Event description"),
|
||||||
|
widget=I18nTextarea,
|
||||||
|
help_text=_(
|
||||||
|
"You can use this to share information with your attendees, such as travel information or the link to a digital event. "
|
||||||
|
"If you keep it empty, we will put a link to the event shop, the admission time, and your organizer name in there. "
|
||||||
|
"We do not allow using placeholders with sensitive person-specific data as calendar entries are often shared with an "
|
||||||
|
"unspecified number of people."
|
||||||
|
),
|
||||||
|
)
|
||||||
|
},
|
||||||
'mail_prefix': {
|
'mail_prefix': {
|
||||||
'default': None,
|
'default': None,
|
||||||
'type': str,
|
'type': str,
|
||||||
|
|||||||
@@ -872,6 +872,8 @@ class MailSettingsForm(SettingsForm):
|
|||||||
'mail_attach_ical',
|
'mail_attach_ical',
|
||||||
'mail_attach_tickets',
|
'mail_attach_tickets',
|
||||||
'mail_attachment_new_order',
|
'mail_attachment_new_order',
|
||||||
|
'mail_attach_ical_paid_only',
|
||||||
|
'mail_attach_ical_description',
|
||||||
]
|
]
|
||||||
|
|
||||||
mail_sales_channel_placed_paid = forms.MultipleChoiceField(
|
mail_sales_channel_placed_paid = forms.MultipleChoiceField(
|
||||||
@@ -1079,7 +1081,8 @@ class MailSettingsForm(SettingsForm):
|
|||||||
'mail_text_download_reminder_attendee': ['event', 'order', 'position'],
|
'mail_text_download_reminder_attendee': ['event', 'order', 'position'],
|
||||||
'mail_text_resend_link': ['event', 'order'],
|
'mail_text_resend_link': ['event', 'order'],
|
||||||
'mail_text_waiting_list': ['event', 'waiting_list_entry'],
|
'mail_text_waiting_list': ['event', 'waiting_list_entry'],
|
||||||
'mail_text_resend_all_links': ['event', 'orders']
|
'mail_text_resend_all_links': ['event', 'orders'],
|
||||||
|
'mail_attach_ical_description': ['event', 'event_or_subevent'],
|
||||||
}
|
}
|
||||||
|
|
||||||
def _set_field_placeholders(self, fn, base_parameters):
|
def _set_field_placeholders(self, fn, base_parameters):
|
||||||
|
|||||||
@@ -14,7 +14,6 @@
|
|||||||
<legend>{% trans "General" %}</legend>
|
<legend>{% trans "General" %}</legend>
|
||||||
{% bootstrap_field form.mail_prefix layout="control" %}
|
{% bootstrap_field form.mail_prefix layout="control" %}
|
||||||
{% bootstrap_field form.mail_attach_tickets layout="control" %}
|
{% bootstrap_field form.mail_attach_tickets layout="control" %}
|
||||||
{% bootstrap_field form.mail_attach_ical layout="control" %}
|
|
||||||
{% url "control:organizer.settings.mail" organizer=request.organizer.slug as org_url %}
|
{% url "control:organizer.settings.mail" organizer=request.organizer.slug as org_url %}
|
||||||
{% propagated request.event org_url "mail_from" "smtp_use_custom" "smtp_host" "smtp_port" "smtp_username" "smtp_password" "smtp_use_tls" "smtp_use_ssl" %}
|
{% propagated request.event org_url "mail_from" "smtp_use_custom" "smtp_host" "smtp_port" "smtp_username" "smtp_password" "smtp_use_tls" "smtp_use_ssl" %}
|
||||||
<div class="form-group">
|
<div class="form-group">
|
||||||
@@ -57,6 +56,12 @@
|
|||||||
{% endpropagated %}
|
{% endpropagated %}
|
||||||
{% bootstrap_field form.mail_sales_channel_placed_paid layout="control" %}
|
{% bootstrap_field form.mail_sales_channel_placed_paid layout="control" %}
|
||||||
</fieldset>
|
</fieldset>
|
||||||
|
<fieldset>
|
||||||
|
<legend>{% trans "Calender invites" %}</legend>
|
||||||
|
{% bootstrap_field form.mail_attach_ical layout="control" %}
|
||||||
|
{% bootstrap_field form.mail_attach_ical_paid_only layout="control" %}
|
||||||
|
{% bootstrap_field form.mail_attach_ical_description layout="control" %}
|
||||||
|
</fieldset>
|
||||||
<fieldset>
|
<fieldset>
|
||||||
<legend>{% trans "E-mail design" %}</legend>
|
<legend>{% trans "E-mail design" %}</legend>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
|
|||||||
@@ -28,11 +28,16 @@ from django.conf import settings
|
|||||||
from django.utils.formats import date_format
|
from django.utils.formats import date_format
|
||||||
from django.utils.translation import gettext as _
|
from django.utils.translation import gettext as _
|
||||||
|
|
||||||
|
from pretix.base.email import get_email_context
|
||||||
from pretix.base.models import Event
|
from pretix.base.models import Event
|
||||||
from pretix.multidomain.urlreverse import build_absolute_uri
|
from pretix.multidomain.urlreverse import build_absolute_uri
|
||||||
|
|
||||||
|
|
||||||
def get_ical(events):
|
def get_public_ical(events):
|
||||||
|
"""
|
||||||
|
Return an ical feed for a sequence of events or subevents. The calendar files will only include public
|
||||||
|
information.
|
||||||
|
"""
|
||||||
cal = vobject.iCalendar()
|
cal = vobject.iCalendar()
|
||||||
cal.add('prodid').value = '-//pretix//{}//'.format(settings.PRETIX_INSTANCE_NAME.replace(" ", "_"))
|
cal.add('prodid').value = '-//pretix//{}//'.format(settings.PRETIX_INSTANCE_NAME.replace(" ", "_"))
|
||||||
creation_time = datetime.datetime.now(pytz.utc)
|
creation_time = datetime.datetime.now(pytz.utc)
|
||||||
@@ -83,3 +88,91 @@ def get_ical(events):
|
|||||||
|
|
||||||
vevent.add('description').value = '\n'.join(descr)
|
vevent.add('description').value = '\n'.join(descr)
|
||||||
return cal
|
return cal
|
||||||
|
|
||||||
|
|
||||||
|
def get_private_icals(event, positions):
|
||||||
|
"""
|
||||||
|
Return a list of ical objects based on a sequence of positions.
|
||||||
|
|
||||||
|
Unlike get_public_ical, this will
|
||||||
|
|
||||||
|
- Generate multiple ical files instead of one (but with deduplication applied)
|
||||||
|
- Respect the mail_attach_ical_description setting
|
||||||
|
|
||||||
|
It is private in the sense that mail_attach_ical_description may contain content not suited for
|
||||||
|
public display.
|
||||||
|
|
||||||
|
We however intentionally do not allow using placeholders based on the order and position
|
||||||
|
specifically. This is for two reasons:
|
||||||
|
|
||||||
|
- In reality, many people will add their invite to their calendar which is shared with a larger
|
||||||
|
team. People are probably not aware that they're sharing sensitive information such as their
|
||||||
|
secret ticket link with everyone they share their calendar with.
|
||||||
|
|
||||||
|
- It would be pretty hard to implement it in a way that doesn't require us to use distinct
|
||||||
|
settings fields for emails to customers and to attendees, which feels like an overcomplication.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from pretix.base.services.mail import TolerantDict
|
||||||
|
|
||||||
|
tz = pytz.timezone(event.settings.timezone)
|
||||||
|
|
||||||
|
creation_time = datetime.datetime.now(pytz.utc)
|
||||||
|
calobjects = []
|
||||||
|
|
||||||
|
evs = set(p.subevent or event for p in positions)
|
||||||
|
for ev in evs:
|
||||||
|
if isinstance(ev, Event):
|
||||||
|
url = build_absolute_uri(event, 'presale:event.index')
|
||||||
|
else:
|
||||||
|
url = build_absolute_uri(event, 'presale:event.index', {
|
||||||
|
'subevent': ev.pk
|
||||||
|
})
|
||||||
|
|
||||||
|
if event.settings.mail_attach_ical_description:
|
||||||
|
ctx = get_email_context(event=event, event_or_subevent=ev)
|
||||||
|
description = str(event.settings.mail_attach_ical_description).format_map(TolerantDict(ctx))
|
||||||
|
else:
|
||||||
|
# Default description
|
||||||
|
descr = []
|
||||||
|
descr.append(_('Tickets: {url}').format(url=url))
|
||||||
|
if ev.date_admission:
|
||||||
|
descr.append(str(_('Admission: {datetime}')).format(
|
||||||
|
datetime=date_format(ev.date_admission.astimezone(tz), 'SHORT_DATETIME_FORMAT')
|
||||||
|
))
|
||||||
|
|
||||||
|
descr.append(_('Organizer: {organizer}').format(organizer=event.organizer.name))
|
||||||
|
description = '\n'.join(descr)
|
||||||
|
|
||||||
|
cal = vobject.iCalendar()
|
||||||
|
cal.add('prodid').value = '-//pretix//{}//'.format(settings.PRETIX_INSTANCE_NAME.replace(" ", "_"))
|
||||||
|
|
||||||
|
vevent = cal.add('vevent')
|
||||||
|
vevent.add('summary').value = str(ev.name)
|
||||||
|
vevent.add('description').value = description
|
||||||
|
vevent.add('dtstamp').value = creation_time
|
||||||
|
if ev.location:
|
||||||
|
vevent.add('location').value = str(ev.location)
|
||||||
|
|
||||||
|
vevent.add('uid').value = 'pretix-{}-{}-{}@{}'.format(
|
||||||
|
event.organizer.slug,
|
||||||
|
event.organizer.slug, event.slug,
|
||||||
|
ev.pk if not isinstance(ev, Event) else '0',
|
||||||
|
urlparse(url).netloc
|
||||||
|
)
|
||||||
|
|
||||||
|
if event.settings.show_times:
|
||||||
|
vevent.add('dtstart').value = ev.date_from.astimezone(tz)
|
||||||
|
else:
|
||||||
|
vevent.add('dtstart').value = ev.date_from.astimezone(tz).date()
|
||||||
|
|
||||||
|
if event.settings.show_date_to and ev.date_to:
|
||||||
|
if event.settings.show_times:
|
||||||
|
vevent.add('dtend').value = ev.date_to.astimezone(tz)
|
||||||
|
else:
|
||||||
|
# with full-day events date_to in pretix is included (e.g. last day)
|
||||||
|
# whereas dtend in vcalendar is non-inclusive => add one day for export
|
||||||
|
vevent.add('dtend').value = ev.date_to.astimezone(tz).date() + datetime.timedelta(days=1)
|
||||||
|
|
||||||
|
calobjects.append(cal)
|
||||||
|
return calobjects
|
||||||
|
|||||||
@@ -70,7 +70,7 @@ from pretix.base.models.items import (
|
|||||||
from pretix.base.services.quotas import QuotaAvailability
|
from pretix.base.services.quotas import QuotaAvailability
|
||||||
from pretix.helpers.compat import date_fromisocalendar
|
from pretix.helpers.compat import date_fromisocalendar
|
||||||
from pretix.multidomain.urlreverse import eventreverse
|
from pretix.multidomain.urlreverse import eventreverse
|
||||||
from pretix.presale.ical import get_ical
|
from pretix.presale.ical import get_public_ical
|
||||||
from pretix.presale.signals import item_description
|
from pretix.presale.signals import item_description
|
||||||
from pretix.presale.views.organizer import (
|
from pretix.presale.views.organizer import (
|
||||||
EventListMixin, add_subevents_for_days, days_for_template,
|
EventListMixin, add_subevents_for_days, days_for_template,
|
||||||
@@ -719,7 +719,7 @@ class EventIcalDownload(EventViewMixin, View):
|
|||||||
raise Http404(pgettext_lazy('subevent', 'Unknown date selected.'))
|
raise Http404(pgettext_lazy('subevent', 'Unknown date selected.'))
|
||||||
|
|
||||||
event = self.request.event
|
event = self.request.event
|
||||||
cal = get_ical([subevent or event])
|
cal = get_public_ical([subevent or event])
|
||||||
|
|
||||||
resp = HttpResponse(cal.serialize(), content_type='text/calendar')
|
resp = HttpResponse(cal.serialize(), content_type='text/calendar')
|
||||||
resp['Content-Disposition'] = 'attachment; filename="{}-{}-{}.ics"'.format(
|
resp['Content-Disposition'] = 'attachment; filename="{}-{}-{}.ics"'.format(
|
||||||
|
|||||||
@@ -67,7 +67,7 @@ from pretix.helpers.formats.en.formats import (
|
|||||||
SHORT_MONTH_DAY_FORMAT, WEEK_FORMAT,
|
SHORT_MONTH_DAY_FORMAT, WEEK_FORMAT,
|
||||||
)
|
)
|
||||||
from pretix.multidomain.urlreverse import eventreverse
|
from pretix.multidomain.urlreverse import eventreverse
|
||||||
from pretix.presale.ical import get_ical
|
from pretix.presale.ical import get_public_ical
|
||||||
from pretix.presale.views import OrganizerViewMixin
|
from pretix.presale.views import OrganizerViewMixin
|
||||||
|
|
||||||
|
|
||||||
@@ -1159,9 +1159,9 @@ class OrganizerIcalDownload(OrganizerViewMixin, View):
|
|||||||
|
|
||||||
if 'locale' in request.GET and request.GET.get('locale') in dict(settings.LANGUAGES):
|
if 'locale' in request.GET and request.GET.get('locale') in dict(settings.LANGUAGES):
|
||||||
with language(request.GET.get('locale'), self.request.organizer.settings.region):
|
with language(request.GET.get('locale'), self.request.organizer.settings.region):
|
||||||
cal = get_ical(events)
|
cal = get_public_ical(events)
|
||||||
else:
|
else:
|
||||||
cal = get_ical(events)
|
cal = get_public_ical(events)
|
||||||
|
|
||||||
resp = HttpResponse(cal.serialize(), content_type='text/calendar')
|
resp = HttpResponse(cal.serialize(), content_type='text/calendar')
|
||||||
resp['Content-Disposition'] = 'attachment; filename="{}.ics"'.format(
|
resp['Content-Disposition'] = 'attachment; filename="{}.ics"'.format(
|
||||||
|
|||||||
Reference in New Issue
Block a user