Files
vergabe-teilnahme/vergabe_teilnahme/apps/feedback/views.py
tegwick 5a231223c0 feat(WP-0012): Querschnitt — Freigaben, Felder, Feedback, Suche, Tests
Implements all 8 tasks of the final cross-cutting workplan:

- T01: Generisches Freigabe-Modal (freigabe_modal, freigabe_erteilen views + templates)
- T02: Freigaben-Übersicht pro Ausschreibung (freigaben_uebersicht view + template)
- T03: EntityFieldConfig Admin-Interface (/felder/<entity_type>/ with HTMX toggle)
- T04: CustomAttribute-Panel (full CRUD with sort, lazy HTMX load)
- T05: Feedback-Backlog mit Statusverwaltung + feedback_success.html template
- T06: End-to-End-Tests in vergabe_teilnahme/tests/test_e2e.py (8 tests)
- T07: Globale Suche erweitert (Dokumente, Nachweise, Referenzen, Marktbegleiter)
- T08: Alle Migrationen sauber, 68/68 Tests grün, Ruff-Fehler in neuem Code behoben

Bugfix: URL-Namespace-Fehler in Abgabe-Templates (ausschreibungen:nachbetrachtung:abgabe → ausschreibungen:abgabe)

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
2026-05-11 17:54:38 +02:00

82 lines
2.9 KiB
Python

from django.shortcuts import get_object_or_404, render
from django.views.decorators.http import require_GET, require_POST
from .models import Feedbackeintrag
@require_GET
def modal(request):
return render(request, 'partials/feedback_modal.html')
@require_POST
def submit(request):
beschreibung = request.POST.get('beschreibung', '').strip()
if not beschreibung:
return render(request, 'partials/feedback_modal.html',
{'error': 'Beschreibung ist erforderlich.'})
entry = Feedbackeintrag(
titel=request.POST.get('titel', 'Ohne Titel') or 'Ohne Titel',
beschreibung=beschreibung,
seite_kontext=request.POST.get('seite_kontext', ''),
kategorie=request.POST.get('kategorie', 'hinweis'),
dringlichkeit=request.POST.get('dringlichkeit', 'mittel'),
)
ausschreibung_pk = request.POST.get('ausschreibung')
if ausschreibung_pk:
try:
from vergabe_teilnahme.apps.ausschreibungen.models import Ausschreibung
entry.ausschreibung = Ausschreibung.objects.get(pk=ausschreibung_pk)
except (Ausschreibung.DoesNotExist, ValueError):
pass
if request.user.is_authenticated:
entry.erfasst_von = request.user
entry.save()
return render(request, 'partials/feedback_success.html', {})
def backlog(request):
qs = Feedbackeintrag.objects.select_related('erfasst_von', 'ausschreibung').all()
status_filter = request.GET.get('status')
if status_filter:
qs = qs.filter(status=status_filter)
kategorie_filter = request.GET.get('kategorie')
if kategorie_filter:
qs = qs.filter(kategorie=kategorie_filter)
dringlichkeit_filter = request.GET.get('dringlichkeit')
if dringlichkeit_filter:
qs = qs.filter(dringlichkeit=dringlichkeit_filter)
ctx = {
'eintraege': qs,
'status_choices': Feedbackeintrag.STATUS_CHOICES,
'kategorie_choices': Feedbackeintrag.KATEGORIE_CHOICES,
'dringlichkeit_choices': Feedbackeintrag.DRINGLICHKEIT_CHOICES,
'current_status': status_filter or '',
'current_kategorie': kategorie_filter or '',
'current_dringlichkeit': dringlichkeit_filter or '',
'breadcrumbs': [{'label': 'Feedback-Backlog', 'url': None}],
}
return render(request, 'feedback/backlog.html', ctx)
@require_POST
def status_aendern(request, pk):
eintrag = get_object_or_404(Feedbackeintrag, pk=pk)
neuer_status = request.POST.get('status')
if neuer_status in dict(Feedbackeintrag.STATUS_CHOICES):
eintrag.status = neuer_status
bewertung = request.POST.get('bewertung')
if bewertung is not None:
eintrag.bewertung = bewertung
entscheidung = request.POST.get('entscheidung')
if entscheidung is not None:
eintrag.entscheidung = entscheidung
eintrag.save()
return render(request, 'feedback/partials/eintrag_zeile.html', {'eintrag': eintrag})