{
  "contract_version": "gab_303_contract_v1",
  "gab_id": "GAB-303",
  "canonical_name": "DocumentLearningObserveDescribe",
  "module_owner": "EdTechDocumentLearning",
  "renderer_key": "annotation_media",
  "required_fields": [
    "gab_id",
    "observe_id",
    "title",
    "document_ref",
    "observation_prompt",
    "modes",
    "observable_zones",
    "description_rules"
  ],
  "optional_fields": [
    "observe_order",
    "source_notice",
    "accessibility",
    "child_safety"
  ],
  "field_types": {
    "document_ref": "object{src,kind:enum['image','document'],alt}",
    "observation_prompt": "string — consigne d'observation (faits, pas interprétation)",
    "modes": "array<{id:enum['obs','int'],label,active:boolean}> — toujours 2 modes : observer (obs) et interpréter (int)",
    "observable_zones": "array<{id,emoji,fact,pos:{top,left|right,width,height}}> — fact = UNIQUEMENT ce qu'on VOIT, jamais une interprétation",
    "observe_order": "array<string> — ids des zones dans l'ordre d'observation suggéré",
    "description_rules": "object{observe_feedback,interpret_block,all_observed}"
  },
  "constraints": [
    "Chaque zone.fact décrit UNIQUEMENT ce qui est visible (fait), jamais une interprétation — l'interprétation est renvoyée à GAB-305.",
    "modes déclaré avec au moins le mode 'obs' ; le mode 'int' bloque l'observation et affiche un rappel pédagogique (pas d'échec dur).",
    "document_ref présent avec src ; si l'image casse, les zones restent cliquables (faits textuels).",
    "observe_order référence des ids présents dans observable_zones."
  ],
  "blocked_conditions": [
    "observable_zones absent ou vide",
    "modes absent ou sans mode 'obs'",
    "document_ref absent ou sans src",
    "une zone.fact manquante ou vide"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "fallback_text"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, 0 erreur" },
    { "case": "champ requis manquant", "expected": "BLOCKED listant le champ" },
    { "case": "clic en mode interpréter", "expected": "affiche interpret_block (panel warn), ne marque pas la zone observée" },
    { "case": "toutes les zones observées", "expected": "affiche all_observed + état 'ready' (prêt pour 304)" },
    { "case": "instance externe injectée", "expected": "le rendu change sans modifier le HTML" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-303",
    "note": "Ce schema VALIDE l'instance. Le contrat pédagogique complet (input_contract/validation_logic/feedback_scoring_logic) vit dans le CORE-GAB officiel (moteur annotation_media, parent GAB-389), pas ici (évite la duplication)."
  }
}
