Merge pull request 'Branch gui-dev auf aktuellen Stand bringen' (#10) from main into gui-dev

Reviewed-on: #10
This commit was merged in pull request #10.
This commit is contained in:
2026-06-26 07:33:26 +00:00
10 changed files with 286 additions and 22 deletions
-1
View File
@@ -1,4 +1,3 @@
__pycache__/ __pycache__/
*.pyc *.pyc
*.sqlite3
+93 -4
View File
@@ -6,12 +6,15 @@ Muster-Implementation durch die Lehrkraft in python3 mit Django
## To get started ## To get started
## Vorbereitung der virtuellen Entwicklungsumgebung
in einem Projekt-Vezeichnis über dem eigentlichen SGTMensa-Projekt-Ordner
```bash ```bash
cd SGTMensa
python3 -m venv .venv python3 -m venv .venv
source .venv/bin/activate source .venv/bin/activate
pip install django pip install django
django-admin startproject mensa-core . pip install pillow
``` ```
Falls eine fish-Shell vorliegt: Falls eine fish-Shell vorliegt:
@@ -19,9 +22,27 @@ Falls eine fish-Shell vorliegt:
python3 -m venv .venv python3 -m venv .venv
source .venv/bin/activate.fish source .venv/bin/activate.fish
pip install django pip install django
django-admin startproject mensa-core . pip install pillow
``` ```
## Klonen des eigentlichen Projekts
```bash
git clone https://git.sgtlernen.de/mputzlocher/SGTMensa.git
```
## Starten des Servers
```bash
cd SGTMensa
python manage.py runserver
```
---
# Erweiterte Erklärung einzelner Schritte
### Schritt 1: Die Arbeitsumgebung isolieren (Virtual Environment) ### Schritt 1: Die Arbeitsumgebung isolieren (Virtual Environment)
Bevor du auch nur eine Zeile Code schreibst, erstelle eine eigene Umgebung. Das verhindert, dass sich verschiedene Projekte auf deinem Rechner gegenseitig stören. Bevor du auch nur eine Zeile Code schreibst, erstelle eine eigene Umgebung. Das verhindert, dass sich verschiedene Projekte auf deinem Rechner gegenseitig stören.
@@ -42,9 +63,10 @@ Installiere nur das Nötigste. Für den Start reicht Django.
```bash ```bash
pip install django pip install django
pip install pillow
``` ```
### Schritt 3: Das Projekt initialisieren ### Schritt 3: Das Projekt initialisieren (nur einmalig, beim ersten Start des Projekts)
Erstelle die Struktur eines Django-Projekts. Erstelle die Struktur eines Django-Projekts.
@@ -119,3 +141,70 @@ python manage.py runserver
Öffne deinen Browser unter `http://127.0.0.1:8000/admin`. Logge dich mit deinem Superuser ein. **Herzlichen Glückwunsch!** Du hast eine voll funktionsfähige Web-Anwendung, mit der du bereits Speisepläne und Bestellungen in einer Datenbank verwalten kannst. Öffne deinen Browser unter `http://127.0.0.1:8000/admin`. Logge dich mit deinem Superuser ein. **Herzlichen Glückwunsch!** Du hast eine voll funktionsfähige Web-Anwendung, mit der du bereits Speisepläne und Bestellungen in einer Datenbank verwalten kannst.
### Schritt 9: Eigentliche Entwicklungsarbeit
Folgende Befehle werden nacheinander ausgeführt:
#### Eigenen Projektbaum auf dem lokalen Rechen auf den aktuellen Stand bringen
Öffne ein Terminal und navigiere in das Projektverzeichnis, das bereits über `git clone` abgeholt worden ist.
zum Beispiel:
```bash
cd ~/projects/SGTMensa
```
Hole den aktuellen Stand ab:
```bash
git pull
```
Aktiviere die virtuelle Entwicklungsumgebung (falls noch nicht geschehen):
```bash
source .venv/bin/activate
```
Führe notwendige Migrationen der bestehenden Datenbank aus:
```bash
python manage.py makemigrations
python manage.py migrate
```
Starte den Server des Projekts:
```bash
python manage.py runserver
```
Jetzt bist Du startklar für die Entwicklungsarbeit und solltest den aktuellen Stand im Browser unter `http://localhost:8000/` abrufen können.
#### Änderungen vornehmen
Django erkennt selbstständig Veränderungen am Quellcode und startet den Webserver neu. Achte auf Fehlermeldungen!
#### Änderungen wieder auf den git-Server einspielen
Nach einem Entwicklungsschritt solltest Du Deine Änderungen von Deinem lokalen Rechner wieder auf den git-Server hochladen,
damit Dein Team auch sieht, was Du gemacht hast, und damit Deine Ergänzungen und Änderungen wieder in den gemeinsamen
Code integriert werden können.
Nachsehen der Änderungen:
```bash
git status
```
Dies zeigt alle veränderten Dateien an.
Veränderte Dateien ins Änderungspaket aufnehmen:
```bash
git add .
```
Veränderungen bestätigen und beschreiben:
```bash
git commit -m "Deine sinnvolle Beschreibung der Änderung in wenigen Worten"
```
Veränderungen auf den git-Server hochladen:
```bash
git push
```
BIN
View File
Binary file not shown.
+33
View File
@@ -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
@@ -0,0 +1,38 @@
# Generated by Django 5.2.15 on 2026-06-12 10:50
import datetime
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('mensa_app', '0010_merge_20260612_0913'),
]
operations = [
migrations.AlterModelOptions(
name='person',
options={},
),
migrations.AlterField(
model_name='gericht',
name='time_creation',
field=models.TimeField(default=datetime.datetime(2026, 6, 12, 10, 50, 49, 206132)),
),
migrations.AlterField(
model_name='gericht',
name='time_last_change',
field=models.TimeField(default=datetime.datetime(2026, 6, 12, 10, 50, 49, 206132)),
),
migrations.AlterField(
model_name='gerichtbild',
name='id',
field=models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID'),
),
migrations.AlterField(
model_name='person',
name='rolle',
field=models.CharField(choices=[('schueler', 'Schüler'), ('eltern', 'Eltern'), ('lehrer', 'Lehrer'), ('mitarbeiter', 'Mensa-Mitarbeiter'), ('chef', 'Mensa-Leitung')], max_length=20),
),
]
+2 -3
View File
@@ -1,10 +1,8 @@
import os
from django.db import models from django.db import models
from django.contrib.auth.models import User from django.contrib.auth.models import User
from django.utils import timezone from django.utils import timezone
import datetime import datetime
import os
from django.db import models
from django.core.exceptions import ValidationError from django.core.exceptions import ValidationError
from django.core.validators import FileExtensionValidator from django.core.validators import FileExtensionValidator
# Pillow ist für die Bildbearbeitung zuständig (installiere per pip install pillow) # Pillow ist für die Bildbearbeitung zuständig (installiere per pip install pillow)
@@ -130,6 +128,7 @@ class Gericht(models.Model):
) )
ist_vegetarisch = models.BooleanField(default=False) ist_vegetarisch = models.BooleanField(default=False)
ist_allergene_frei = 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="") allergene = models.TextField(blank=True, default="")
preis = models.DecimalField(max_digits=5, decimal_places=2, default=0.00) 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">
&larr; Zurück zur Speisekarte &amp; Verwalten
</a>
</div>
</div>
</body>
</html>
+16
View File
@@ -8,3 +8,19 @@ urlpatterns = [
path('speiseplan/', SpeiseplanView.as_view(), name='speiseplan'), path('speiseplan/', SpeiseplanView.as_view(), name='speiseplan'),
path('bestellungen/summary/', BestellSummaryView.as_view(), name='bestell_summary'), path('bestellungen/summary/', BestellSummaryView.as_view(), name='bestell_summary'),
] ]
# 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'),
]
+42 -13
View File
@@ -1,18 +1,20 @@
# views.py
from django.db.models import Count, Sum
from django.shortcuts import render 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): class GerichtListView(ListView):
model = Gericht model = Gericht
template_name = 'mensa_app/gericht_liste.html' # Der Pfad zum Template template_name = 'mensa_app/gericht_liste.html' # Der Pfad zum Template
context_object_name = 'alle_gerichte' # Der Name, den wir im Template nutzen 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): class SpeiseplanView(TemplateView):
template_name = 'mensa_app/speiseplan.html' template_name = 'mensa_app/speiseplan.html'
@@ -36,6 +38,7 @@ class SpeiseplanView(TemplateView):
context['target_date'] = target_date context['target_date'] = target_date
context['menues_day'] = Menue.objects.filter(tag__datum=target_date) context['menues_day'] = Menue.objects.filter(tag__datum=target_date)
# 3. Dauerangebote laden (unabhängig vom Tag) # 3. Dauerangebote laden (unabhängig vom Tag)
context['dauerangebote'] = Gericht.objects.filter(ist_dauerangebot=True) context['dauerangebote'] = Gericht.objects.filter(ist_dauerangebot=True)
@@ -50,12 +53,6 @@ class SpeiseplanView(TemplateView):
return context 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): class BestellSummaryView(TemplateView):
template_name = 'mensa_app/bestell_summary.html' template_name = 'mensa_app/bestell_summary.html'
@@ -99,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 context['total_umsatz'] = Bestellung.objects.filter(menue__tag__datum=target_date).aggregate(Sum('menue__preis'))['menue__preis__sum'] or 0
return context 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