some missing contrib files
[wolnelektury.git] / src / club / views.py
index 9816b4b..e80fcf9 100644 (file)
-from django.contrib.auth.decorators import login_required
+# 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, 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 django.views import View
 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):
     template_name = 'club/index.html'
 
+    def get_context_data(self, *args, **kwargs):
+        ctx = super().get_context_data(*args, **kwargs)
+        ctx['active_menu_item'] = 'club'
+        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
+
+    def get_context_data(self, **kwargs):
+        c = super().get_context_data(**kwargs)
+        c['club'] = models.Club.objects.first()
+        return c
+
+
+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):
+        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())
+
         if self.is_app():
             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, form=None):
-        c = super(JoinView, self).get_context_data()
+    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:
@@ -57,13 +121,28 @@ class JoinView(CreateView):
         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'
+        return ctx
 
     def post(self, request, key):
         schedule = self.get_object()
@@ -96,11 +175,20 @@ class DummyPaymentView(TemplateView):
     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):
@@ -110,23 +198,87 @@ 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']
+class ScheduleThanksView(DetailView):
+    model = models.Schedule
+    slug_field = slug_url_kwarg = 'key'
+    step = 4
 
-    def get_success_url(self):
-        # TODO: get only current schedule if multiple.
-        return self.object.schedule_set.first().get_absolute_url()
+    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
 
-    def get_object(self):
-        return self.request.user.membership
+
+class YearSummaryView(DetailView):
+    model = models.Schedule
+    slug_field = slug_url_kwarg = 'key'
+    template_name = 'club/year_summary.html'
+
+    def get_context_data(self, *args, **kwargs):
+        ctx = super().get_context_data(*args, **kwargs)
+        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
+        }
+    )