{
  "contract_version": "gab_139_contract_v1",
  "gab_id": "GAB-139",
  "canonical_name": "InteractiveClickableDiagram",
  "module_owner": "EdTechInteractiveLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "diagram_id",
    "title",
    "items"
  ],
  "optional_fields": [
    "instruction",
    "_note_dev"
  ],
  "field_types": {
    "diagram_id": "string — identifiant unique de l'instance de schéma",
    "title": "string — titre affiché dans le hero",
    "instruction": "string — texte d'invite dans la zone d'explication avant tout clic",
    "items": "array<{id:string, label:string, title:string, description:string, position:{top:string,left:string}}> — points chauds du schéma"
  },
  "constraints": [
    "items doit contenir au moins 1 élément ; si vide → BLOCKED.",
    "Chaque item doit avoir id, title et description ; label et position sont fortement recommandés.",
    "position.top et position.left sont des valeurs CSS en % pour positionnement absolu dans le canvas 16:10.",
    "Le moteur positionne les boutons dynamiquement — pas de classes CSS h1/h2/h3 en dur dans le JSON."
  ],
  "blocked_conditions": [
    "items vide ou absent",
    "gab_id absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "aria_label_par_item",
    "prefers_reduced_motion",
    "fallback_instruction_text"
  ],
  "qa_cases": [
    { "case": "instance conforme 3 items", "expected": "3 points cliquables, texte change au clic, 0 erreur" },
    { "case": "items:[] vide", "expected": "BLOCKED listé dans le moteur, pas de rendu partiel" },
    { "case": "gab_id absent", "expected": "BLOCKED listé dans le moteur" },
    { "case": "instance externe injectée via ENGINE.init(ext)", "expected": "rendu change sans modifier le HTML" },
    { "case": "navigation clavier (Tab + Entrée)", "expected": "focus visible, description affichée" },
    { "case": "responsive 375/768/1024", "expected": "canvas 16:10 respecté, aucun débordement" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-139",
    "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)."
  }
}
