{
  "contract_version": "gab_028_contract_v1",
  "gab_id": "GAB-028",
  "canonical_name": "CompletionBlock",
  "module_owner": "EdTechPlayKit",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "completion_block_id",
    "title",
    "body",
    "primary_cta"
  ],
  "optional_fields": [
    "icon_symbol"
  ],
  "field_types": {
    "completion_block_id": "string — identifiant unique de l'instance",
    "title": "string — titre de complétion (ex : 'Étape terminée')",
    "body": "string — message de confirmation + invitation à continuer",
    "primary_cta": "object{label:string, action:string} — bouton structurel, libellé et action viennent du JSON",
    "icon_symbol": "string — symbole affiché dans le cercle checkmark (défaut : '✓')"
  },
  "constraints": [
    "primary_cta.label non vide : libellé du bouton toujours fourni par l'instance.",
    "body ne doit pas être vide : message de confirmation obligatoire.",
    "title ne doit pas être vide : titre de complétion obligatoire.",
    "CompletionBlock = complétion d'UNE étape/action ; pour une célébration forte → GAB-022 ; pour un récap complet → GAB-021."
  ],
  "blocked_conditions": [
    "title absent",
    "body absent",
    "primary_cta absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "aria-label sur le bouton CTA"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, animation pop au chargement, 0 erreur" },
    { "case": "champ requis manquant (title/body/primary_cta)", "expected": "BLOCKED listant le champ manquant" },
    { "case": "clic sur le bouton CTA", "expected": "animation pop rejouée sur l'icône checkmark" },
    { "case": "instance externe injectée via init(ext)", "expected": "le rendu change sans modifier le HTML" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-028",
    "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)."
  }
}
