{
  "contract_version": "gab_035_contract_v1",
  "gab_id": "GAB-035",
  "canonical_name": "UnlockBlock",
  "module_owner": "EdTechPlayKit",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "unlock_block_id",
    "kicker",
    "title",
    "body",
    "primary_cta"
  ],
  "optional_fields": [
    "icon",
    "highlight"
  ],
  "field_types": {
    "kicker": "string — libellé court affiché au-dessus du titre (ex : « Nouveau déblocage »)",
    "title": "string — titre principal du déblocage",
    "body": "string — description de ce qui a été débloqué",
    "primary_cta": "object{label:string, action:string}",
    "icon": "string — emoji ou identifiant pictogramme (défaut : 🔓)",
    "highlight": "enum['gold','violet','coral','mint'] — couleur d'accentuation du kicker"
  },
  "constraints": [
    "primary_cta.label : libellé affiché dans le bouton ; action transmis au handler parent.",
    "kicker : texte court (< 40 car.), en majuscules dans le rendu via CSS letter-spacing.",
    "body : décrit le contenu débloqué, jamais vide si title est présent.",
    "icon : si absent le rendu utilise 🔓 par défaut.",
    "highlight : si absent le rendu utilise la couleur gold par défaut."
  ],
  "blocked_conditions": [
    "kicker absent",
    "title absent",
    "body absent",
    "primary_cta absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "aria_live_region_on_unlock"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet avec animation unlock, 0 erreur" },
    { "case": "champ requis manquant (ex: title absent)", "expected": "BLOCKED listant le champ manquant" },
    { "case": "instance externe injectée via ENGINE.init(ext)", "expected": "le rendu change sans modifier le HTML" },
    { "case": "highlight absent", "expected": "fallback couleur gold appliqué" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal" },
    { "case": "CTA cliqué", "expected": "feedback textuel visible (rejouer l'animation)" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-035",
    "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)."
  }
}
