{
  "contract_version": "gab_033_contract_v1",
  "gab_id": "GAB-033",
  "canonical_name": "ProgressSummaryBlock",
  "module_owner": "EdTechPlayKit",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "progress_summary_id",
    "title",
    "progress",
    "steps"
  ],
  "optional_fields": [
    "current_index",
    "total_steps",
    "replay_label"
  ],
  "field_types": {
    "progress": "number(0..100) — pourcentage complété",
    "steps": "array<{label:string, status:enum['done','current','pending']}> — liste ordonnée des étapes",
    "current_index": "number — index (0-based) de l'étape en cours",
    "total_steps": "number — nombre total d'étapes",
    "replay_label": "string — libellé du bouton rejouer (ex : '↻ Rejouer l'animation')"
  },
  "constraints": [
    "progress doit être un nombre entre 0 et 100.",
    "steps doit contenir au moins 1 élément.",
    "Chaque step doit avoir un label (string non vide) et un status parmi ['done','current','pending'].",
    "Le renderer anime l'anneau SVG de 0 à progress — ne pas pré-animer côté instance.",
    "replay_label : libellé textuel du bouton replay, venant du JSON."
  ],
  "blocked_conditions": [
    "title absent",
    "progress absent ou non numérique",
    "steps vides"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "aria_live_on_percentage",
    "role_list_on_steps"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "anneau animé jusqu'à progress%, étapes rendues avec statuts corrects, 0 erreur" },
    { "case": "champ requis manquant", "expected": "BLOCKED listant le champ manquant" },
    { "case": "steps vides ([])", "expected": "BLOCKED 'steps vides'" },
    { "case": "progress = 0", "expected": "anneau à 0%, étapes affichées" },
    { "case": "progress = 100", "expected": "anneau plein, toutes étapes 'done'" },
    { "case": "instance externe injectée via ENGINE.init(ext)", "expected": "rendu change sans modifier le HTML" },
    { "case": "bouton replay cliqué", "expected": "animation recommence depuis 0%" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal, flex-direction:column sur mobile" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-033",
    "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)."
  }
}
