Feature: QR-Kontingente für Partner — Bestellung, Übergabe, Rückverfolgung
Partner verteilen gedruckte QR-Codes (Sticker/Flyer); jeder physische Code
ist einzeln rückverfolgbar von Scan bis Registrierung.
Backend:
- partner_qr_batches + partner_qr_codes (Token 8-stellig, ohne 0/O/1/l/I),
users.referred_qr, partner_codes.owner_user_id (+Backfill über referred_by)
- /q/{token}: Scan zählen (scans, first/last_scan_at) → Redirect
/?ref=CODE&qr=TOKEN — dockt am bestehenden Referral-Flow an
- Registrierung: qr_token wird nur zugeordnet, wenn er zum eingelösten
Partner-Code gehört (Manipulationsschutz)
- Admin: Kontingent bestellen (max 500), Liste mit Scans/Registrierungen,
Löschen (Zweiklick), druckfertiges A4-PDF (segno+fpdf2, 3×4 Grid mit
Kurz-URL + laufender Nummer), Code-Besitzer zuordnen
- Partner-Self-Service: /partner/my-qr (+PDF) für Code-Besitzer
Frontend:
- Admin-Partner-Tab: Karte 'QR-Kontingente' (Bestellung, Stats, PDF, Besitzer)
- Partner-Profil: 'Meine QR-Codes' mit Scans/Registrierungen + PDF-Download
- boot.js/app.js speichern ?qr=, Registrierung schickt qr_token mit
Neu: segno==1.6.6 (pure-python QR). Tests: 5 neue (PDF, Scan-Zählung,
Attribution, Fremd-Token-Schutz, Self-Service). Suite: 51 passed.
This commit is contained in:
parent
cadfb24a8d
commit
f604ab7c4f
16 changed files with 621 additions and 23 deletions
|
|
@ -86,14 +86,14 @@
|
|||
<title>Ban Yaro</title>
|
||||
|
||||
<!-- Theme + theme-color Statusleiste vor CSS setzen -->
|
||||
<script src="/js/boot-early.js?v=1256"></script>
|
||||
<script src="/js/boot-early.js?v=1257"></script>
|
||||
|
||||
<!-- CSS: Reihenfolge ist wichtig — ?v= zwingt Browser zur Neuladung -->
|
||||
<link rel="stylesheet" href="/css/design-system.css?v=1256">
|
||||
<link rel="stylesheet" href="/css/layout.css?v=1256">
|
||||
<link rel="stylesheet" href="/css/components.css?v=1256">
|
||||
<link rel="stylesheet" href="/css/utilities.css?v=1256">
|
||||
<link rel="stylesheet" href="/css/lists.css?v=1256">
|
||||
<link rel="stylesheet" href="/css/design-system.css?v=1257">
|
||||
<link rel="stylesheet" href="/css/layout.css?v=1257">
|
||||
<link rel="stylesheet" href="/css/components.css?v=1257">
|
||||
<link rel="stylesheet" href="/css/utilities.css?v=1257">
|
||||
<link rel="stylesheet" href="/css/lists.css?v=1257">
|
||||
</head>
|
||||
<body>
|
||||
|
||||
|
|
@ -612,11 +612,11 @@
|
|||
<div id="modal-container"></div>
|
||||
|
||||
<!-- JS: Reihenfolge ist wichtig — erst Basis, dann Features -->
|
||||
<script src="/js/api.js?v=1256"></script>
|
||||
<script src="/js/ui.js?v=1256"></script>
|
||||
<script src="/js/app.js?v=1256"></script>
|
||||
<script src="/js/worlds.js?v=1256"></script>
|
||||
<script src="/js/offline-indicator.js?v=1256"></script>
|
||||
<script src="/js/api.js?v=1257"></script>
|
||||
<script src="/js/ui.js?v=1257"></script>
|
||||
<script src="/js/app.js?v=1257"></script>
|
||||
<script src="/js/worlds.js?v=1257"></script>
|
||||
<script src="/js/offline-indicator.js?v=1257"></script>
|
||||
|
||||
<!-- Feature-Seiten werden lazy geladen -->
|
||||
|
||||
|
|
@ -626,7 +626,7 @@
|
|||
|
||||
|
||||
<!-- Boot: Offline-Banner + SW-Registration (extrahiert für CSP) -->
|
||||
<script src="/js/boot.js?v=1256"></script>
|
||||
<script src="/js/boot.js?v=1257"></script>
|
||||
|
||||
|
||||
</body>
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue