{
  "contract_version": "gab_091_contract_v1",
  "gab_id": "GAB-091",
  "canonical_name": "VisualFormulaBreakdown",
  "module_owner": "EdTechVisualLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "formula_id",
    "formula_display",
    "formula_parts"
  ],
  "optional_fields": [
    "title",
    "condition",
    "primary_cta",
    "use_when",
    "do_not_use_when",
    "_note_dev"
  ],
  "field_types": {
    "formula_id": "string — identifiant unique de l'instance formule",
    "formula_display": "string — texte brut de la formule, tokens des parts inclus",
    "formula_parts": "array<{sym:string, token:string, label:string, description:string}> — chaque symbole de la formule et sa définition",
    "condition": "string — condition d'application de la formule (optionnel)",
    "primary_cta": "object{label:string, action:string}",
    "use_when": "array<string>",
    "do_not_use_when": "array<string>"
  },
  "constraints": [
    "Chaque entrée de formula_parts doit avoir : sym (clé de liaison), token (fragment exact de formula_display), label (symbole affiché), description (définition).",
    "formula_display doit contenir les tokens de toutes les formula_parts pour que le moteur puisse les rendre cliquables.",
    "primary_cta.label vient du JSON — jamais hardcodé dans le HTML.",
    "condition : signal pédagogique affiché tel quel, jamais inventé."
  ],
  "blocked_conditions": [
    "formula_id absent",
    "formula_display absent",
    "formula_parts vide"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "aria-label sur chaque sym et fb-leg"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, 0 erreur" },
    { "case": "champ requis manquant (formula_display)", "expected": "BLOCKED listant le champ" },
    { "case": "formula_parts vide", "expected": "BLOCKED — formula_parts vide" },
    { "case": "clic symbole formule", "expected": "le symbole et la ligne de légende correspondante passent en état actif" },
    { "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-091",
    "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)."
  }
}
