{
  "contract_version": "gab_320_contract_v1",
  "gab_id": "GAB-320",
  "canonical_name": "RevisionLearningEmptyState",
  "module_owner": "EdTechRevisionLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "empty_state_id",
    "actions"
  ],
  "optional_fields": [
    "title",
    "subtitle",
    "cause_diagnostic",
    "prompt_label",
    "fallback_text_summary"
  ],
  "field_types": {
    "empty_state_id": "string — identifiant unique de l'occurrence d'état vide",
    "title": "string — titre affiché dans l'écran vide",
    "subtitle": "string — texte explicatif sous le titre",
    "cause_diagnostic": "string — description de la cause de l'état vide (aucun item, niveau non couvert, etc.)",
    "prompt_label": "string — libellé de l'invite au-dessus des chips d'action",
    "actions": "array<{id:string, label:string, feedback:string, is_report?:boolean}> — liste des actions utiles proposées",
    "fallback_text_summary": "string — résumé textuel de l'état (accessibilité, fallback)"
  },
  "constraints": [
    "actions doit contenir au moins 1 élément — sans action, l'écran vide est interdit (do_not_use: laisser un écran vide sans action).",
    "is_report:true déclenche un style feedback distinct (warn vs ok) — signale l'admin sans paniquer l'élève.",
    "Ne pas générer de cartes sans prudence ni validation (is_report ou feedback explicite requis si génération).",
    "Si des items de révision existent, utiliser GAB-312 — ce GAB est réservé à l'état vide avéré."
  ],
  "blocked_conditions": [
    "actions vides",
    "empty_state_id absent",
    "gab_id absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "aria-label sur chaque bouton chip",
    "fallback_text_summary disponible"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, chips actionnables, 0 erreur" },
    { "case": "champ requis manquant (actions vides)", "expected": "BLOCKED listant le champ" },
    { "case": "empty_state_id absent", "expected": "BLOCKED listant le champ" },
    { "case": "clic 'Activité libre'", "expected": "chip sélectionnée, panel ok avec feedback correspondant" },
    { "case": "clic 'Signaler à l'admin' (is_report:true)", "expected": "chip report-sel, panel note avec feedback admin" },
    { "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, chips empilées sur mobile" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-320",
    "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)."
  }
}
