{
  "contract_version": "gab_148_contract_v1",
  "gab_id": "GAB-148",
  "canonical_name": "InteractiveScenarioBranch",
  "module_owner": "EdTechInteractiveLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "scenario_id",
    "instruction",
    "choices",
    "branches"
  ],
  "optional_fields": [
    "title",
    "branch_heading",
    "branch_placeholder",
    "feedback"
  ],
  "field_types": {
    "instruction": "string — question pédagogique centrale affichée en haut",
    "choices": "array<{id:string, label:string, description?:string}> — liste des options mutuellement exclusives (min 2)",
    "branches": "object{[choiceId:string]:string(html)} — map id → texte de conséquence (HTML inline autorisé pour gras/em)",
    "branch_heading": "string — libellé du titre de la zone branche (ex : '📌 Conséquence sur ton plan')",
    "branch_placeholder": "string — texte affiché dans la zone branche avant tout choix",
    "feedback": "string — message pédagogique global affiché après toute sélection"
  },
  "constraints": [
    "choices doit contenir au moins 2 éléments.",
    "Chaque choices[].id doit avoir une clé correspondante dans branches.",
    "HTML dans branches limité aux balises inline (<b>, <em>, <strong>) — pas de scripts ni d'événements.",
    "instruction doit être une vraie question pédagogique, pas un titre."
  ],
  "blocked_conditions": [
    "instruction absente",
    "choices vides",
    "branches absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "aria-label sur chaque bouton de choix",
    "prefers_reduced_motion",
    "fallback_text via branch_placeholder"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, choix cliquables, branche affichée au clic" },
    { "case": "champ requis manquant (instruction)", "expected": "BLOCKED listant le champ" },
    { "case": "choices vides ([])", "expected": "BLOCKED listant le champ" },
    { "case": "branches absent ou null", "expected": "BLOCKED listant le champ" },
    { "case": "clic choix A", "expected": "branche branches.a affichée, libellé branch_heading visible" },
    { "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" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-148",
    "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)."
  }
}
