{
  "contract_version": "gab_004_contract_v1",
  "gab_id": "GAB-004",
  "canonical_name": "SessionProgressHeader",
  "module_owner": "EdTechPlayEngine",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "session_progress_id",
    "title",
    "current_index",
    "total_steps",
    "progress_percent"
  ],
  "optional_fields": [
    "icon",
    "step_label_template",
    "cta_label"
  ],
  "field_types": {
    "title": "string — libellé de la session affiché dans le header",
    "icon": "string — emoji ou identifiant d'icône affiché dans le badge gauche",
    "current_index": "integer >= 1 — numéro de l'étape courante",
    "total_steps": "integer >= 2 — nombre total d'étapes de la session",
    "progress_percent": "number(0..100) — pourcentage de progression pour la barre",
    "step_label_template": "string — gabarit du libellé étape, variables {current} et {total}",
    "cta_label": "string — libellé du bouton CTA si présent"
  },
  "constraints": [
    "current_index doit être dans [1, total_steps].",
    "progress_percent dans [0, 100].",
    "total_steps >= 2 (header inutile pour session à une seule étape).",
    "Les dots de navigation sont générés dynamiquement à partir de total_steps et current_index — jamais listés en dur dans l'instance."
  ],
  "blocked_conditions": [
    "title absent",
    "current_index absent",
    "total_steps absent",
    "progress_percent absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "aria_labels_sur_dots",
    "fallback_text"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, barre à progress_percent%, dot current_index en violet" },
    { "case": "champ requis manquant", "expected": "BLOCKED listant le champ absent" },
    { "case": "current_index = 1", "expected": "aucun dot done, premier dot current, barre à progress_percent%" },
    { "case": "current_index = total_steps", "expected": "tous dots précédents done, dernier dot current" },
    { "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, dots wrappent proprement" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-004",
    "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)."
  }
}
