{
  "contract_version": "gab_307_contract_v1",
  "gab_id": "GAB-307",
  "canonical_name": "DocumentLearningMapAnalysis",
  "module_owner": "EdTechDocumentLearning",
  "renderer_key": "annotation_media",
  "required_fields": [
    "gab_id",
    "map_analysis_id",
    "document_ref",
    "map_title",
    "scale",
    "legend"
  ],
  "optional_fields": [
    "title",
    "map_kind_label",
    "orientation",
    "instruction",
    "fallback_text_summary",
    "accessibility",
    "child_safety"
  ],
  "field_types": {
    "document_ref": "object{src,kind,alt}",
    "map_title": "string — titre complet affiché sur la carte (échelle et orientation inclus si connus)",
    "scale": "string — ex: '1:500 000'",
    "orientation": "string — ex: 'Nord (↑)', optionnel si non précisé sur la carte",
    "legend": "array<{id:string, label:string, color:string, feedback:string, is_trap:boolean}>"
  },
  "constraints": [
    "legend doit contenir au moins 1 item.",
    "is_trap:true déclenche un feedback d'avertissement (⚠) — le piège doit être signalé, jamais masqué.",
    "document_ref.src peut contenir '_TODO:...' si l'asset n'est pas encore disponible ; le renderer affiche un placeholder.",
    "Pas d'inférence de zones non présentes dans la légende source — anti-invention.",
    "scale et orientation sont affichées telles quelles, jamais devinées."
  ],
  "blocked_conditions": [
    "legend vide ou absente",
    "document_ref absent",
    "map_title absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "fallback_text"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, légende cliquable, 0 erreur" },
    { "case": "champ requis manquant (legend vide)", "expected": "BLOCKED listant le champ" },
    { "case": "item is_trap:true cliqué", "expected": "feedback ⚠ affiché, item coloré en avertissement" },
    { "case": "tous items légendés trouvés", "expected": "état 'ready' — tous validés" },
    { "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-307",
    "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)."
  }
}
