{
  "contract_version": "gab_130_contract_v1",
  "gab_id": "GAB-130",
  "canonical_name": "ExerciseRetryPrompt",
  "module_owner": "EdTechExerciseLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "retry_prompt_id",
    "title",
    "body",
    "retry_cta"
  ],
  "optional_fields": [
    "emoji",
    "hint",
    "variant_cta",
    "variants_pool"
  ],
  "field_types": {
    "gab_id": "string — identifiant du gabarit, toujours 'GAB-130'",
    "retry_prompt_id": "string — identifiant unique de cette instance d'invitation à réessayer",
    "title": "string — titre court bienveillant affiché en grand",
    "body": "string — texte d'encouragement principal (1-2 phrases)",
    "retry_cta": "object{label:string, action:string} — bouton principal relancer même exercice",
    "emoji": "string — emoji décoratif affiché en header de carte (défaut : '💪')",
    "hint": "object{icon:string, label:string, text:string} — indice contextuel optionnel, affiché dans bandeau doré",
    "variant_cta": "object{label:string, action:string} — bouton optionnel pour proposer une variante d'énoncé",
    "variants_pool": "array<string> — liste d'énoncés alternatifs à rotation (utilisé par load_variant)"
  },
  "constraints": [
    "title et body sont des textes pédagogiques bienveillants, jamais accusatoires.",
    "retry_cta.action doit être 'retry_same' ou une action reconnue par le moteur appelant.",
    "variant_cta et variants_pool sont liés : si variant_cta présent, variants_pool doit contenir au moins 1 item.",
    "hint.text ne doit pas révéler la réponse directe, uniquement un amorçage méthodologique.",
    "emoji est cosmétique ; ne pas encoder de sens pédagogique critique dans l'emoji seul."
  ],
  "blocked_conditions": [
    "title absent",
    "body absent",
    "retry_cta absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "buttons_labeled_by_data",
    "hint_section_hidden_when_absent"
  ],
  "qa_cases": [
    { "case": "instance conforme complète", "expected": "rendu complet : emoji, titre, body, hint, 2 boutons actifs" },
    { "case": "champ requis manquant (title absent)", "expected": "BLOCKED listant le champ manquant" },
    { "case": "champ requis manquant (retry_cta absent)", "expected": "BLOCKED listant le champ manquant" },
    { "case": "hint absent (champ non fourni)", "expected": "bandeau hint masqué (hidden), pas d'erreur" },
    { "case": "variant_cta absent", "expected": "seul le bouton retry affiché, pas d'erreur" },
    { "case": "variants_pool épuisé (cycle)", "expected": "rotation cyclique sur les variantes disponibles" },
    { "case": "instance externe injectée via ENGINE.init(ext)", "expected": "rendu change sans modifier le HTML" },
    { "case": "responsive 375px", "expected": "boutons en colonne, aucun débordement horizontal" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-130",
    "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)."
  }
}
