{
  "contract_version": "gab_327_contract_v1",
  "gab_id": "GAB-327",
  "canonical_name": "ResearchLearningCitationCapture",
  "module_owner": "EdTechResearchLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "citation_capture_id",
    "instruction",
    "options"
  ],
  "optional_fields": [
    "title",
    "primary_cta",
    "slots",
    "use_when",
    "do_not_use_when",
    "fallback_text_summary"
  ],
  "field_types": {
    "instruction": "string — consigne affichée à l'élève au-dessus des choix",
    "options": "array<{id:string, label:string, kind:enum['ok','long','halluc'], feedback:string, panel_style:enum['ok','warn','bad']}> — exactement 3 items attendus (1 correct, 1 trop long, 1 sans source)",
    "primary_cta": "object{label:string, action:string}",
    "slots": "array<string> — noms des slots pédagogiques (quote, source, length_limit, exact_vs_paraphrase, hallucination_guard)",
    "use_when": "array<string>",
    "do_not_use_when": "array<string>"
  },
  "constraints": [
    "options doit contenir exactement 1 item kind='ok' (citation valide).",
    "options doit contenir exactement 1 item kind='long' (extrait trop long).",
    "options doit contenir exactement 1 item kind='halluc' (source non vérifiée).",
    "panel_style doit correspondre au kind : ok→ok, long→warn, halluc→bad.",
    "La sélection d'une option désélectionne les autres (single-choice).",
    "Le feedback de l'option kind='halluc' doit inclure un avertissement anti-invention de source."
  ],
  "blocked_conditions": [
    "options absent ou tableau vide",
    "citation_capture_id absent",
    "instruction absente"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "fallback_text"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, 3 chips affichées, 0 erreur" },
    { "case": "champ requis manquant (options absent)", "expected": "BLOCKED listant le champ" },
    { "case": "clic option ok", "expected": "chip verte + panel ok avec feedback validé" },
    { "case": "clic option long", "expected": "chip non sélectionnée + panel warn avec feedback too_long" },
    { "case": "clic option halluc", "expected": "chip rouge + panel bad avec feedback source introuvable" },
    { "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-327",
    "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)."
  }
}
