# Copyright (C) 2018-2024 by BDE ENS Paris-Saclay # SPDX-License-Identifier: GPL-3.0-or-later from datetime import timedelta from django.db import transaction from django.contrib.auth.mixins import LoginRequiredMixin from django.http import HttpResponseRedirect from django_tables2.views import SingleTableView from django.urls import reverse from django.utils.translation import gettext_lazy as _ from django.utils import timezone from django.views.generic import DetailView, UpdateView, TemplateView from django.views.generic.edit import FormView from permission.views import ProtectQuerysetMixin, ProtectedCreateView from .forms import AddIngredientForms, BasicFoodForms, QRCodeForms, TransformedFoodForms from .models import BasicFood, Food, QRCode, TransformedFood from .tables import TransformedFoodTable class AddIngredientView(ProtectQuerysetMixin, FormView): """ A view to see a qrcode """ model = Food template_name = 'food/add_ingredient_form.html' extra_context = {"title": _("Add the ingredient")} form_class = AddIngredientForms def get_context_data(self, **kwargs): context = super().get_context_data(**kwargs) context["pk"] = self.kwargs["pk"] return context @transaction.atomic def form_valid(self, form): form.instance.creater = self.request.user add_ingredient_form = AddIngredientForms(data=self.request.POST) if not add_ingredient_form.is_valid(): return self.form_invalid(form) food = Food.objects.get(pk=self.kwargs['pk']) # Save the aliment and the allergens associed for transformed_pk in self.request.POST.getlist('ingredient'): transformed = TransformedFood.objects.get(pk=transformed_pk) transformed.ingredient.add(food) transformed.update() return super().form_valid(form) def get_success_url(self, **kwargs): return reverse('food:food_list') def get_sample_object(self): return TransformedFood( name="", creation_date=timezone.now(), ) class BasicFoodUpdateView(ProtectQuerysetMixin, LoginRequiredMixin, UpdateView): """ A view to add a basic food """ model = BasicFood form_class = BasicFoodForms template_name = 'food/basic_food_form.html' extra_context = {"title": _("Add a new aliment")} @transaction.atomic def form_valid(self, form): form.instance.creater = self.request.user basic_food_form = BasicFoodForms(data=self.request.POST) if not basic_food_form.is_valid(): return self.form_invalid(form) ans = super().form_valid(form) form.instance.update() return ans def get_success_url(self, **kwargs): self.object.refresh_from_db() return reverse('food:food_view', kwargs={"pk": self.object.pk}) class FoodCreateView(ProtectQuerysetMixin, LoginRequiredMixin, TemplateView): """ A view to add a new aliment """ template_name = 'food/create_food_form.html' extra_context = {"title": _("Add a new aliment")} class FoodView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView): """ A view to see a food """ model = Food extra_context = {"title": _("Details")} context_object_name = "food" class QRCodeBasicFoodCreateView(ProtectQuerysetMixin, ProtectedCreateView): ##################################################################### # TO DO # - fix picture save # - implement solution crop and convert image (reuse or recode ImageForm from members apps) ##################################################################### """ A view to add a basic food with a qrcode """ model = BasicFood form_class = BasicFoodForms template_name = 'food/basic_food_form.html' extra_context = {"title": _("Add a new basic food with QRCode")} @transaction.atomic def form_valid(self, form): form.instance.creater = self.request.user basic_food_form = BasicFoodForms(data=self.request.POST) if not basic_food_form.is_valid(): return self.form_invalid(form) # Save the aliment and the allergens associed basic_food = form.save(commit=False) # We assume the date of labeling and the same as the date of arrival basic_food.arrival_date = timezone.now() basic_food.is_ready = True basic_food._force_save = True basic_food.save() basic_food.refresh_from_db() qrcode = QRCode() qrcode.qr_code_number = self.kwargs['slug'] qrcode.food_container = basic_food qrcode.save() return super().form_valid(form) def get_success_url(self, **kwargs): self.object.refresh_from_db() return reverse('food:qrcode_view', kwargs={"slug": self.kwargs['slug']}) def get_sample_object(self): return BasicFood( name="", expiry_date=timezone.now(), ) class QRCodeCreateView(ProtectQuerysetMixin, ProtectedCreateView): """ A view to add a new qrcode """ model = QRCode template_name = 'food/create_qrcode_form.html' form_class = QRCodeForms extra_context = {"title": _("Add a new QRCode")} def get(self, *args, **kwargs): qrcode = kwargs["slug"] if self.model.objects.filter(qr_code_number=qrcode).count() > 0: return HttpResponseRedirect(reverse("food:qrcode_view", kwargs=kwargs)) else: return super().get(*args, **kwargs) def get_context_data(self, **kwargs): context = super().get_context_data(**kwargs) context["slug"] = self.kwargs["slug"] return context @transaction.atomic def form_valid(self, form): form.instance.creater = self.request.user qrcode_food_form = QRCodeForms(data=self.request.POST) if not qrcode_food_form.is_valid(): return self.form_invalid(form) # Save the qrcode qrcode = form.save(commit=False) qrcode.qr_code_number = self.kwargs["slug"] qrcode._force_save = True qrcode.save() qrcode.refresh_from_db() qrcode.food_container.is_ready = True qrcode.food_container.save() return super().form_valid(form) def get_success_url(self, **kwargs): self.object.refresh_from_db() return reverse('food:qrcode_view', kwargs={"slug": self.kwargs['slug']}) def get_sample_object(self): return QRCode( qr_code_number=self.kwargs["slug"], ) class QRCodeView(ProtectQuerysetMixin, LoginRequiredMixin, DetailView): """ A view to see a qrcode """ model = QRCode extra_context = {"title": _("QRCode")} context_object_name = "qrcode" slug_field = "qr_code_number" def get(self, *args, **kwargs): qrcode = kwargs["slug"] if self.model.objects.filter(qr_code_number=qrcode).count() > 0: return super().get(*args, **kwargs) else: return HttpResponseRedirect(reverse("food:qrcode_create", kwargs=kwargs)) class TransformedFoodFormView(ProtectQuerysetMixin): """ A view to add a tranformed food """ model = TransformedFood template_name = 'food/transformed_food_form.html' form_class = TransformedFoodForms extra_context = {"title": _("Add a new meal")} @transaction.atomic def form_valid(self, form): form.instance.creater = self.request.user transformed_food_form = TransformedFoodForms(data=self.request.POST) if not transformed_food_form.is_valid(): return self.form_invalid(form) # Save the aliment and allergens associated transformed_food = form.save(commit=False) # Without microbiological analyzes, the storage time is 3 days transformed_food.expiry_date = transformed_food.creation_date + timedelta(days=3) transformed_food._force_save = True transformed_food.save() transformed_food.refresh_from_db() return super().form_valid(form) def get_success_url(self, **kwargs): self.object.refresh_from_db() return reverse('food:food_view', kwargs={"pk": self.object.pk}) class TransformedFoodUpdateView(TransformedFoodFormView, LoginRequiredMixin, UpdateView): pass class TransformedFoodCreateView(TransformedFoodFormView, ProtectedCreateView): def get_sample_object(self): return TransformedFood( name="", creation_date=timezone.now(), ) class TransfomedListView(ProtectQuerysetMixin, LoginRequiredMixin, SingleTableView): """ Displays all Activities, and classify if they are on-going or upcoming ones. """ model = TransformedFood table_class = TransformedFoodTable ordering = ('-name',) extra_context = {"title": _("Transformed food")} def get_queryset(self, **kwargs): return super().get_queryset(**kwargs)\ .filter(is_ready=False)\ .distinct()