/* ============================================================
   MentivisOS - Design System (v0.23.0)
   Source de verite : design-lightmode.md + design-darkmode.md
   (livres par le graphiste). Tokens --ds-* dans design-tokens.css.
   Light-first, accent bleu, typo Geist Sans + Geist Mono.
   Anciens tokens (--m-*, --color-*, --text, --bg-elev, etc.)
   conserves comme alias remappes sur --ds-* pour ne rien casser.
   ROLLBACK : git revert <commit v0.23.0> + redeploy sc14.
   ============================================================ */

/* ---------- 0. Tokens DS + polices (embarquees localement) ---------- */
@import url('./design-tokens.css');
@import url('../fonts/fonts.css');

/* ---------- 1. Variables : alias retro-compatibles vers --ds-* ----------
   Les anciens tokens (--m-*, --color-*, --bg, --text, --shadow*, --r-*, --t-*)
   sont remappes sur les tokens du design system (design-tokens.css). Tout le
   CSS aval (3500+ lignes) continue de fonctionner sans changement, mais
   herite automatiquement des nouvelles couleurs (light/dark via prefers OS),
   typo Geist, ombres et motion. */
:root {
  /* === Accent : la palette --m-coral / --m-purple est remappee sur le bleu
     officiel du DS pour respecter le DS Mentivis (un seul accent par vue). */
  --m-coral:         var(--ds-blue-700);
  --m-coral-hover:   var(--ds-blue-600);
  --m-coral-soft:    var(--ds-blue-200);
  --m-coral-tint:    var(--ds-blue-100);
  --m-purple:        var(--ds-blue-700);
  --m-purple-mid:    var(--ds-blue-700);
  --m-purple-deep:   var(--ds-blue-800);
  --m-purple-soft:   var(--ds-blue-200);
  --m-purple-bright: var(--ds-blue-600);
  --m-purple-tint:   var(--ds-blue-100);

  /* === Encres (textes) = text hierarchy DS === */
  --m-ink:    var(--ds-text-primary);
  --m-ink-2:  var(--ds-text-secondary);
  --m-ink-3:  var(--ds-text-tertiary);
  --m-ink-4:  var(--ds-text-disabled);

  /* === Lignes / bordures === */
  --m-line:   var(--ds-border-default);
  --m-line-2: var(--ds-gray-200);

  /* === Fonds === */
  --m-bg:            var(--ds-bg-100);
  --m-bg-soft:       var(--ds-bg-200);
  --m-bg-warm:       var(--ds-bg-300);
  --m-bg-warm-trans: color-mix(in srgb, var(--ds-bg-300) 80%, transparent);
  --m-bg-near-white: var(--ds-bg-200);

  /* === Etats semantiques === */
  --m-success:      var(--ds-green-700);
  --m-success-soft: var(--ds-green-200);
  --m-success-text: var(--ds-green-900);
  --m-warning:      var(--ds-amber-700);
  --m-warning-soft: var(--ds-amber-200);
  --m-danger:       var(--ds-red-700);
  --m-danger-soft:  var(--ds-red-200);

  /* === Rayons === */
  --r-sm:     var(--ds-radius-sm);
  --r-md:     var(--ds-radius-md);
  --r-lg:     var(--ds-radius-lg);
  --r-xl:     20px;
  --r-warm:   var(--ds-radius-full);
  --r-pill:   var(--ds-radius-full);
  --r-card:   var(--ds-radius-lg);
  --r-module: 20px;

  /* === Ombres === */
  --shadow-inset:         0 0 0 .5px rgba(0,0,0,0.075) inset;
  --shadow-outline:       0 0 0 1px var(--ds-border-default);
  --shadow-soft:          var(--ds-shadow-sm);
  --shadow-card:          var(--ds-shadow-md);
  --shadow-warm:          var(--ds-shadow-md);
  --shadow-card-full:     var(--ds-shadow-md);
  --shadow-1:             var(--ds-shadow-sm);
  --shadow-2:             var(--ds-shadow-md);
  --shadow-header:        var(--ds-shadow-sm);
  --shadow-header-scroll: var(--ds-shadow-md);
  --shadow-sm:            var(--ds-shadow-sm);
  --shadow:               var(--ds-shadow-md);
  --shadow-lg:            var(--ds-shadow-lg);

  /* === Typographie (Geist Sans + Geist Mono) === */
  --f-display: var(--ds-font-sans);
  --f-sans:    var(--ds-font-sans);
  --f-mono:    var(--ds-font-mono);
  --font-sans: var(--ds-font-sans);
  --font-mono: var(--ds-font-mono);

  /* === Echelles typographiques (conservees, consommees par le CSS aval) === */
  --text-hero:    clamp(32px, 5vw, 56px);
  --text-display: clamp(36px, 4.5vw, 56px);
  --text-title:   clamp(24px, 3vw, 36px);
  --text-heading: clamp(18px, 2vw, 28px);
  --text-body-lg: 20px;
  --text-body:    18px;
  --text-body-sm: 16px;
  --text-nav:     15px;
  --text-button:  15px;
  --text-caption: 14px;
  --text-small:   13px;
  --text-micro:   12px;
  --text-tiny:    10px;

  /* === Layout === */
  --container:         var(--ds-layout-max);
  --container-wide:    1440px;
  --gutter:            var(--ds-layout-padding);
  --grid-margin:       var(--ds-layout-padding);
  --section-padding-y: clamp(96px, 12vw, 160px);
  --section-gap:       clamp(96px, 12vw, 160px);
  --section-gap-sm:    clamp(64px, 8vw, 96px);

  /* === Focus + easing === */
  --focus-ring:    var(--ds-blue-400);
  --ease-mentivis: var(--ds-easing-default);

  /* === Couleurs semantiques (alias) === */
  --color-primary:      var(--ds-blue-700);
  --color-primary-dark: var(--ds-blue-800);
  --color-primary-light: var(--ds-blue-200);
  --color-accent:       var(--ds-blue-700);
  --color-success:      var(--ds-success);
  --color-success-soft: var(--ds-green-200);
  --color-warning:      var(--ds-warning);
  --color-warning-soft: var(--ds-amber-200);
  --color-danger:       var(--ds-error);
  --color-danger-soft:  var(--ds-red-200);

  /* === Fonds / textes / bordures (alias top-level) === */
  --bg:          var(--ds-bg-100);
  --bg-elev:     var(--ds-bg-100);
  --bg-soft:     var(--ds-bg-200);
  --border:      var(--ds-border-default);
  --border-soft: var(--ds-gray-200);
  --text:        var(--ds-text-primary);
  --text-soft:   var(--ds-text-secondary);
  --text-muted:  var(--ds-text-tertiary);

  /* === Grades pedagogiques (semantique success / warning / danger DS) === */
  --grade-a-bg: var(--ds-green-200);  --grade-a-fg: var(--ds-green-900);
  --grade-b-bg: var(--ds-green-100);  --grade-b-fg: var(--ds-green-800);
  --grade-c-bg: var(--ds-amber-200);  --grade-c-fg: var(--ds-amber-900);
  --grade-d-bg: var(--ds-red-200);    --grade-d-fg: var(--ds-red-900);
  --grade-e-bg: var(--ds-red-300);    --grade-e-fg: var(--ds-red-1000);

  /* === Rayons alias secondaires === */
  --radius-sm: var(--ds-radius-sm);
  --radius:    var(--ds-radius-md);
  --radius-lg: var(--ds-radius-lg);
  --radius-xl: 20px;

  /* === Transitions === */
  --t-fast: var(--ds-duration-fast)   var(--ds-easing-default);
  --t:      var(--ds-duration-normal) var(--ds-easing-default);
  --t-slow: var(--ds-duration-slow)   var(--ds-easing-default);
}

/* Note : aucun bloc [data-theme="dark"] manuel ici. Les tokens --ds-* changent
   automatiquement en dark via @media (prefers-color-scheme: dark) defini
   dans design-tokens.css, et tous les alias ci-dessus repointent vers eux. */

/* Selection brand (alias DS) */
::selection { background: var(--ds-blue-700); color: var(--ds-text-on-accent); }

/* ---------- 2. Reset ---------- */
*, *::before, *::after { box-sizing: border-box; margin: 0; padding: 0; }
html { scroll-behavior: smooth; }
body {
  font-family: var(--f-sans);
  background: var(--bg);
  color: var(--text);
  min-height: 100dvh;
  line-height: 1.55;
  -webkit-font-smoothing: antialiased;
  text-rendering: optimizeLegibility;
  font-feature-settings: 'ss01', 'cv01';
}
h1, h2, h3, h4, h5, h6 {
  font-family: var(--f-display);
  letter-spacing: -0.01em;
  color: var(--m-ink);
  line-height: 1.18;
}
h1 { font-weight: 700; }
h2, h3 { font-weight: 700; }
h4, h5, h6 { font-weight: 600; }
.t-eyebrow {
  font-size: 12px; font-weight: 600; text-transform: uppercase;
  letter-spacing: 0.06em; color: var(--m-ink-3);
}
.t-mono { font-family: var(--f-mono); letter-spacing: 0.04em; }
.t-lead { font-size: 18px; color: var(--m-ink-3); line-height: 1.5; }
@media (max-width: 720px) { .t-lead { font-size: 16px; } }
img, svg { max-width: 100%; display: block; }
button { font: inherit; cursor: pointer; border: none; background: none; color: inherit; }
a { color: var(--color-primary); text-decoration: none; transition: color var(--t-fast); }
a:hover { color: var(--color-primary-dark); }
input, textarea, select { font: inherit; color: inherit; }

/* ---------- 3. Header (Mentivis pill flottant) ---------- */
.site-header {
  position: fixed;
  top: 12px;
  left: 50%;
  transform: translateX(-50%);
  width: calc(100% - 24px);
  /* v3.4.51 : aligne avec .module-layout elargi (sidebar + main + marge bulles symetrique) */
  max-width: 1632px;
  z-index: 50;
  background: var(--ds-header-bg);
  color: var(--ds-header-text);
  border-radius: var(--r-xl);
  box-shadow: var(--shadow-header);
  transition: background .35s ease, box-shadow .35s ease, backdrop-filter .35s ease;
  animation: headerDropIn 1.1s var(--ease-mentivis) both;
}
.site-header.is-scrolled {
  /* Fond translucide qui suit le mode via le token configurable du bandeau. */
  background: color-mix(in srgb, var(--ds-header-bg) 88%, transparent);
  box-shadow: var(--shadow-header-scroll);
  backdrop-filter: blur(12px);
  -webkit-backdrop-filter: blur(12px);
}
/* Dark gere par --ds-header-bg / --ds-header-text (design-tokens.css) :
   plus d'override [data-theme="dark"] en dur (qui delavait le bandeau). */
body { padding-top: 80px; } /* compense header fixe */
@media (max-width: 720px) { body { padding-top: 76px; } }

.header-inner {
  padding: 10px 16px;
  height: 56px;
  display: flex; align-items: center; justify-content: space-between;
  gap: 12px;
}
.logo-link {
  display: flex; align-items: center; gap: .55rem;
  color: var(--m-ink); min-width: 150px;
  font-family: var(--f-display);
}
.logo-link:hover { color: var(--m-ink); }
.logo-text { font-weight: 700; font-size: 1rem; letter-spacing: -0.01em; }
.logo-by { color: var(--m-ink-3); font-weight: 400; font-size: .8rem; }

/* Logo d en-tete : deux variantes baked permutees selon le theme.
   --clair = logo noir (mode jour), --sombre = logo blanc (mode nuit).
   Aucune dependance a currentColor : le basculement est garanti. */
.logo-img { height: 28px; width: auto; display: block; }
.logo-img--sombre { display: none; }
html[data-theme="dark"] .logo-img--clair  { display: none; }
html[data-theme="dark"] .logo-img--sombre { display: block; }
/* Avant l execution du JS (pas encore de data-theme), on suit l OS. */
@media (prefers-color-scheme: dark) {
  html:not([data-theme="light"]) .logo-img--clair  { display: none; }
  html:not([data-theme="light"]) .logo-img--sombre { display: block; }
}

/* Logo wordmark des pages auth (img) : inversion en dark mode OS. */
@media (prefers-color-scheme: dark) {
  .auth-page img[src*="mentivisos-logo-wordmark"] {
    filter: invert(1) brightness(2);
  }
}

.header-nav {
  display: flex; align-items: center; gap: 8px; font-size: 14px;
}
.header-nav a {
  color: var(--m-ink-2);
  font-weight: 500;
  padding: 8px 14px;
  border-radius: var(--r-pill);
  transition: background var(--t-fast), color var(--t-fast);
}
.header-nav a:hover { background: rgba(0,0,0,0.05); color: var(--m-ink); }
[data-theme="dark"] .header-nav a:hover { background: rgba(255,255,255,0.06); color: var(--m-ink); }
.nav-user {
  color: var(--m-ink-3); font-size: .8rem;
  padding: 0 8px; font-family: var(--f-mono);
}
.theme-toggle {
  display: inline-flex; align-items: center; justify-content: center;
  width: 36px; height: 36px;
  border-radius: var(--r-pill);
  color: var(--m-ink-2);
  background: var(--m-bg-soft);
  border: 1px solid var(--m-line-2);
  transition: all var(--t-fast);
}
.theme-toggle:hover {
  background: var(--m-coral-tint);
  color: var(--m-coral);
  border-color: var(--m-coral-soft);
}
[data-theme="dark"] .theme-toggle { background: var(--m-bg-warm); border-color: var(--m-line); }

@keyframes headerDropIn {
  from { transform: translate(-50%, -16px); opacity: 0; }
  to   { transform: translate(-50%, 0);     opacity: 1; }
}
@media (max-width: 720px) {
  .site-header { top: 8px; width: calc(100% - 16px); border-radius: var(--r-lg); }
  .header-nav { gap: 4px; font-size: 13px; }
  .header-nav a { padding: 6px 10px; }
  .nav-user { display: none; }
  .logo-link { min-width: 0; }
  .logo-by { display: none; }
}

/* ---------- 4. Layouts Mentivis ---------- */
.container {
  max-width: var(--container); margin: 0 auto;
  padding: 32px var(--gutter);
}
.container-sm { max-width: 760px; margin: 0 auto; padding: 32px var(--gutter); }
@media (max-width: 900px) {
  .container, .container-sm { padding: 24px 20px; }
}
.center-screen {
  min-height: 100dvh; display: flex; align-items: center; justify-content: center;
  padding: 24px;
  background:
    radial-gradient(1200px 600px at 50% -100px, color-mix(in srgb, var(--color-primary) 8%, transparent), transparent 60%),
    var(--m-bg-soft);
}
body.auth-page { padding-top: 0; }
body.auth-page .site-header { display: none; }

.grid { display: grid; gap: 1.25rem; }
.grid-cards { grid-template-columns: repeat(auto-fill, minmax(280px, 1fr)); }

.flex { display: flex; gap: 1rem; align-items: center; }
.flex-between { display: flex; align-items: center; justify-content: space-between; gap: 1rem; flex-wrap: wrap; }

.stack > * + * { margin-top: var(--gap, 1rem); }
.stack-sm > * + * { margin-top: .5rem; }
.stack-lg > * + * { margin-top: 2rem; }

/* ---------- 5. Boutons Mentivis (pill) ---------- */
/* v2.1.2 : Boutons conformes spec MentivisOS-final-design
   - btn-primary : NOIR (#000), hover #222 - role principal de l'app
   - btn-secondary : WARM (#f5f2ef), hover plus fonce - role secondaire
   - btn-coral : CORAL (#E8726A) - reserve aux CTAs accent (rares, type "Composants")
   - btn-outline : outline noir sur fond transparent
   - btn-ghost : sans fond ni bordure (link-like)
   - Radius unifie : 8px (--r-md) partout (plus de pill par defaut) */
.btn {
  display: inline-flex; align-items: center; justify-content: center;
  gap: 8px;
  padding: 12px 20px;
  border-radius: var(--r-md);
  font-family: var(--f-sans);
  font-size: 15px; font-weight: 500;
  letter-spacing: 0; line-height: 1.4;
  cursor: pointer; transition: background var(--t-fast), border-color var(--t-fast), color var(--t-fast), transform var(--t-fast), box-shadow var(--t-fast);
  white-space: nowrap; text-decoration: none;
  border: 1px solid transparent;
}
.btn:hover { text-decoration: none; }
.btn:active { transform: translateY(1px); }

/* Primaire : tokens dedies configurables (super-admin), corrects clair ET dark
   (repos + survol). Remplace --ds-gray-1000/-900/-800 qui s'inversaient en dark
   sans contraste de texte (bouton delave). */
.btn-primary {
  background: var(--ds-btn-bg); color: var(--ds-btn-text);
  border-color: var(--ds-btn-bg);
}
.btn-primary:hover { background: var(--ds-btn-bg-hover); border-color: var(--ds-btn-bg-hover); color: var(--ds-btn-text-hover); }
.btn-primary:active { background: var(--ds-btn-bg-hover); border-color: var(--ds-btn-bg-hover); }

/* Secondaire : tokens dedies configurables (super-admin), repos + survol, clair ET
   dark. Defauts = bg-100 / texte principal / bg-200 au survol (specs DS d'origine). */
.btn-secondary {
  background: var(--ds-btn2-bg); color: var(--ds-btn2-text);
  border-color: var(--ds-border-default);
}
.btn-secondary:hover {
  background: var(--ds-btn2-bg-hover); border-color: var(--ds-border-hover); color: var(--ds-btn2-text-hover);
}

/* Outline : bordure noire sur fond transparent */
.btn-outline {
  background: transparent; color: var(--m-ink);
  border-color: var(--m-line);
}
.btn-outline:hover {
  background: var(--m-ink); color: var(--m-bg); border-color: var(--m-ink);
}
[data-theme="dark"] .btn-outline {
  color: var(--m-ink); border-color: var(--m-line);
}
[data-theme="dark"] .btn-outline:hover {
  background: var(--m-ink); color: var(--m-bg); border-color: var(--m-ink);
}

/* Coral : reserve aux CTAs accent (utilisation parcimonieuse) */
.btn-coral {
  background: var(--m-coral); color: var(--ds-text-on-accent);
  border-color: var(--m-coral);
}
.btn-coral:hover { background: var(--m-coral-hover); border-color: var(--m-coral-hover); color: var(--ds-text-on-accent); }
.btn-coral:active { background: var(--m-coral-hover); border-color: var(--m-coral-hover); }

/* Ghost : sans fond, link-like */
.btn-ghost {
  background: transparent; color: var(--m-ink-2);
  border-color: transparent;
}
.btn-ghost:hover { background: var(--m-bg-soft); color: var(--m-ink); }
[data-theme="dark"] .btn-ghost:hover { background: var(--m-bg-warm); color: var(--m-ink); }

.btn-danger { background: var(--m-danger); color: var(--ds-text-on-accent); border-color: var(--m-danger); }
.btn-danger:hover { background: var(--ds-red-900); border-color: var(--ds-red-900); color: var(--ds-text-on-accent); }

/* Variante douce : utilisee en liste pour ne pas saturer l'oeil */
.btn-danger-soft { background: transparent; color: var(--ds-red-700); border-color: var(--ds-red-700); }
.btn-danger-soft:hover { background: var(--ds-red-700); color: var(--ds-text-on-accent); border-color: var(--ds-red-700); }

.btn-success { background: var(--m-success); color: var(--ds-text-on-accent); border-color: var(--m-success); }
.btn-success:hover { background: var(--m-success-text); border-color: var(--m-success-text); }

.btn-sm { padding: 8px 14px; font-size: 13px; }
.btn-lg { padding: 16px 26px; font-size: 15px; }
.btn:disabled { opacity: .5; cursor: not-allowed; pointer-events: none; }
.btn-block { width: 100%; }

/* Icone Material Symbols dans bouton (chevron_right par defaut) */
.btn .material-symbols-outlined {
  font-size: 18px;
  font-variation-settings: 'FILL' 0, 'wght' 500, 'GRAD' 0, 'opsz' 20;
}

/* ---------- 6. Cards Mentivis ---------- */
.card {
  background: var(--ds-bg-100);
  border: 1px solid var(--ds-border-default);
  border-radius: var(--r-xl);
  padding: 28px;
  box-shadow: var(--shadow-1);
  transition: box-shadow var(--t), border-color var(--t), transform var(--t);
}
.card-hover { cursor: pointer; }
.card-hover:hover {
  box-shadow: var(--shadow-2);
  border-color: var(--m-purple-soft);
  transform: translateY(-2px);
}
.card-pad-sm { padding: 18px; }
.card-pad-lg { padding: 36px 32px; }
.card-accent { background: var(--m-bg-soft); }

/* ---------- 7. Forms ---------- */
.form-group { margin-bottom: 1.25rem; }
.form-group label, label.form-label {
  display: block; margin-bottom: .4rem;
  font-size: .875rem; font-weight: 600; color: var(--text);
}
.form-help { font-size: .8rem; color: var(--text-muted); margin-top: .35rem; }
/* v2.0.23 : indication d'exemples à côté du label (poids normal, gris) */
.form-label-hint {
  font-weight: 400;
  color: var(--text-muted);
  font-size: .82rem;
  margin-left: .25rem;
}
.form-input, .form-textarea, .form-select {
  width: 100%; padding: 13px 16px;
  background: var(--ds-bg-100); color: var(--ds-text-primary);
  border: 1px solid var(--ds-border-default); border-radius: var(--r-lg);
  font-family: var(--f-sans);
  font-size: 16px;
  transition: border-color var(--t-fast), box-shadow var(--t-fast);
  outline: none;
}
.form-input::placeholder, .form-textarea::placeholder { color: var(--ds-text-tertiary); }
.form-input:focus, .form-textarea:focus, .form-select:focus {
  border-color: var(--ds-border-focus);
  box-shadow: 0 0 0 3px color-mix(in srgb, var(--color-primary) 10%, transparent);
}
.form-textarea { min-height: 140px; resize: vertical; line-height: 1.5; font-family: var(--f-sans); }
.form-textarea-mono { font-family: var(--f-mono); font-size: 14px; min-height: 220px; letter-spacing: 0.02em; }
.form-group { margin-bottom: 18px; }
.form-group label, label.form-label {
  display: block; margin-bottom: 8px;
  font-size: 14px; font-weight: 600; color: var(--m-ink-2);
}

/* ---------- 8. Alerts Mentivis ---------- */
.alert {
  padding: 14px 18px; border-radius: var(--r-lg);
  font-size: 14px; margin-bottom: 16px;
  border: 1px solid transparent;
  line-height: 1.5;
}
.alert-success { background: var(--m-success-soft); color: var(--m-success-text); border-color: color-mix(in srgb, var(--m-success) 32%, transparent); }
.alert-error   { background: var(--m-danger-soft); color: var(--ds-red-900); border-color: color-mix(in srgb, var(--m-danger) 32%, transparent); }
.alert-info    { background: var(--m-purple-tint); color: var(--m-purple-deep); border-color: var(--m-purple-soft); }
.alert-warning { background: var(--m-warning-soft); color: var(--ds-amber-900); border-color: var(--ds-amber-700); }
[data-theme="dark"] .alert-info { color: var(--m-ink); }

/* ---------- 9. Badges & Grades ---------- */
.badge {
  display: inline-flex; align-items: center; gap: .3rem;
  padding: 4px 12px; font-size: 12px; font-weight: 600;
  border-radius: var(--r-pill); background: var(--m-bg-soft);
  color: var(--m-ink-2); border: 1px solid var(--m-line-2);
  font-family: var(--f-sans);
}
.badge-primary { background: var(--m-purple-tint); color: var(--m-purple); border-color: var(--m-purple-soft); }
.badge-success { background: var(--m-success-soft); color: var(--m-success-text); border-color: color-mix(in srgb, var(--m-success) 32%, transparent); }
.badge-warning { background: var(--m-warning-soft); color: var(--ds-amber-900); border-color: var(--ds-amber-300); }
.badge-danger  { background: var(--m-danger-soft); color: var(--ds-red-900); border-color: color-mix(in srgb, var(--m-danger) 32%, transparent); }
.badge-grade {
  display: inline-flex; align-items: center; justify-content: center;
  width: 32px; height: 32px; border-radius: 9999px;
  font-weight: 800; font-size: 1rem;
}
.badge-grade.grade-a { background: var(--grade-a-bg); color: var(--grade-a-fg); }
.badge-grade.grade-b { background: var(--grade-b-bg); color: var(--grade-b-fg); }
.badge-grade.grade-c { background: var(--grade-c-bg); color: var(--grade-c-fg); }
.badge-grade.grade-d { background: var(--grade-d-bg); color: var(--grade-d-fg); }
.badge-grade.grade-e { background: var(--grade-e-bg); color: var(--grade-e-fg); }
.badge-grade-lg { width: 96px; height: 96px; font-size: 3rem; box-shadow: var(--shadow-lg); }

/* ---------- 10. Progress ---------- */
.progress-bar {
  height: 8px; background: var(--bg-soft); border-radius: 9999px;
  overflow: hidden; position: relative;
}
.progress-bar > span {
  display: block; height: 100%;
  background: linear-gradient(90deg, var(--color-primary), var(--color-primary-dark));
  border-radius: 9999px; transition: width var(--t-slow);
}

/* ---------- 11. Module sidebar ---------- */
.module-layout {
  display: grid;
  /* v3.4.51 : reserve droite symetrique a la sidebar gauche (280px + gap 2rem).
     Le bloc texte est ainsi centre visuellement entre sidebar et bulles. */
  grid-template-columns: 280px 1fr;
  gap: 2rem;
  max-width: 1632px; margin: 0 auto;
  padding: 2rem 1.25rem;
  /* 280px (largeur sidebar) + 2rem (gap) reserves a droite pour les bulles */
  padding-right: calc(1.25rem + 280px + 2rem);
}
.module-sidebar {
  position: sticky; top: 80px; align-self: start;
  background: var(--bg-elev); border: 1px solid var(--border);
  border-radius: var(--radius-lg); padding: 1.25rem;
  max-height: calc(100dvh - 100px); overflow-y: auto;
}
.module-sidebar h3 {
  font-size: .75rem; font-weight: 700; text-transform: uppercase;
  letter-spacing: .5px; color: var(--text-muted); margin: 1rem 0 .5rem;
}
.module-sidebar h3:first-child { margin-top: 0; }

.section-tab {
  display: flex; align-items: center; gap: .5rem;
  padding: .5rem .75rem; border-radius: var(--radius);
  font-size: .85rem; color: var(--text-soft);
  cursor: pointer; transition: all var(--t-fast);
  width: 100%; text-align: left;
}
.section-tab:hover { background: var(--bg-soft); color: var(--text); }
.section-tab.active { background: var(--m-purple-tint); color: var(--m-purple); font-weight: 600; border-left: 3px solid var(--m-purple); padding-left: calc(.75rem - 3px); }
.section-tab.completed { color: var(--color-success); }
/* N21 (v0.14.9.135) : la coche est deja portee par .sec-num (pastille verte
   avec checkmark blanc, posee dans show.php pour les sections validees). Le
   ::before ajoutait un second ✓ devant le badge, d ou un doublon visuel coche
   + badge. On retire la pseudo pour garder un seul indicateur (le badge). */
.section-tab.locked { opacity: .55; cursor: not-allowed; }
.section-tab.locked::after { content: '🔒'; margin-left: auto; font-size: .75rem; }
.section-tab .tab-score {
  margin-left: auto; font-size: .7rem; padding: .1rem .4rem;
  border-radius: 9999px; background: var(--bg-soft); font-weight: 700;
}

/* ---------- 11b. N13 / N38 : sidebar masquable / rail compact ----------
   Toggle utilisateur qui reduit la sidebar 280px en un rail compact d environ
   56px ne montrant que les pastilles numerotees colorees. L etat est gere par
   module-sidebar-toggle.js (.is-rail pose sur .module-layout). Depuis N39
   (v0.14.9.149), il n est plus persiste : la sidebar est toujours depliee a
   l arrivee, seul un clic explicite la rabat pour la session courante.
   Le rail reste cliquable : un clic sur une pastille scrolle vers la section
   (handler data-target deja gere par module-show.js).
   N38 (v0.14.9.154) : le bouton chevron est rendu sticky a l interieur de la
   sidebar scrollable pour rester accessible meme si le contenu defile, et le
   resume textuel (paragraphe competences cibles) est masque proprement en
   mode rail (regle .module-sidebar > p qui manquait). */
.module-layout {
  transition: grid-template-columns .28s ease;
}
.module-layout.is-rail {
  grid-template-columns: 56px 1fr;
  /* Plus besoin de reserver 280px a droite pour les bulles : le gain de place
     profite au contenu. La reserve droite reste geree par les media queries. */
}
.module-sidebar { transition: padding .28s ease; }

/* Bouton toggle : chevron sticky en haut a droite de la sidebar.
   N38 (v0.14.9.154) : passe de position:absolute a position:sticky pour
   rester visible quand la sidebar scrolle son contenu interne (sections
   nombreuses qui deborderaient calc(100dvh - 100px)).
   N38-r1 (v0.14.9.157) : couleurs inversees pour rendre le bouton bien
   visible sur le fond clair de la sidebar (auparavant fond blanc sur fond
   blanc + chevron gris a peine visible). On utilise --text comme fond et
   --bg-elev comme couleur d icone : contraste fort et automatique en theme
   sombre (les variables CSS s inversent). */
.module-sidebar-toggle {
  position: sticky;
  top: 0;
  z-index: 3;
  align-self: flex-end;
  margin-left: auto;
  margin-bottom: .25rem;
  width: 28px;
  height: 28px;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  padding: 0;
  border: 1px solid var(--text, #111);
  border-radius: 7px;
  background: var(--text, #111);
  color: var(--bg-elev, #fff);
  cursor: pointer;
  line-height: 1;
  transition: background var(--t-fast, .15s) ease, color var(--t-fast, .15s) ease, border-color var(--t-fast, .15s) ease;
}
/* Pour que align-self:flex-end fonctionne, la sidebar doit etre en flex column.
   Elle ne l etait pas (block par defaut). On bascule en colonne, ce qui ne
   change rien au rendu des enfants (chacun occupe deja toute la largeur). */
.module-sidebar { display: flex; flex-direction: column; }
.module-sidebar-toggle:hover {
  background: var(--m-purple, #6366f1);
  border-color: var(--m-purple, #6366f1);
  color: var(--ds-text-on-accent);
}
.module-sidebar-toggle:focus-visible { outline: 2px solid var(--m-purple, #6366f1); outline-offset: 2px; }
.module-sidebar-toggle-chevron {
  font-size: 1.25rem;
  transition: transform .28s ease;
}
/* En mode rail, le chevron pointe vers la droite (deplier) et le bouton se
   recentre dans le rail etroit (align-self:center plutot que flex-end). */
.is-rail .module-sidebar-toggle { align-self: center; margin-left: 0; }
.is-rail .module-sidebar-toggle-chevron { transform: rotate(180deg); }

/* La sidebar reste sticky ; on resserre son padding en mode rail. */
.is-rail .module-sidebar { padding: 1.25rem .5rem; overflow-x: hidden; }

/* En mode rail : on masque tout le contenu textuel, on ne garde que les
   pastilles numerotees cliquables (acces direct + scan visuel).
   N38 (v0.14.9.154) : ajout de "> p" pour masquer le paragraphe competences
   cibles (resume du module) qui restait visible en rail et debordait. */
.is-rail .module-sidebar h3,
.is-rail .module-sidebar > a,
.is-rail .module-sidebar > p,
.is-rail .module-sidebar > div[style],
.is-rail .module-progression,
.is-rail .section-tab-label {
  display: none;
}
/* On reaffiche le conteneur des sections (masque par la regle div[style]
   ci-dessus s il porte un style inline) : #sidebarSections garde ses tabs. */
.is-rail #sidebarSections { display: block; }
/* Les tabs en rail : centrees, sans fond hover envahissant, juste la pastille. */
.is-rail .section-tab {
  justify-content: center;
  padding: .35rem 0;
  gap: 0;
}
.is-rail .section-tab .sec-num { margin-right: 0; }
.is-rail .section-tab.locked::after { content: none; }
/* Le bouton "Regenerer ce module" (bloc en bas) n a pas de sens en rail. */
.is-rail .module-sidebar > div[style*="border-top"] { display: none; }

/* Le toggle n a de sens que sur la mise en page deux colonnes (> 900px).
   Sous 900px la sidebar est empilee (cf media query responsive) : on masque
   le toggle et on neutralise un eventuel etat rail memorise pour preserver le
   comportement mobile actuel intact. */
@media (max-width: 900px) {
  .module-sidebar-toggle { display: none; }
  .module-layout.is-rail { grid-template-columns: 1fr; }
  .is-rail .module-sidebar { padding: 1.25rem; }
  .is-rail .module-sidebar h3,
  .is-rail .module-sidebar > a,
  .is-rail .module-sidebar > p,
  .is-rail .module-sidebar > div[style],
  .is-rail .module-progression,
  .is-rail .section-tab-label { display: revert; }
  .is-rail .section-tab { justify-content: flex-start; padding: .5rem .75rem; gap: .5rem; }
}

/* Respect de prefers-reduced-motion : pas d animation de largeur ni de chevron. */
@media (prefers-reduced-motion: reduce) {
  .module-layout,
  .module-sidebar,
  .module-sidebar-toggle-chevron { transition: none; }
}

.module-content { min-width: 0; }
.module-content h1 {
  font-family: var(--f-display);
  font-size: clamp(1.75rem, 3vw, 2.25rem); font-weight: 700;
  margin-bottom: .75rem; letter-spacing: -0.015em; color: var(--m-ink);
}
.module-content h2 {
  font-family: var(--f-display);
  font-size: 1.4rem; font-weight: 700; margin: 2rem 0 .75rem;
  padding-bottom: .5rem; border-bottom: 1px solid var(--m-line);
  letter-spacing: -0.01em; color: var(--m-ink);
}
.module-content h3 { font-family: var(--f-display); font-size: 1.1rem; font-weight: 600; margin: 1.5rem 0 .5rem; color: var(--m-ink); }
.module-content p, .module-content li { margin-bottom: .6rem; color: var(--m-ink-2); }
.module-content a { color: var(--color-primary-dark); text-decoration: underline; text-underline-offset: 3px; }
.module-content blockquote { border-left: 3px solid var(--m-purple); background: var(--m-bg-soft); padding: 12px 16px; margin: 16px 0; border-radius: 0 var(--r-md) var(--r-md) 0; font-style: italic; color: var(--m-ink-2); }
.module-content strong, .module-content b { color: var(--m-ink); font-weight: 600; }
.module-content ul, .module-content ol { margin: .5rem 0 1rem 1.5rem; }
.module-content table {
  width: 100%; border-collapse: collapse; margin: 1rem 0;
  background: var(--bg-elev); border-radius: var(--radius);
  overflow: hidden; border: 1px solid var(--border);
}
.module-content th, .module-content td {
  padding: .6rem .8rem; text-align: left; border-bottom: 1px solid var(--border-soft);
}
.module-content th { background: var(--bg-soft); font-weight: 700; }

/* ---------- 12. Code blocks ---------- */
.code-block {
  position: relative; background: #1e1e1e; color: #e8e4dc;
  border-radius: var(--radius); margin: 1rem 0;
  font-family: var(--font-mono); font-size: .85rem;
  overflow: hidden;
}
.code-block .code-header {
  display: flex; justify-content: space-between; align-items: center;
  padding: .5rem .9rem; background: #14140f; font-size: .75rem;
  color: #a0a0a0; border-bottom: 1px solid #2a2a25;
}
.code-block pre { padding: 1rem; overflow-x: auto; line-height: 1.5; }
.code-block code { font-family: inherit; }
.code-block .btn-copy {
  background: #2a2a25; color: #d8d4cc; padding: .25rem .65rem;
  border-radius: var(--radius-sm); font-size: .7rem; transition: all var(--t-fast);
}
.code-block .btn-copy:hover { background: var(--color-primary); color: var(--ds-text-on-accent); }
.code-block .btn-copy.copied { background: var(--color-success); color: var(--ds-text-on-accent); }

.module-content code:not(pre code) {
  background: var(--bg-soft); padding: .1rem .4rem;
  border-radius: var(--radius-sm); font-family: var(--font-mono);
  font-size: .85em; color: var(--color-primary-dark);
}
/* Le code inline suit --color-primary-dark (qui s'inverse seul en dark) :
   l'override [data-theme="dark"] en dur (#7ec5c8) est redondant -> supprime. */

/* ---------- 13. Correction panel ---------- */
.correction-panel {
  background: var(--bg-soft); border-radius: var(--radius-lg);
  padding: 1.5rem; margin-top: 1.5rem;
  border: 1px solid var(--border);
}
.correction-panel h3 { font-size: 1rem; font-weight: 700; margin-bottom: .75rem; }
.zone-positifs {
  background: var(--color-success-soft); padding: 1rem;
  border-radius: var(--radius); border-left: 3px solid var(--color-success);
  margin-bottom: 1rem;
}
.zone-positifs h4 { color: var(--color-success); margin-bottom: .5rem; }
.zone-amelioration {
  background: var(--color-warning-soft); padding: 1rem;
  border-radius: var(--radius); border-left: 3px solid var(--color-warning);
  margin-bottom: 1rem;
}
.zone-amelioration h4 { color: var(--color-warning); margin-bottom: .5rem; }
.zone-conseils {
  background: var(--color-primary-light); padding: 1rem;
  border-radius: var(--radius); border-left: 3px solid var(--color-primary);
}

/* ---------- 14. Score circle SVG ---------- */
.score-circle-wrap {
  display: inline-flex; flex-direction: column; align-items: center; gap: .5rem;
}
.score-circle {
  width: 110px; height: 110px; transform: rotate(-90deg);
}
.score-circle .bg-track { stroke: var(--border-soft); }
.score-circle .progress {
  stroke: var(--color-primary);
  stroke-dasharray: 314; stroke-dashoffset: 314;
  transition: stroke-dashoffset 1.4s cubic-bezier(.4, 0, .2, 1);
}
.score-circle-label {
  font-size: 1.5rem; font-weight: 800; color: var(--color-primary);
  display: flex; align-items: center; justify-content: center;
  width: 110px; height: 110px; margin-top: -110px;
  position: relative; z-index: 1;
  pointer-events: none;
}

/* ---------- 15. Timer (contrôle) ---------- */
/* v0.22.24 : bandeau sticky plein largeur, MM:SS bien visible.
   .timer-display (petite pastille legacy) reste defini pour compat. */
.timer-bar {
  position: fixed; top: 0; left: 0; right: 0;
  background: var(--bg-elev);
  border-bottom: 2px solid var(--border);
  box-shadow: var(--shadow);
  z-index: 200;
  transition: background var(--t-fast), border-color var(--t-fast), color var(--t-fast);
}
.timer-bar-inner {
  max-width: 720px; margin: 0 auto;
  display: flex; align-items: baseline; justify-content: center;
  gap: .75rem;
  padding: .85rem 1rem;
}
.timer-bar-label {
  font-size: .85rem; font-weight: 600;
  color: var(--text-soft, var(--text));
  text-transform: uppercase; letter-spacing: .04em;
}
.timer-bar-value {
  font-family: var(--font-mono);
  font-size: 2rem; font-weight: 800;
  font-variant-numeric: tabular-nums;
  color: var(--text);
  line-height: 1;
}
.timer-bar-total {
  font-size: .9rem; color: var(--text-soft, var(--text)); font-weight: 500;
}
.timer-bar.warn {
  background: color-mix(in srgb, var(--color-warning) 8%, transparent);
  border-bottom-color: var(--color-warning);
}
.timer-bar.warn .timer-bar-value { color: var(--color-warning); }
.timer-bar.danger {
  background: color-mix(in srgb, var(--color-danger) 10%, transparent);
  border-bottom-color: var(--color-danger);
  animation: timer-bar-pulse 1.3s ease-in-out infinite;
}
.timer-bar.danger .timer-bar-value { color: var(--color-danger); }
@keyframes timer-bar-pulse {
  0%, 100% { box-shadow: 0 0 0 0 color-mix(in srgb, var(--color-danger) 35%, transparent), var(--shadow); }
  50%      { box-shadow: 0 0 0 6px color-mix(in srgb, var(--color-danger) 0%, transparent),  var(--shadow); }
}
/* Legacy : ancienne pastille fixed top-right (encore utilisee ailleurs ?) */
.timer-display {
  position: fixed; top: 14px; right: 18px;
  background: var(--bg-elev); border: 2px solid var(--border);
  padding: .55rem 1rem; border-radius: var(--radius);
  font-family: var(--font-mono); font-size: 1.15rem; font-weight: 800;
  color: var(--text); z-index: 100;
  box-shadow: var(--shadow); transition: all var(--t-fast);
}
.timer-display.warn { color: var(--color-warning); border-color: var(--color-warning); }
.timer-display.danger {
  color: var(--color-danger); border-color: var(--color-danger);
  animation: pulse 1.3s ease-in-out infinite;
}
@keyframes pulse {
  0%, 100% { transform: scale(1); box-shadow: 0 0 0 0 color-mix(in srgb, var(--color-danger) 50%, transparent); }
  50% { transform: scale(1.05); box-shadow: 0 0 0 8px color-mix(in srgb, var(--color-danger) 0%, transparent); }
}

/* ---------- 16. QCM ---------- */
/* v0.14.9.122 (N12) : le conteneur .qcm-question ne doit PAS imposer font-weight
   a toute sa descendance (sinon labels d options et feedback heritent du gras).
   Le selecteur p.qcm-question cible uniquement l enonce de la page controle.php
   ou .qcm-question est un <p>. Dans section-article.php, .qcm-question est un
   <div> conteneur : son enonce porte la classe .qcm-enonce (cf bloc plus bas). */
p.qcm-question { margin-bottom: 1rem; font-size: 1.05rem; font-weight: 600; }
div.qcm-question { margin-bottom: 1rem; }
.qcm-options { display: grid; gap: .65rem; }
.qcm-option {
  display: flex; align-items: center; gap: .85rem;
  padding: 14px 16px; border: 1.5px solid var(--m-line);
  border-radius: var(--r-lg); background: var(--bg-elev);
  cursor: pointer; transition: all var(--t-fast); text-align: left;
  font-size: .95rem;
}
.qcm-option:hover { border-color: var(--m-purple-soft); background: var(--m-bg-soft); }
.qcm-option .key {
  display: inline-flex; align-items: center; justify-content: center;
  width: 28px; height: 28px; border-radius: var(--r-pill);
  background: var(--m-bg-soft); border: 1px solid var(--m-line); font-weight: 700;
  flex-shrink: 0;
  font-family: var(--f-mono); font-size: .8rem;
}
.qcm-option.selected {
  border-color: var(--m-purple); background: var(--m-purple-tint);
}
.qcm-option.selected .key { background: var(--m-purple); color: var(--ds-text-on-accent); border-color: var(--m-purple); }
.qcm-option.locked { cursor: not-allowed; opacity: .85; }
.qcm-option.correct { border-color: var(--color-success); background: var(--color-success-soft); }
.qcm-option.wrong { border-color: var(--color-danger); background: var(--color-danger-soft); }
.qcm-nav { display: flex; justify-content: space-between; margin-top: 1.25rem; }
.qcm-progress { font-size: .85rem; color: var(--text-muted); text-align: center; margin-bottom: 1rem; }

/* ---------- 17. Skeleton / Loading ---------- */
.skeleton {
  background: linear-gradient(90deg, var(--bg-soft) 25%, var(--border-soft) 50%, var(--bg-soft) 75%);
  background-size: 200% 100%;
  animation: shimmer 1.5s infinite;
  border-radius: var(--radius);
}
@keyframes shimmer {
  0% { background-position: 200% 0; }
  100% { background-position: -200% 0; }
}
.spinner {
  display: inline-block; width: 18px; height: 18px;
  border: 2px solid var(--bg-soft); border-top-color: var(--color-primary);
  border-radius: 50%; animation: spin .8s linear infinite;
}
@keyframes spin { to { transform: rotate(360deg); } }
.loading-block {
  display: flex; flex-direction: column; align-items: center;
  gap: 1rem; padding: 3rem; text-align: center;
  color: var(--text-muted);
}

/* ---------- Barre de progression IA ---------- */
.progress-block {
  padding: 2rem 1.75rem;
  background: var(--bg-soft);
  border: 1px solid var(--m-line);
  border-radius: var(--radius-lg, 12px);
}
.progress-header {
  display: flex;
  align-items: center;
  gap: 1rem;
  margin-bottom: 1.25rem;
}
.progress-header .spinner {
  width: 24px; height: 24px;
  border-width: 3px;
  flex-shrink: 0;
}
.progress-title {
  font-weight: 700;
  font-size: 1rem;
  color: var(--m-ink);
  margin-bottom: .25rem;
}
.progress-step {
  font-size: .875rem;
  transition: opacity .3s;
}
.progress-bar-wrap {
  width: 100%;
  height: 10px;
  background: var(--m-bg-soft);
  border-radius: 999px;
  overflow: hidden;
  position: relative;
}
.progress-bar-fill {
  height: 100%;
  background: linear-gradient(90deg, var(--color-primary) 0%, var(--color-primary-dark) 100%);
  border-radius: 999px;
  transition: width .4s ease-out;
  position: relative;
  overflow: hidden;
}
.progress-bar-fill::after {
  content: '';
  position: absolute; inset: 0;
  background: linear-gradient(90deg,
    transparent 0%,
    rgba(255,255,255,.25) 50%,
    transparent 100%);
  background-size: 200% 100%;
  animation: shimmer 2s linear infinite;
}
.progress-bar-fail {
  background: linear-gradient(90deg, var(--m-danger) 0%, var(--ds-red-900) 100%) !important;
}
.progress-bar-fail::after { display: none; }
.progress-meta {
  display: flex;
  justify-content: space-between;
  margin-top: .6rem;
  font-size: .8rem;
  font-variant-numeric: tabular-nums;
}

/* ---------- Barre step-by-step (analyse profil v2.0.43) ---------- */
.analyse-steps {
  list-style: none;
  padding: 0;
  margin: 1.25rem 0 0 0;
  display: flex;
  flex-direction: column;
  gap: .5rem;
}
.analyse-step {
  display: flex;
  align-items: center;
  gap: .75rem;
  padding: .65rem .85rem;
  border: 1px solid var(--m-line);
  border-radius: 8px;
  background: var(--ds-bg-100);
  font-size: .9rem;
  transition: background .25s, border-color .25s, opacity .25s;
}
.analyse-step .step-icon {
  width: 22px;
  height: 22px;
  border-radius: 50%;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  font-size: .75rem;
  font-weight: 700;
  flex-shrink: 0;
  background: var(--m-bg-soft);
  color: var(--text-muted, #666);
}
.analyse-step .step-body {
  flex: 1;
  display: flex;
  flex-direction: column;
  gap: .35rem;
  min-width: 0;
}
.analyse-step .step-label {
  color: var(--m-ink);
}
/* v2.0.46 - barre de progression interne par etape (visible si is-active uniquement) */
.analyse-step .step-progress {
  display: none;
  width: 100%;
  height: 4px;
  background: color-mix(in srgb, var(--color-primary) 15%, transparent);
  border-radius: 2px;
  overflow: hidden;
}
.analyse-step.is-active .step-progress { display: block; }
.analyse-step.is-fail .step-progress { display: block; background: color-mix(in srgb, var(--m-danger) 15%, transparent); }
.analyse-step .step-progress-fill {
  height: 100%;
  width: 0%;
  background: var(--color-primary);
  border-radius: 2px;
  transition: width .15s linear;
}
.analyse-step.is-fail .step-progress-fill { background: var(--m-danger); }
.analyse-step .step-meta {
  font-size: .75rem;
  color: var(--text-muted, #888);
  font-variant-numeric: tabular-nums;
  /* v0.17.7 : empeche le step-meta de comprimer step-body / mini-barre.
     Sans flex-shrink:0, un libelle long ("Attente de reponse - 18s
     restantes") faisait passer la mini-barre interne a 30% de largeur
     visuelle. */
  flex-shrink: 0;
  text-align: right;
  white-space: nowrap;
}
.analyse-step.is-pending { opacity: .6; }
.analyse-step.is-active {
  border-color: var(--color-primary);
  background: color-mix(in srgb, var(--color-primary) 6%, transparent);
}
.analyse-step.is-active .step-icon {
  background: var(--color-primary);
  color: var(--ds-text-on-accent);
}
.analyse-step.is-done {
  border-color: var(--m-success);
  background: color-mix(in srgb, var(--m-success) 6%, transparent);
}
.analyse-step.is-done .step-icon {
  background: var(--m-success);
  color: var(--ds-text-on-accent);
}
.analyse-step.is-fail {
  border-color: var(--m-danger);
  background: color-mix(in srgb, var(--m-danger) 6%, transparent);
}
.analyse-step.is-fail .step-icon {
  background: var(--m-danger);
  color: var(--ds-text-on-accent);
}
.analyse-step .step-retry {
  font-size: .8rem;
  padding: .25rem .65rem;
  border: 1px solid var(--m-danger);
  background: var(--ds-bg-100);
  color: var(--m-danger);
  border-radius: 6px;
  cursor: pointer;
  font-weight: 600;
}
.analyse-step .step-retry:hover { background: var(--m-danger); color: var(--ds-text-on-accent); }
.analyse-spinner {
  width: 14px;
  height: 14px;
  border: 2px solid color-mix(in srgb, var(--ds-text-on-accent) 40%, transparent);
  border-top-color: var(--ds-text-on-accent);
  border-radius: 50%;
  animation: spin 1s linear infinite;
}
@keyframes spin { to { transform: rotate(360deg); } }

/* ---------- Assistant chat (bulle + panneau) ---------- */
/* v1.0.1 : FAB compactes (-20%) - padding .85/.1.2 -> .65/1, font .9 -> .82,
   gap .5 -> .4. Cible visuelle : boutons moins "pates" sur la colonne droite. */
.chat-toggle {
  position: fixed;
  right: 1.5rem;
  bottom: 1.5rem;
  z-index: 90;
  display: flex;
  align-items: center;
  gap: .4rem;
  background: var(--ds-fab-assistant);
  color: var(--ds-text-on-accent);
  border: none;
  padding: .65rem 1rem;
  border-radius: 999px;
  font-weight: 600;
  font-size: .82rem;
  cursor: pointer;
  box-shadow: 0 8px 24px rgba(0,0,0,.18);
  transition: transform .2s, box-shadow .2s;
}
.chat-toggle:hover {
  transform: translateY(-2px);
  box-shadow: 0 12px 28px rgba(0,0,0,.22);
}
.chat-toggle svg { flex-shrink: 0; }
.chat-toggle-label { white-space: nowrap; }
/* Masque la bulle flottante lorsque le panneau est ouvert (évite chevauchement avec le bouton Envoyer) */
body.chat-open .chat-toggle {
  opacity: 0;
  visibility: hidden;
  pointer-events: none;
  transition: opacity .2s, visibility .2s;
}
@media (max-width: 600px) {
  /* v1.0.3 : on ne reaffirme PAS bottom ici, c est bloc-notes.css qui
     pilote l empilement mobile via :
       bottom: calc(1rem + 3 * var(--fab-row-h) + var(--fab-feedback-gap, 14px));
     Ecraser avec bottom:1rem ramenait Assistant sur Feedback. */
  .chat-toggle { right: 1rem; padding: .6rem; }
  .chat-toggle-label { display: none; }
}

.chat-panel {
  position: fixed;
  top: 0;
  right: 0;
  bottom: 0;
  width: 420px;
  max-width: 100%;
  background: var(--bg);
  border-left: 1px solid var(--border);
  box-shadow: -12px 0 40px rgba(0,0,0,.22);
  display: flex;
  flex-direction: column;
  transform: translateX(100%);
  transition: transform .35s cubic-bezier(.16,1,.3,1);
  z-index: 100;
}
.chat-panel.open { transform: translateX(0); }
@media (max-width: 600px) {
  .chat-panel { width: 100%; }
}

.chat-header {
  display: flex;
  align-items: center;
  justify-content: space-between;
  padding: 1rem 1.25rem;
  border-bottom: 1px solid var(--border);
  background: var(--bg-soft);
}
.chat-title {
  font-weight: 700;
  font-size: 1rem;
  color: var(--text);
}
.chat-subtitle {
  font-size: .8rem;
  color: var(--text-muted);
  margin-top: .15rem;
}
.chat-close {
  background: transparent;
  border: none;
  cursor: pointer;
  color: var(--text-muted);
  padding: .4rem;
  border-radius: 8px;
  display: flex;
}
.chat-close:hover { background: var(--border); color: var(--text); }

.chat-messages {
  flex: 1;
  overflow-y: auto;
  padding: 1rem 1.25rem;
  display: flex;
  flex-direction: column;
  gap: .75rem;
  background: var(--bg);
}
.chat-msg {
  max-width: 85%;
  padding: .65rem .9rem;
  border-radius: 14px;
  font-size: .9rem;
  line-height: 1.5;
  word-wrap: break-word;
}
.chat-msg p { margin: 0 0 .5rem 0; }
.chat-msg p:last-child { margin-bottom: 0; }
.chat-msg ul, .chat-msg ol { margin: .25rem 0 .25rem 1.25rem; padding: 0; }
.chat-msg code {
  background: rgba(0,0,0,.08);
  padding: .1rem .3rem;
  border-radius: 4px;
  font-size: .85em;
}
.chat-msg pre {
  background: rgba(0,0,0,.06);
  padding: .6rem;
  border-radius: 6px;
  overflow-x: auto;
  font-size: .8em;
  margin: .5rem 0;
}
.chat-msg-user {
  align-self: flex-end;
  background: var(--color-primary);
  color: var(--ds-text-on-accent);
  border-bottom-right-radius: 4px;
}
.chat-msg-user code, .chat-msg-user pre { background: rgba(255,255,255,.18); }
.chat-msg-bot {
  align-self: flex-start;
  background: var(--bg-soft);
  color: var(--text);
  border: 1px solid var(--border);
  border-bottom-left-radius: 4px;
}

.chat-typing {
  display: inline-flex;
  gap: 4px;
  align-items: center;
  padding: .25rem 0;
}
.chat-typing span {
  width: 7px; height: 7px;
  background: var(--text-muted);
  border-radius: 50%;
  animation: chat-bounce 1.2s infinite ease-in-out;
}
.chat-typing span:nth-child(2) { animation-delay: .2s; }
.chat-typing span:nth-child(3) { animation-delay: .4s; }
@keyframes chat-bounce {
  0%, 60%, 100% { transform: translateY(0); opacity: .4; }
  30% { transform: translateY(-5px); opacity: 1; }
}

.chat-form {
  display: flex;
  gap: .5rem;
  padding: .9rem 1rem;
  border-top: 1px solid var(--border);
  background: var(--bg);
}
.chat-form textarea {
  flex: 1;
  resize: none;
  border: 1px solid var(--border);
  border-radius: 10px;
  padding: .55rem .75rem;
  font: inherit;
  font-size: .9rem;
  background: var(--bg);
  color: var(--text);
  outline: none;
}
.chat-form textarea:focus {
  border-color: var(--color-primary);
  box-shadow: 0 0 0 3px color-mix(in srgb, var(--color-primary) 18%, transparent);
}
.chat-send {
  background: var(--color-primary);
  color: var(--ds-text-on-accent);
  border: none;
  width: 42px;
  height: 42px;
  border-radius: 10px;
  cursor: pointer;
  display: flex;
  align-items: center;
  justify-content: center;
  flex-shrink: 0;
  align-self: flex-end;
}
.chat-send:hover { background: var(--color-primary-dark); }
.chat-send:disabled { opacity: .5; cursor: not-allowed; }

.chat-footer {
  font-size: .7rem;
  text-align: center;
  padding: .4rem 1rem .8rem;
  border-top: 1px solid var(--m-line);
}

/* ---------- Pour aller plus loin (dans cours) ---------- */
.section-resources {
  margin-top: 1.5rem;
  padding: 1rem 1.25rem;
  background: linear-gradient(135deg, color-mix(in srgb, var(--color-primary) 6%, transparent), color-mix(in srgb, var(--color-primary) 2%, transparent));
  border-left: 3px solid var(--color-primary);
  border-radius: 8px;
}
.section-resources-title {
  font-weight: 700;
  color: var(--color-primary-dark);
  margin-bottom: .5rem;
  font-size: .9rem;
  display: flex;
  align-items: center;
  gap: .4rem;
}
.section-resources ul {
  list-style: none;
  padding: 0;
  margin: 0;
  display: flex;
  flex-direction: column;
  gap: .4rem;
}
.section-resources li {
  font-size: .88rem;
  line-height: 1.45;
}
.section-resources a {
  color: var(--color-primary-dark);
  text-decoration: none;
  font-weight: 600;
}
.section-resources a:hover { text-decoration: underline; }
.section-resources .res-desc { color: var(--text-muted); }

/* ---------- 18. Toasts ---------- */
/* v0.14.9.89 (A57) : repositionnement centre-bas au-dessus du FAB Feedback
   (z=9998) et de la modale Feedback (overlay z=9999). Avant : bottom-right
   1.5rem, recouvert par le FAB. Le toast glisse desormais depuis le bas
   au centre de l ecran, avec un z-index 10001 garantissant la visibilite
   meme quand la modale Feedback est ouverte. */
.toast-stack {
  position: fixed; bottom: 1.5rem; left: 50%;
  transform: translateX(-50%);
  display: flex; flex-direction: column; gap: .5rem;
  z-index: 10001; pointer-events: none;
  width: min(420px, calc(100vw - 2rem));
  align-items: stretch;
}
.toast {
  background: var(--bg-elev); color: var(--text);
  border: 1px solid var(--border); border-radius: var(--radius);
  padding: .75rem 1rem; box-shadow: 0 10px 28px rgba(0,0,0,.22);
  min-width: 240px; font-size: .875rem;
  transform: translateY(120%); opacity: 0;
  transition: transform var(--t), opacity var(--t);
  pointer-events: auto;
  text-align: center;
}
.toast.show { transform: translateY(0); opacity: 1; }
.toast.success { border-left: 3px solid var(--color-success); }
.toast.error   { border-left: 3px solid var(--color-danger); }
.toast.info    { border-left: 3px solid var(--color-primary); }

/* ---------- 19. Wizard (nouveau parcours) ---------- */
.wizard-steps {
  display: flex; gap: 0; margin-bottom: 2rem;
  border-radius: var(--radius); overflow: hidden;
  border: 1px solid var(--border);
}
.wizard-steps > div {
  flex: 1; padding: .75rem 1rem; text-align: center;
  background: var(--bg-elev); font-size: .85rem; color: var(--text-muted);
  position: relative;
}
.wizard-steps > div + div { border-left: 1px solid var(--border); }
.wizard-steps > div.active {
  background: var(--color-primary); color: var(--ds-text-on-accent); font-weight: 600;
}
.wizard-steps > div.done {
  background: var(--color-primary-light); color: var(--color-primary-dark);
}
.wizard-step { display: none; animation: fadeIn .35s ease; }
.wizard-step.active { display: block; }

/* v3.4.70 : etape "wide" - casse la contrainte container-sm (760px)
   pour atteindre la largeur de la vue cours (var(--container) = 1240px).
   N'agit que sur cette etape, l'etape 1 (saisie) reste a 760px.
   v3.4.77 : approche viewport-relative pure. On retire l'element du flow
   du parent en le positionnant via 'position: relative; left: 50%;
   margin-left: -W/2'. Calcul en une seule passe, pas de transform donc
   pas de double rendu, pas de flash. */
.wizard-step-wide.active {
  position: relative;
  width: min(var(--container), calc(100vw - 2 * var(--gutter, 24px)));
  left: 50%;
  margin-left: calc(min(var(--container), calc(100vw - 2 * var(--gutter, 24px))) / -2);
  box-sizing: border-box;
}
@media (max-width: 900px) {
  .wizard-step-wide.active {
    width: calc(100vw - 40px);
    margin-left: calc((100vw - 40px) / -2);
  }
}
/* v3.4.78 : pendant l'analyse en cours (UI .analyse-steps presente),
   on reduit la largeur de l'etape elargie a 960px. La liste compacte
   des 8 etapes + barre de progression n'a pas besoin de 1240px et
   parait visuellement trop massive. Des que l'analyse est terminee
   (renderAnalyseEditPreview retire .analyse-steps), la regle ne
   matche plus et la largeur revient automatiquement a 1240px pour
   accueillir la synthese et les tableaux. */
.wizard-step-wide.active:has(.analyse-steps) {
  width: min(960px, calc(100vw - 2 * var(--gutter, 24px)));
  margin-left: calc(min(960px, calc(100vw - 2 * var(--gutter, 24px))) / -2);
}
@media (max-width: 900px) {
  .wizard-step-wide.active:has(.analyse-steps) {
    width: calc(100vw - 40px);
    margin-left: calc((100vw - 40px) / -2);
  }
}

/* v3.4.75 : etape "medium" - largeur intermediaire (960px) pour la saisie.
   Le container-sm (760px) etait trop etroit et laissait beaucoup d'espace
   blanc autour de la card. 960px reste confortable pour la lecture des
   formulaires sans rendre les champs trop larges.
   v3.4.77 : meme approche 'left:50% + margin-left:-W/2' (sans transform)
   pour eviter le flash. */
.wizard-step-medium.active {
  position: relative;
  width: min(960px, calc(100vw - 2 * var(--gutter, 24px)));
  left: 50%;
  margin-left: calc(min(960px, calc(100vw - 2 * var(--gutter, 24px))) / -2);
  box-sizing: border-box;
}
@media (max-width: 900px) {
  .wizard-step-medium.active {
    width: calc(100vw - 40px);
    margin-left: calc((100vw - 40px) / -2);
  }
}
@keyframes fadeIn { from { opacity: 0; transform: translateY(8px); } to { opacity: 1; transform: none; } }

/* ---------- 20. Result page ---------- */
.result-hero {
  text-align: center; padding: 2.5rem 1.5rem;
  background: var(--bg-elev); border: 1px solid var(--border);
  border-radius: var(--radius-xl); margin-bottom: 2rem;
}
.result-hero .badge-grade-lg {
  margin: 0 auto 1.25rem;
  animation: scaleIn .6s cubic-bezier(.34, 1.56, .64, 1);
}
@keyframes scaleIn { from { transform: scale(.4); opacity: 0; } to { transform: scale(1); opacity: 1; } }
.score-breakdown {
  display: grid; grid-template-columns: repeat(auto-fit, minmax(160px, 1fr));
  gap: 1rem; margin-top: 1.5rem;
}
.score-item {
  background: var(--bg-soft); padding: 1rem; border-radius: var(--radius);
  text-align: center;
}
.score-item .label { font-size: .75rem; color: var(--text-muted); text-transform: uppercase; letter-spacing: .5px; }
.score-item .value { font-size: 1.5rem; font-weight: 800; color: var(--color-primary); margin-top: .25rem; }

.next-cours-box {
  background: var(--color-primary-light);
  border-left: 4px solid var(--color-primary);
  padding: 1.25rem; border-radius: var(--radius);
  margin: 1.5rem 0;
}
.next-cours-box h3 { color: var(--color-primary-dark); margin-bottom: .5rem; }

/* ---------- 21. Accordion ---------- */
.accordion-item {
  border: 1px solid var(--border); border-radius: var(--radius);
  margin-bottom: .5rem; background: var(--bg-elev); overflow: hidden;
}
.accordion-header {
  width: 100%; padding: .85rem 1rem; text-align: left;
  font-weight: 600; display: flex; justify-content: space-between; align-items: center;
  background: transparent; transition: background var(--t-fast);
}
.accordion-header:hover { background: var(--bg-soft); }
.accordion-header .chev { transition: transform var(--t); }
.accordion-item.open .accordion-header .chev { transform: rotate(180deg); }
.accordion-body {
  padding: 0 1rem; max-height: 0; overflow: hidden;
  transition: max-height var(--t), padding var(--t);
}
.accordion-item.open .accordion-body {
  padding: 1rem; max-height: 4000px;
}

/* ---------- 22. Empty state ---------- */
.empty-state {
  text-align: center; padding: 4rem 2rem;
  background: var(--bg-elev); border: 1px solid var(--border);
  border-radius: var(--radius-lg);
}
.empty-state svg { margin: 0 auto 1rem; color: var(--text-muted); opacity: .6; }
.empty-state h2 { font-size: 1.15rem; font-weight: 700; margin-bottom: .5rem; }
.empty-state p { color: var(--text-muted); max-width: 42ch; margin: 0 auto 1.5rem; }

/* ---------- 23. Responsive ---------- */
@media (max-width: 900px) {
  .module-layout { grid-template-columns: 1fr; padding: 1rem; }
  .module-sidebar {
    position: relative; top: 0; max-height: none;
    margin-bottom: 1rem;
  }
  .header-nav .nav-user { display: none; }
}
@media (max-width: 600px) {
  .container, .container-sm { padding: 1.25rem .75rem; }
  .card { padding: 1rem; }
  .timer-display { top: 8px; right: 10px; padding: .35rem .65rem; font-size: 1rem; }
  .timer-bar-inner { padding: .65rem .75rem; gap: .5rem; }
  .timer-bar-label { font-size: .75rem; }
  .timer-bar-value { font-size: 1.6rem; }
  .timer-bar-total { font-size: .8rem; }
  .wizard-steps > div { font-size: .7rem; padding: .5rem .25rem; }
  .header-nav { gap: .5rem; }
  .header-nav a { font-size: .8rem; }
}

/* ---------- 24. Print (résultat) ---------- */
@media print {
  .site-header, .btn, .accordion-header .chev { display: none !important; }
  body { background: #fff; color: #000; }
  .card { box-shadow: none; border: 1px solid var(--m-line); }
  .accordion-body { max-height: none !important; padding: 1rem !important; }
}

/* ---------- 25. Misc ---------- */
.hidden { display: none !important; }
.text-muted { color: var(--text-muted); }
.text-soft  { color: var(--text-soft); }
.text-center { text-align: center; }
.mt-1 { margin-top: .5rem; }
.mt-2 { margin-top: 1rem; }
.mt-3 { margin-top: 1.5rem; }
.mt-4 { margin-top: 2rem; }
.mb-2 { margin-bottom: 1rem; }
.gap-1 { gap: .5rem; }
.gap-2 { gap: 1rem; }
.flex-col { display: flex; flex-direction: column; }

/* ============================================================
   v2.0.7 - GRADE F + VAGUES (REMEDIATION / COMPLEMENT)
   ============================================================ */
:root {
  --grade-f-bg: #e89a92; --grade-f-fg: #5a0d0d;
}
[data-theme="dark"] {
  --grade-f-bg: #4a0a0a; --grade-f-fg: #f4a8a0;
}
.badge-grade.grade-f { background: var(--grade-f-bg); color: var(--grade-f-fg); font-weight: 800; }

/* Bandeaux d'action après contrôle */
.action-banner { border-left: 4px solid var(--color-primary); padding-left: 1.25rem; }
.action-banner-fail {
  border-left-color: var(--color-danger);
  background: var(--color-danger-soft);
}
.action-banner-fail h3 { color: var(--ds-red-900); }

.action-banner-complement {
  border-left-color: var(--color-primary);
  background: var(--color-primary-light);
}
.action-banner-complement h3 { color: var(--color-primary-dark); }

.action-banner-success {
  border-left-color: var(--color-success);
  background: var(--color-success-soft);
}
.action-banner-success h3 { color: var(--ds-green-900); }

/* Badge de vague (sidebar module.php) */
.vague-badge-remediation {
  background: var(--color-danger-soft);
  color: var(--ds-red-900);
  border: 1px solid var(--ds-red-300);
}
.vague-badge-complement {
  background: var(--color-primary-light);
  color: var(--color-primary-dark);
  border: 1px solid var(--color-primary-light);
}
.vague-badge-initial {
  background: var(--bg-soft);
  color: var(--text-muted);
  border: 1px solid var(--m-line);
}

/* Conteneur de progression injecté dynamiquement (génération de vague) */
.vague-progress {
  margin-top: 1rem;
  padding: 1rem;
  background: var(--bg-soft);
  border-radius: 8px;
  border: 1px solid rgba(0,0,0,.06);
}

/* ============================================================
   v2.0.13 - Illustrations typées et lexique
   ============================================================ */

/* Encarts illustrations - couleur de fond et bordure selon type */
.illustration {
  border-radius: 10px;
  padding: 1rem 1.25rem;
  margin: 1.25rem 0;
  border: 1px solid rgba(0,0,0,.06);
  background: var(--bg-soft);
  border-left: 4px solid var(--color-primary);
}
.illustration-header {
  display: flex;
  align-items: baseline;
  gap: .75rem;
  margin-bottom: .5rem;
  flex-wrap: wrap;
}
.illustration-tag {
  display: inline-block;
  font-size: .7rem;
  text-transform: uppercase;
  letter-spacing: .04em;
  padding: .15rem .55rem;
  border-radius: 999px;
  background: var(--color-primary);
  color: var(--ds-text-on-accent);
  font-weight: 600;
  white-space: nowrap;
}
.illustration-title {
  margin: 0;
  font-size: 1.05rem;
  font-weight: 600;
  color: var(--m-ink);
}
.illustration-body {
  font-size: .96rem;
  line-height: 1.6;
}
.illustration-body table {
  border-collapse: collapse;
  width: 100%;
  margin: .5rem 0;
}
.illustration-body th, .illustration-body td {
  border: 1px solid rgba(0,0,0,.12);
  padding: .5rem .75rem;
  text-align: left;
}
.illustration-body th {
  background: rgba(0,0,0,.04);
  font-weight: 600;
}
.illustration-explication {
  margin: .75rem 0 0 0;
  font-size: .9rem;
  color: var(--text-muted);
}

/* Variantes par type */
.illustration-code            { border-left-color: var(--m-purple); }
.illustration-code .illustration-tag { background: var(--m-purple); }

.illustration-schema_texte    { border-left-color: var(--ds-teal-700); background: color-mix(in srgb, var(--ds-teal-700) 4%, transparent); }
.illustration-schema_texte .illustration-tag { background: var(--ds-teal-700); }
.illustration-schema_texte pre { background: var(--m-bg-soft); color: var(--m-ink); padding: 1rem; border-radius: 8px; overflow-x: auto; font-family: ui-monospace, "SF Mono", Menlo, monospace; font-size: .85rem; }

.illustration-tableau         { border-left-color: var(--ds-purple-700); background: color-mix(in srgb, var(--ds-purple-700) 4%, transparent); }
.illustration-tableau .illustration-tag { background: var(--ds-purple-700); }

.illustration-cas_concret     { border-left-color: var(--ds-green-700); background: color-mix(in srgb, var(--ds-green-700) 5%, transparent); }
.illustration-cas_concret .illustration-tag { background: var(--ds-green-700); }

.illustration-analogie        { border-left-color: var(--ds-amber-700); background: color-mix(in srgb, var(--ds-amber-700) 5%, transparent); }
.illustration-analogie .illustration-tag { background: var(--ds-amber-700); }

.illustration-exemple_chiffre { border-left-color: var(--ds-teal-700); background: color-mix(in srgb, var(--ds-teal-700) 5%, transparent); }
.illustration-exemple_chiffre .illustration-tag { background: var(--ds-teal-700); }

.illustration-image_suggeree  { border-left-color: var(--ds-pink-700); background: color-mix(in srgb, var(--ds-pink-700) 4%, transparent); }
.illustration-image_suggeree .illustration-tag { background: var(--ds-pink-700); }
.illustration-image_suggeree::before {
  content: "🖼";
  display: inline-block;
  margin-right: .35rem;
  opacity: .7;
}

/* Mode sombre : ajustements */
[data-theme="dark"] .illustration { background: rgba(255,255,255,.03); border-color: rgba(255,255,255,.08); }
[data-theme="dark"] .illustration-schema_texte { background: color-mix(in srgb, var(--ds-teal-700) 8%, transparent); }
[data-theme="dark"] .illustration-tableau     { background: color-mix(in srgb, var(--ds-purple-700) 8%, transparent); }
[data-theme="dark"] .illustration-cas_concret { background: color-mix(in srgb, var(--ds-green-700) 8%, transparent); }
[data-theme="dark"] .illustration-analogie    { background: color-mix(in srgb, var(--ds-amber-700) 8%, transparent); }
[data-theme="dark"] .illustration-exemple_chiffre { background: color-mix(in srgb, var(--ds-teal-700) 8%, transparent); }
[data-theme="dark"] .illustration-image_suggeree { background: color-mix(in srgb, var(--ds-pink-700) 6%, transparent); }
[data-theme="dark"] .illustration-body th { background: rgba(255,255,255,.04); }

/* Section introduction du cours */
.course-intro {
  font-size: 1.02rem;
  line-height: 1.65;
}

/* Lexique en bas de cours
 * v3.4.36 : la grille s'applique sur .lexique-item (wrapper de chaque paire
 * terme+definition), pas sur les dt/dd directement. Auparavant <dl> en grid
 * separait dt et dd dans des cellules independantes, donc un terme pouvait
 * apparaitre cote a cote avec la definition d'un autre terme.
 */
.lexique-list {
  display: grid;
  grid-template-columns: repeat(auto-fit, minmax(280px, 1fr));
  gap: .75rem;
  margin: 1rem 0 0 0;
}
.lexique-list .lexique-item {
  display: flex;
  flex-direction: column;
  background: color-mix(in srgb, var(--ds-purple-700) 5%, transparent);
  border-left: 3px solid var(--ds-purple-700);
  border-radius: 8px;
  overflow: hidden;
}
.lexique-list .lexique-item dt {
  font-weight: 700;
  color: var(--ds-purple-900);
  margin: 0;
  padding: .65rem .85rem .25rem .85rem;
  background: color-mix(in srgb, var(--ds-purple-700) 7%, transparent);
}
.lexique-list .lexique-item dd {
  margin: 0;
  padding: 0 .85rem .65rem .85rem;
  color: var(--m-ink);
  font-size: .92rem;
  line-height: 1.55;
}
[data-theme="dark"] .lexique-list .lexique-item {
  background: color-mix(in srgb, var(--ds-purple-700) 7%, transparent);
}
[data-theme="dark"] .lexique-list .lexique-item dt {
  color: var(--ds-purple-900);
  background: color-mix(in srgb, var(--ds-purple-700) 12%, transparent);
}

/* v3.4.31 : encadre dore pour les blocs 'A retenir' en fin de section */
.callout-retenir {
  margin: 1.5rem 0 .5rem 0;
  padding: 1rem 1.25rem;
  background: linear-gradient(135deg, color-mix(in srgb, var(--ds-amber-600) 8%, transparent), color-mix(in srgb, var(--ds-amber-600) 3%, transparent));
  border: 1px solid color-mix(in srgb, var(--ds-amber-600) 25%, transparent);
  border-left: 4px solid var(--ds-amber-600);
  border-radius: 10px;
  box-shadow: 0 1px 3px color-mix(in srgb, var(--ds-amber-600) 8%, transparent);
}
.callout-retenir-title {
  display: flex;
  align-items: center;
  gap: .5rem;
  margin: 0 0 .65rem 0;
  font-size: 1.05rem;
  font-weight: 700;
  color: var(--ds-amber-900);
}
.callout-retenir-title::before {
  content: "";
  display: inline-block;
  width: 1.4rem;
  height: 1.4rem;
  background: var(--ds-amber-600);
  -webkit-mask: url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' fill='white'><path d='M12 2L9.19 8.62 2 9.27l5.46 4.73L5.82 21 12 17.27 18.18 21l-1.64-7 5.46-4.73-7.19-.65L12 2z'/></svg>") center/contain no-repeat;
          mask: url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' fill='white'><path d='M12 2L9.19 8.62 2 9.27l5.46 4.73L5.82 21 12 17.27 18.18 21l-1.64-7 5.46-4.73-7.19-.65L12 2z'/></svg>") center/contain no-repeat;
  flex-shrink: 0;
}
.callout-retenir-body {
  margin: 0;
  font-size: .96rem;
  line-height: 1.6;
}
.callout-retenir-body ul,
.callout-retenir-body ol {
  margin: .25rem 0 0 0;
  padding-left: 1.4rem;
}
.callout-retenir-body li {
  margin: .35rem 0;
}
.callout-retenir-body p:last-child { margin-bottom: 0; }
[data-theme="dark"] .callout-retenir {
  background: linear-gradient(135deg, color-mix(in srgb, var(--ds-amber-600) 12%, transparent), color-mix(in srgb, var(--ds-amber-600) 5%, transparent));
  border-color: color-mix(in srgb, var(--ds-amber-600) 35%, transparent);
}
[data-theme="dark"] .callout-retenir-title { color: var(--ds-amber-900); }

/* ============================================================
   v2.0.15 - Mode SÉQUENTIEL : sections grisées, badges, plan
   ============================================================ */

/* États de la carte de section dans le plan */
.card-locked {
  opacity: .58;
  background: var(--bg-soft, #f6f7f9);
  border-style: dashed;
  position: relative;
}
.card-locked h3 { color: var(--text-muted, #6b7280); }
.card-active {
  border-left: 4px solid var(--color-primary);
  box-shadow: 0 2px 10px rgba(0,0,0,.04);
}
.card-done {
  border-left: 4px solid var(--color-success, #16a34a);
  background: color-mix(in srgb, var(--m-success) 4%, transparent);
}

/* Pastille numérotée à gauche du titre de section */
.sec-badge {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  min-width: 2rem;
  height: 2rem;
  padding: 0 .5rem;
  margin-right: .6rem;
  border-radius: 999px;
  background: var(--color-primary);
  color: var(--ds-text-on-accent);
  font-weight: 700;
  font-size: .9rem;
  vertical-align: middle;
  box-shadow: 0 1px 3px rgba(0,0,0,.12);
}
/* v3.4.31 : palette alternee pour donner du rythme visuel aux sections */
.sec-badge--c1 { background: var(--ds-teal-800); } /* teal Mentivis */
.sec-badge--c2 { background: var(--ds-purple-700); } /* violet */
.sec-badge--c3 { background: var(--ds-amber-700); } /* orange */
.sec-badge--c4 { background: var(--ds-teal-700); } /* cyan */
.sec-badge--c5 { background: var(--ds-pink-700); } /* rose */
.card-locked .sec-badge,
.card-locked .sec-badge--c1,
.card-locked .sec-badge--c2,
.card-locked .sec-badge--c3,
.card-locked .sec-badge--c4,
.card-locked .sec-badge--c5 {
  background: var(--ds-gray-400);
  color: var(--ds-text-on-accent);
  box-shadow: none;
}
.card-done .sec-badge,
.card-done .sec-badge--c1,
.card-done .sec-badge--c2,
.card-done .sec-badge--c3,
.card-done .sec-badge--c4,
.card-done .sec-badge--c5 {
  background: var(--color-success, #16a34a);
}

/* Numéro dans la sidebar (sections-tab) */
.sec-num {
  display: inline-block;
  min-width: 1.5rem;
  padding: 0 .35rem;
  margin-right: .35rem;
  border-radius: 4px;
  background: rgba(0,0,0,.06);
  font-weight: 600;
  font-size: .8rem;
  text-align: center;
}
/* v3.4.31 : palette alternee pour les numeros de sections dans la sidebar */
.sec-num--c1 { background: color-mix(in srgb, var(--ds-teal-800) 15%, transparent); color: var(--ds-teal-900); }
.sec-num--c2 { background: color-mix(in srgb, var(--ds-purple-700) 15%, transparent); color: var(--ds-purple-900); }
.sec-num--c3 { background: color-mix(in srgb, var(--ds-amber-700) 15%, transparent); color: var(--ds-amber-900); }
.sec-num--c4 { background: color-mix(in srgb, var(--ds-teal-700) 15%, transparent); color: var(--ds-teal-900); }
.sec-num--c5 { background: color-mix(in srgb, var(--ds-pink-700) 15%, transparent); color: var(--ds-pink-900); }
[data-theme="dark"] .sec-num--c1 { background: color-mix(in srgb, var(--ds-teal-800) 28%, transparent); color: var(--ds-teal-900); }
[data-theme="dark"] .sec-num--c2 { background: color-mix(in srgb, var(--ds-purple-700) 28%, transparent); color: var(--ds-purple-900); }
[data-theme="dark"] .sec-num--c3 { background: color-mix(in srgb, var(--ds-amber-700) 28%, transparent); color: var(--ds-amber-900); }
[data-theme="dark"] .sec-num--c4 { background: color-mix(in srgb, var(--ds-teal-700) 28%, transparent); color: var(--ds-teal-900); }
[data-theme="dark"] .sec-num--c5 { background: color-mix(in srgb, var(--ds-pink-700) 28%, transparent); color: var(--ds-pink-900); }
.section-tab.locked {
  opacity: .55;
  font-style: italic;
}
.section-tab.locked .sec-num { background: rgba(0,0,0,.04); }
.section-tab.completed .sec-num {
  background: color-mix(in srgb, var(--m-success) 18%, transparent);
  color: var(--ds-green-800);
}

/* N32 (v0.14.9.146) : sidebar section en cours de generation. Pastille avec
   spinner pour signaler que la section n est plus "future" mais "active".
   Aligne le ressenti avec le bloc principal qui montre "Generation...". */
.section-tab.in-progress { color: var(--m-purple, #6366f1); font-weight: 600; }
.section-tab.in-progress .sec-num,
.sec-num--gen {
  background: color-mix(in srgb, var(--m-purple) 18%, transparent);
  color: var(--ds-purple-900);
  position: relative;
  display: inline-flex;
  align-items: center;
  justify-content: center;
}
.sec-num-spinner {
  width: 12px;
  height: 12px;
  border: 2px solid color-mix(in srgb, var(--m-purple) 30%, transparent);
  border-top-color: var(--ds-purple-900);
  border-radius: 50%;
  animation: sec-num-spin .8s linear infinite;
}
@keyframes sec-num-spin { to { transform: rotate(360deg); } }

/* N32 (v0.14.9.146) : sidebar section deja generee mais pas encore validee.
   Pastille bleue avec le numero, visuellement distincte de la section future
   (chiffre colore pastel) ET de la section validee (checkmark vert). */
.section-tab.available { color: var(--text, #1f2937); }
.section-tab.available .sec-num,
.sec-num--ready {
  background: color-mix(in srgb, var(--color-primary) 18%, transparent);
  color: var(--color-primary-dark);
  font-weight: 700;
}

[data-theme="dark"] .section-tab.in-progress .sec-num,
[data-theme="dark"] .sec-num--gen {
  background: color-mix(in srgb, var(--m-purple) 28%, transparent);
  color: var(--ds-purple-900);
}
[data-theme="dark"] .section-tab.available .sec-num,
[data-theme="dark"] .sec-num--ready {
  background: color-mix(in srgb, var(--color-primary) 28%, transparent);
  color: var(--color-primary-dark);
}

/* Badges d'état (À générer / Validée) */
.badge-locked {
  background: var(--m-bg-soft);
  color: var(--m-ink-2);
}
.badge-success {
  background: color-mix(in srgb, var(--m-success) 15%, transparent);
  color: var(--ds-green-800);
  border: 1px solid color-mix(in srgb, var(--m-success) 30%, transparent);
}

/* Bouton 'Régénérer ce module' dans la sidebar */
#regenModuleBtn { margin-top: 1rem; }

/* Texte adouci sous le titre de section */
.text-soft {
  color: var(--text-muted, #6b7280);
  font-size: .95rem;
  line-height: 1.55;
}

/* Distinction visuelle remédiation locale */
.card.remediation-locale {
  border-left: 4px dashed var(--color-warning, #f59e0b);
  background: color-mix(in srgb, var(--ds-amber-600) 5%, transparent);
}

/* Mode sombre */
[data-theme="dark"] .card-locked {
  background: rgba(255,255,255,.03);
}
[data-theme="dark"] .card-done {
  background: color-mix(in srgb, var(--m-success) 8%, transparent);
}
[data-theme="dark"] .badge-locked {
  background: rgba(255,255,255,.08);
  color: var(--m-ink-2);
}
[data-theme="dark"] .badge-success {
  background: color-mix(in srgb, var(--m-success) 18%, transparent);
  color: var(--ds-green-800);
  border-color: color-mix(in srgb, var(--m-success) 40%, transparent);
}
[data-theme="dark"] .sec-num { background: rgba(255,255,255,.06); }
[data-theme="dark"] .section-tab.completed .sec-num {
  background: color-mix(in srgb, var(--m-success) 22%, transparent);
  color: var(--ds-green-800);
}

/* v2.0.19.2 : synthèse globale dépliable (parcours.php) */
.synthese-globale {
  padding: 1rem 1.25rem;
}
.synthese-globale > summary {
  list-style: none;
  user-select: none;
  color: var(--text);
  padding: .25rem 0;
  outline: none;
}
.synthese-globale > summary::-webkit-details-marker { display: none; }
.synthese-globale > summary svg {
  transition: transform .2s ease;
  color: var(--color-primary);
}
.synthese-globale[open] > summary svg { transform: rotate(90deg); }
.synthese-globale > summary:hover { color: var(--color-primary); }

/* v2.0.19.3 : zone de génération en cours observée (au retour sur la page,
   quand une section est encore 'en_generation' côté BDD sans génération active). */
.sp-gen-pending {
  background: var(--ds-blue-100);
  border: 1px solid var(--ds-blue-300);
  border-left: 4px solid var(--ds-blue-600);
  border-radius: 8px;
  padding: 1rem 1.25rem;
}
.sp-gen-pending .sp-gen-pending-head {
  display: flex;
  align-items: center;
  gap: .6rem;
  font-weight: 600;
  color: var(--color-primary-dark);
  margin-bottom: .35rem;
}
.sp-gen-pending .sp-gen-pending-head svg {
  animation: spin 1.4s linear infinite;
  color: var(--ds-blue-600);
  flex-shrink: 0;
}
.sp-gen-pending .sp-gen-pending-msg {
  font-size: .9rem;
  color: var(--text-soft);
  margin: 0 0 .65rem 0;
}
.sp-gen-pending .sp-gen-pending-bar {
  position: relative;
  width: 100%;
  height: 8px;
  background: var(--ds-blue-200);
  border-radius: 999px;
  overflow: hidden;
}
.sp-gen-pending .sp-gen-pending-bar::before {
  content: '';
  position: absolute;
  inset: 0;
  width: 35%;
  background: linear-gradient(90deg, var(--ds-blue-600), var(--ds-blue-500));
  border-radius: 999px;
  animation: sp-indeterminate 1.6s ease-in-out infinite;
}
.sp-gen-pending .sp-gen-pending-foot {
  display: flex;
  justify-content: space-between;
  align-items: center;
  margin-top: .65rem;
  gap: .75rem;
  flex-wrap: wrap;
}
.sp-gen-pending .sp-gen-pending-chrono {
  font-size: .8rem;
  color: var(--text-muted);
  font-variant-numeric: tabular-nums;
}
.sp-gen-pending .sp-gen-pending-cancel {
  background: transparent;
  border: 1px solid var(--m-line);
  color: var(--text-soft);
  padding: .35rem .75rem;
  border-radius: 6px;
  font-size: .8rem;
  cursor: pointer;
  transition: all .15s;
}
.sp-gen-pending .sp-gen-pending-cancel:hover {
  background: var(--m-danger-soft);
  border-color: var(--ds-red-300);
  color: var(--ds-red-900);
}
@keyframes sp-indeterminate {
  0%   { left: -35%; }
  100% { left: 100%; }
}

/* v3.4.27 : suppression du CSS .sp-cascade-banner (cascade arrière-plan retirée) */

/* ============================================================
   v2.0.20 - Modale régénération motivée
   ============================================================ */
.sp-modal {
  position: fixed;
  inset: 0;
  z-index: 9100;
  display: none;
  align-items: center;
  justify-content: center;
  padding: 1rem;
}
.sp-modal[style*="flex"] { display: flex !important; }
.sp-modal-backdrop {
  position: absolute;
  inset: 0;
  background: color-mix(in srgb, var(--m-ink) 55%, transparent);
  backdrop-filter: blur(2px);
  cursor: pointer;
}
.sp-modal-card {
  position: relative;
  background: var(--bg-elev, #ffffff);
  border: 1px solid var(--border, #dcd9d5);
  border-radius: var(--radius-lg, .75rem);
  box-shadow: 0 24px 60px rgba(0, 0, 0, .25), 0 8px 16px rgba(0, 0, 0, .12);
  width: min(34rem, 100%);
  max-height: calc(100vh - 2rem);
  display: flex;
  flex-direction: column;
  animation: sp-modal-in .2s ease-out;
}
@keyframes sp-modal-in {
  from { opacity: 0; transform: scale(.96) translateY(.5rem); }
  to   { opacity: 1; transform: scale(1) translateY(0); }
}
.sp-modal-header {
  display: flex;
  align-items: center;
  justify-content: space-between;
  padding: 1rem 1.25rem;
  border-bottom: 1px solid var(--border-soft, #e8e6e1);
}
.sp-modal-title {
  margin: 0;
  font-size: 1.1rem;
  font-weight: 700;
  color: var(--text, #28251d);
}
.sp-modal-close {
  background: transparent;
  border: none;
  color: var(--text-muted, #7a7974);
  cursor: pointer;
  padding: .25rem;
  border-radius: 6px;
  display: flex;
  align-items: center;
  justify-content: center;
  transition: all .15s;
}
.sp-modal-close:hover {
  background: var(--bg-soft, #f3f0ec);
  color: var(--text, #28251d);
}
.sp-modal-body {
  padding: 1.1rem 1.25rem;
  overflow-y: auto;
  flex: 1 1 auto;
}
.sp-modal-body > p { margin: 0 0 1rem 0; }
.sp-modal-footer {
  display: flex;
  justify-content: flex-end;
  gap: .6rem;
  padding: .9rem 1.25rem;
  border-top: 1px solid var(--border-soft, #e8e6e1);
  background: var(--bg-soft, #f3f0ec);
  border-radius: 0 0 var(--radius-lg, .75rem) var(--radius-lg, .75rem);
}

/* v0.14.9.139 (N31b) : fieldset choix mode regeneration (conserver / zero). */
.sp-regen-mode {
  margin: 0 0 1rem 0;
  padding: .8rem 1rem .9rem;
  border: 1px solid var(--border-soft, #e8e6e1);
  border-radius: var(--radius, .5rem);
  background: var(--bg, #f7f6f2);
}
.sp-regen-radio {
  display: flex;
  align-items: flex-start;
  gap: .55rem;
  padding: .35rem .15rem;
  font-size: .9rem;
  line-height: 1.35;
  color: var(--text, #28251d);
  cursor: pointer;
  border-radius: 4px;
  transition: background .12s;
}
.sp-regen-radio:hover {
  background: color-mix(in srgb, var(--color-primary) 4%, transparent);
}
.sp-regen-radio input[type="radio"] {
  margin-top: .2rem;
  flex: 0 0 auto;
  accent-color: var(--color-primary, #01696f);
}
.sp-regen-reasons {
  margin: 0 0 1rem 0;
  padding: .8rem 1rem .9rem;
  border: 1px solid var(--border-soft, #e8e6e1);
  border-radius: var(--radius, .5rem);
  background: var(--bg, #f7f6f2);
}
.sp-regen-legend {
  font-size: .8rem;
  font-weight: 700;
  color: var(--text-soft, #5a5852);
  padding: 0 .35rem;
}
.sp-regen-check {
  display: flex;
  align-items: flex-start;
  gap: .55rem;
  padding: .35rem .15rem;
  font-size: .9rem;
  line-height: 1.35;
  color: var(--text, #28251d);
  cursor: pointer;
  border-radius: 4px;
  transition: background .12s;
}
.sp-regen-check:hover {
  background: color-mix(in srgb, var(--color-primary) 4%, transparent);
}
.sp-regen-check input[type="checkbox"] {
  margin-top: .2rem;
  flex: 0 0 auto;
  accent-color: var(--color-primary, #01696f);
}
.sp-regen-textarea-label {
  display: flex;
  flex-direction: column;
  gap: .35rem;
  font-size: .85rem;
  color: var(--text-soft, #5a5852);
  font-weight: 600;
}
.sp-regen-textarea-label textarea {
  width: 100%;
  border: 1px solid var(--border, #dcd9d5);
  border-radius: var(--radius, .5rem);
  padding: .6rem .75rem;
  font-family: inherit;
  font-size: .9rem;
  line-height: 1.45;
  color: var(--text, #28251d);
  background: var(--bg-elev, #fff);
  resize: vertical;
  min-height: 90px;
  transition: border-color .15s, box-shadow .15s;
}
.sp-regen-textarea-label textarea:focus {
  outline: none;
  border-color: var(--color-primary, #01696f);
  box-shadow: 0 0 0 3px color-mix(in srgb, var(--color-primary) 15%, transparent);
}

/* Responsive : mobile */
@media (max-width: 600px) {
  /* v3.4.27 : .sp-cascade-banner retiré */
  .sp-modal { padding: .5rem; }
  .sp-modal-card { max-height: calc(100vh - 1rem); }
}

/* ============================================================
   v3.4.0 - Documents enrichis du parcours (widget)
   ============================================================ */
.mvs-docs-drop {
  border: 2px dashed var(--border, #dcd9d5);
  border-radius: var(--radius, .5rem);
  padding: 1.5rem 1rem;
  text-align: center;
  background: var(--bg-soft, #faf8f4);
  cursor: pointer;
  transition: background .15s, border-color .15s;
}
.mvs-docs-drop:hover,
.mvs-docs-drop:focus {
  border-color: var(--color-primary, #01696f);
  background: var(--bg, #fff);
  outline: none;
}
.mvs-docs-drop.is-dragover {
  border-color: var(--color-primary, #01696f);
  background: var(--color-primary-light, #e6f4f5);
}
.mvs-docs-drop-inner { display: flex; flex-direction: column; align-items: center; gap: .5rem; color: var(--text, #28251d); }
.mvs-docs-drop-inner svg { color: var(--color-primary, #01696f); }
.mvs-docs-drop-title { font-weight: 600; margin: 0; }
.mvs-docs-drop-hint { font-size: .85rem; color: var(--text-muted, #6b6660); margin: 0; }
.mvs-docs-browse-btn {
  background: none; border: none; padding: 0;
  color: var(--color-primary, #01696f); text-decoration: underline; cursor: pointer; font: inherit;
}

.mvs-docs-list { margin-top: 1rem; display: flex; flex-direction: column; gap: .75rem; }
.mvs-docs-list-title { font-weight: 600; margin: 0 0 .25rem; font-size: .9rem; color: var(--text, #28251d); }

.mvs-doc-card {
  border: 1px solid var(--border, #dcd9d5);
  border-radius: var(--radius, .5rem);
  padding: .75rem 1rem;
  background: var(--bg-elev, #fff);
  display: flex; flex-direction: column; gap: .4rem;
}
.mvs-doc-card-pending { opacity: .7; }
.mvs-doc-card-head { display: flex; justify-content: space-between; align-items: center; gap: .5rem; flex-wrap: wrap; }
.mvs-doc-card-name { font-weight: 600; word-break: break-word; }
.mvs-doc-card-meta { font-size: .85rem; color: var(--text-muted, #6b6660); }
.mvs-doc-card-ctx { font-size: .85rem; color: var(--text-muted, #6b6660); }
.mvs-doc-card-ctx p { margin: .1rem 0; }
.mvs-doc-card-actions { display: flex; gap: .5rem; justify-content: flex-end; margin-top: .25rem; }

.mvs-doc-status {
  font-size: .78rem; padding: .15rem .55rem; border-radius: 9999px; font-weight: 600;
  white-space: nowrap;
}
.mvs-doc-status-ok      { color: var(--color-success, #2f7a4a); background: var(--color-success-soft, #e2f3e9); }
.mvs-doc-status-pending { color: var(--color-warning, #a37214); background: var(--color-warning-soft, #fdf2dd); }
.mvs-doc-status-fail    { color: var(--color-danger,  #b13a3a); background: var(--color-danger-soft,  #fbe6e6); }

/* v3.4.9 : recap pre-generation du programme (etape 3 du wizard nouveau-parcours) */
.mvs-recap {
  background: var(--m-bg-soft);
  border: 1px solid var(--m-line);
  border-radius: 12px;
  padding: 1rem 1.1rem;
  margin-bottom: 1.25rem;
}
.mvs-recap-block {
  margin-bottom: 1rem;
}
.mvs-recap-block:last-of-type {
  margin-bottom: .5rem;
}
.mvs-recap-title {
  font-size: .95rem;
  font-weight: 700;
  margin: 0 0 .5rem;
  color: var(--m-ink);
  border-bottom: 1px solid var(--m-line);
  padding-bottom: .35rem;
}
.mvs-recap-line {
  margin: .25rem 0;
  font-size: .9rem;
  line-height: 1.45;
}
.mvs-recap-list {
  margin: .25rem 0 .25rem 1.1rem;
  padding: 0;
  font-size: .9rem;
}
.mvs-recap-list li {
  margin: .2rem 0;
  line-height: 1.45;
}
.mvs-recap-docs {
  list-style: none;
  margin: .25rem 0;
  padding: 0;
  display: flex;
  flex-direction: column;
  gap: .6rem;
}
.mvs-recap-doc {
  background: var(--ds-bg-100);
  border: 1px solid var(--m-line);
  border-radius: 8px;
  padding: .6rem .75rem;
  font-size: .88rem;
  line-height: 1.45;
}
.mvs-recap-doc-head {
  display: flex;
  justify-content: space-between;
  align-items: center;
  gap: .5rem;
  flex-wrap: wrap;
  margin-bottom: .25rem;
}
.mvs-recap-doc-role {
  color: var(--m-ink);
  font-weight: 600;
  font-size: .85rem;
}
.mvs-recap-doc-help {
  color: var(--text-muted, #6b6660);
  font-size: .82rem;
  margin-bottom: .15rem;
}
.mvs-recap-doc-ctx {
  margin-top: .25rem;
  font-size: .85rem;
  color: var(--m-ink-2);
}
.mvs-recap-note {
  margin: .5rem 0 0;
  font-size: .85rem;
  color: var(--text-muted, #6b6660);
  font-style: italic;
  border-top: 1px dashed var(--m-line);
  padding-top: .5rem;
}

/* v3.4.8 : barre de progression de distillation IA (card temporaire pendant l'upload) */
.mvs-doc-progress-wrap {
  display: flex; align-items: center; gap: .6rem; margin-top: .35rem;
}
.mvs-doc-progress-bar {
  flex: 1 1 auto; height: 6px; background: var(--color-warning-soft, #fdf2dd);
  border-radius: 9999px; overflow: hidden; position: relative;
}
.mvs-doc-progress-fill {
  height: 100%; width: 0%;
  background: var(--color-warning, #a37214);
  border-radius: 9999px;
  transition: width .35s ease-out;
}
.mvs-doc-progress-label {
  font-size: .75rem; font-variant-numeric: tabular-nums;
  color: var(--color-warning, #a37214); font-weight: 600;
  min-width: 2.5rem; text-align: right;
}

/* Modale documents */
.mvs-modal-overlay {
  position: fixed; inset: 0; background: rgba(0,0,0,.45);
  display: flex; align-items: center; justify-content: center;
  z-index: 1000; padding: 1rem;
}
.mvs-modal {
  background: var(--bg-elev, #fff);
  border-radius: var(--radius, .5rem);
  padding: 1.25rem;
  width: 100%; max-width: 540px;
  max-height: calc(100vh - 2rem); overflow: auto;
  box-shadow: 0 10px 40px rgba(0,0,0,.25);
}
.mvs-modal-head { font-weight: 700; font-size: 1.05rem; margin-bottom: 1rem; word-break: break-word; }
.mvs-modal-label { display: block; font-weight: 600; font-size: .9rem; margin: .75rem 0 .35rem; color: var(--text, #28251d); }
.mvs-modal-actions { display: flex; gap: .5rem; justify-content: flex-end; margin-top: 1.25rem; }

@media (max-width: 600px) {
  .mvs-modal { padding: 1rem; }
  .mvs-modal-actions { flex-direction: column-reverse; }
  .mvs-modal-actions .btn { width: 100%; }
}

/* ============================================================
   MERMAID - Schemas pedagogiques (v3.4.18)
   ============================================================ */
.mermaid-block {
  position: relative;
  margin: 1.5rem 0;
  padding: 1.25rem 1rem 1rem;
  background: var(--m-bg-soft);
  border: 1px solid var(--m-line);
  border-radius: 12px;
  overflow: hidden;
}
.mermaid-block .mermaid {
  display: flex;
  justify-content: center;
  align-items: center;
  min-height: 80px;
}
.mermaid-block .mermaid svg {
  max-width: 100%;
  height: auto;
}
.mermaid-actions {
  position: absolute;
  top: .5rem;
  right: .5rem;
  display: flex;
  gap: .25rem;
  opacity: 0;
  transition: opacity .15s ease;
}
.mermaid-block:hover .mermaid-actions,
.mermaid-block:focus-within .mermaid-actions {
  opacity: 1;
}
.mermaid-fullscreen-btn {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 32px;
  height: 32px;
  padding: 0;
  background: var(--m-bg);
  color: var(--m-ink-2);
  border: 1px solid var(--m-line);
  border-radius: 8px;
  cursor: pointer;
  transition: background .15s ease, color .15s ease, border-color .15s ease;
}
.mermaid-fullscreen-btn:hover {
  background: var(--m-coral-soft);
  color: var(--m-coral);
  border-color: var(--m-coral);
}
.mermaid-error {
  padding: .75rem 1rem;
  background: var(--m-danger-soft);
  color: var(--m-danger);
  border-radius: 8px;
  font-size: .9rem;
  font-family: ui-monospace, "SF Mono", Menlo, monospace;
}

/* Overlay fullscreen */
.mermaid-fullscreen {
  position: fixed;
  inset: 0;
  background: rgba(0, 0, 0, 0.78);
  display: flex;
  align-items: center;
  justify-content: center;
  z-index: 9999;
  padding: 2rem;
  animation: mermaidFadeIn .18s ease;
}
.mermaid-fullscreen-inner {
  width: 100%;
  height: 100%;
  display: flex;
  align-items: center;
  justify-content: center;
  background: var(--m-bg);
  border-radius: 12px;
  padding: 2rem;
  box-shadow: 0 20px 80px rgba(0, 0, 0, 0.4);
  overflow: auto;
}
.mermaid-fullscreen-inner svg {
  max-width: 100%;
  max-height: 100%;
}
.mermaid-fullscreen-close {
  position: absolute;
  top: 1rem;
  right: 1rem;
  width: 40px;
  height: 40px;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  background: var(--m-bg);
  color: var(--m-ink);
  border: 1px solid var(--m-line);
  border-radius: 50%;
  cursor: pointer;
  transition: background .15s ease;
}
.mermaid-fullscreen-close:hover {
  background: var(--m-coral-soft);
  color: var(--m-coral);
}
@keyframes mermaidFadeIn {
  from { opacity: 0; }
  to   { opacity: 1; }
}

/* Demo admin : grille code / rendu */
.mermaid-demo-grid {
  display: grid;
  grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
  gap: 1rem;
  margin: 1rem 0 2rem;
}
.mermaid-demo-grid pre {
  margin: 0;
  padding: 1rem;
  background: var(--m-bg-warm);
  border: 1px solid var(--m-line);
  border-radius: 12px;
  overflow: auto;
  font-size: .8rem;
  font-family: ui-monospace, "SF Mono", Menlo, monospace;
  max-height: 360px;
}
.mermaid-demo-grid .mermaid-block {
  margin: 0;
}
@media (max-width: 900px) {
  .mermaid-demo-grid { grid-template-columns: 1fr; }
}

/* ============================================================
   KaTeX - Formules mathematiques (v3.4.21)
   ============================================================ */
/* KaTeX inline : ajuster legerement la taille pour s'aligner avec le texte courant */
.katex { font-size: 1.05em; }
/* KaTeX display (centre, sur sa propre ligne) : scroll horizontal sur mobile */
.katex-display {
  margin: 1rem 0;
  padding: .5rem 0;
  overflow-x: auto;
  overflow-y: hidden;
}
.katex-display > .katex {
  white-space: nowrap;
  font-size: 1.15em;
}
/* En mode sombre : KaTeX prend automatiquement la couleur du texte parent */
[data-theme="dark"] .katex { color: var(--m-ink); }

/* ============================================================
   Chart.js - Graphiques de donnees (v3.4.21)
   ============================================================ */
.chart-block {
  position: relative;
  margin: 1.5rem 0;
  padding: 1.25rem 1rem 1rem;
  background: var(--m-bg-soft);
  border: 1px solid var(--m-line);
  border-radius: 12px;
  overflow: hidden;
}
.chart-canvas-wrap {
  position: relative;
  width: 100%;
  height: 340px;
}
.chart-canvas-wrap canvas {
  max-width: 100%;
  height: 100% !important;
}
.chart-actions {
  position: absolute;
  top: .5rem;
  right: .5rem;
  display: flex;
  gap: .25rem;
  opacity: 0;
  transition: opacity .15s ease;
  z-index: 2;
}
.chart-block:hover .chart-actions,
.chart-block:focus-within .chart-actions {
  opacity: 1;
}
.chart-fullscreen-btn {
  display: inline-flex;
  align-items: center;
  justify-content: center;
  width: 32px;
  height: 32px;
  padding: 0;
  background: var(--m-bg);
  color: var(--m-ink-2);
  border: 1px solid var(--m-line);
  border-radius: 8px;
  cursor: pointer;
  transition: background .15s ease, color .15s ease, border-color .15s ease;
}
.chart-fullscreen-btn:hover {
  background: var(--m-coral-soft);
  color: var(--m-coral);
  border-color: var(--m-coral);
}
.chart-error {
  padding: .75rem 1rem;
  margin: 1rem 0;
  background: var(--m-danger-soft);
  color: var(--m-danger);
  border-radius: 8px;
  font-size: .9rem;
  font-family: ui-monospace, "SF Mono", Menlo, monospace;
}
/* Overlay fullscreen chart */
.chart-fullscreen {
  position: fixed;
  inset: 0;
  background: rgba(0, 0, 0, 0.78);
  display: flex;
  align-items: center;
  justify-content: center;
  z-index: 9999;
  padding: 2rem;
  animation: chartFadeIn .18s ease;
}
.chart-fullscreen-inner {
  width: 100%;
  height: 100%;
  background: var(--m-bg);
  border-radius: 12px;
  padding: 2rem;
  box-shadow: 0 20px 80px rgba(0, 0, 0, 0.4);
  position: relative;
}
.chart-fullscreen-inner canvas {
  width: 100% !important;
  height: 100% !important;
}
.chart-fullscreen-close {
  position: absolute;
  top: 1rem;
  right: 1rem;
  width: 40px;
  height: 40px;
  display: inline-flex;
  align-items: center;
  justify-content: center;
  background: var(--m-bg);
  color: var(--m-ink);
  border: 1px solid var(--m-line);
  border-radius: 50%;
  cursor: pointer;
}
.chart-fullscreen-close:hover {
  background: var(--m-coral-soft);
  color: var(--m-coral);
}
@keyframes chartFadeIn {
  from { opacity: 0; }
  to   { opacity: 1; }
}

/* Grille demo (formules / graphiques) */
.demo-grid {
  display: grid;
  grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
  gap: 1rem;
  margin: 1rem 0 2rem;
  align-items: start;
}
.demo-grid pre {
  margin: 0;
  padding: 1rem;
  background: var(--m-bg-warm);
  border: 1px solid var(--m-line);
  border-radius: 12px;
  overflow: auto;
  font-size: .8rem;
  font-family: ui-monospace, "SF Mono", Menlo, monospace;
  max-height: 360px;
}
.demo-grid .chart-block,
.demo-grid .formula-render {
  margin: 0;
}
.formula-render {
  padding: 1.25rem 1.25rem;
  background: var(--m-bg-soft);
  border: 1px solid var(--m-line);
  border-radius: 12px;
  min-height: 80px;
  line-height: 1.7;
  /* Pas de flex : on veut un flux texte normal pour que les formules
     inline ($...$) restent dans la ligne et que $$...$$ se centre via
     son .katex-display natif. */
}
.formula-render p { margin: 0 0 .5rem; }
.formula-render p:last-child { margin-bottom: 0; }
@media (max-width: 900px) {
  .demo-grid { grid-template-columns: 1fr; }
}

/* ============================================================
   v3.4.28 - Widget d'évaluation (section / module / parcours)
   ============================================================ */
.mvs-eval-card {
  background: var(--bg-elev, #ffffff);
  border: 1px solid var(--border, #dcd9d5);
  border-left: 4px solid var(--color-primary, #01696f);
  border-radius: var(--radius-lg, .75rem);
  padding: 1rem 1.1rem;
  margin-top: 1rem;
  box-shadow: 0 1px 2px rgba(0,0,0,.04);
}
.mvs-eval-header {
  display: flex;
  flex-wrap: wrap;
  align-items: center;
  gap: .5rem;
  margin-bottom: .75rem;
}
.mvs-eval-title {
  margin: 0;
  font-size: 1rem;
  font-weight: 700;
  color: var(--text, #28251d);
}
.mvs-eval-sub {
  width: 100%;
  margin: .15rem 0 0;
  font-size: .85rem;
  color: var(--text-muted, #7a7974);
}
.mvs-eval-optional {
  margin-left: auto;
  font-size: .7rem;
  font-weight: 600;
  text-transform: uppercase;
  letter-spacing: .03em;
  color: var(--text-muted, #7a7974);
  background: var(--bg-soft, #f3f0ec);
  padding: .15rem .5rem;
  border-radius: 999px;
}
.mvs-eval-row {
  display: flex;
  flex-wrap: wrap;
  align-items: center;
  gap: .6rem;
  margin-top: .55rem;
}
.mvs-eval-row-comment {
  flex-direction: column;
  align-items: stretch;
  gap: .3rem;
}
.mvs-eval-label {
  font-size: .85rem;
  font-weight: 600;
  color: var(--text, #28251d);
  margin: 0;
}
.mvs-eval-stars {
  display: inline-flex;
  gap: .15rem;
}
.mvs-eval-star {
  background: transparent;
  border: 0;
  padding: .15rem;
  cursor: pointer;
  color: var(--border, #dcd9d5);
  transition: color .15s, transform .1s;
  line-height: 0;
}
.mvs-eval-star:hover,
.mvs-eval-star.hover {
  color: var(--ds-amber-600);
  transform: scale(1.08);
}
.mvs-eval-star[data-active="1"] {
  color: var(--ds-amber-600);
}
.mvs-eval-star:focus-visible {
  outline: 2px solid var(--color-primary, #01696f);
  outline-offset: 2px;
  border-radius: 4px;
}
.mvs-eval-status {
  font-size: .8rem;
  color: var(--text-muted, #7a7974);
  margin-left: auto;
  min-height: 1.1em;
  transition: color .2s;
}
.mvs-eval-status[data-kind="ok"]    { color: var(--ds-green-700); }
.mvs-eval-status[data-kind="saving"]{ color: var(--text-muted, #7a7974); font-style: italic; }
.mvs-eval-status[data-kind="error"] { color: var(--m-danger); }
.mvs-eval-yesno {
  display: inline-flex;
  gap: .35rem;
}
.mvs-eval-yesno-btn {
  background: var(--bg-soft, #f3f0ec);
  border: 1px solid var(--border, #dcd9d5);
  color: var(--text, #28251d);
  font-size: .85rem;
  font-weight: 600;
  padding: .35rem .9rem;
  border-radius: 6px;
  cursor: pointer;
  transition: all .15s;
}
.mvs-eval-yesno-btn:hover {
  border-color: var(--color-primary, #01696f);
}
.mvs-eval-yesno-btn[data-active="1"] {
  background: var(--color-primary, #01696f);
  color: var(--ds-text-on-accent);
  border-color: var(--color-primary, #01696f);
}
.mvs-eval-textarea {
  width: 100%;
  min-height: 2.5rem;
  border: 1px solid var(--border, #dcd9d5);
  border-radius: 6px;
  padding: .5rem .65rem;
  font-family: inherit;
  font-size: .9rem;
  color: var(--text, #28251d);
  background: var(--bg, #ffffff);
  resize: vertical;
  transition: border-color .15s, box-shadow .15s;
}
.mvs-eval-textarea:focus {
  outline: none;
  border-color: var(--color-primary, #01696f);
  box-shadow: 0 0 0 3px color-mix(in srgb, var(--color-primary) 12%, transparent);
}
.mvs-eval-row-actions {
  display: flex;
  justify-content: flex-end;
  margin-top: .25rem;
}
.mvs-eval-save-btn {
  background: var(--color-primary, #01696f);
  color: var(--ds-text-on-accent);
  border: 1px solid var(--color-primary, #01696f);
  border-radius: 6px;
  padding: .45rem .9rem;
  font-size: .9rem;
  font-weight: 600;
  cursor: pointer;
  transition: background .15s, transform .1s, opacity .15s;
}
.mvs-eval-save-btn:hover:not(:disabled) {
  background: var(--color-primary-dark);
  transform: translateY(-1px);
}
.mvs-eval-save-btn:disabled {
  opacity: .5;
  cursor: not-allowed;
}
@media (max-width: 600px) {
  .mvs-eval-card { padding: .85rem .9rem; }
  .mvs-eval-status { margin-left: 0; width: 100%; }
  .mvs-eval-row-actions { justify-content: stretch; }
  .mvs-eval-save-btn { width: 100%; }
}

/* ============================================================
 * v3.4.40 - Surlignages + annotations (polish visuel)
 * ============================================================ */

/* Surlignages : 5 couleurs claires, lisibles sur fond blanc */
.mvs-hl {
  border-radius: 2px;
  padding: 0 1px;
  box-decoration-break: clone;
  -webkit-box-decoration-break: clone;
  cursor: context-menu;
}
/* Fonds de surlignage : versions pastel tres pales des 7 couleurs
   (rouge, bleu, vert, jaune, orange, rose, noir/gris). v0.14.9.124 (N3a) :
   adoucies vers les teintes Tailwind 100/50 pour ne pas casser le contraste
   du texte ni des blocs de fond colore (ex. .callout-retenir dore). */
.mvs-hl-rouge  { background-color: #fee2e2; }
.mvs-hl-bleu   { background-color: #dbeafe; }
.mvs-hl-vert   { background-color: #dcfce7; }
.mvs-hl-jaune  { background-color: #fef9c3; }
.mvs-hl-orange { background-color: #ffedd5; }
.mvs-hl-rose   { background-color: #fce7f3; }
.mvs-hl-noir   { background-color: #e5e7eb; }
/* Regle defensive : sur le bloc "A retenir" (fond dore rgba(245,158,11,.08),
   cf a-retenir.css), un surlignage jaune devient jaune-sur-jaune illisible.
   On bascule alors le jaune vers un rose pale neutre qui reste lisible. */
.callout-retenir .mvs-hl-jaune { background-color: #fce7f3; }

/* ----------------------------------------------------------------------
   N22 (v0.14.9.130) : bulle ronde de couleur de fond, presente sur toutes
   les sections. Remplace le bouton rond degrade N3b (.mvs-hl-launcher, retire
   car il empietait sur le chevron de collapse et le badge de statut). La bulle
   ne contient rien (pas de croix, pas d icone) : sa couleur de remplissage
   suit la couleur de fond appliquee a la section. Au clic, un menu de pastilles
   pastel (la premiere etant Blanc) permet de changer ce fond.
   ---------------------------------------------------------------------- */
.cours-section { position: relative; }
/* Reserve un peu d espace a droite du header pour que la pastille de statut
   et le chevron ne passent pas sous la bulle (petite, dans le coin).
   N27 (v0.14.9.134) : bulle reduite a 18px, on peut donc reduire la reserve. */
.cours-section > .cours-section-header { padding-right: 2rem; }

/* Couleur de fond appliquee a la section via la bulle (pilotee par
   data-bg-couleur, pose par annotations.js). Pastels clairs uniquement. La
   couleur cohabite avec le halo box-shadow N3c (effet temporaire distinct).
   N27 (v0.14.9.134) : ajout d une bordure legerement plus foncee autour de
   la section quand une couleur de fond est appliquee, pour mieux distinguer
   la zone mise en evidence (reprend le comportement d avant N22, ou la croix
   avait une bordure coloree). La couleur de bordure est derivee du pastel. */
.cours-section[data-bg-couleur="jaune"]  { background-color: #fef3c7; border: 1px solid #fbbf24; }
.cours-section[data-bg-couleur="vert"]   { background-color: #d1fae5; border: 1px solid #34d399; }
.cours-section[data-bg-couleur="bleu"]   { background-color: #dbeafe; border: 1px solid #60a5fa; }
.cours-section[data-bg-couleur="rose"]   { background-color: #fce7f3; border: 1px solid #f472b6; }
.cours-section[data-bg-couleur="mauve"]  { background-color: #ede9fe; border: 1px solid #a78bfa; }
.cours-section[data-bg-couleur="peche"]  { background-color: #fed7aa; border: 1px solid #fb923c; }
.cours-section[data-bg-couleur="grisbleu"] { background-color: #e0f2fe; border: 1px solid #7dd3fc; }

/* La bulle elle-meme. N27 (v0.14.9.134) : taille reduite a 18px (au lieu de
   24px) pour reprendre l empreinte visuelle de l ancienne croix dismiss. La
   couleur de remplissage --bg-bulle suit la couleur de bordure de la section
   (pas le fond pastel) afin que la bulle se voie sur le fond pastel.
   --bg-bordure est mise a jour par annotations.js en meme temps. */
.mvs-section-bg-bubble {
  position: absolute;
  top: 10px;
  right: 12px;
  z-index: 5;
  width: 18px;
  height: 18px;
  padding: 0;
  border: 1px solid var(--bg-bordure, #cbd5e1);
  border-radius: 50%;
  background-color: var(--bg-bulle, #ffffff);
  cursor: pointer;
  box-shadow: 0 1px 2px rgba(0, 0, 0, .14);
  transition: transform .12s ease, box-shadow .12s ease;
}
.mvs-section-bg-bubble:hover {
  transform: scale(1.12);
  box-shadow: 0 2px 8px rgba(0, 0, 0, .22);
}
.mvs-section-bg-bubble:focus-visible {
  outline: 2px solid var(--m-purple);
  outline-offset: 2px;
}

/* Menu de la bulle : popover blanc, liste horizontale de pastilles 22px. */
.mvs-section-bg-menu {
  position: fixed;
  z-index: 10001;
  background: var(--ds-bg-100);
  border: 1px solid var(--m-line);
  border-radius: 12px;
  box-shadow: 0 10px 28px rgba(0, 0, 0, .2);
  padding: 12px;
}
.mvs-section-bg-menu-title {
  font-size: 12px;
  font-weight: 600;
  text-transform: uppercase;
  letter-spacing: .04em;
  color: var(--m-ink-3);
  margin: 0 0 8px 2px;
}
.mvs-section-bg-grid {
  display: flex;
  flex-wrap: wrap;
  gap: 10px;
  max-width: 220px;
}
.mvs-section-bg-swatch {
  width: 22px;
  height: 22px;
  border-radius: 50%;
  border: 2px solid rgba(0, 0, 0, .08);
  padding: 0;
  cursor: pointer;
  transition: transform .1s ease, box-shadow .12s ease;
}
/* La pastille Blanc (retirer la couleur) : bordure plus marquee pour la
   distinguer du fond blanc du menu. */
.mvs-section-bg-swatch--blanc {
  background-color: #ffffff;
  border-color: #94a3b8;
}
.mvs-section-bg-swatch:hover {
  transform: scale(1.12);
  box-shadow: 0 2px 8px rgba(0, 0, 0, .18);
}
.mvs-section-bg-swatch:focus-visible {
  outline: 2px solid var(--m-purple);
  outline-offset: 2px;
}
.mvs-section-bg-swatch[aria-checked="true"] {
  box-shadow: 0 0 0 2px var(--m-purple);
}
@media (prefers-reduced-motion: reduce) {
  .cours-section,
  .mvs-section-bg-bubble,
  .mvs-section-bg-swatch {
    transition: none;
  }
}

/* Menu pastels : 7 pastilles rondes cliquables + bouton Supprimer separe. */
.mvs-hl-pastels-menu {
  position: fixed;
  z-index: 10001;
  background: var(--ds-bg-100);
  border: 1px solid var(--m-line);
  border-radius: 12px;
  box-shadow: 0 10px 28px rgba(0, 0, 0, .2);
  padding: 12px;
}
.mvs-hl-pastels-title {
  font-size: 12px;
  font-weight: 600;
  text-transform: uppercase;
  letter-spacing: .04em;
  color: var(--m-ink-3);
  margin: 0 0 8px 2px;
}
.mvs-hl-pastels-grid {
  display: flex;
  gap: 10px;
}
.mvs-hl-pastille {
  width: 30px;
  height: 30px;
  border-radius: 50%;
  border: 2px solid rgba(0, 0, 0, .08);
  padding: 0;
  cursor: pointer;
  transition: transform .1s ease, box-shadow .12s ease;
}
.mvs-hl-pastille:hover {
  transform: scale(1.12);
  box-shadow: 0 2px 8px rgba(0, 0, 0, .18);
}
.mvs-hl-pastille:focus-visible {
  outline: 2px solid var(--m-purple);
  outline-offset: 2px;
}
.mvs-hl-pastille--rouge  { background-color: #fee2e2; }
.mvs-hl-pastille--bleu   { background-color: #dbeafe; }
.mvs-hl-pastille--vert   { background-color: #dcfce7; }
.mvs-hl-pastille--jaune  { background-color: #fef9c3; }
.mvs-hl-pastille--orange { background-color: #ffedd5; }
.mvs-hl-pastille--rose   { background-color: #fce7f3; }
.mvs-hl-pastille--noir   { background-color: #e5e7eb; }
.mvs-hl-pastels-sep {
  height: 1px;
  background: var(--m-line);
  margin: 12px 0;
}
.mvs-hl-pastels-delete {
  display: flex;
  align-items: center;
  gap: 8px;
  width: 100%;
  background: transparent;
  border: 0;
  border-radius: 6px;
  padding: 8px 10px;
  text-align: left;
  font-size: 14px;
  color: var(--ds-red-900);
  cursor: pointer;
}
.mvs-hl-pastels-delete:hover { background: var(--m-danger-soft); }
/* Soulignage ondule sur une annotation (style correcteur), couleur de
   l'annotation. Pas de fond colore : seul l'ondule signale l'annotation.
   Si l'annotation est posee sur un surlignage existant, les spans sont
   imbriques et le fond du surlignage reste visible sous l'ondule. */
.mvs-hl.mvs-has-annot {
  text-decoration-line: underline;
  text-decoration-style: wavy;
  text-decoration-thickness: 1.5px;
  text-underline-offset: 3px;
}
/* Ondules : teintes saturees, max contraste mutuel */
.mvs-has-annot-rouge  { text-decoration-color: #dc2626; }
.mvs-has-annot-bleu   { text-decoration-color: #2563eb; }
.mvs-has-annot-vert   { text-decoration-color: #16a34a; }
.mvs-has-annot-jaune  { text-decoration-color: #eab308; }
.mvs-has-annot-orange { text-decoration-color: #ea580c; }
.mvs-has-annot-rose   { text-decoration-color: #ec4899; }
.mvs-has-annot-noir   { text-decoration-color: #1f2937; }

/* Marge droite : bulles d'annotation editables */
.mvs-annot-marge {
  position: absolute;
  top: 0;
  right: -320px;
  width: 300px;
  height: 100%;
  pointer-events: none;
  z-index: 2;
}

.mvs-annot-bubble {
  pointer-events: auto;
  position: absolute;
  left: 0;
  width: 280px;
  background: var(--ds-bg-100);
  border: 1px solid var(--m-line);
  border-left-width: 6px;
  border-radius: 6px;
  box-shadow: 0 1px 3px rgba(0,0,0,0.08);
  padding: 6px 8px 8px 10px;
  font-size: 13px;
  color: var(--m-ink);
  transition: top .15s ease-out, box-shadow .15s, transform .12s;
}
.mvs-annot-bubble:hover,
.mvs-annot-bubble.is-editing {
  box-shadow: 0 4px 12px rgba(0,0,0,0.14);
  transform: translateX(-2px);
}
/* Bordure gauche bulle : meme teinte saturee que l'ondule */
.mvs-annot-bubble-rouge  { border-left-color: #dc2626; }
.mvs-annot-bubble-bleu   { border-left-color: #2563eb; }
.mvs-annot-bubble-vert   { border-left-color: #16a34a; }
.mvs-annot-bubble-jaune  { border-left-color: #eab308; }
.mvs-annot-bubble-orange { border-left-color: #ea580c; }
.mvs-annot-bubble-rose   { border-left-color: #ec4899; }
.mvs-annot-bubble-noir   { border-left-color: #1f2937; }
.mvs-annot-bubble-orphan { border-left-color: var(--m-line); background: var(--m-bg-soft); }

.mvs-annot-bubble-head {
  display: flex;
  align-items: center;
  gap: 6px;
  margin-bottom: 4px;
}
.mvs-annot-bubble-dot {
  width: 10px;
  height: 10px;
  border-radius: 50%;
  flex-shrink: 0;
  border: 1px solid rgba(0,0,0,0.1);
}
.mvs-annot-bubble-quote {
  flex: 1;
  font-size: 11px;
  color: var(--m-ink-3);
  font-style: italic;
  overflow: hidden;
  text-overflow: ellipsis;
  white-space: nowrap;
}
.mvs-annot-bubble-toggle,
.mvs-annot-bubble-del {
  background: transparent;
  border: 0;
  cursor: pointer;
  color: var(--m-ink-3);
  font-size: 14px;
  line-height: 1;
  padding: 2px 6px;
  border-radius: 4px;
  flex-shrink: 0;
}
.mvs-annot-bubble-toggle:hover { background: var(--m-bg-soft); color: var(--m-ink); }
.mvs-annot-bubble-del:hover    { background: var(--m-danger-soft); color: var(--ds-red-900); }

.mvs-annot-bubble-body {
  /* Visible par defaut */
}
.mvs-annot-bubble.is-collapsed .mvs-annot-bubble-body {
  display: none;
}
.mvs-annot-bubble-ta {
  width: 100%;
  min-height: 28px;
  resize: none;
  border: 0;
  outline: 0;
  background: transparent;
  font: inherit;
  color: inherit;
  padding: 2px 0;
  overflow: hidden;
}
.mvs-annot-bubble-ta::placeholder {
  color: var(--m-ink-3);
  font-style: italic;
}
.mvs-annot-bubble.is-saving {
  border-right: 2px solid var(--color-primary);
}
.mvs-annot-bubble.is-error {
  border-color: var(--m-danger);
  background: var(--m-danger-soft);
}

/* Sur ecrans moyens : marge plus etroite collee a droite */
/* v3.4.51 : adaptation du padding-right reserve aux bulles selon palier ecran
   On garde la symetrie cote gauche (sidebar 280px + gap 2rem) cote droit. */
@media (max-width: 1400px) {
  .module-layout {
    padding-right: calc(1.25rem + 260px + 2rem);
  }
  .mvs-annot-marge {
    right: -260px;
    width: 240px;
  }
  .mvs-annot-bubble {
    width: 220px;
  }
}
@media (max-width: 1100px) {
  /* Pas la place a l'exterieur : on superpose en transparence sur la droite du container.
     On retire la reserve d'espace droite pour redonner toute la place au texte. */
  .module-layout {
    padding-right: 1.25rem;
  }
  .mvs-annot-marge {
    right: 0;
    width: 220px;
    background: rgba(255,255,255,0.85);
    backdrop-filter: blur(2px);
    border-left: 1px solid var(--m-line);
  }
  .mvs-annot-bubble {
    width: 200px;
  }
}
@media (max-width: 720px) {
  /* Sur mobile, on masque les bulles (les surlignages restent visibles) */
  .mvs-annot-marge {
    display: none;
  }
}

/* Menu contextuel (clic droit) */
.mvs-annot-ctxmenu {
  z-index: 10000;
  background: var(--ds-bg-100);
  border: 1px solid var(--m-line);
  border-radius: 8px;
  box-shadow: 0 8px 24px rgba(0,0,0,0.18);
  padding: 6px;
  min-width: 220px;
  font-size: 14px;
  color: var(--m-ink);
}
.mvs-annot-ctxmenu-row {
  display: flex;
  align-items: center;
  width: 100%;
  background: transparent;
  border: 0;
  text-align: left;
  padding: 7px 10px;
  border-radius: 5px;
  cursor: pointer;
  font-size: 14px;
  color: inherit;
  gap: 8px;
}
.mvs-annot-ctxmenu-row:hover {
  background: var(--m-bg-soft);
}
.mvs-annot-ctxmenu-row.is-danger {
  color: var(--ds-red-900);
}
.mvs-annot-ctxmenu-row.is-danger:hover {
  background: var(--m-danger-soft);
}
.mvs-annot-ctxmenu-sep {
  height: 1px;
  background: var(--m-line);
  margin: 4px 0;
}
/* Mini-confirmation inline (suppression d'annotation) */
.mvs-confirm-inline {
  z-index: 10000;
  display: inline-flex;
  align-items: center;
  gap: 8px;
  background: var(--ds-bg-100);
  border: 1px solid var(--m-line);
  border-radius: 8px;
  box-shadow: 0 8px 24px rgba(0,0,0,0.18);
  padding: 8px 10px;
  font-size: 13px;
  color: var(--m-ink);
  max-width: 280px;
}
.mvs-confirm-inline-msg {
  flex: 1 1 auto;
  white-space: nowrap;
}
.mvs-confirm-inline-yes,
.mvs-confirm-inline-no {
  flex: 0 0 auto;
  border: 0;
  border-radius: 5px;
  padding: 5px 10px;
  font-size: 13px;
  font-weight: 600;
  cursor: pointer;
  line-height: 1.2;
}
.mvs-confirm-inline-yes {
  background: var(--m-danger);
  color: var(--ds-text-on-accent);
}
.mvs-confirm-inline-yes:hover {
  background: var(--ds-red-900);
}
.mvs-confirm-inline-no {
  background: var(--m-bg-soft);
  color: var(--m-ink);
}
.mvs-confirm-inline-no:hover {
  background: var(--m-bg-warm);
}
.mvs-annot-swatch {
  display: inline-block;
  width: 16px;
  height: 16px;
  border-radius: 4px;
  border: 1px solid rgba(0,0,0,0.1);
}
/* Pastilles du menu contextuel : teintes saturees pour bien distinguer
   les choix dans la liste. */
.mvs-annot-swatch-rouge  { background-color: #dc2626; }
.mvs-annot-swatch-bleu   { background-color: #2563eb; }
.mvs-annot-swatch-vert   { background-color: #16a34a; }
.mvs-annot-swatch-jaune  { background-color: #eab308; }
.mvs-annot-swatch-orange { background-color: #ea580c; }
.mvs-annot-swatch-rose   { background-color: #ec4899; }
.mvs-annot-swatch-noir   { background-color: #1f2937; }
.mvs-annot-swatch-aucun {
  background-color: #ffffff;
  background-image: linear-gradient(135deg, transparent 0%, transparent 45%, #dc2626 45%, #dc2626 55%, transparent 55%, transparent 100%);
  border: 1px dashed #9ca3af;
}

/* Drag manuel des bulles (v3.4.42) */
.mvs-annot-bubble .mvs-annot-bubble-head {
  user-select: none;
  cursor: grab;
}
.mvs-annot-bubble.is-dragging {
  box-shadow: 0 6px 18px rgba(0,0,0,0.18);
  opacity: 0.95;
  z-index: 50;
}
.mvs-annot-bubble.is-dragging .mvs-annot-bubble-head {
  cursor: grabbing;
}

/* ============================================================
   v3.4.52 - QCM intermediaires dans les sections (theorie)
   Cliquables, correction locale instantanee.
   ============================================================ */
.exo-inter.exo-format-qcm {
  border-left: 4px solid var(--m-purple, #6366f1);
}
.qcm-questions {
  display: flex;
  flex-direction: column;
  gap: .6rem;
}
.qcm-question {
  padding: .6rem .75rem;
  background: var(--bg-soft, rgba(0,0,0,.025));
  border-radius: 8px;
  border: 1px solid var(--border, rgba(0,0,0,.08));
}
.qcm-enonce {
  color: var(--m-ink, #1f2937);
  font-size: 1rem;
  font-weight: 600;
  line-height: 1.4;
  margin: .75rem 0 .5rem;
}
.qcm-option {
  display: flex;
  align-items: flex-start;
  gap: .55rem;
  padding: .45rem .6rem;
  margin: .2rem 0;
  border: 1px solid var(--border, rgba(0,0,0,.1));
  border-radius: 6px;
  background: var(--ds-bg-100);
  cursor: pointer;
  transition: background .12s, border-color .12s;
  font-size: .9rem;
}
.qcm-option:hover {
  background: var(--bg-soft, rgba(0,0,0,.04));
  border-color: var(--m-purple, #6366f1);
}
.qcm-option input[type="radio"] {
  margin-top: .15rem;
  flex-shrink: 0;
  cursor: pointer;
}
.qcm-option input[type="radio"]:disabled {
  cursor: default;
}
.qcm-option input[type="radio"]:disabled + .qcm-letter {
  opacity: .8;
}
.qcm-letter {
  font-weight: 700;
  color: var(--m-purple, #6366f1);
  flex-shrink: 0;
  width: 1.2rem;
}
.qcm-text {
  color: var(--m-ink, #1f2937);
  line-height: 1.35;
}
/* Correction : bonne reponse */
.qcm-option.is-correct {
  background: color-mix(in srgb, var(--m-success) 8%, transparent);
  border-color: var(--color-success, #16a34a);
}
.qcm-option.is-correct .qcm-letter {
  color: var(--color-success, #16a34a);
}
/* Correction : reponse choisie mauvaise */
.qcm-option.is-wrong {
  background: color-mix(in srgb, var(--m-danger) 8%, transparent);
  border-color: var(--color-danger);
}
.qcm-option.is-wrong .qcm-letter {
  color: var(--color-danger);
}
/* Marque visuelle pour la reponse selectionnee */
.qcm-option.is-chosen {
  font-weight: 500;
}
.qcm-option.is-chosen.is-correct::after {
  content: "Votre reponse - correcte";
  font-size: .7rem;
  margin-left: auto;
  color: var(--color-success, #16a34a);
  font-weight: 600;
}
.qcm-option.is-chosen.is-wrong::after {
  content: "Votre reponse";
  font-size: .7rem;
  margin-left: auto;
  color: var(--color-danger);
  font-weight: 600;
}
