{
  "contract_version": "gab_097_contract_v1",
  "gab_id": "GAB-097",
  "canonical_name": "VisualProgressiveReveal",
  "module_owner": "EdTechVisualLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "reveal_id",
    "title",
    "steps"
  ],
  "optional_fields": [
    "cta_label",
    "cta_done_label",
    "caption"
  ],
  "field_types": {
    "steps": "array<{num:integer, text:string, pre_revealed:boolean}>  — au moins 2 étapes ; la première porte pre_revealed:true",
    "cta_label": "string — libellé du bouton de révélation (défaut : 'Révéler l'étape suivante')",
    "cta_done_label": "string — libellé quand toutes les étapes sont révélées",
    "caption": "string|null — légende facultative sous la carte"
  },
  "constraints": [
    "steps doit contenir au minimum 2 éléments.",
    "Exactement 1 étape porte pre_revealed:true (l'amorce visible, typiquement num:1).",
    "Le num de chaque step doit être unique et croissant.",
    "cta_label et cta_done_label viennent du JSON ; jamais en dur dans le HTML moteur."
  ],
  "blocked_conditions": [
    "steps absent",
    "steps vide (longueur 0)",
    "reveal_id absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "aria_live_region_sur_reveal"
  ],
  "qa_cases": [
    { "case": "instance conforme 4 étapes", "expected": "rendu complet, bouton actif, 1 étape visible" },
    { "case": "champ steps absent", "expected": "BLOCKED listant le champ manquant" },
    { "case": "steps vide []", "expected": "BLOCKED — steps vide" },
    { "case": "reveal_id absent", "expected": "BLOCKED — reveal_id absent" },
    { "case": "clic bouton jusqu'à la fin", "expected": "bouton se désactive avec cta_done_label, toutes étapes visibles" },
    { "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-097",
    "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)."
  }
}
