{
  "contract_version": "gab_180_contract_v1",
  "gab_id": "GAB-180",
  "canonical_name": "LevelTestRecommendationBridge",
  "module_owner": "EdTechLevelTest",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "bridge_id",
    "title",
    "summary",
    "items",
    "primary_cta"
  ],
  "optional_fields": [
    "banner_text",
    "_note_dev"
  ],
  "field_types": {
    "bridge_id": "string — identifiant unique de ce pont de recommandations",
    "title": "string — titre du plan personnalisé affiché en en-tête de carte",
    "summary": "string — sous-titre descriptif (ex: nombre de recommandations, base profil)",
    "banner_text": "string — message contextuel transversal affiché en bannière (optionnel)",
    "items": "array<{priority:integer(1..n), priority_label:string, type:enum['fiche','exo','mem','story'], icon:string, title:string, detail:string, confirm_feedback:string}> — ordonné par priorité croissante",
    "primary_cta": "object{label:string, action:string, confirm_feedback:string} — déclenche la priorité 1"
  },
  "constraints": [
    "items doit contenir au moins 1 entrée.",
    "items[].priority est un entier >= 1, ordonné croissant.",
    "items[].type est un enum strict : 'fiche' | 'exo' | 'mem' | 'story'.",
    "primary_cta.label et primary_cta.action sont obligatoires dans l'objet primary_cta.",
    "Le contenu pédagogique (titres, détails) vient exclusivement du JSON — zéro texte hardcodé dans le HTML."
  ],
  "blocked_conditions": [
    "items absent ou tableau vide (BLOCKED)",
    "primary_cta absent (BLOCKED)",
    "bridge_id absent (BLOCKED)"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "fallback_text"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet avec 3 items et bouton CTA, 0 erreur" },
    { "case": "items vide []", "expected": "BLOCKED listant 'items absent ou tableau vide'" },
    { "case": "primary_cta absent", "expected": "BLOCKED listant 'primary_cta absent'" },
    { "case": "bridge_id absent", "expected": "BLOCKED listant 'bridge_id absent'" },
    { "case": "instance externe injectée via init(ext)", "expected": "rendu change sans modifier le HTML" },
    { "case": "clic sur un item", "expected": "confirm_feedback de l'item affiché" },
    { "case": "clic sur primary_cta", "expected": "primary_cta.confirm_feedback affiché" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-180",
    "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)."
  },
  "external_refs": {
    "dependencies": [
      "GAB-179 (LevelTestProfileResult) — ce pont est activé après affichage du profil",
      "GAB-131 (remédiation post-erreur — ne pas confondre)",
      "GAB-160 (pont récompense PlayKit — ne pas confondre)",
      "GAB-163 (débrief mini-jeu — ne pas confondre)"
    ]
  }
}
