{
  "contract_version": "gab_257_contract_v1",
  "gab_id": "GAB-257",
  "canonical_name": "CollaborationLearningExplainToPeer",
  "module_owner": "EdTechCollaborationLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "explain_to_peer_id",
    "title",
    "summary",
    "roles",
    "feynman_steps",
    "primary_cta"
  ],
  "optional_fields": [
    "explain_target",
    "explain_target_enum",
    "estimated_duration",
    "mastery_label",
    "subject_label",
    "reason_title",
    "reason_body",
    "secondary_cta",
    "target_variants",
    "accessibility",
    "child_safety"
  ],
  "field_types": {
    "explain_to_peer_id": "string — identifiant unique de l'instance",
    "title": "string — titre de la notion à expliquer",
    "summary": "string — accroche méthode Feynman",
    "explain_target": "enum['concept','method','definition','chronology'] — type de notion",
    "explain_target_enum": "array<string> — valeurs proposées dans le sélecteur",
    "estimated_duration": "string — ex: '8 min'",
    "mastery_label": "string — ex: 'maîtrise active'",
    "subject_label": "string — ex: 'maths · 3e'",
    "roles": "array<{key:string, label:string, instruction:string}> — exactement 2 rôles attendus (expliqueur + ecouteur)",
    "feynman_steps": "array<{num:integer, verb:string, detail:string}> — 4 étapes Feynman structurelles",
    "reason_title": "string — titre bloc pédagogique 'Pourquoi'",
    "reason_body": "string — explication de l'effet Feynman (HTML léger autorisé: <b>)",
    "primary_cta": "object{label:string, action:string}",
    "secondary_cta": "object{label:string, action:string}",
    "target_variants": "object<enum_key, {title,summary,role_expliqueur,role_ecouteur,cta_label}> — swap dynamique par explain_target"
  },
  "constraints": [
    "roles doit contenir au moins 2 éléments (expliqueur + ecouteur).",
    "feynman_steps doit contenir au moins 1 étape ; 4 étapes est la valeur canonique.",
    "primary_cta.label est affiché sur le bouton structurel — libellé depuis JSON uniquement.",
    "target_variants clés doivent correspondre aux valeurs de explain_target_enum.",
    "explain_target si présent doit être une valeur de explain_target_enum."
  ],
  "blocked_conditions": [
    "title absent",
    "summary absent",
    "roles vides",
    "feynman_steps vides",
    "primary_cta absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "role_buttons_labelled"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, 0 erreur, sélecteur enum actif" },
    { "case": "champ requis manquant (ex: title)", "expected": "BLOCKED listant le champ" },
    { "case": "clic enum chip 'method'", "expected": "titre/summary/roles/cta swappés sans modifier le HTML" },
    { "case": "instance externe injectée via ENGINE.init(ext)", "expected": "rendu change sans modifier le HTML" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal, roles-grid passe en 1 col" },
    { "case": "secondary_cta absent", "expected": "bouton secondaire masqué (hidden)" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-257",
    "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)."
  }
}
