{
  "contract_version": "gab_222_contract_v1",
  "gab_id": "GAB-222",
  "canonical_name": "SmartSelectLevelFit",
  "module_owner": "EdTechSmartSelect",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "level_fit_id",
    "selected_level",
    "levels",
    "reason"
  ],
  "optional_fields": [
    "title",
    "summary",
    "safety_notice",
    "secondary_cta_label",
    "fallback_text_summary"
  ],
  "field_types": {
    "selected_level": "enum['discovery','guided','standard','exam','challenge']",
    "levels": "array<{key:enum,icon:string,label:string,description:string,explanation:string,cta_label:string}> — exactement 5 paliers dans l'ordre discovery→guided→standard→exam→challenge",
    "reason": "string — justification du niveau sélectionné, en langage élève",
    "safety_notice": "string — garde-fou affiché (ne jamais modifier session notée en cours)",
    "secondary_cta_label": "string — libellé du bouton secondaire"
  },
  "constraints": [
    "selected_level doit correspondre à un key présent dans levels[].",
    "levels[] doit contenir exactement 5 entrées avec les keys discovery, guided, standard, exam, challenge.",
    "reason : jamais de raw data technique (scores bruts, logs) — langage élève uniquement.",
    "SmartSelect choisit le niveau de DÉPART (pré-activité) — ne pas confondre avec GAB-214 AdaptiveDifficultyAdjuster (adapte pendant/après activité).",
    "Ne s'affiche pas si évaluation officielle verrouillée en cours."
  ],
  "blocked_conditions": [
    "selected_level absent",
    "levels vides",
    "reason absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "aria_pressed_sur_paliers",
    "fallback_text"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, palier actif mis en évidence, explication affichée, 0 erreur" },
    { "case": "selected_level absent", "expected": "BLOCKED listant le champ manquant" },
    { "case": "levels vide", "expected": "BLOCKED listant levels vides" },
    { "case": "reason absent", "expected": "BLOCKED listant reason absent" },
    { "case": "clic sur palier différent", "expected": "chip + explication + CTA se mettent à jour sans reload" },
    { "case": "instance externe injectée via init(ext)", "expected": "le rendu change sans modifier le HTML" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-222",
    "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)."
  }
}
