{
  "contract_version": "gab_167_contract_v1",
  "gab_id": "GAB-167",
  "canonical_name": "AudioListenAndRepeat",
  "module_owner": "EdTechAudioLearning",
  "renderer_key": "media_viewer",
  "required_fields": [
    "gab_id",
    "listen_repeat_id",
    "title",
    "instruction",
    "phrase"
  ],
  "optional_fields": [
    "tts_lang",
    "tts_rate",
    "btn_listen_label",
    "btn_record_label",
    "status_initial",
    "status_listening",
    "status_after_listen",
    "status_recording",
    "status_done",
    "record_auto_stop_ms",
    "rgpd_notice",
    "accessibility",
    "child_safety"
  ],
  "field_types": {
    "listen_repeat_id": "string — identifiant unique de l'exercice",
    "title": "string — titre affiché dans le hero",
    "instruction": "string — texte d'instruction affiché en label",
    "phrase": "string — phrase courte à écouter puis répéter",
    "tts_lang": "string — code BCP47 (défaut 'fr-FR')",
    "tts_rate": "number(0.1..2.0) — vitesse TTS (défaut 0.9)",
    "btn_listen_label": "string — libellé du bouton Écouter",
    "btn_record_label": "string — libellé du bouton Enregistrer",
    "status_initial": "string — texte de statut initial",
    "status_listening": "string — texte affiché pendant l'écoute TTS",
    "status_after_listen": "string — texte affiché après écoute (invitation à répéter)",
    "status_recording": "string — texte affiché pendant l'enregistrement local",
    "status_done": "string — texte affiché après enregistrement terminé",
    "record_auto_stop_ms": "integer — durée max enregistrement en ms (défaut 4000)",
    "rgpd_notice": "string — mention RGPD affichée dans la bannière",
    "accessibility": "object{phrase_role,phrase_aria_label,btn_listen_aria,btn_record_aria,status_aria_live}",
    "child_safety": "object{recording_local_only:boolean,no_server_transmission:boolean,rgpd_banner_visible:boolean}"
  },
  "constraints": [
    "phrase : texte court (recommandé < 120 caractères) pour faciliter la répétition orale.",
    "tts_rate dans [0.1, 2.0] si présent ; défaut 0.9 pour lisibilité enfant.",
    "record_auto_stop_ms : plafond recommandé 8000ms ; valeur < 1000 ignorée (défaut 4000).",
    "child_safety.recording_local_only DOIT être true — l'enregistrement ne peut pas être transmis.",
    "rgpd_notice DOIT être visible si child_safety.rgpd_banner_visible est true."
  ],
  "blocked_conditions": [
    "phrase absent",
    "instruction absent",
    "title absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "aria_live_status",
    "region_role_on_phrase"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, hero + phrase + 2 boutons + bannière RGPD" },
    { "case": "champ requis manquant (phrase)", "expected": "BLOCKED listant le champ manquant" },
    { "case": "champ requis manquant (instruction)", "expected": "BLOCKED listant le champ manquant" },
    { "case": "instance externe injectée via ENGINE.init(ext)", "expected": "le rendu change sans modifier le HTML" },
    { "case": "clic Écouter → TTS disponible", "expected": "bouton actif (violet), statut 'Écoute attentivement…'" },
    { "case": "clic Enregistrer → auto-stop 4s", "expected": "bouton actif (coral), statut done après record_auto_stop_ms" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal, boutons pleine largeur sur mobile" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-167",
    "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)."
  }
}
