{
  "contract_version": "gab_026_contract_v1",
  "gab_id": "GAB-026",
  "canonical_name": "CalloutLearningNote",
  "module_owner": "EdTechPlayKit",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "callout_id",
    "default_tone",
    "tones"
  ],
  "optional_fields": [
    "accessibility"
  ],
  "field_types": {
    "default_tone": "enum['exam_focus','calm','warning','encouraging']",
    "tones": "object{ exam_focus?, calm?, warning?, encouraging? } — chaque ton: {icon:string, title:string, body:string}",
    "accessibility": "object{keyboard_navigable:boolean, focus_visible:boolean, prefers_reduced_motion:boolean}"
  },
  "constraints": [
    "default_tone doit exister comme clé dans tones.",
    "Chaque ton présent doit contenir icon, title et body.",
    "Au moins 1 ton doit être défini dans tones.",
    "Aucun contenu pédagogique ne doit être codé en dur dans le renderer — tout vient de l'instance."
  ],
  "blocked_conditions": [
    "tones absent ou vide",
    "default_tone absent",
    "callout_id absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "role=note sur le callout"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "callout rendu avec le ton default_tone, 0 erreur" },
    { "case": "champ requis manquant (tones absent)", "expected": "BLOCKED listant le champ manquant" },
    { "case": "clic bouton de ton", "expected": "icône + titre + corps + couleur changent sans modifier le HTML" },
    { "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-026",
    "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). Contenu extrait du HTML source INDEX-300-playkit-GAB-026-030-PLAYABLE.html, stage data-tpl='26', objet calloutData."
  }
}
