{
  "contract_version": "gab_309_contract_v1",
  "gab_id": "GAB-309",
  "canonical_name": "DocumentLearningImageAnalysis",
  "module_owner": "EdTechDocumentLearning",
  "renderer_key": "annotation_media",
  "required_fields": [
    "gab_id",
    "image_analysis_id",
    "zones",
    "mode_labels"
  ],
  "optional_fields": [
    "title",
    "image_ref",
    "observe_instruction",
    "interpret_guard",
    "min_zones_for_success",
    "use_when",
    "do_not_use_when",
    "accessibility",
    "child_safety"
  ],
  "field_types": {
    "image_analysis_id": "string — identifiant unique de l'instance",
    "image_ref": "object{src,alt,caption} — src peut être _TODO si image non fournie",
    "mode_labels": "object{observe:string, interpret:string} — libellés des deux boutons de mode",
    "observe_instruction": "string — texte d'aide affiché au-dessus de la scène",
    "interpret_guard": "string — message affiché si l'élève clique en mode interpréter sans avoir observé",
    "zones": "array<{zone_id:string, kind:enum['composition','symbole','ocr','couleur','texte','forme','autre'], emoji:string, style:string(CSS inline position), feedback_observe:string, is_trap:boolean}>",
    "min_zones_for_success": "integer — nombre minimal de zones non-piège à trouver pour valider l'étape",
    "use_when": "array<string>",
    "do_not_use_when": "array<string>",
    "accessibility": "object{keyboard_navigable,focus_visible,prefers_reduced_motion,alt_text_required}",
    "child_safety": "object{anti_invention:string, ocr_trap_labeled:string}"
  },
  "constraints": [
    "zones ne doit pas être vide : au moins 1 zone est requise.",
    "Toute zone is_trap:true doit avoir un feedback_observe commençant par ⚠ et ne doit JAMAIS deviner le contenu incertain.",
    "Le champ image_ref.src peut porter le préfixe _TODO si l'image réelle n'est pas encore liée — l'instance reste valide mais doit être complétée avant déploiement.",
    "min_zones_for_success doit être ≤ nombre de zones avec is_trap:false.",
    "mode_labels.observe et mode_labels.interpret doivent tous deux être présents."
  ],
  "blocked_conditions": [
    "image_analysis_id absent",
    "zones vides",
    "mode_labels incomplet"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "alt_text sur image_ref obligatoire en instance finale"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, zones cliquables, 0 erreur" },
    { "case": "champ requis manquant (zones vides)", "expected": "BLOCKED listant le champ" },
    { "case": "clic zone is_trap:true en mode observe", "expected": "panel warn avec message ⚠, zone non marquée 'found'" },
    { "case": "clic zone en mode interpret", "expected": "panel warn interpret_guard, aucune zone marquée" },
    { "case": "instance externe injectée via ENGINE.init(ext)", "expected": "le rendu change sans modifier le HTML" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal, mode-row en colonne sur mobile" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-309",
    "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). Source HTML : INDEX-300-documentlearning-GAB-306-310-PLAYABLE.html, stage data-tpl='309', handlers d309Mode / d309Zone."
  }
}
