{
  "contract_version": "gab_111_contract_v1",
  "gab_id": "GAB-111",
  "canonical_name": "MemoryDailyGoal",
  "module_owner": "EdTechMemoryLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "goal_id",
    "title",
    "subtitle",
    "progress_done",
    "progress_total",
    "primary_cta"
  ],
  "optional_fields": [
    "progress_unit",
    "duration_label",
    "streak_label",
    "accessibility"
  ],
  "field_types": {
    "goal_id": "string — identifiant unique de l'objectif",
    "title": "string — titre principal affiché sous l'anneau",
    "subtitle": "string — description courte de l'objectif (ex : '8 cartes : dates et vocabulaire')",
    "progress_done": "integer >= 0 — cartes déjà révisées",
    "progress_total": "integer >= 1 — total cartes de l'objectif du jour",
    "progress_unit": "string — libellé de l'unité (ex : 'cartes')",
    "duration_label": "string — durée estimée (ex : '~4 min')",
    "streak_label": "string — libellé de la série quotidienne (ex : 'série de 3 jours')",
    "primary_cta": "object{label,action}",
    "accessibility": "object{ring_aria_label}"
  },
  "constraints": [
    "progress_done doit être dans [0, progress_total].",
    "progress_total doit être >= 1.",
    "primary_cta.label : libellé du bouton venant du JSON ; le moteur calcule 'restantes' dynamiquement.",
    "L'arc SVG est calculé côté moteur à partir de progress_done / progress_total (stroke-dashoffset).",
    "Quand progress_done == progress_total, le bouton passe en état 'atteint' (style mint, désactivé)."
  ],
  "blocked_conditions": [
    "title absent",
    "progress_total absent ou < 1",
    "primary_cta absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "ring_aria_label depuis instance"
  ],
  "qa_cases": [
    { "case": "instance conforme (progress_done=5, total=8)", "expected": "anneau à 62.5 %, bouton 'Continuer (3 restantes)', 0 erreur" },
    { "case": "progress_done == progress_total", "expected": "anneau plein, bouton 'Objectif du jour atteint !' état mint désactivé" },
    { "case": "champ requis manquant (title absent)", "expected": "BLOCKED listant le champ manquant" },
    { "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, anneau centré" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-111",
    "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). Source HTML : INDEX-300-memorylearning-GAB-111-115-PLAYABLE.html, stage data-tpl='111', handler dgContinue()."
  }
}
