mirror of
https://gitlab.crans.org/bde/nk20
synced 2025-06-21 18:08:21 +02:00
Compare commits
13 Commits
New_permis
...
a3073ba5a5
Author | SHA1 | Date | |
---|---|---|---|
a3073ba5a5 | |||
9b9fa0bcfe | |||
b1d0cf92b1 | |||
0c3e712f8f | |||
708216a67f | |||
c27a8fefe5 | |||
c8afee91d2 | |||
aaa6076e9b | |||
77233e995e | |||
c9980b0bd1 | |||
4e6ec16e94 | |||
89785ce632 | |||
b636ca49d1 |
0
apps/food/__init__.py
Normal file
0
apps/food/__init__.py
Normal file
32
apps/food/admin.py
Normal file
32
apps/food/admin.py
Normal file
@ -0,0 +1,32 @@
|
||||
# Copyright (C) 2018-2024 by BDE ENS Paris-Saclay
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
from django.contrib import admin
|
||||
from note_kfet.admin import admin_site
|
||||
|
||||
from .models import QR_code, BasicFood, TransformedFood, Allergen
|
||||
|
||||
|
||||
@admin.register(QR_code, site = admin_site)
|
||||
class QR_codeAdmin(admin.ModelAdmin):
|
||||
"""
|
||||
TEMPORARY
|
||||
"""
|
||||
|
||||
@admin.register(BasicFood, site = admin_site)
|
||||
class Basic_foodAdmin(admin.ModelAdmin):
|
||||
"""
|
||||
TEMPORARY
|
||||
"""
|
||||
|
||||
@admin.register(TransformedFood, site = admin_site)
|
||||
class Transformed_foodAdmin(admin.ModelAdmin):
|
||||
"""
|
||||
TEMPORARY
|
||||
"""
|
||||
|
||||
@admin.register(Allergen, site = admin_site)
|
||||
class AllergenAdmin(admin.ModelAdmin):
|
||||
"""
|
||||
TEMPORARY
|
||||
"""
|
10
apps/food/apps.py
Normal file
10
apps/food/apps.py
Normal file
@ -0,0 +1,10 @@
|
||||
# Copyright (C) 2018-2024 by BDE ENS Paris-Saclay
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
|
||||
from django.apps import AppConfig
|
||||
|
||||
|
||||
class FoodkfetConfig(AppConfig):
|
||||
name = 'food'
|
||||
verbose_name = _('food')
|
92
apps/food/forms.py
Normal file
92
apps/food/forms.py
Normal file
@ -0,0 +1,92 @@
|
||||
# Copyright (C) 2018-2024 by BDE ENS Paris-Saclay
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
from random import shuffle
|
||||
|
||||
from django import forms
|
||||
from django.utils.translation import gettext_lazy as _
|
||||
from django.utils import timezone
|
||||
from member.models import Club
|
||||
from note_kfet.inputs import Autocomplete, DatePickerInput, DateTimePickerInput
|
||||
from note_kfet.middlewares import get_current_request
|
||||
from permission.backends import PermissionBackend
|
||||
|
||||
from .models import QR_code, Allergen, BasicFood, TransformedFood
|
||||
|
||||
|
||||
class BasicFoodForms(forms.ModelForm):
|
||||
"""
|
||||
Form for add non-transformed food
|
||||
"""
|
||||
def __init__(self, *args, **kwargs):
|
||||
super().__init__(*args, **kwargs)
|
||||
self.fields['name'].widget.attrs.update({"autofocus": "autofocus"})
|
||||
self.fields['name'].required = True
|
||||
self.fields['owner'].required = True
|
||||
self.fields['label'].help_text = _('The lot number must be contained in the picture')
|
||||
|
||||
# Some example
|
||||
self.fields['name'].widget.attrs.update({"placeholder": _("pasta")})
|
||||
clubs = list(Club.objects.filter(PermissionBackend.filter_queryset(get_current_request(), Club, "change")).all())
|
||||
shuffle(clubs)
|
||||
self.fields['owner'].widget.attrs["placeholder"] = ", ".join(club.name for club in clubs[:4]) + ", ..."
|
||||
def clean_dlm_or_dlc(self):
|
||||
is_dlc = self.cleaned_data["is_DLC"]
|
||||
is_ddm = self.cleaned_data["is_DDM"]
|
||||
if is_dlc and is_ddm:
|
||||
self.add_error("is_ddm", _("the product cannot be a DLC and a DDM"))
|
||||
return is_ddm
|
||||
|
||||
class Meta:
|
||||
model = BasicFood
|
||||
fields = ('name', 'owner', 'is_DLC', 'is_DDM', 'expiry_date', 'label')
|
||||
widgets = {
|
||||
"owner": Autocomplete(
|
||||
model = Club,
|
||||
attrs = {"api_url": "/api/members/club/"},
|
||||
),
|
||||
'expiry_date': DatePickerInput(),
|
||||
}
|
||||
|
||||
|
||||
|
||||
class TransformedFoodForms(forms.ModelForm):
|
||||
"""
|
||||
Form for add transformed food
|
||||
"""
|
||||
def __init__(self, *args, **kwargs):
|
||||
super().__init__(*args, **kwargs)
|
||||
self.fields['name'].widget.attrs.update({"autofocus": "autofocus"})
|
||||
self.fields['name'].required = True
|
||||
self.fields['owner'].required = True
|
||||
self.fields['creation_date'].required = True
|
||||
self.fields['creation_date'].initial = timezone.now
|
||||
self.fields['is_active'].initial = True
|
||||
|
||||
# Some example
|
||||
self.fields['name'].widget.attrs.update({"placeholder": _("lasagna")})
|
||||
clubs = list(Club.objects.filter(PermissionBackend.filter_queryset(get_current_request(), Club, "change")).all())
|
||||
shuffle(clubs)
|
||||
self.fields['owner'].widget.attrs["placeholder"] = ", ".join(club.name for club in clubs[:4]) + ", ..."
|
||||
|
||||
|
||||
class Meta:
|
||||
model = TransformedFood
|
||||
fields = ('name', 'creation_date', 'owner', 'is_active',)
|
||||
widgets = {
|
||||
"owner": Autocomplete(
|
||||
model = Club,
|
||||
attrs = {"api_url": "/api/members/club/"},
|
||||
),
|
||||
'creation_date': DateTimePickerInput(),
|
||||
}
|
||||
class AllergenForms(forms.ModelForm):
|
||||
"""
|
||||
Form for allergen
|
||||
"""
|
||||
def __init__(self, *args, **kwargs):
|
||||
super().__init__(*args, **kwargs)
|
||||
|
||||
class Meta:
|
||||
model = Allergen
|
||||
exclude = ['basic_food', 'transformed_food']
|
97
apps/food/migrations/0001_initial.py
Normal file
97
apps/food/migrations/0001_initial.py
Normal file
@ -0,0 +1,97 @@
|
||||
# Generated by Django 2.2.28 on 2024-05-25 20:32
|
||||
|
||||
from django.db import migrations, models
|
||||
import django.db.models.deletion
|
||||
import django.utils.timezone
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
initial = True
|
||||
|
||||
dependencies = [
|
||||
('member', '0011_profile_vss_charter_read'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.CreateModel(
|
||||
name='BasicFood',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('name', models.CharField(max_length=255, verbose_name='name')),
|
||||
('is_DLC', models.BooleanField(default=False, verbose_name='is DLC')),
|
||||
('is_DDM', models.BooleanField(default=False, verbose_name='is DDM')),
|
||||
('arrival_date', models.DateTimeField(default=django.utils.timezone.now, verbose_name='arrival date')),
|
||||
('expiry_date', models.DateTimeField(blank=True, null=True, verbose_name='expiry date')),
|
||||
('label', models.ImageField(max_length=255, upload_to='label/', verbose_name='food label')),
|
||||
('was_eaten', models.BooleanField(default=False, verbose_name='was eaten')),
|
||||
('owner', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='+', to='member.Club', verbose_name='owner')),
|
||||
],
|
||||
options={
|
||||
'verbose_name': 'Basic food',
|
||||
'verbose_name_plural': 'Basic foods',
|
||||
},
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='TransformedFood',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('name', models.CharField(max_length=255, verbose_name='name')),
|
||||
('creation_date', models.DateTimeField(verbose_name='creation date')),
|
||||
('expiry_date', models.DateTimeField(verbose_name='expiry date')),
|
||||
('is_active', models.BooleanField(default=True, verbose_name='is active')),
|
||||
('was_eaten', models.BooleanField(default=False, verbose_name='was eaten')),
|
||||
('owner', models.ForeignKey(on_delete=django.db.models.deletion.PROTECT, related_name='+', to='member.Club', verbose_name='owner')),
|
||||
('transformed_ingredient', models.ManyToManyField(blank=True, related_name='transformed_ingredient_inv', to='food.TransformedFood', verbose_name='transformed ingredient')),
|
||||
],
|
||||
options={
|
||||
'verbose_name': 'Transformed food',
|
||||
'verbose_name_plural': 'Transformed foods',
|
||||
},
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='QR_code',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('qr_code_number', models.PositiveIntegerField(verbose_name='QR-code number')),
|
||||
('basic_food', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.PROTECT, related_name='QR_code', to='food.BasicFood', verbose_name='basic food')),
|
||||
('transformed_food_container', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.PROTECT, related_name='QR_code', to='food.TransformedFood', verbose_name='transformed food container')),
|
||||
],
|
||||
options={
|
||||
'verbose_name': 'QR-code',
|
||||
'verbose_name_plural': 'QR-codes',
|
||||
},
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='basicfood',
|
||||
name='transformed_food',
|
||||
field=models.ManyToManyField(blank=True, related_name='BasicFood', to='food.TransformedFood', verbose_name='transformed food'),
|
||||
),
|
||||
migrations.CreateModel(
|
||||
name='Allergen',
|
||||
fields=[
|
||||
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('gluten', models.BooleanField(default=False, verbose_name='gluten')),
|
||||
('nut', models.BooleanField(default=False, verbose_name='nut')),
|
||||
('crustecean', models.BooleanField(default=False, verbose_name='crustacean')),
|
||||
('celery', models.BooleanField(default=False, verbose_name='celery')),
|
||||
('egg', models.BooleanField(default=False, verbose_name='egg')),
|
||||
('mustard', models.BooleanField(default=False, verbose_name='mustard')),
|
||||
('fish', models.BooleanField(default=False, verbose_name='fish')),
|
||||
('soy', models.BooleanField(default=False, verbose_name='soy')),
|
||||
('milk', models.BooleanField(default=False, verbose_name='milk')),
|
||||
('sulphite', models.BooleanField(default=False, verbose_name='sulphite')),
|
||||
('lupine', models.BooleanField(default=False, verbose_name='lupine')),
|
||||
('mollusc', models.BooleanField(default=False, verbose_name='mollusc')),
|
||||
('groundnut', models.BooleanField(default=False, verbose_name='groundnut')),
|
||||
('sesame', models.BooleanField(default=False, verbose_name='sesame')),
|
||||
('alcohol', models.BooleanField(default=False, verbose_name='alcohol')),
|
||||
('basic_food', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='Allergen', to='food.BasicFood', verbose_name='basic food')),
|
||||
('transformed_food', models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='Allergen', to='food.TransformedFood', verbose_name='transformed food')),
|
||||
],
|
||||
options={
|
||||
'verbose_name': 'Allergen',
|
||||
'verbose_name_plural': 'Allergens',
|
||||
},
|
||||
),
|
||||
]
|
0
apps/food/migrations/__init__.py
Normal file
0
apps/food/migrations/__init__.py
Normal file
289
apps/food/models.py
Normal file
289
apps/food/models.py
Normal file
@ -0,0 +1,289 @@
|
||||
# Copyright (C) 2018-2024 by BDE ENS Paris-Saclay
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
from datetime import date
|
||||
|
||||
from django.conf import settings
|
||||
from django.contrib.auth.models import User
|
||||
from django.core.exceptions import ValidationError
|
||||
from django.core.validators import MinValueValidator
|
||||
from django.db import models, transaction
|
||||
from django.db.models import Q
|
||||
from django.utils import timezone
|
||||
from django.utils.translation import gettext_lazy as _
|
||||
from member.models import Club
|
||||
|
||||
#################################################################
|
||||
# TO DO
|
||||
# - link allergen with one food (basic or transformed) with check
|
||||
# - check on basic food
|
||||
# - check on transformed food
|
||||
#################################################################
|
||||
|
||||
class QR_code(models.Model):
|
||||
"""
|
||||
An QR_code model
|
||||
"""
|
||||
qr_code_number = models.PositiveIntegerField(
|
||||
verbose_name=_("QR-code number"),
|
||||
)
|
||||
|
||||
transformed_food_container = models.ForeignKey(
|
||||
'TransformedFood',
|
||||
on_delete = models.PROTECT,
|
||||
related_name = 'QR_code',
|
||||
null = True,
|
||||
blank = True,
|
||||
verbose_name = _('transformed food container'),
|
||||
)
|
||||
|
||||
basic_food = models.ForeignKey(
|
||||
'BasicFood',
|
||||
on_delete = models.PROTECT,
|
||||
related_name = 'QR_code',
|
||||
null = True,
|
||||
blank = True,
|
||||
verbose_name = _('basic food'),
|
||||
)
|
||||
|
||||
class Meta:
|
||||
verbose_name = _("QR-code")
|
||||
verbose_name_plural = _("QR-codes")
|
||||
|
||||
def __str__(self):
|
||||
return _("QR-code number {qr_code_number}").format(qr_code_number=self.qr_code_number)
|
||||
|
||||
class Allergen(models.Model):
|
||||
"""
|
||||
A list of allergen and alimentary restrictions
|
||||
"""
|
||||
|
||||
gluten = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('gluten'),
|
||||
)
|
||||
|
||||
nut = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('nut'),
|
||||
)
|
||||
|
||||
crustecean = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('crustacean'),
|
||||
)
|
||||
|
||||
celery = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('celery'),
|
||||
)
|
||||
|
||||
egg = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('egg'),
|
||||
)
|
||||
|
||||
mustard = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('mustard'),
|
||||
)
|
||||
|
||||
fish = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('fish'),
|
||||
)
|
||||
|
||||
soy = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('soy'),
|
||||
)
|
||||
|
||||
milk = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('milk'),
|
||||
)
|
||||
|
||||
sulphite = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('sulphite'),
|
||||
)
|
||||
|
||||
lupine = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('lupine'),
|
||||
)
|
||||
|
||||
mollusc = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('mollusc'),
|
||||
)
|
||||
|
||||
groundnut = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('groundnut'),
|
||||
)
|
||||
|
||||
sesame = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('sesame'),
|
||||
)
|
||||
|
||||
alcohol = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('alcohol'),
|
||||
)
|
||||
|
||||
transformed_food = models.ForeignKey(
|
||||
'TransformedFood',
|
||||
on_delete = models.CASCADE,
|
||||
related_name = 'Allergen',
|
||||
blank = True,
|
||||
null = True,
|
||||
verbose_name = _('transformed food'),
|
||||
)
|
||||
|
||||
basic_food = models.ForeignKey(
|
||||
'BasicFood',
|
||||
on_delete = models.CASCADE,
|
||||
related_name = 'Allergen',
|
||||
blank = True,
|
||||
null = True,
|
||||
verbose_name = _('basic food'),
|
||||
)
|
||||
|
||||
class Meta:
|
||||
verbose_name = _('Allergen')
|
||||
verbose_name_plural = _('Allergens')
|
||||
|
||||
def __str__(self):
|
||||
return _('Allergens of #{id}').format(id=self.id)
|
||||
|
||||
|
||||
|
||||
|
||||
class BasicFood(models.Model):
|
||||
"""
|
||||
Food which has been directly buy on supermarket
|
||||
"""
|
||||
name = models.CharField(
|
||||
verbose_name=_('name'),
|
||||
max_length=255,
|
||||
)
|
||||
|
||||
is_DLC = models.BooleanField(
|
||||
verbose_name=_("is DLC"),
|
||||
default=False,
|
||||
)
|
||||
|
||||
is_DDM = models.BooleanField(
|
||||
verbose_name=_("is DDM"),
|
||||
default=False,
|
||||
)
|
||||
|
||||
arrival_date = models.DateTimeField(
|
||||
verbose_name=_('arrival date'),
|
||||
default=timezone.now,
|
||||
)
|
||||
|
||||
expiry_date = models.DateTimeField(
|
||||
verbose_name=_('expiry date'),
|
||||
blank=True,
|
||||
null = True,
|
||||
)
|
||||
|
||||
owner = models.ForeignKey(
|
||||
Club,
|
||||
on_delete=models.PROTECT,
|
||||
related_name= '+',
|
||||
verbose_name=_('owner'),
|
||||
)
|
||||
|
||||
label = models.ImageField(
|
||||
verbose_name=_('food label'),
|
||||
max_length=255,
|
||||
blank=False,
|
||||
null=False,
|
||||
upload_to='label/',
|
||||
)
|
||||
|
||||
was_eaten = models.BooleanField(
|
||||
verbose_name=_('was eaten'),
|
||||
default = False,
|
||||
)
|
||||
|
||||
transformed_food = models.ManyToManyField(
|
||||
'TransformedFood',
|
||||
related_name= 'BasicFood',
|
||||
blank = True,
|
||||
verbose_name = _('transformed food'),
|
||||
)
|
||||
|
||||
|
||||
class Meta:
|
||||
verbose_name=_('Basic food')
|
||||
verbose_name_plural=_('Basic foods')
|
||||
|
||||
def __str__(self):
|
||||
return self.name
|
||||
|
||||
@transaction.atomic
|
||||
def save(self, force_insert=False, force_update=False, using= None, update_fields=None):
|
||||
# Check if is_DLC and is DDM are not both True
|
||||
if self.is_DLC and self.is_DDM:
|
||||
raise ValidationError("The product cannot be a DLC and a DDM")
|
||||
return super().save(force_insert, force_update, using, update_fields)
|
||||
|
||||
|
||||
class TransformedFood(models.Model):
|
||||
"""
|
||||
Transformed food are a mix between basic food and meal
|
||||
"""
|
||||
name = models.CharField(
|
||||
max_length = 255,
|
||||
verbose_name =_('name'),
|
||||
)
|
||||
|
||||
creation_date = models.DateTimeField(
|
||||
verbose_name =_('creation date'),
|
||||
)
|
||||
|
||||
expiry_date = models.DateTimeField(
|
||||
verbose_name =_('expiry date'),
|
||||
)
|
||||
|
||||
owner = models.ForeignKey(
|
||||
Club,
|
||||
on_delete = models.PROTECT,
|
||||
related_name = '+',
|
||||
verbose_name =_('owner'),
|
||||
)
|
||||
|
||||
transformed_ingredient = models.ManyToManyField(
|
||||
"self",
|
||||
blank = True,
|
||||
symmetrical = False,
|
||||
related_name = 'transformed_ingredient_inv',
|
||||
verbose_name = _('transformed ingredient'),
|
||||
)
|
||||
|
||||
is_active = models.BooleanField(
|
||||
default = True,
|
||||
verbose_name = _('is active'),
|
||||
)
|
||||
|
||||
was_eaten = models.BooleanField(
|
||||
default = False,
|
||||
verbose_name = _('was eaten'),
|
||||
)
|
||||
|
||||
|
||||
class Meta:
|
||||
verbose_name = _('Transformed food')
|
||||
verbose_name_plural = _('Transformed foods')
|
||||
|
||||
def __str__(self):
|
||||
return self.name
|
||||
|
||||
@transaction.atomic
|
||||
def save(self, force_insert=False, force_update=False, using= None, update_fields=None):
|
||||
return super().save(force_insert, force_update, using, update_fields)
|
22
apps/food/templates/food/basic_food_form.html
Normal file
22
apps/food/templates/food/basic_food_form.html
Normal file
@ -0,0 +1,22 @@
|
||||
{% extends "base.html" %}
|
||||
{% comment %}
|
||||
SPDX-License-Identifier: GPL-3.0-or-later
|
||||
{% endcomment %}
|
||||
{% load i18n crispy_forms_tags %}
|
||||
|
||||
{% block content %}
|
||||
<div class="card bg-white mb-3">
|
||||
<h3 class="card-header text-center">
|
||||
HTML not finished <br>
|
||||
{{ title }}
|
||||
</h3>
|
||||
<div class="card-body">
|
||||
<form method="post">
|
||||
{% csrf_token %}
|
||||
{{ form|crispy }}
|
||||
{{ allergenform|crispy }}
|
||||
<button class="btn btn-primary" type="submit">{% trans "Submit"%}</button>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
21
apps/food/templates/food/transformed_food_form.html
Normal file
21
apps/food/templates/food/transformed_food_form.html
Normal file
@ -0,0 +1,21 @@
|
||||
{% extends "base.html" %}
|
||||
{% comment %}
|
||||
SPDX-License-Identifier: GPL-3.0-or-later
|
||||
{% endcomment %}
|
||||
{% load i18n crispy_forms_tags %}
|
||||
|
||||
{% block content %}
|
||||
<div class="card bg-white mb-3">
|
||||
<h3 class="card-header text-center">
|
||||
HTML not finished <br>
|
||||
{{ title }}
|
||||
</h3>
|
||||
<div class="card-body">
|
||||
<form method="post">
|
||||
{% csrf_token %}
|
||||
{{ form|crispy }}
|
||||
<button class="btn btn-primary" type="submit">{% trans "Submit"%}</button>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
3
apps/food/tests.py
Normal file
3
apps/food/tests.py
Normal file
@ -0,0 +1,3 @@
|
||||
from django.test import TestCase
|
||||
|
||||
# Create your tests here.
|
19
apps/food/urls.py
Normal file
19
apps/food/urls.py
Normal file
@ -0,0 +1,19 @@
|
||||
# Copyright (C) 2018-2024 by BDE ENS Paris-Saclay
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
from django.urls import path
|
||||
|
||||
from . import views
|
||||
|
||||
###############################
|
||||
# TO DO
|
||||
# - name url correctly, thinking about the scheme of the app
|
||||
###############################
|
||||
|
||||
app_name = 'food'
|
||||
|
||||
urlpatterns = [
|
||||
path('0', views.BasicFoodCreateView.as_view(), name = 'basic_food'),
|
||||
path('1', views.TransformedFoodCreateView.as_view(), name = 'transformed_food'),
|
||||
]
|
||||
|
104
apps/food/views.py
Normal file
104
apps/food/views.py
Normal file
@ -0,0 +1,104 @@
|
||||
# Copyright (C) 2018-2024 by BDE ENS Paris-Saclay
|
||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||
|
||||
from django.urls import reverse_lazy
|
||||
from django.db import transaction
|
||||
from django.utils.translation import gettext_lazy as _
|
||||
from django.utils import timezone
|
||||
from datetime import timedelta
|
||||
from permission.views import ProtectQuerysetMixin, ProtectedCreateView
|
||||
from django.shortcuts import render
|
||||
|
||||
from .forms import BasicFoodForms, TransformedFoodForms, AllergenForms
|
||||
from .models import BasicFood, TransformedFood, Allergen
|
||||
|
||||
|
||||
class BasicFoodCreateView(ProtectQuerysetMixin, ProtectedCreateView):
|
||||
#####################################################################
|
||||
# TO DO
|
||||
# - fix picture save
|
||||
# - implement solution crop and convert image (reuse or recode ImageForm from members apps
|
||||
# - implement AllergenForms
|
||||
# - redirect to another view after the poll is submitted
|
||||
#####################################################################
|
||||
"""
|
||||
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")}
|
||||
|
||||
def get_sample_object(self):
|
||||
return BasicFood(
|
||||
name="",
|
||||
is_DLC=False,
|
||||
is_DDM=False,
|
||||
expiry_date=timezone.now(),
|
||||
label='pic/default.png',
|
||||
)
|
||||
|
||||
@transaction.atomic
|
||||
def form_valid(self, form):
|
||||
form.instance.creater = self.request.user
|
||||
basic_food_form = BasicFoodForms(data=self.request.POST)
|
||||
allergen_form = AllergenForms(data=self.request.POST)
|
||||
if not basic_food_form.is_valid() or not allergen_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._force_save = True
|
||||
basic_food.save()
|
||||
basic_food.refresh_from_db()
|
||||
return super().form_valid(form)
|
||||
|
||||
def get_success_url(self, **kwargs):
|
||||
self.object.refresh_from_db()
|
||||
# TEMPORARY, I create a fonctionnal view before
|
||||
# return reverse_lazy('food:basicfood', kwargs={"pk": self.object.pk})
|
||||
return '0'
|
||||
|
||||
class TransformedFoodCreateView(ProtectQuerysetMixin, ProtectedCreateView):
|
||||
###############################################
|
||||
# TO DO
|
||||
# -redirect to another view after submit
|
||||
###############################################
|
||||
"""
|
||||
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")}
|
||||
|
||||
def get_sample_object(self):
|
||||
return TransformedFood(
|
||||
name="",
|
||||
creation_date=timezone.now(),
|
||||
)
|
||||
|
||||
@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()
|
||||
# TEMPORARY, I create a fonctionnal view before
|
||||
# return reverse_lazy('food:tranformed_food', kwargs={"pk": self.object.pk})
|
||||
return '1'
|
@ -77,6 +77,7 @@ INSTALLED_APPS = [
|
||||
'scripts',
|
||||
'treasury',
|
||||
'wei',
|
||||
'food',
|
||||
]
|
||||
|
||||
MIDDLEWARE = [
|
||||
|
@ -21,6 +21,7 @@ urlpatterns = [
|
||||
path('activity/', include('activity.urls')),
|
||||
path('treasury/', include('treasury.urls')),
|
||||
path('wei/', include('wei.urls')),
|
||||
path('food/',include('food.urls')),
|
||||
|
||||
# Include Django Contrib and Core routers
|
||||
path('i18n/', include('django.conf.urls.i18n')),
|
||||
|
Reference in New Issue
Block a user