feat(loyalty): cross-persona page alignment with shared components

Align loyalty pages across admin, merchant, and store personas so each
sees the same page set scoped to their access level. Admin acts as a
superset of merchant with "on behalf" capabilities.

New pages:
- Store: Staff PINs management (CRUD)
- Merchant: Cards, Card Detail, Transactions, Staff PINs (CRUD), Settings (read-only)
- Admin: Merchant Cards, Card Detail, Transactions, PINs (read-only)

Architecture:
- 4 shared Jinja2 partials (cards-list, card-detail, transactions, pins)
- 4 shared JS factory modules parameterized by apiPrefix/scope
- Persona templates are thin wrappers including shared partials
- PinDetailResponse schema for cross-store PIN listings

API: 17 new endpoints (11 merchant, 6 admin on-behalf)
Tests: 38 new integration tests, arch-check green
i18n: ~130 new keys across en/fr/de/lb
Docs: pages-and-navigation.md with full page matrix

Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
This commit is contained in:
2026-03-22 19:28:07 +01:00
parent f41f72b86f
commit 6161d69ba2
49 changed files with 4385 additions and 14 deletions

View File

@@ -81,7 +81,9 @@
"program": "Programme",
"overview": "Aperçu",
"settings": "Paramètres",
"wallet_debug": "Wallet Debug"
"wallet_debug": "Wallet Debug",
"staff_pins": "PINs du personnel",
"transactions": "Transactions"
},
"permissions": {
"view_programs": "Voir les programmes",
@@ -242,6 +244,105 @@
"terms_conditions": "Conditions Générales",
"privacy_policy_url": "URL politique de confidentialité"
},
"cards": {
"total_members": "Membres totaux",
"active_30d": "Actifs (30j)",
"new_this_month": "Nouveaux ce mois",
"total_points_balance": "Solde total des points",
"search_placeholder": "Rechercher par nom, email, téléphone ou numéro de carte...",
"all_status": "Tous les statuts",
"all_stores": "Tous les magasins",
"col_member": "Membre",
"col_card_number": "Numéro de carte",
"col_points_balance": "Solde de points",
"col_last_activity": "Dernière activité",
"col_status": "Statut",
"col_actions": "Actions",
"no_members": "Aucun membre trouvé",
"adjust_search": "Essayez de modifier vos critères de recherche"
},
"card_detail": {
"title": "Détail de la carte",
"loading": "Chargement des détails...",
"error_loading": "Erreur lors du chargement",
"points_balance": "Solde de points",
"total_earned": "Total gagné",
"total_redeemed": "Total échangé",
"member_since": "Membre depuis",
"customer_information": "Informations client",
"name": "Nom",
"email": "E-mail",
"phone": "Téléphone",
"birthday": "Anniversaire",
"card_details": "Détails de la carte",
"card_number": "Numéro de carte",
"status": "Statut",
"last_activity": "Dernière activité",
"enrolled_at": "Inscrit à",
"transaction_history": "Historique des transactions",
"col_date": "Date",
"col_type": "Type",
"col_points": "Points",
"col_location": "Emplacement",
"col_notes": "Notes",
"no_transactions": "Aucune transaction trouvée"
},
"transactions": {
"title": "Transactions",
"subtitle": "Voir toutes les transactions",
"loading": "Chargement des transactions...",
"error_loading": "Erreur lors du chargement",
"search_placeholder": "Rechercher des transactions...",
"all_types": "Tous les types",
"all_stores": "Tous les magasins",
"col_date": "Date",
"col_customer": "Client",
"col_type": "Type",
"col_points": "Points",
"col_location": "Emplacement",
"col_notes": "Notes",
"no_transactions": "Aucune transaction trouvée"
},
"pins": {
"title": "PINs du personnel",
"subtitle": "Gérer les PINs d'authentification",
"loading": "Chargement des PINs...",
"error_loading": "Erreur lors du chargement",
"total_pins": "Total PINs",
"active_pins": "Actifs",
"locked_pins": "Verrouillés",
"all_stores": "Tous les magasins",
"all_status": "Tous les statuts",
"status_active": "Actif",
"status_inactive": "Inactif",
"status_locked": "Verrouillé",
"col_name": "Nom",
"col_staff_id": "ID employé",
"col_store": "Magasin",
"col_status": "Statut",
"col_locked": "Verrouillé",
"col_last_used": "Dernière utilisation",
"col_actions": "Actions",
"no_pins": "Aucun PIN trouvé",
"create_pin": "Créer un PIN",
"edit_pin": "Modifier le PIN",
"delete_pin": "Supprimer le PIN",
"unlock_pin": "Déverrouiller",
"confirm_delete": "Êtes-vous sûr de vouloir supprimer ce PIN ?",
"pin_name": "Nom de l'employé",
"pin_staff_id": "ID employé (optionnel)",
"pin_code": "Code PIN",
"pin_code_hint": "PIN à 4-6 chiffres",
"pin_store": "Magasin",
"select_store": "Sélectionner un magasin",
"pin_created": "PIN créé avec succès",
"pin_updated": "PIN modifié avec succès",
"pin_deleted": "PIN supprimé avec succès",
"pin_unlocked": "PIN déverrouillé avec succès",
"save": "Enregistrer",
"cancel": "Annuler",
"read_only_notice": "Les PINs sont en lecture seule en mode admin"
},
"program_form": {
"program_type": "Type de programme",
"points_type_desc": "Gagner des points par EUR dépensé",
@@ -360,7 +461,31 @@
"cross_location_redemption": "Échange inter-points de vente",
"allowed": "Autorisé",
"disabled": "Désactivé",
"modify_policy": "Modifier la politique admin"
"modify_policy": "Modifier la politique admin",
"view_cards": "Voir les cartes",
"view_transactions": "Voir les transactions",
"view_pins": "Voir les PINs"
},
"merchant_cards": {
"title": "Cartes du commerçant",
"subtitle": "Voir les cartes de ce commerçant",
"loading": "Chargement des cartes...",
"error_loading": "Erreur lors du chargement"
},
"merchant_card_detail": {
"title": "Détail de la carte"
},
"merchant_transactions": {
"title": "Transactions du commerçant",
"subtitle": "Voir toutes les transactions de ce commerçant",
"loading": "Chargement des transactions...",
"error_loading": "Erreur lors du chargement"
},
"merchant_pins": {
"title": "PINs du commerçant",
"subtitle": "Voir les PINs de ce commerçant (lecture seule)",
"loading": "Chargement des PINs...",
"error_loading": "Erreur lors du chargement"
},
"merchant_settings": {
"title": "Paramètres de fidélité du commerçant",
@@ -445,6 +570,34 @@
"delete_message": "Cela supprimera définitivement votre programme de fidélité et toutes les données associées (cartes, transactions, récompenses). Cette action est irréversible.",
"delete_confirm": "Supprimer le programme"
},
"cards": {
"title": "Cartes clients",
"subtitle": "Voir et gérer les membres fidélité sur tous les sites"
},
"card_detail": {
"title": "Détail de la carte"
},
"transactions": {
"title": "Transactions",
"subtitle": "Voir toutes les transactions sur tous les sites"
},
"pins": {
"title": "PINs du personnel",
"subtitle": "Gérer les PINs sur tous les sites"
},
"settings": {
"title": "Paramètres fidélité",
"subtitle": "Voir les paramètres du programme",
"admin_controlled": "Ces paramètres sont gérés par l'administrateur",
"staff_pin_policy": "Politique PIN personnel",
"self_enrollment": "Auto-inscription",
"cross_location": "Échange multi-sites",
"void_transactions": "Autoriser les annulations",
"enabled": "Activé",
"disabled": "Désactivé",
"required": "Obligatoire",
"optional": "Optionnel"
},
"analytics": {
"title": "Analytique fidélité",
"subtitle": "Statistiques de fidélité pour tous vos magasins",
@@ -609,6 +762,12 @@
"create_program": "Créer un programme",
"contact_admin": "Contactez votre administrateur pour configurer un programme de fidélité."
},
"pins": {
"title": "PINs du personnel",
"subtitle": "Gérer les PINs d'authentification pour ce magasin",
"loading": "Chargement des PINs...",
"error_loading": "Erreur lors du chargement des PINs"
},
"settings": {
"title": "Paramètres de fidélité",
"page_title": "Paramètres du programme de fidélité",