+# This file is part of Wolnelektury, licensed under GNU Affero GPLv3 or later.
+# Copyright © Fundacja Nowoczesna Polska. See NOTICE for more information.
+#
from django.conf import settings
-from django.contrib.auth.decorators import login_required
+from django.contrib.auth.decorators import login_required, permission_required
+from django.db.models import Sum
from django.http import HttpResponseRedirect
from django.shortcuts import get_object_or_404, render
+from django.urls import reverse
from django.utils.decorators import method_decorator
from django.views.decorators.cache import never_cache
from django.views.generic import FormView, CreateView, TemplateView, DetailView, UpdateView
from .payu import POSS
from .payu import views as payu_views
from .forms import ScheduleForm, PayUCardTokenForm
+from . import forms
from . import models
from .helpers import get_active_schedule
-from .payment_methods import payure_method
+from .payment_methods import recurring_payment_method
class ClubView(TemplateView):
return ctx
-class JoinView(CreateView):
- form_class = ScheduleForm
- template_name = 'club/membership_form.html'
+class DonationStep1(UpdateView):
+ queryset = models.Schedule.objects.filter(payed_at=None)
+ form_class = forms.DonationStep1Form
+ slug_field = slug_url_kwarg = 'key'
+ template_name = 'club/2022/donation_step1.html'
+ step = 1
+
+ def get_context_data(self, **kwargs):
+ c = super().get_context_data(**kwargs)
+ c['club'] = models.Club.objects.first()
+ return c
+
+ def get_success_url(self):
+ return reverse('donation_step2', args=[self.object.key])
+
+
+class DonationStep2(UpdateView):
+ queryset = models.Schedule.objects.filter(payed_at=None)
+ form_class = forms.DonationStep2Form
+ slug_field = slug_url_kwarg = 'key'
+ template_name = 'club/2022/donation_step2.html'
+ step = 2
+
+
+class JoinView(CreateView):
+ @property
+ def club(self):
+ return models.Club.objects.first()
+
+ @property
+ def new_layout(self):
+ return self.request.EXPERIMENTS['layout'].value
+
+ def get_template_names(self):
+ if self.new_layout:
+ return 'club/2022/donation_step1.html'
+ return 'club/membership_form.html'
+
+ def get_form_class(self):
+ if self.new_layout:
+ return forms.DonationStep1Form
+ return ScheduleForm
+
def is_app(self):
return self.request.GET.get('app')
def get(self, request):
- # TODO: configure as app-allowed hosts.
if settings.CLUB_APP_HOST and self.is_app() and request.META['HTTP_HOST'] != settings.CLUB_APP_HOST:
return HttpResponseRedirect("https://" + settings.CLUB_APP_HOST + request.get_full_path())
request.session['from_app'] = True
elif request.session and 'from_app' in request.session:
del request.session['from_app']
- schedule = get_active_schedule(request.user)
- if schedule is not None:
- return HttpResponseRedirect(schedule.get_absolute_url())
- else:
- return super(JoinView, self).get(request)
- def get_form_kwargs(self):
- kwargs = super().get_form_kwargs()
- kwargs['request'] = self.request
- return kwargs
+ return super().get(request)
def get_context_data(self, **kwargs):
c = super(JoinView, self).get_context_data(**kwargs)
c['membership'] = getattr(self.request.user, 'membership', None)
c['active_menu_item'] = 'club'
+ c['club'] = models.Club.objects.first()
+
+ c['ambassador'] = models.Ambassador.objects.all().order_by('?').first()
return c
def get_initial(self):
if self.request.user.is_authenticated and self.request.user.email:
return {
- 'email': self.request.user.email,
+ 'email': self.request.user.email
}
+ def get_form_kwargs(self):
+ kwargs = super().get_form_kwargs()
+ if not self.new_layout:
+ kwargs['referer'] = self.request.META.get('HTTP_REFERER', '')
+ return kwargs
+
def form_valid(self, form):
retval = super(JoinView, self).form_valid(form)
if self.request.user.is_authenticated:
return retval
def get_success_url(self):
+ if self.new_layout:
+ return reverse('donation_step2', args=[self.object.key])
return self.object.initiate_payment(self.request)
@method_decorator(never_cache, name='dispatch')
class ScheduleView(DetailView):
- model = models.Schedule
+ queryset = models.Schedule.objects.exclude(email='')
slug_field = slug_url_kwarg = 'key'
template_name = 'club/schedule.html'
-
+ step = 3
+
+ def get_template_names(self):
+ if self.request.EXPERIMENTS['layout'].value:
+ if not self.object.payed_at:
+ return 'club/2022/donation_step3.html'
+ return 'club/schedule.html'
+
def get_context_data(self, *args, **kwargs):
ctx = super().get_context_data(*args, **kwargs)
ctx['active_menu_item'] = 'club'
def post(self, request, key):
schedule = models.Schedule.objects.get(key=key)
schedule.create_payment()
+
return HttpResponseRedirect(schedule.get_absolute_url())
-class PayUPayment(payu_views.Payment):
- pass
+class PayUPayment(DetailView):
+ model = models.Schedule
+ slug_field = slug_url_kwarg = 'key'
+
+ def get(self, request, key):
+ schedule = self.get_object()
+ schedule.method = 'payu'
+ schedule.save(update_fields=['method'])
+ return HttpResponseRedirect(schedule.initiate_payment(request))
+
class PayURecPayment(payu_views.RecPayment):
return get_object_or_404(models.Schedule, key=self.kwargs['key'])
def get_pos(self):
- pos_id = payure_method.pos_id
+ pos_id = recurring_payment_method.pos_id
return POSS[pos_id]
def get_success_url(self):
- return self.get_schedule().pay(self.request)
+ schedule = self.get_schedule()
+ schedule.method = 'payu-re'
+ schedule.save(update_fields=['method'])
+ return schedule.pay(self.request)
class PayUNotifyView(payu_views.NotifyView):
order_model = models.PayUOrder
-class MembershipView(UpdateView):
- fields = ['name']
-
- def get_success_url(self):
- # TODO: get only current schedule if multiple.
- return self.object.schedule_set.first().get_absolute_url()
+class ScheduleThanksView(DetailView):
+ model = models.Schedule
+ slug_field = slug_url_kwarg = 'key'
+ step = 4
- def get_object(self):
- return self.request.user.membership
+ def get_template_names(self):
+ if self.request.EXPERIMENTS['layout'].value:
+ return 'club/2022/donation_step4.html'
+ return 'club/thanks.html'
+
+ def get_context_data(self, *args, **kwargs):
+ ctx = super().get_context_data(*args, **kwargs)
+ ctx['active_menu_item'] = 'club'
+ return ctx
-class ScheduleThanksView(DetailView):
+class YearSummaryView(DetailView):
model = models.Schedule
slug_field = slug_url_kwarg = 'key'
- template_name = 'club/thanks.html'
+ template_name = 'club/year_summary.html'
def get_context_data(self, *args, **kwargs):
ctx = super().get_context_data(*args, **kwargs)
- ctx['active_menu_item'] = 'club'
+ ctx['payments'] = models.PayUOrder.objects.filter(
+ status='COMPLETED',
+ completed_at__year=self.kwargs['year'],
+ schedule__email=self.object.email,
+ ).order_by('completed_at')
+ ctx['total_amount'] = ctx['payments'].aggregate(s=Sum('schedule__amount'))['s']
return ctx
+
+@permission_required('club.schedule_view')
+def member_verify(request):
+ if request.method == 'POST':
+ emails = request.POST.get('emails').strip().split('\n')
+ rows = ['email;członek;nazwa użytkownika;aktywny;co najmniej do']
+ for email in emails:
+ email = email.strip()
+ row = [email]
+ schedules = models.Schedule.objects.filter(email=email).exclude(payed_at=None)
+ if schedules.exists():
+ row.append('tak')
+ akt = False
+ unames = set()
+ exp = None
+ for s in schedules:
+ if s.is_active():
+ akt = True
+ if s.membership:
+ unames.add(s.membership.user.username)
+ if exp is None or s.expires_at > exp:
+ exp = s.expires_at
+ row.append(','.join(sorted(unames)))
+ row.append('tak' if akt else 'nie')
+ row.append(exp.date().isoformat())
+ else:
+ row.append('nie')
+ rows.append(';'.join(row))
+ rows = '\n'.join(rows)
+ else:
+ rows = ''
+
+ return render(
+ request,
+ 'club/member_verify.html',
+ {
+ 'result': rows
+ }
+ )