{
  "contract_version": "gab_295_contract_v1",
  "gab_id": "GAB-295",
  "canonical_name": "OralLearningSelfReview",
  "module_owner": "EdTechOralLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "self_review_id",
    "oral_attempt_ref",
    "review_prompts",
    "review_goal",
    "next_action"
  ],
  "optional_fields": [
    "title",
    "subtitle",
    "non_judgment_rule",
    "rubric_ref",
    "reflection_privacy",
    "primary_cta",
    "secondary_cta",
    "save_policy",
    "why_feeling_first",
    "confidence_scale",
    "recording_ref",
    "source_metadata"
  ],
  "field_types": {
    "review_prompts": "array<{prompt_id:string, question:string, type:enum['scale','text'], scale_options?:array<{emoji:string,label:string}>, placeholder?:string}>",
    "review_goal": "string — identifiant de l'objectif de la réflexion",
    "next_action": "string — suggestion d'étape suivante basée sur le ressenti",
    "primary_cta": "object{label:string, action:string}",
    "secondary_cta": "object{label:string, action:string}",
    "save_policy": "enum['local_only_no_share_default','share_on_toggle','auto_share_prof'] — défaut : local_only_no_share_default"
  },
  "constraints": [
    "review_prompts doit contenir au moins 1 entrée de type 'scale' ou 'text'.",
    "primary_cta.label vient du JSON — jamais hardcodé dans le HTML.",
    "next_action est un texte de suggestion, jamais une injonction.",
    "reflection_privacy : si absent, afficher un message de confidentialité par défaut.",
    "Non-jugement absolu : aucune réponse n'est qualifiée de 'bonne' ou 'mauvaise' dans le rendu."
  ],
  "blocked_conditions": [
    "review_prompts absent ou tableau vide",
    "self_review_id absent",
    "oral_attempt_ref absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "scale_buttons_aria_labelled",
    "text_input_with_placeholder"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, 0 erreur, tous les prompts affichés" },
    { "case": "review_prompts vide ou absent", "expected": "BLOCKED affiché, rendu stoppé" },
    { "case": "self_review_id absent", "expected": "BLOCKED listant le champ manquant" },
    { "case": "oral_attempt_ref absent", "expected": "BLOCKED listant le champ manquant" },
    { "case": "instance externe injectée via init(ext)", "expected": "le rendu change sans modifier le HTML" },
    { "case": "scale à 4 options (plan)", "expected": "4 boutons rendus, 5 boutons pour stress" },
    { "case": "prompt type text", "expected": "input text rendu avec placeholder" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-295",
    "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)."
  }
}
