{
  "contract_version": "gab_156_contract_v1",
  "gab_id": "GAB-156",
  "canonical_name": "GameLevelSelect",
  "module_owner": "EdTechGameLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "level_select_id",
    "title",
    "levels",
    "primary_cta"
  ],
  "optional_fields": [
    "_note_dev",
    "renderer_key"
  ],
  "field_types": {
    "gab_id": "string — doit valoir 'GAB-156'",
    "level_select_id": "string — identifiant unique de l'instance",
    "title": "string — titre affiché en hero (ex : 'Choisis ton niveau')",
    "levels": "array<{id:string, stars:string, label:string, description:string}> — 2 à 4 éléments",
    "primary_cta": "object{label_idle:string, label_selected_prefix:string} — libellés du bouton",
    "renderer_key": "string — toujours 'text_cta'"
  },
  "constraints": [
    "levels doit contenir entre 2 et 4 éléments.",
    "primary_cta.label_idle : affiché quand aucun niveau n'est sélectionné (bouton désactivé).",
    "primary_cta.label_selected_prefix : concaténé avec level.label au clic pour former le libellé final.",
    "L'ordre des levels dans le tableau détermine l'ordre d'affichage dans la grille.",
    "Les classes CSS de couleur (easy/medium/hard) sont assignées par position (index 0/1/2) ; au-delà de l'index 2 aucune classe de couleur supplémentaire n'est appliquée."
  ],
  "blocked_conditions": [
    "title absent",
    "levels vides ou absents",
    "primary_cta.label_idle absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "aria_label_per_level"
  ],
  "qa_cases": [
    { "case": "instance conforme (3 niveaux)", "expected": "grille 3 colonnes, CTA désactivé au chargement, activé après sélection" },
    { "case": "title absent", "expected": "BLOCKED — message d'erreur affiché, grille non rendue" },
    { "case": "levels vide ([])", "expected": "BLOCKED — message d'erreur affiché" },
    { "case": "primary_cta.label_idle absent", "expected": "BLOCKED — message d'erreur affiché" },
    { "case": "instance externe injectée via ENGINE.init(ext)", "expected": "rendu change sans modifier le HTML" },
    { "case": "2 niveaux seulement", "expected": "grille 2 colonnes, aucune erreur" },
    { "case": "responsive 375/768/1024", "expected": "grille en colonne unique sur mobile, aucun débordement" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-156",
    "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)."
  }
}
