{
  "contract_version": "gab_323_contract_v1",
  "gab_id": "GAB-323",
  "canonical_name": "ResearchLearningSearchQueryBuilder",
  "module_owner": "EdTechResearchLearning",
  "renderer_key": "text_cta",
  "required_fields": [
    "gab_id",
    "query_builder_id",
    "keywords",
    "min_terms",
    "feedback_too_vague",
    "feedback_ok_template",
    "primary_cta"
  ],
  "optional_fields": [
    "title",
    "instruction"
  ],
  "field_types": {
    "keywords": "array<{id:string, label:string, type:enum['keyword','filter','operator']}>",
    "min_terms": "integer — nombre minimum de termes requis pour valider la requête",
    "feedback_too_vague": "string — message affiché si requête trop vague (BLOCKED)",
    "feedback_ok_template": "string — gabarit avec {count} remplacé dynamiquement",
    "primary_cta": "object{label:string, action:string}"
  },
  "constraints": [
    "keywords doit contenir au moins 1 élément.",
    "min_terms >= 1 ; une requête avec strictement moins que min_terms termes sélectionnés déclenche le feedback_too_vague (BLOCKED).",
    "feedback_ok_template doit contenir le placeholder {count}.",
    "primary_cta.label vient du JSON, jamais en dur dans le HTML."
  ],
  "blocked_conditions": [
    "keywords vide ou absent",
    "min_terms absent",
    "feedback_too_vague absent",
    "feedback_ok_template absent",
    "primary_cta absent"
  ],
  "accessibility": [
    "keyboard_navigable",
    "focus_visible",
    "prefers_reduced_motion",
    "aria_pressed_on_chips"
  ],
  "qa_cases": [
    { "case": "instance conforme", "expected": "rendu complet, 0 erreur" },
    { "case": "champ requis manquant", "expected": "BLOCKED listant le champ" },
    { "case": "0 chip sélectionné + clic tester", "expected": "panel warn avec feedback_too_vague" },
    { "case": "1 chip sélectionné + clic tester (min_terms=2)", "expected": "panel warn avec feedback_too_vague (BLOCKED)" },
    { "case": "2+ chips sélectionnés + clic tester", "expected": "panel ok avec feedback_ok_template interpolé" },
    { "case": "instance externe injectée", "expected": "le rendu change sans modifier le HTML" },
    { "case": "responsive 375/768/1024", "expected": "aucun débordement horizontal" }
  ],
  "traceability": {
    "derived_from_core_gab": "GAB-323",
    "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)."
  }
}
