{
  "contract_version": "gab_003_contract_v1",
  "gab_id": "GAB-003",
  "canonical_name": "SessionStepPlayerShell",
  "module_owner": "EdTechPlayEngine",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "session_step_id",
    "session_title",
    "total_steps",
    "steps"
  ],
  "optional_fields": [
    "current_step_index",
    "back_cta",
    "next_cta"
  ],
  "field_types": {
    "session_title": "string — titre de la session affiché dans la topbar",
    "current_step_index": "number(0-based) — index de départ (défaut 0 si absent)",
    "total_steps": "number — nombre total d'étapes de la session",
    "steps": "array<{badge:string, content:string, mod:string}> — séquence des étapes jouables",
    "back_cta": "string — libellé bouton Précédent (défaut '← Précédent')",
    "next_cta": "string — libellé bouton Suivant (défaut 'Suivant →')"
  },
  "constraints": [
    "steps doit contenir au moins 1 élément, sinon BLOCKED.",
    "total_steps doit être un entier >= 1, sinon BLOCKED.",
    "current_step_index doit être dans [0, steps.length-1] ; clamp silencieux à 0 si hors borne.",
    "Le shell est un conteneur : il ne contient aucun contenu pédagogique en dur.",
    "Jamais utiliser ce gabarit hors session (→ bloc autonome) ou pour l'écran de fin (→ GAB-013)."
  ],
  "blocked_conditions": [
    "session_title absent",
    "steps vides ou absents",
    "total_steps invalide ou absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "aria-label sur bouton retour topbar"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, navigation Précédent/Suivant fonctionnelle, 0 erreur" },
    { "case": "champ requis manquant (ex: session_title)", "expected": "BLOCKED listant le champ manquant" },
    { "case": "steps vide []", "expected": "BLOCKED 'steps vides'" },
    { "case": "current_step_index hors borne", "expected": "clamp silencieux à 0, rendu de la 1ère étape" },
    { "case": "instance externe injectée via ENGINE.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-003",
    "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)."
  }
}
