{
  "contract_version": "gab_040_contract_v1",
  "gab_id": "GAB-040",
  "canonical_name": "XPGainAnimation",
  "module_owner": "EdTechPlayKit",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "xp_gain_id",
    "xp_gained",
    "level_label",
    "xp_after",
    "xp_max"
  ],
  "optional_fields": [
    "context_note",
    "xp_icon",
    "xp_before",
    "replay_cta_label",
    "footer_note"
  ],
  "field_types": {
    "xp_gained": "number — points XP gagnés (déjà calculés par GamificationAdapter)",
    "xp_icon": "string — emoji ou caractère représentant le XP (défaut : '⚡')",
    "level_label": "string — libellé du niveau courant (ex: 'Niveau 4')",
    "xp_before": "number — XP avant le gain (point de départ de l'animation barre)",
    "xp_after": "number — XP après le gain (point d'arrivée de l'animation barre)",
    "xp_max": "number — seuil de passage au niveau suivant",
    "context_note": "string — note pédagogique affichée au-dessus du badge (lecture seule)",
    "replay_cta_label": "string — libellé du bouton replay",
    "footer_note": "string — note de bas de page (rôle PlayKit vs GamificationAdapter)"
  },
  "constraints": [
    "PlayKit AFFICHE uniquement — jamais calculer ni écrire le XP (responsabilité GamificationAdapter).",
    "xp_gained doit être un nombre positif fourni par GamificationAdapter.",
    "xp_after <= xp_max (une valeur supérieure tronque l'affichage à 100%).",
    "replay_cta_label : libellé venant du JSON, jamais hardcodé dans le HTML.",
    "context_note : message informatif read-only, pas interactif."
  ],
  "blocked_conditions": [
    "xp_gained absent ou non numérique",
    "level_label absent",
    "xp_after absent",
    "xp_max absent",
    "instance absente"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "bouton replay accessible au clavier"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "badge animé, barre de progression, 0 erreur" },
    { "case": "xp_gained absent", "expected": "BLOCKED listant le champ" },
    { "case": "xp_max absent", "expected": "BLOCKED listant le champ" },
    { "case": "instance externe injectée", "expected": "le rendu change sans modifier le HTML" },
    { "case": "bouton replay cliqué", "expected": "badge pop + barre animée depuis xp_before vers xp_after" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-040",
    "note": "Ce schema VALIDE l'instance. Le contrat pédagogique complet (input_contract/validation_logic/feedback_scoring_logic) vit dans le CORE-GAB officiel, pas ici (évite la duplication)."
  }
}
