{
  "contract_version": "gab_021_contract_v1",
  "gab_id": "GAB-021",
  "canonical_name": "SessionRecapChecklist",
  "module_owner": "EdTechPlayKit",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "recap_id",
    "title",
    "key_points",
    "primary_cta"
  ],
  "optional_fields": [
    "icon",
    "summary",
    "secondary_cta"
  ],
  "field_types": {
    "recap_id": "string — identifiant unique de l'instance de récap",
    "title": "string — titre affiché sous le trophée",
    "icon": "string — emoji affiché dans le trophée (défaut : 🏆)",
    "key_points": "array<string> — liste des points clés appris (non vide)",
    "primary_cta": "object{label:string, action:string, confirm_message?:string}",
    "summary": "string|null — texte résumé optionnel (non affiché dans le rendu de base)",
    "secondary_cta": "object{label,action}|null — CTA secondaire optionnel"
  },
  "constraints": [
    "key_points doit contenir au moins 1 entrée (array non vide).",
    "primary_cta.label est le libellé du bouton CTA — vient du JSON, jamais hardcodé.",
    "primary_cta.confirm_message est affiché au clic comme feedback de confirmation.",
    "icon est un emoji — défaut 🏆 si absent.",
    "Le HTML (layout, animation, barre, bouton) ne change pas selon l'instance."
  ],
  "blocked_conditions": [
    "gab_id absent",
    "recap_id absent",
    "title absent",
    "key_points vides",
    "primary_cta absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "bouton CTA accessible au clavier"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, animation staggered, 0 erreur" },
    { "case": "champ requis manquant (recap_id)", "expected": "BLOCKED avec message lisible" },
    { "case": "key_points vides []", "expected": "BLOCKED listant key_points vides" },
    { "case": "primary_cta absent", "expected": "BLOCKED listant primary_cta absent" },
    { "case": "instance externe injectée via ENGINE.init(ext)", "expected": "le rendu change sans modifier le HTML" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal" },
    { "case": "clic sur le bouton CTA", "expected": "confirm_message affiché sous le bouton" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-021",
    "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)."
  }
}
