Vorbereitung für Erfassung neuer Gerichte
This commit is contained in:
@@ -0,0 +1,33 @@
|
||||
# forms.py
|
||||
|
||||
from django import forms
|
||||
from .models import Gericht
|
||||
# Importiere alle Modelle, die für das Formular relevant sind
|
||||
from .models import Kategorie
|
||||
|
||||
class GerichtForm(forms.ModelForm):
|
||||
"""Definiert die Struktur und Validierungsregeln für das Gericht-Modell."""
|
||||
|
||||
class Meta:
|
||||
model = Gericht
|
||||
fields = [
|
||||
'name',
|
||||
'kategorie',
|
||||
'preis',
|
||||
'ist_vegetarisch',
|
||||
'ist_allergene_frei',
|
||||
'allergene',
|
||||
'beschreibung', # Neues Feld für Zutaten/Rohstoffe
|
||||
'bilder', # Das Bild-Feld
|
||||
'ist_dauerangebot'
|
||||
]
|
||||
|
||||
def __init__(self, *args, **kwargs):
|
||||
super().__init__(*args, **kwargs)
|
||||
# Wir können hier zusätzliche Validierungen oder Widgets einbinden
|
||||
# Beispiel: Erzwinge das Betreten der Kategorie-Auswahl
|
||||
self.fields['kategorie'].required = True
|
||||
|
||||
# Hier könntest du auch einen Fallback für den Preis hinzufügen, falls er nicht gesetzt ist
|
||||
if 'initial' in kwargs: # Wird bei GET-Requests genutzt
|
||||
self.initial['preis'] = 0.00
|
||||
@@ -128,6 +128,7 @@ class Gericht(models.Model):
|
||||
)
|
||||
ist_vegetarisch = models.BooleanField(default=False)
|
||||
ist_allergene_frei = models.BooleanField(default=False)
|
||||
beschreibung = models.TextField(blank=True, null=True) # Zutaten/Rohstoffe Beschreibung (Textfeld für V1)
|
||||
allergene = models.TextField(blank=True, default="")
|
||||
preis = models.DecimalField(max_digits=5, decimal_places=2, default=0.00)
|
||||
|
||||
|
||||
@@ -0,0 +1,32 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="de">
|
||||
<head>
|
||||
<!-- (Standard Head und Bootstrap CDN einbinden) -->
|
||||
</head>
|
||||
<body class="bg-light">
|
||||
<div class="container mt-5">
|
||||
<h1>Neue Speiseart hinzufügen</h1>
|
||||
|
||||
{# !!! KRITISCH WICHTIG: enctype muss für Dateiuploads stehen !!! #}
|
||||
<form method="post" enctype="multipart/form-data">
|
||||
{% csrf_token %}
|
||||
|
||||
<!-- Der Standardway, alle Felder mit Bootstrap zu stylen (für Anfänger am besten) -->
|
||||
{# Man kann die komplexe Struktur durch eine einfache Schleife ersetzen #}
|
||||
{% for field in form %}
|
||||
<div class="mb-3">
|
||||
<label for="{{ field.id_for_label }}" class="form-label">{{ field.label }}</label>
|
||||
{{ field }}
|
||||
|
||||
<!-- Fehlerhinweis des Feldes wird hier automatisch gerendert -->
|
||||
{% if field.errors %}
|
||||
<p class="text-danger small mt-1">{{ field.errors }}</p>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endfor %}
|
||||
|
||||
<button type="submit" class="btn btn-success mt-4">Gericht Speichern & Veröffentlichen</button>
|
||||
</form>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
@@ -0,0 +1,29 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="de">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>Erfolg</title>
|
||||
<link href="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/css/bootstrap.min.css" rel="stylesheet">
|
||||
</head>
|
||||
|
||||
<body>
|
||||
<div class="container mt-5">
|
||||
<div class="alert alert-success shadow p-5 text-center">
|
||||
<h2 class="mb-3">🎉 Erfolgreich hinzugefügt! 🎉</h2>
|
||||
<p class="lead">
|
||||
Das Gericht <strong>{{ objekt_name }}</strong> wurde erfolgreich in das Menüsystem übernommen.
|
||||
</p>
|
||||
<p>
|
||||
Du kannst nun die Details überprüfen und weitere Gerichte hinzufügen.
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<div class="mt-4 text-center">
|
||||
<!-- Der wichtigste Button, der zum Hauptbereichen zurückführt -->
|
||||
<a href="{% url 'speisekarte' %}" class="btn btn-primary btn-lg">
|
||||
← Zurück zur Speisekarte & Verwalten
|
||||
</a>
|
||||
</div>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
+16
-1
@@ -7,5 +7,20 @@ urlpatterns = [
|
||||
path('speisekarte/', GerichtListView.as_view(), name='speisekarte'),
|
||||
path('speiseplan/', SpeiseplanView.as_view(), name='speiseplan'),
|
||||
path('bestellungen/summary/', BestellSummaryView.as_view(), name='bestell_summary'),
|
||||
path('userlist/', {},"" ),
|
||||
]
|
||||
|
||||
# ACHTUNG: Hier benötigen wir jetzt eine separate View für die Erfolgsmeldung!
|
||||
# Wir nutzen hierfür ein sehr einfaches TemplateView.
|
||||
from django.views.generic import TemplateView # Muss oben importiert werden
|
||||
from .models import Gericht
|
||||
|
||||
class ErfolgTemplateView(TemplateView):
|
||||
"""Zeigt einfach nur die Bestätigung an."""
|
||||
template_name = 'mensa_app/add_gericht_success.html'
|
||||
|
||||
urlpatterns += [
|
||||
# ... (vorherige Pfade bleiben) ...
|
||||
path('gerichte/neu/', GerichtCreateView.as_view(), name='add_gericht'),
|
||||
# NEU: Erfolgsmeldung nach erfolgreichem POST/redirect
|
||||
path('erfolg/', ErfolgTemplateView.as_view(), name='success_message'),
|
||||
]
|
||||
|
||||
+41
-13
@@ -1,18 +1,20 @@
|
||||
# views.py
|
||||
from django.db.models import Count, Sum
|
||||
from django.shortcuts import render
|
||||
from django.utils import timezone
|
||||
from datetime import datetime
|
||||
from django.views.generic import ListView, TemplateView
|
||||
from django.views.generic.edit import CreateView
|
||||
from django.shortcuts import redirect # Importiere die Redirect-Funktion
|
||||
from .forms import GerichtForm # Importiere dein neues Formular
|
||||
from .models import Gericht, Menue, SpeiseplanTag, Bestellung
|
||||
|
||||
# Create your views here.
|
||||
from django.views.generic import ListView
|
||||
from .models import Gericht
|
||||
|
||||
class GerichtListView(ListView):
|
||||
model = Gericht
|
||||
template_name = 'mensa_app/gericht_liste.html' # Der Pfad zum Template
|
||||
context_object_name = 'alle_gerichte' # Der Name, den wir im Template nutzen
|
||||
|
||||
from django.views.generic import TemplateView
|
||||
from django.utils import timezone
|
||||
from datetime import datetime
|
||||
from .models import Menue, Gericht, SpeiseplanTag
|
||||
|
||||
class SpeiseplanView(TemplateView):
|
||||
template_name = 'mensa_app/speiseplan.html'
|
||||
@@ -51,12 +53,6 @@ class SpeiseplanView(TemplateView):
|
||||
return context
|
||||
|
||||
|
||||
from django.views.generic import TemplateView
|
||||
from django.utils import timezone
|
||||
from django.db.models import Count, Sum
|
||||
from datetime import datetime
|
||||
from .models import Bestellung, SpeiseplanTag, Gericht, Menue
|
||||
|
||||
class BestellSummaryView(TemplateView):
|
||||
template_name = 'mensa_app/bestell_summary.html'
|
||||
|
||||
@@ -100,3 +96,35 @@ class BestellSummaryView(TemplateView):
|
||||
context['total_umsatz'] = Bestellung.objects.filter(menue__tag__datum=target_date).aggregate(Sum('menue__preis'))['menue__preis__sum'] or 0
|
||||
|
||||
return context
|
||||
|
||||
class GerichtCreateView(CreateView):
|
||||
"""
|
||||
View zur Erstellung eines neuen Gerichts.
|
||||
Nutzt das benutzerdefinierte GerichtForm für die gesamte Steuerung.
|
||||
"""
|
||||
model = Gericht
|
||||
form_class = GerichtForm # Verwendet unser Custom-Formular
|
||||
template_name = 'mensa_app/add_gericht.html'
|
||||
|
||||
def get_success_url(self):
|
||||
# Die Methode wird ausgeführt, nachdem die View erfolgreich ist
|
||||
objekt = self.object # Gibt uns das gerade gespeicherte Objekt zurück
|
||||
return reverse('speisekarte')
|
||||
|
||||
def form_valid(self, form):
|
||||
"""
|
||||
Übersteuert die Standard-Logik:
|
||||
1. Speichert zuerst das Gericht (über super()).
|
||||
2. Leitet danach auf ein success/confirmation Pattern um.
|
||||
"""
|
||||
super().form_valid() # Das eigentliche Speichern des Objekts
|
||||
|
||||
# Hier ist der Haken an Django: Die View wird beim Erfolg automatisch weitergeleitet.
|
||||
# Wir verwenden einen kleinen Trick, damit wir den Namen im Kontext haben.
|
||||
|
||||
return redirect('success_nach_gerichterstellung') # Leiten auf eine temporäre URL um
|
||||
|
||||
# Optional: Setzen des initialen Datenzustands (z.B. Datum, falls relevant)
|
||||
def get_context_data(self, **kwargs):
|
||||
context = super().get_context_data(**kwargs)
|
||||
return context
|
||||
|
||||
Reference in New Issue
Block a user