# -*- coding:utf-8 -*- # # Copyright 2023 © AE UTBM # ae@utbm.fr / ae.info@utbm.fr # All contributors are listed in the CONTRIBUTORS file. # # This file is part of the website of the UTBM Student Association (AE UTBM), # https://ae.utbm.fr. # # You can find the whole source code at https://github.com/ae-utbm/sith3 # # LICENSED UNDER THE GNU GENERAL PUBLIC LICENSE VERSION 3 (GPLv3) # SEE : https://raw.githubusercontent.com/ae-utbm/sith3/master/LICENSE # OR WITHIN THE LOCAL FILE "LICENSE" # # PREVIOUSLY LICENSED UNDER THE MIT LICENSE, # SEE : https://raw.githubusercontent.com/ae-utbm/sith3/master/LICENSE.old # OR WITHIN THE LOCAL FILE "LICENSE.old" # from ast import literal_eval from enum import Enum from django import forms from django.http.response import HttpResponseRedirect from django.urls import reverse from django.utils.translation import gettext_lazy as _ from django.views.generic import ListView, View from django.views.generic.detail import SingleObjectMixin from django.views.generic.edit import FormView from phonenumber_field.widgets import PhoneNumberInternationalFallbackWidget from core.models import User from core.views import FormerSubscriberMixin, search_user from core.views.forms import SelectDate # Enum to select search type class SearchType(Enum): NORMAL = 1 REVERSE = 2 QUICK = 3 # Custom form class SearchForm(forms.ModelForm): class Meta: model = User fields = [ "first_name", "last_name", "nick_name", "role", "department", "semester", "promo", "date_of_birth", "phone", ] widgets = { "date_of_birth": SelectDate, "phone": PhoneNumberInternationalFallbackWidget, } quick = forms.CharField(label=_("Last/First name or nickname"), max_length=255) def __init__(self, *args, **kwargs): super(SearchForm, self).__init__(*args, **kwargs) for key in self.fields.keys(): self.fields[key].required = False @property def cleaned_data_json(self): data = self.cleaned_data for key in data.keys(): if key in ("date_of_birth", "phone") and data[key] is not None: data[key] = str(data[key]) return data # Views class SearchFormListView(FormerSubscriberMixin, SingleObjectMixin, ListView): model = User ordering = ["-id"] paginate_by = 12 template_name = "matmat/search_form.jinja" def dispatch(self, request, *args, **kwargs): self.form_class = kwargs["form"] self.search_type = kwargs["search_type"] self.session = request.session self.last_search = self.session.get("matmat_search_result", str([])) self.last_search = literal_eval(self.last_search) if "valid_form" in kwargs.keys(): self.valid_form = kwargs["valid_form"] else: self.valid_form = None self.init_query = self.model.objects self.can_see_hidden = True if not (request.user.is_board_member or request.user.is_root): self.can_see_hidden = False self.init_query = self.init_query.exclude(is_subscriber_viewable=False) return super(SearchFormListView, self).dispatch(request, *args, **kwargs) def post(self, request, *args, **kwargs): return self.get(request, *args, **kwargs) def get_context_data(self, **kwargs): self.object = None kwargs = super(SearchFormListView, self).get_context_data(**kwargs) kwargs["form"] = self.form_class kwargs["result_exists"] = self.result_exists return kwargs def get_queryset(self): q = self.init_query if self.valid_form is not None: if self.search_type == SearchType.REVERSE: q = q.filter(phone=self.valid_form["phone"]).all() elif self.search_type == SearchType.QUICK: if self.valid_form["quick"].strip(): q = search_user(self.valid_form["quick"]) else: q = [] if not self.can_see_hidden and len(q) > 0: q = [user for user in q if user.is_subscriber_viewable] else: search_dict = {} for key, value in self.valid_form.items(): if key not in ("phone", "quick") and not ( value == "" or value is None ): search_dict[key + "__icontains"] = value q = q.filter(**search_dict).all() else: q = q.filter(pk__in=self.last_search).all() if isinstance(q, list): self.result_exists = len(q) > 0 else: self.result_exists = q.exists() self.last_search = [] for user in q: self.last_search.append(user.id) self.session["matmat_search_result"] = str(self.last_search) return q class SearchFormView(FormerSubscriberMixin, FormView): """ Allows users to search inside the user list """ form_class = SearchForm def dispatch(self, request, *args, **kwargs): self.session = request.session self.init_query = User.objects kwargs["form"] = self.get_form() kwargs["search_type"] = self.search_type return super(SearchFormView, self).dispatch(request, *args, **kwargs) def get(self, request, *args, **kwargs): view = SearchFormListView.as_view() return view(request, *args, **kwargs) def post(self, request, *args, **kwargs): form = self.get_form() view = SearchFormListView.as_view() if form.is_valid(): kwargs["valid_form"] = form.clean() request.session["matmat_search_form"] = form.cleaned_data_json return view(request, *args, **kwargs) def get_initial(self): init = self.session.get("matmat_search_form", {}) if not init: init["department"] = "" return init class SearchNormalFormView(SearchFormView): search_type = SearchType.NORMAL class SearchReverseFormView(SearchFormView): search_type = SearchType.REVERSE class SearchQuickFormView(SearchFormView): search_type = SearchType.QUICK class SearchClearFormView(FormerSubscriberMixin, View): """ Clear SearchFormView and redirect to it """ def dispatch(self, request, *args, **kwargs): super(SearchClearFormView, self).dispatch(request, *args, **kwargs) if "matmat_search_form" in request.session.keys(): request.session.pop("matmat_search_form") if "matmat_search_result" in request.session.keys(): request.session.pop("matmat_search_result") return HttpResponseRedirect(reverse("matmat:search"))