Feature: Partner-Dashboard (#partner-dashboard) — operative Daten raus aus dem Profil-Editor

Rene: QR-Stats gehören nicht ins öffentliche Profil, eigene Seite fehlte.
Neue Seite 'Partner-Bereich' (Welten-Chip 🤝 zwischen Moderation und Admin,
role:partner — sichtbar für is_partner + Admin; _mergeDefaults reicht den
Chip an bestehende Welt-Configs nach):
- Einladungscode groß + Link-kopieren-Button
- Kacheln: Registrierungen gesamt / diesen Monat / unbestätigt
- QR-Kontingente mit Einzel-Code-Status (aus partner-profil.js hierher verschoben)
- Profil-Status-Karte (Entwurf/Prüfung/frei) mit Sprung zum Editor

Backend: GET /partner/my-stats (Codes mit Zahlen + Profil-Status).
Settings-Partner-Karte: zwei Buttons (Partner-Bereich primär, Profil sekundär);
Dank-Mail-CTA zeigt auf #partner-dashboard. Suite: 52 passed.
This commit is contained in:
rene 2026-06-07 19:06:51 +02:00
parent 3d7d5dc1c4
commit 0a262989f3
12 changed files with 287 additions and 96 deletions

View file

@ -750,6 +750,37 @@ def qr_batch_pdf_admin(batch_id: int, user=Depends(require_admin)):
headers={"Content-Disposition": f'attachment; filename="banyaro-qr-{batch_id}.pdf"'})
@router.get("/partner/my-stats")
def my_partner_stats(user=Depends(require_partner)):
"""Dashboard-Zahlen für den Partner: eigene Codes mit Registrierungen/Versuchen
+ Status des öffentlichen Profils."""
with db() as conn:
codes = conn.execute(
"""SELECT pc.id, pc.code, pc.label, pc.uses, pc.grants_founder,
(SELECT COUNT(*) FROM users u
WHERE u.referred_by = -pc.id AND u.email_verified = 1) AS registrations,
(SELECT COUNT(*) FROM users u
WHERE u.referred_by = -pc.id AND u.email_verified = 0) AS attempts,
(SELECT COUNT(*) FROM users u
WHERE u.referred_by = -pc.id AND u.email_verified = 1
AND strftime('%Y-%m', u.created_at) = strftime('%Y-%m', 'now')) AS registrations_month
FROM partner_codes pc
WHERE pc.owner_user_id = ?
ORDER BY pc.created_at""",
(user["id"],)
).fetchall()
profile = _pp_get_or_empty(conn, user["id"])
return {
"codes": [dict(c) for c in codes],
"profile": {
"exists": bool(profile),
"approved": profile.get("approved", 0),
"submitted_at": profile.get("submitted_at"),
"display_name": profile.get("display_name"),
},
}
@router.get("/partner/my-qr")
def my_qr_batches(user=Depends(require_partner)):
"""Übergabe/Self-Service: eigene Kontingente mit Stats (Code-Besitzer)."""