{
  "contract_version": "gab_173_contract_v1",
  "gab_id": "GAB-173",
  "canonical_name": "AudioVocabularyListen",
  "module_owner": "EdTechAudioLearning",
  "renderer_key": "media_viewer",
  "required_fields": [
    "gab_id",
    "vocabulary_listen_id",
    "title",
    "words"
  ],
  "optional_fields": [
    "subject",
    "chapter",
    "accessibility",
    "child_safety"
  ],
  "field_types": {
    "words": "array<{word:string, phonetic:string, definition:string, example:string, audio_rate?:number}> — minimum 1 entrée",
    "audio_rate": "number(0.5..2.0) — taux de synthèse vocale, défaut 0.85",
    "example": "string HTML — balises <b> autorisées pour mettre le mot en évidence"
  },
  "constraints": [
    "words doit contenir au moins 1 entrée.",
    "word et definition sont obligatoires dans chaque entrée de words.",
    "phonetic est recommandé mais non bloquant si absent.",
    "example peut contenir des balises <b> pour mettre en évidence le mot.",
    "audio_rate dans [0.5, 2.0] si fourni ; défaut 0.85.",
    "La navigation cyclique est gérée côté renderer (index % length)."
  ],
  "blocked_conditions": [
    "words absent ou tableau vide",
    "gab_id absent",
    "vocabulary_listen_id absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "tts_web_speech_api_with_fallback"
  ],
  "qa_cases": [
    { "case": "instance conforme 3 mots", "expected": "rendu complet, navigation 1/3 → 2/3 → 3/3, 0 erreur" },
    { "case": "words absent ou vide", "expected": "BLOCKED listant le champ" },
    { "case": "gab_id absent", "expected": "BLOCKED listant le champ" },
    { "case": "vocabulary_listen_id absent", "expected": "BLOCKED listant le champ" },
    { "case": "instance externe injectée via init(ext)", "expected": "le rendu change sans modifier le HTML" },
    { "case": "1 seul mot — navigation Précédent/Suivant", "expected": "reste sur le même mot (cyclique), counter 1/1" },
    { "case": "bouton speaker", "expected": "SpeechSynthesisUtterance déclenché ou fallback animation 1200ms" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-173",
    "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 extraite de INDEX-300-audiolearning-GAB-171-175-PLAYABLE.html, stage data-tpl='173', handlers vlRender/vlPlay/vlPrev/vlNext."
  }
}
