{
  "contract_version": "gab_383_contract_v1",
  "gab_id": "GAB-383",
  "canonical_name": "MediaLearningVideoPlayer",
  "module_owner": "EdTechMediaLearning",
  "renderer_key": "media_viewer",
  "required_fields": [
    "gab_id",
    "video_player_id",
    "title",
    "video_ref",
    "no_autoplay",
    "transcript",
    "chapters",
    "controls"
  ],
  "optional_fields": [
    "source_rights",
    "fallback",
    "_note_dev"
  ],
  "field_types": {
    "video_ref": "object{src,kind,poster?,alt_summary?}",
    "no_autoplay": "boolean — doit être true ; le lecteur ne se déclenche jamais automatiquement",
    "transcript": "object{label,text,accessible_note?}",
    "chapters": "array<{label,target_time,description}>  — tableau ordonné de repères temporels",
    "controls": "object{play_label,pause_label,replay_label,transcript_label}"
  },
  "constraints": [
    "no_autoplay DOIT être true ; autoplay interdit (accessibilité + sécurité mineur).",
    "transcript requis : la vidéo doit toujours avoir un fallback texte accessible.",
    "chapters : chaque entrée possède un label court, un target_time (format m:ss) et une description.",
    "controls : les 4 libellés sont requis pour alimenter les boutons sans hardcoder."
  ],
  "blocked_conditions": [
    "video_ref absent",
    "transcript absent",
    "no_autoplay manquant ou false"
  ],
  "accessibility": [
    "no_autoplay",
    "transcript_text_fallback",
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, barre de progression, chapitres cliquables, 0 erreur" },
    { "case": "champ requis manquant (video_ref absent)", "expected": "BLOCKED listant le champ" },
    { "case": "transcript absent", "expected": "BLOCKED — accessibilité non garantie" },
    { "case": "no_autoplay manquant ou false", "expected": "BLOCKED — autoplay interdit" },
    { "case": "clic chapitre", "expected": "chip sélectionnée, panneau indique le target_time et la description" },
    { "case": "bouton Transcript", "expected": "panneau affiche transcript.text + accessible_note" },
    { "case": "instance externe injectée via init(ext)", "expected": "rendu change sans modifier le HTML" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-383",
    "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)."
  }
}
