{
  "contract_version": "gab_103_contract_v1",
  "gab_id": "GAB-103",
  "canonical_name": "MemorySpacedReviewPrompt",
  "module_owner": "EdTechMemoryLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "review_prompt_id",
    "heading",
    "card_count",
    "primary_cta"
  ],
  "optional_fields": [
    "title",
    "spr_label",
    "card_count_label",
    "topic",
    "schedule",
    "_note_dev"
  ],
  "field_types": {
    "heading": "string — titre principal de l'invitation (ex : 'C'est le moment de réviser !')",
    "card_count": "number — nombre de cartes dues à revoir",
    "card_count_label": "string — libellé de l'unité (ex : 'cartes à revoir aujourd'hui')",
    "topic": "string — description du jeu de cartes concerné",
    "primary_cta": "object{label:string, action:string, icon?:string}",
    "schedule": "array<{day_label:string, status:enum['done','now','future']}> — timeline de la répétition espacée"
  },
  "constraints": [
    "card_count doit être un nombre entier >= 1 ; si 0 carte à revoir, ne pas afficher ce GAB (do_not_use_when).",
    "primary_cta.label vient exclusivement du JSON — jamais codé en dur dans le HTML.",
    "schedule : exactement 1 item avec status='now' représente l'étape courante.",
    "spr_label est le sous-titre badge (ex : 'Répétition espacée') — libellé court."
  ],
  "blocked_conditions": [
    "heading absent",
    "primary_cta.label absent",
    "card_count absent ou non numérique"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "button disabled state after click"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, cloche animée, timeline, bouton actif — 0 erreur" },
    { "case": "champ requis manquant (heading)", "expected": "BLOCKED listant le champ manquant" },
    { "case": "card_count non numérique", "expected": "BLOCKED card_count" },
    { "case": "instance externe injectée via ENGINE.init(ext)", "expected": "rendu change sans modifier le HTML" },
    { "case": "clic bouton CTA", "expected": "bouton disabled + icône '✓' + libellé 'Révision en cours — N cartes'" },
    { "case": "schedule absent ou vide", "expected": "timeline non rendue, reste des champs OK" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-103",
    "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)."
  }
}
