{
  "contract_version": "gab_143_contract_v1",
  "gab_id": "GAB-143",
  "canonical_name": "InteractiveTimelineExplorer",
  "module_owner": "EdTechInteractiveLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "timeline_id",
    "title",
    "events"
  ],
  "optional_fields": [
    "instruction",
    "primary_cta"
  ],
  "field_types": {
    "timeline_id": "string — identifiant unique de l'instance (slug)",
    "title": "string — titre affiché dans le hero",
    "instruction": "string — consigne affichée au-dessus de la frise",
    "events": "array<{date:string, title:string, detail:string}> — liste ordonnée des événements",
    "primary_cta": "object{label:string, action:string} — libellé et action du bouton principal"
  },
  "constraints": [
    "events doit contenir au moins 1 événement.",
    "Chaque événement doit avoir date, title et detail non vides.",
    "primary_cta.label vient du JSON — jamais hardcodé dans le HTML.",
    "Les événements sont affichés dans l'ordre du tableau (ordre chronologique attendu)."
  ],
  "blocked_conditions": [
    "events absent ou vide (BLOCKED)",
    "title absent (BLOCKED)",
    "timeline_id absent (BLOCKED)"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "accordion_aria_expanded"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, frise avec 4 événements, accordéon fonctionnel" },
    { "case": "events vide []", "expected": "BLOCKED listant 'events absent ou vide'" },
    { "case": "title absent", "expected": "BLOCKED listant 'title absent'" },
    { "case": "timeline_id absent", "expected": "BLOCKED listant 'timeline_id absent'" },
    { "case": "clic sur un événement", "expected": "détail s'affiche (accordéon open), clic à nouveau = ferme" },
    { "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, frise lisible" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-143",
    "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)."
  }
}
